use std::ffi::OsString; use std::fs; use std::fs::{File, OpenOptions}; use std::io::BufReader; use std::path::{Path, PathBuf}; use std::time::SystemTime; #[cfg(feature = "heif")] use anyhow::Result; use directories_next::ProjectDirs; use image::{DynamicImage, ImageBuffer, Rgb}; use imagepipe::{ImageSource, Pipeline}; #[cfg(feature = "heif")] use libheif_rs::{Channel, ColorSpace, HeifContext, RgbChroma}; static NUMBER_OF_THREADS: state::Storage = state::Storage::new(); pub fn get_number_of_threads() -> usize { let data = NUMBER_OF_THREADS.get(); if *data >= 1 { *data } else { num_cpus::get() } } pub fn set_default_number_of_threads() { set_number_of_threads(num_cpus::get()); } #[must_use] pub fn get_default_number_of_threads() -> usize { num_cpus::get() } pub fn set_number_of_threads(thread_number: usize) { NUMBER_OF_THREADS.set(thread_number); rayon::ThreadPoolBuilder::new().num_threads(get_number_of_threads()).build_global().unwrap(); } /// Class for common functions used across other class/functions pub const RAW_IMAGE_EXTENSIONS: &[&str] = &[ ".mrw", ".arw", ".srf", ".sr2", ".mef", ".orf", ".srw", ".erf", ".kdc", ".kdc", ".dcs", ".rw2", ".raf", ".dcr", ".dng", ".pef", ".crw", ".iiq", ".3fr", ".nrw", ".nef", ".mos", ".cr2", ".ari", ]; pub const IMAGE_RS_EXTENSIONS: &[&str] = &[ ".jpg", ".jpeg", ".png", ".bmp", ".tiff", ".tif", ".tga", ".ff", ".jif", ".jfi", ".webp", ".gif", ".ico", ".exr", ]; pub const IMAGE_RS_SIMILAR_IMAGES_EXTENSIONS: &[&str] = &[".jpg", ".jpeg", ".png", ".tiff", ".tif", ".tga", ".ff", ".jif", ".jfi", ".bmp", ".webp", ".exr"]; pub const IMAGE_RS_BROKEN_FILES_EXTENSIONS: &[&str] = &[ ".jpg", ".jpeg", ".png", ".tiff", ".tif", ".tga", ".ff", ".jif", ".jfi", ".gif", ".bmp", ".ico", ".jfif", ".jpe", ".pnz", ".dib", ".webp", ".exr", ]; pub const HEIC_EXTENSIONS: &[&str] = &[".heif", ".heifs", ".heic", ".heics", ".avci", ".avcs", ".avif", ".avifs"]; pub const ZIP_FILES_EXTENSIONS: &[&str] = &[".zip"]; pub const PDF_FILES_EXTENSIONS: &[&str] = &[".pdf"]; pub const AUDIO_FILES_EXTENSIONS: &[&str] = &[ ".mp3", ".flac", ".wav", ".ogg", ".m4a", ".aac", ".aiff", ".pcm", ".aif", ".aiff", ".aifc", ".m3a", ".mp2", ".mp4a", ".mp2a", ".mpga", ".wave", ".weba", ".wma", ".oga", ]; pub const VIDEO_FILES_EXTENSIONS: &[&str] = &[ ".mp4", ".mpv", ".flv", ".mp4a", ".webm", ".mpg", ".mp2", ".mpeg", ".m4p", ".m4v", ".avi", ".wmv", ".qt", ".mov", ".swf", ".mkv", ]; pub const LOOP_DURATION: u32 = 200; //ms pub struct Common(); pub fn open_cache_folder(cache_file_name: &str, save_to_cache: bool, use_json: bool, warnings: &mut Vec) -> Option<((Option, PathBuf), (Option, PathBuf))> { if let Some(proj_dirs) = ProjectDirs::from("pl", "Qarmin", "Czkawka") { let cache_dir = PathBuf::from(proj_dirs.cache_dir()); let cache_file = cache_dir.join(cache_file_name); let cache_file_json = cache_dir.join(cache_file_name.replace(".bin", ".json")); let mut file_handler_default = None; let mut file_handler_json = None; if save_to_cache { if cache_dir.exists() { if !cache_dir.is_dir() { warnings.push(format!("Config dir {} is a file!", cache_dir.display())); return None; } } else if let Err(e) = fs::create_dir_all(&cache_dir) { warnings.push(format!("Cannot create config dir {}, reason {}", cache_dir.display(), e)); return None; } file_handler_default = Some(match OpenOptions::new().truncate(true).write(true).create(true).open(&cache_file) { Ok(t) => t, Err(e) => { warnings.push(format!("Cannot create or open cache file {}, reason {}", cache_file.display(), e)); return None; } }); if use_json { file_handler_json = Some(match OpenOptions::new().truncate(true).write(true).create(true).open(&cache_file_json) { Ok(t) => t, Err(e) => { warnings.push(format!("Cannot create or open cache file {}, reason {}", cache_file_json.display(), e)); return None; } }); } } else { if let Ok(t) = OpenOptions::new().read(true).open(&cache_file) { file_handler_default = Some(t); } else { if use_json { file_handler_json = Some(match OpenOptions::new().read(true).open(&cache_file_json) { Ok(t) => t, Err(_) => return None, }); } else { // messages.push(format!("Cannot find or open cache file {}", cache_file.display())); // No error or warning return None; } } }; return Some(((file_handler_default, cache_file), (file_handler_json, cache_file_json))); } None } #[cfg(feature = "heif")] pub fn get_dynamic_image_from_heic(path: &str) -> Result { let im = HeifContext::read_from_file(path)?; let handle = im.primary_image_handle()?; let image = handle.decode(ColorSpace::Rgb(RgbChroma::Rgb), false)?; let width = image.width(Channel::Interleaved).map_err(|e| anyhow::anyhow!("{}", e))?; let height = image.height(Channel::Interleaved).map_err(|e| anyhow::anyhow!("{}", e))?; let planes = image.planes(); let interleaved_plane = planes.interleaved.unwrap(); ImageBuffer::from_raw(width, height, interleaved_plane.data.to_owned()) .map(DynamicImage::ImageRgb8) .ok_or_else(|| anyhow::anyhow!("Failed to create image buffer")) } pub fn get_dynamic_image_from_raw_image(path: impl AsRef + std::fmt::Debug) -> Option { let file_handler = match OpenOptions::new().read(true).open(&path) { Ok(t) => t, Err(_e) => { return None; } }; let mut reader = BufReader::new(file_handler); let raw = match rawloader::decode(&mut reader) { Ok(raw) => raw, Err(_e) => { return None; } }; let source = ImageSource::Raw(raw); let mut pipeline = match Pipeline::new_from_source(source) { Ok(pipeline) => pipeline, Err(_e) => { return None; } }; pipeline.run(None); let image = match pipeline.output_8bit(None) { Ok(image) => image, Err(_e) => { return None; } }; let image = match ImageBuffer::, Vec>::from_raw(image.width as u32, image.height as u32, image.data) { Some(image) => image, None => { return None; } }; // println!("Properly hashed {:?}", path); Some(DynamicImage::ImageRgb8(image)) } #[must_use] pub fn split_path(path: &Path) -> (String, String) { match (path.parent(), path.file_name()) { (Some(dir), Some(file)) => (dir.display().to_string(), file.to_string_lossy().into_owned()), (Some(dir), None) => (dir.display().to_string(), String::new()), (None, _) => (String::new(), String::new()), } } #[must_use] pub fn create_crash_message(library_name: &str, file_path: &str, home_library_url: &str) -> String { format!("{library_name} library crashed when opening \"{file_path}\", please check if this is fixed with the latest version of {library_name} (e.g. with https://github.com/qarmin/crates_tester) and if it is not fixed, please report bug here - {home_library_url}") } impl Common { /// Printing time which took between start and stop point and prints also function name #[allow(unused_variables)] pub fn print_time(start_time: SystemTime, end_time: SystemTime, function_name: &str) { #[cfg(debug_assertions)] println!( "Execution of function \"{}\" took {:?}", function_name, end_time.duration_since(start_time).expect("Time cannot go reverse.") ); } #[must_use] pub fn delete_multiple_entries(entries: &[String]) -> Vec { let mut path: &Path; let mut warnings: Vec = Vec::new(); for entry in entries { path = Path::new(entry); if path.is_dir() { if let Err(e) = fs::remove_dir_all(entry) { warnings.push(format!("Failed to remove folder {entry}, reason {e}")); } } else if let Err(e) = fs::remove_file(entry) { warnings.push(format!("Failed to remove file {entry}, reason {e}")); } } warnings } #[must_use] pub fn delete_one_entry(entry: &str) -> String { let path: &Path = Path::new(entry); let mut warning: String = String::new(); if path.is_dir() { if let Err(e) = fs::remove_dir_all(entry) { warning = format!("Failed to remove folder {entry}, reason {e}"); } } else if let Err(e) = fs::remove_file(entry) { warning = format!("Failed to remove file {entry}, reason {e}"); } warning } /// Function to check if directory match expression pub fn regex_check(expression: &str, directory: impl AsRef) -> bool { if expression == "*" { return true; } let temp_splits: Vec<&str> = expression.split('*').collect(); let mut splits: Vec<&str> = Vec::new(); for i in temp_splits { if !i.is_empty() { splits.push(i); } } if splits.is_empty() { return false; } // Get rid of non unicode characters let directory = directory.as_ref().to_string_lossy(); // Early checking if directory contains all parts needed by expression for split in &splits { if !directory.contains(split) { return false; } } let mut position_of_splits: Vec = Vec::new(); // `git*` shouldn't be true for `/gitsfafasfs` if !expression.starts_with('*') && directory.find(splits[0]).unwrap() > 0 { return false; } // `*home` shouldn't be true for `/homeowner` if !expression.ends_with('*') && !directory.ends_with(splits.last().unwrap()) { return false; } // At the end we check if parts between * are correctly positioned position_of_splits.push(directory.find(splits[0]).unwrap()); let mut current_index: usize; let mut found_index: usize; for i in splits[1..].iter().enumerate() { current_index = *position_of_splits.get(i.0).unwrap() + i.1.len(); found_index = match directory[current_index..].find(i.1) { Some(t) => t, None => return false, }; position_of_splits.push(found_index + current_index); } true } pub fn normalize_windows_path(path_to_change: impl AsRef) -> PathBuf { let path = path_to_change.as_ref(); // Don't do anything, because network path may be case intensive if path.to_string_lossy().starts_with('\\') { return path.to_path_buf(); } match path.to_str() { Some(path) if path.is_char_boundary(1) => { let replaced = path.replace('/', "\\"); let mut new_path = OsString::new(); if replaced[1..].starts_with(':') { new_path.push(replaced[..1].to_ascii_uppercase()); new_path.push(replaced[1..].to_ascii_lowercase()); } else { new_path.push(replaced.to_ascii_lowercase()); } PathBuf::from(new_path) } _ => path.to_path_buf(), } } } #[cfg(test)] mod test { use std::path::PathBuf; use crate::common::Common; #[test] fn test_regex() { assert!(Common::regex_check("*home*", "/home/rafal")); assert!(Common::regex_check("*home", "/home")); assert!(Common::regex_check("*home/", "/home/")); assert!(Common::regex_check("*home/*", "/home/")); assert!(Common::regex_check("*.git*", "/home/.git")); assert!(Common::regex_check("*/home/rafal*rafal*rafal*rafal*", "/home/rafal/rafalrafalrafal")); assert!(Common::regex_check("AAA", "AAA")); assert!(Common::regex_check("AAA*", "AAABDGG/QQPW*")); assert!(!Common::regex_check("*home", "/home/")); assert!(!Common::regex_check("*home", "/homefasfasfasfasf/")); assert!(!Common::regex_check("*home", "/homefasfasfasfasf")); assert!(!Common::regex_check("rafal*afal*fal", "rafal")); assert!(!Common::regex_check("rafal*a", "rafal")); assert!(!Common::regex_check("AAAAAAAA****", "/AAAAAAAAAAAAAAAAA")); assert!(!Common::regex_check("*.git/*", "/home/.git")); assert!(!Common::regex_check("*home/*koc", "/koc/home/")); assert!(!Common::regex_check("*home/", "/home")); assert!(!Common::regex_check("*TTT", "/GGG")); #[cfg(target_family = "windows")] { assert!(Common::regex_check("*\\home", "C:\\home")); assert!(Common::regex_check("*/home", "C:\\home")); } } #[test] fn test_windows_path() { assert_eq!(PathBuf::from("C:\\path.txt"), Common::normalize_windows_path("c:/PATH.tXt")); assert_eq!(PathBuf::from("H:\\reka\\weza\\roman.txt"), Common::normalize_windows_path("h:/RekA/Weza\\roMan.Txt")); assert_eq!(PathBuf::from("T:\\a"), Common::normalize_windows_path("T:\\A")); assert_eq!(PathBuf::from("\\\\aBBa"), Common::normalize_windows_path("\\\\aBBa")); assert_eq!(PathBuf::from("a"), Common::normalize_windows_path("a")); assert_eq!(PathBuf::from(""), Common::normalize_windows_path("")); } }