From 820fcf5c12fe69e41f6fc480f0fbcb1d66aeeac2 Mon Sep 17 00:00:00 2001 From: thevickypedia Date: Fri, 15 Mar 2024 11:10:07 -0500 Subject: [PATCH] Allow users to delete file/folder with right click Create a dedicated crate to handle deletions Renames to be implemented --- src/lib.rs | 1 + src/routes/fileIO.rs | 133 +++++++++++++++++++++++++++++++++++++++ src/routes/media.rs | 7 ++- src/routes/mod.rs | 6 +- src/squire/ascii_art.rs | 4 +- src/templates/listing.rs | 114 ++++++++++++++++++++++++++++++++- 6 files changed, 257 insertions(+), 8 deletions(-) create mode 100644 src/routes/fileIO.rs diff --git a/src/lib.rs b/src/lib.rs index 398d2b9..fb137ec 100644 --- a/src/lib.rs +++ b/src/lib.rs @@ -85,6 +85,7 @@ pub async fn start() -> io::Result<()> { .service(routes::auth::logout) .service(routes::auth::home) .service(routes::basics::profile) + .service(routes::fileio::edit) .service(routes::auth::error) .service(routes::media::track) .service(routes::media::stream) diff --git a/src/routes/fileIO.rs b/src/routes/fileIO.rs new file mode 100644 index 0000000..fb84a97 --- /dev/null +++ b/src/routes/fileIO.rs @@ -0,0 +1,133 @@ +use std::fs::{remove_dir_all, remove_file}; + +use std::path::{Path, PathBuf}; +use std::sync::Arc; + +use actix_web::{HttpRequest, HttpResponse, web}; +use fernet::Fernet; +use serde::Deserialize; + +use crate::{constant, routes, squire}; + +/// Struct to represent the payload data with both the URL locator and path locator +#[derive(Debug, Deserialize)] +struct Payload { + url_locator: Option, + path_locator: Option, +} + +/// Extracts the path the file/directory that has to be deleted from the payload received. +/// +/// # Arguments +/// +/// * `payload` - Payload received from the UI as JSON body. +/// * `media_source` - Media source configured for the server. +/// +/// # Returns +/// +/// Returns a result object to describe the status of the extraction. +/// +/// * `Ok(PathBuf)` - If the extraction was successful and the path exists in the server. +/// * `Err(String)` - If the extraction has failed or if the path doesn't exist in the server. +fn extract_media_path(payload: web::Json, media_source: &Path) -> Result { + let url_locator = payload.url_locator.as_deref(); + let path_locator = payload.path_locator.as_deref(); + if let (Some(url_str), Some(path_str)) = (url_locator, path_locator) { + // Create a collection since a tuple is a fixed-size collection in rust and doesn't allow iteration + for locator in &[url_str, path_str] { + if let Some(media_path) = locator.split("stream").nth(1) { + // Without stripping the '/' in front of the path, Rust will assume that's a root path + // This will overwrite media_source and render the joined path instead of combining the two + let path = media_source.join(media_path.strip_prefix('/').unwrap()); + if path.exists() { + log::debug!("Extracted from '{}'", locator); + return Ok(path); + } + } + } + return Err(String::from("Unable to extract path from either of the parameters")); + } + Err(String::from("Both URL locator and path locator must be provided")) +} + +/// Handles requests for the `/edit` endpoint, to delete/rename media files and directories. +/// +/// # Arguments +/// +/// * `request` - A reference to the Actix web `HttpRequest` object. +/// * `payload` - JSON payload with `url_path` and `true_path` received from the UI. +/// * `fernet` - Fernet object to encrypt the auth payload that will be set as `session_token` cookie. +/// * `session` - Session struct that holds the `session_mapping` and `session_tracker` to handle sessions. +/// * `metadata` - Struct containing metadata of the application. +/// * `config` - Configuration data for the application. +/// * `template` - Configuration container for the loaded templates. +/// +/// # Returns +/// +/// * `200` - HttpResponse with a `session_token` and redirect URL to the `/home` entrypoint. +/// * `400` - HttpResponse with an error message for invalid action or incorrect payload. +/// * `401` - HttpResponse with an error message for failed authentication. +#[post("/edit")] +pub async fn edit(request: HttpRequest, + payload: web::Json, + fernet: web::Data>, + session: web::Data>, + metadata: web::Data>, + config: web::Data>, + template: web::Data>>) -> HttpResponse { + let auth_response = squire::authenticator::verify_token(&request, &config, &fernet, &session); + if !auth_response.ok { + return routes::auth::failed_auth(auth_response, &config); + } + let (_host, _last_accessed) = squire::logger::log_connection(&request, &session); + log::debug!("{}", auth_response.detail); + let extracted = extract_media_path(payload, &config.media_source); + // todo: pop up doesn't always occur next to the mouse + // styling of the pop up is very basic + // make custom error responses generic + let media_path: PathBuf = match extracted { + Ok(path) => { + path + }, + Err(msg) => { + return HttpResponse::BadRequest().body(msg); + } + }; + if !squire::authenticator::verify_secure_index(&PathBuf::from(&media_path), &auth_response.username) { + return squire::responses::restricted( + template.get_template("error").unwrap(), + &auth_response.username, + &metadata.pkg_version, + ); + } + if let Some(edit_action) = request.headers().get("edit-action") { + let action = edit_action.to_str().unwrap(); + return if action == "delete" { + log::info!("{} requested to delete {:?}", &auth_response.username, &media_path); + if media_path.is_file() { + if let Err(error) = remove_file(media_path) { + let reason = format!("Error deleting file: {}", error); + HttpResponse::InternalServerError().body(reason) + } else { + HttpResponse::Ok().finish() + } + } else if media_path.is_dir() { + if let Err(error) = remove_dir_all(media_path) { + let reason = format!("Error deleting directory: {}", error); + HttpResponse::InternalServerError().body(reason) + } else { + HttpResponse::Ok().finish() + } + } else { + let reason = format!("{:?} was neither a file nor a directory", media_path); + log::warn!("{}", reason); + HttpResponse::BadRequest().body(reason) + } + } else { + log::warn!("Unsupported action: {} requested to {} {:?}", &auth_response.username, action, &media_path); + HttpResponse::BadRequest().body("Unsupported action!") + }; + } + log::warn!("No action received for: {:?}", media_path); + HttpResponse::BadRequest().body("No action received!") +} diff --git a/src/routes/media.rs b/src/routes/media.rs index 419f96d..9e0d63e 100644 --- a/src/routes/media.rs +++ b/src/routes/media.rs @@ -35,7 +35,7 @@ struct Subtitles { /// * `path` - The input path string to be URL encoded. /// /// ## References -/// - [RustJobs](https://rustjobs.dev/blog/how-to-url-encode-strings-in-rust/) +/// - [rustjobs.dev](https://rustjobs.dev/blog/how-to-url-encode-strings-in-rust/) /// /// # Returns /// @@ -174,6 +174,8 @@ pub async fn stream(request: HttpRequest, &metadata.pkg_version ); } + let secure_path = if filepath.contains(constant::SECURE_INDEX) { "true" } else { "false" }; + let secure_flag = secure_path.to_string(); // True path of the media file let __target = config.media_source.join(&filepath); if !__target.exists() { @@ -251,7 +253,8 @@ pub async fn stream(request: HttpRequest, user => auth_response.username, secure_index => constant::SECURE_INDEX, directories => listing_page.directories, - secured_directories => listing_page.secured_directories + secured_directories => listing_page.secured_directories, + secure_path => &secure_flag )).unwrap()); } log::error!("Something went horribly wrong"); diff --git a/src/routes/mod.rs b/src/routes/mod.rs index ce44db9..f843a9b 100644 --- a/src/routes/mod.rs +++ b/src/routes/mod.rs @@ -1,8 +1,10 @@ -/// Module for the primary and health check entry points. +/// Module for `/`, `/health` and `/profile` entrypoints. pub mod basics; /// Module for all the rendering based entry points. pub mod media; /// Module for `/home`, `/login`, `/logout` and `/error` entrypoints. pub mod auth; -/// Module to handle upload entrypoint. +/// Module for `/upload` entrypoint that handles the file uploads. pub mod upload; +/// Module for `/edit` entrypoint that handles delete/rename actions. +pub mod fileio; diff --git a/src/squire/ascii_art.rs b/src/squire/ascii_art.rs index 45d7daa..cb5a60f 100644 --- a/src/squire/ascii_art.rs +++ b/src/squire/ascii_art.rs @@ -3,8 +3,8 @@ use rand::prelude::SliceRandom; /// Prints random ASCII art of a horse, dog or a dolphin. /// /// ## References -/// - [https://www.asciiart.eu](https://www.asciiart.eu) -/// - [https://asciiart.cc](https://asciiart.cc) +/// - [asciiart.eu](https://www.asciiart.eu) +/// - [asciiart.cc](https://asciiart.cc) pub fn random() { let horse = r" # # diff --git a/src/templates/listing.rs b/src/templates/listing.rs index 556e0cb..1bf303b 100644 --- a/src/templates/listing.rs +++ b/src/templates/listing.rs @@ -129,6 +129,26 @@ pub fn get_content() -> String { margin-right: 0.5rem; } +