use crate::paste::PasteType; use actix_web::{error, Error as ActixError}; use glob::glob; use lazy_regex::{lazy_regex, Lazy, Regex}; use ring::digest::{Context, SHA256}; use std::io::{BufReader, Read}; use std::path::{Path, PathBuf}; use std::time::Duration; use std::time::{SystemTime, UNIX_EPOCH}; /// Regex for matching the timestamp extension of a path. pub static TIMESTAMP_EXTENSION_REGEX: Lazy = lazy_regex!(r#"\.[0-9]{10,}$"#); /// Returns the system time as [`Duration`](Duration). pub fn get_system_time() -> Result { SystemTime::now() .duration_since(UNIX_EPOCH) .map_err(error::ErrorInternalServerError) } /// Returns the first _unexpired_ path matched by a custom glob pattern. /// /// The file extension is accepted as a timestamp that points to the expiry date. pub fn glob_match_file(mut path: PathBuf) -> Result { path = PathBuf::from( TIMESTAMP_EXTENSION_REGEX .replacen( path.to_str().ok_or_else(|| { error::ErrorInternalServerError("path contains invalid characters") })?, 1, "", ) .to_string(), ); if let Some(glob_path) = glob(&format!("{}.[0-9]*", path.to_string_lossy())) .map_err(error::ErrorInternalServerError)? .next() { let glob_path = glob_path.map_err(error::ErrorInternalServerError)?; if let Some(extension) = glob_path .extension() .and_then(|v| v.to_str()) .and_then(|v| v.parse().ok()) { if get_system_time()? < Duration::from_millis(extension) { path = glob_path; } } } Ok(path) } /// Returns the found expired files in the possible upload locations. /// /// Fail-safe, omits errors. pub fn get_expired_files(base_path: &Path) -> Vec { [ PasteType::File, PasteType::Oneshot, PasteType::Url, PasteType::OneshotUrl, ] .into_iter() .filter_map(|v| glob(&v.get_path(base_path).join("*.[0-9]*").to_string_lossy()).ok()) .flat_map(|glob| glob.filter_map(|v| v.ok()).collect::>()) .filter(|path| { if let Some(extension) = path .extension() .and_then(|v| v.to_str()) .and_then(|v| v.parse().ok()) { get_system_time() .map(|system_time| system_time > Duration::from_millis(extension)) .unwrap_or(false) } else { false } }) .collect() } /// Returns the SHA256 digest of the given input. pub fn sha256_digest(input: R) -> Result { let mut reader = BufReader::new(input); let mut context = Context::new(&SHA256); let mut buffer = [0; 1024]; loop { let bytes_read = reader.read(&mut buffer)?; if bytes_read != 0 { context.update(&buffer[..bytes_read]); } else { break; } } Ok(context .finish() .as_ref() .iter() .collect::>() .iter() .map(|byte| format!("{byte:02x}")) .collect::()) } #[cfg(test)] mod tests { use super::*; use std::env; use std::fs; use std::thread; #[test] fn test_system_time() -> Result<(), ActixError> { let system_time = get_system_time()?.as_millis(); thread::sleep(Duration::from_millis(1)); assert!(system_time < get_system_time()?.as_millis()); Ok(()) } #[test] fn test_glob_match() -> Result<(), ActixError> { let path = PathBuf::from(format!( "expired.file1.{}", get_system_time()?.as_millis() + 50 )); fs::write(&path, String::new())?; assert_eq!(path, glob_match_file(PathBuf::from("expired.file1"))?); thread::sleep(Duration::from_millis(75)); assert_eq!( PathBuf::from("expired.file1"), glob_match_file(PathBuf::from("expired.file1"))? ); fs::remove_file(path)?; Ok(()) } #[test] fn test_sha256sum() -> Result<(), ActixError> { assert_eq!( "9f86d081884c7d659a2feaa0c55ad015a3bf4f1b2b0b822cd15d6c15b0f00a08", sha256_digest(String::from("test").as_bytes())? ); assert_eq!( "2fc36f72540bb9145e95e67c41dccdc440c95173257032e32e111ebd7b6df960", sha256_digest(env!("CARGO_PKG_NAME").as_bytes())? ); Ok(()) } #[test] fn test_get_expired_files() -> Result<(), ActixError> { let current_dir = env::current_dir()?; let expiration_time = get_system_time()?.as_millis() + 50; let path = PathBuf::from(format!("expired.file2.{expiration_time}")); fs::write(&path, String::new())?; assert_eq!(Vec::::new(), get_expired_files(¤t_dir)); thread::sleep(Duration::from_millis(75)); assert_eq!( vec![current_dir.join(&path)], get_expired_files(¤t_dir) ); fs::remove_file(path)?; assert_eq!(Vec::::new(), get_expired_files(¤t_dir)); Ok(()) } }