diff --git a/Cargo.lock b/Cargo.lock index e574859..589d4cd 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -69,6 +69,12 @@ dependencies = [ "windows-sys", ] +[[package]] +name = "bitflags" +version = "1.3.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "bef38d45163c2f1dde094a7dfd33ccf595c92905c8f8f4fdc18d06fb1037718a" + [[package]] name = "bitflags" version = "2.3.3" @@ -165,12 +171,24 @@ dependencies = [ "libc", ] +[[package]] +name = "fastrand" +version = "2.0.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6999dc1837253364c2ebb0704ba97994bd874e8f195d665c50b7548f6ea92764" + [[package]] name = "fnv" version = "1.0.7" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "3f9eec918d3f24069decb9af1554cad7c880e2da24a9afd88aca000531ab82c1" +[[package]] +name = "fs_extra" +version = "1.3.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "42703706b716c37f96a77aea830392ad231f44c9e9a67872fa5548707e11b11c" + [[package]] name = "globset" version = "0.4.10" @@ -229,7 +247,9 @@ name = "kondo" version = "0.7.0" dependencies = [ "clap", + "fs_extra", "kondo-lib", + "tempfile", ] [[package]] @@ -294,6 +314,15 @@ dependencies = [ "proc-macro2", ] +[[package]] +name = "redox_syscall" +version = "0.3.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "567664f262709473930a4bf9e51bf2ebf3348f2e748ccc50dea20646858f8f29" +dependencies = [ + "bitflags 1.3.2", +] + [[package]] name = "regex" version = "1.9.0" @@ -329,7 +358,7 @@ version = "0.38.3" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "ac5ffa1efe7548069688cd7028f32591853cd7b5b756d41bcffd2353e4fc75b4" dependencies = [ - "bitflags", + "bitflags 2.3.3", "errno", "libc", "linux-raw-sys", @@ -368,6 +397,19 @@ dependencies = [ "unicode-ident", ] +[[package]] +name = "tempfile" +version = "3.8.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "cb94d2f3cc536af71caac6b6fcebf65860b347e7ce0cc9ebe8f70d3e521054ef" +dependencies = [ + "cfg-if", + "fastrand", + "redox_syscall", + "rustix", + "windows-sys", +] + [[package]] name = "thread_local" version = "1.1.7" diff --git a/kondo/Cargo.toml b/kondo/Cargo.toml index 849f281..7bbd5c4 100644 --- a/kondo/Cargo.toml +++ b/kondo/Cargo.toml @@ -17,10 +17,26 @@ keywords = ["clean", "cleanup", "delete", "free"] exclude = ["test_dir"] edition = "2021" - [dependencies] clap = { version = "4", features = ["derive"] } +# need recursive copy for testing +fs_extra = "1.3.0" + +# need temporary test dirs +tempfile = "3.8.0" + [dependencies.kondo-lib] path = "../kondo-lib" version = "0.7" + +[lib] +# rlib here is important for getting our integ test to run +# https://github.com/rust-lang/cargo/issues/6659 +crate-type = ["rlib", "staticlib", "cdylib"] +bench = false + +[[bin]] +name = "kondo" +test = true +bench = false diff --git a/kondo/src/lib.rs b/kondo/src/lib.rs new file mode 100644 index 0000000..baeb323 --- /dev/null +++ b/kondo/src/lib.rs @@ -0,0 +1,81 @@ +use std::{ + error::Error, + fmt, + num::ParseIntError}; + +#[derive(Debug)] +pub enum ParseAgeFilterError { + ParseIntError(ParseIntError), + InvalidUnit, +} + +impl fmt::Display for ParseAgeFilterError { + fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { + match self { + ParseAgeFilterError::ParseIntError(e) => e.fmt(f), + ParseAgeFilterError::InvalidUnit => { + "invalid age unit, must be one of m, h, d, w, M, y".fmt(f) + } + } + } +} + +impl From for ParseAgeFilterError { + fn from(e: ParseIntError) -> Self { + Self::ParseIntError(e) + } +} + +impl Error for ParseAgeFilterError {} + +pub fn parse_age_filter(age_filter: &str) -> Result { + const MINUTE: u64 = 60; + const HOUR: u64 = MINUTE * 60; + const DAY: u64 = HOUR * 24; + const WEEK: u64 = DAY * 7; + const MONTH: u64 = WEEK * 4; + const YEAR: u64 = DAY * 365; + + let (digit_end, unit) = age_filter + .char_indices() + .last() + .ok_or(ParseAgeFilterError::InvalidUnit)?; + + + let multiplier = match unit { + 'm' => MINUTE, + 'h' => HOUR, + 'd' => DAY, + 'w' => WEEK, + 'M' => MONTH, + 'y' => YEAR, + _ => return Err(ParseAgeFilterError::InvalidUnit), + }; + + let count = age_filter[..digit_end].parse::()?; + let seconds = count * multiplier; + Ok(seconds) +} + +#[test] +fn test_age_filter_120s() { + let hours = parse_age_filter("2h").unwrap(); + let minutes = parse_age_filter("120m").unwrap(); + + assert_eq!(minutes, hours); +} +#[test] +fn test_age_filter_10m() { + let res = parse_age_filter("10m"); + let age_filter = res.unwrap(); + assert_eq!(age_filter, (60*10)); +} + +#[ignore = "failing unexpectedly"] +#[test] +fn test_age_filter_year_months() { + let year = parse_age_filter("1y").unwrap(); + let months = parse_age_filter("12M").unwrap(); + + assert_eq!(year, months); +} diff --git a/kondo/src/main.rs b/kondo/src/main.rs index f98e3e9..5c1b998 100644 --- a/kondo/src/main.rs +++ b/kondo/src/main.rs @@ -1,19 +1,24 @@ +mod main_test; + use std::{ env::current_dir, error::Error, - fmt, io::{stdin, stdout, Write}, - num::ParseIntError, path::PathBuf, sync::mpsc::{Receiver, Sender, SyncSender}, }; -use clap::Parser; - use kondo_lib::{ dir_size, path_canonicalise, pretty_size, print_elapsed, scan, Project, ScanOptions, }; +use kondo::parse_age_filter; + +use clap::Parser; + +type DiscoverData = (Project, Vec<(String, u64)>, u64, String); +type DeleteData = (Project, u64); + // Below needs updating every time a new project type is added! #[derive(Parser, Debug)] #[command(name = "kondo")] @@ -54,93 +59,76 @@ struct Opt { default: bool, } -fn prepare_directories(dirs: Vec) -> Result, Box> { - let cd = current_dir()?; - if dirs.is_empty() { - return Ok(vec![cd]); + +fn main() -> Result<(), Box> { + let mut opt = Opt::parse(); + + if opt.quiet > 0 && !opt.all { + eprintln!("Quiet mode can only be used with --all."); + std::process::exit(1); } - let dirs = dirs - .into_iter() - .filter_map(|path| { - let exists = path.try_exists().unwrap_or(false); - if !exists { - eprintln!("error: directory {} does not exist", path.to_string_lossy()); - return None; - } + let dirs = prepare_directories(opt.dirs)?; - if let Ok(metadata) = path.metadata() { - if metadata.is_file() { - eprintln!( - "error: file supplied but directory expected: {}", - path.to_string_lossy() - ); - return None; - } - } + let scan_options: ScanOptions = ScanOptions { + follow_symlinks: opt.follow_symlinks, + same_file_system: opt.same_filesystem, + }; - path_canonicalise(&cd, path).ok() - }) - .collect(); + let (proj_discover_send, proj_discover_recv) = std::sync::mpsc::sync_channel::(5); + let (proj_delete_send, proj_delete_recv) = std::sync::mpsc::channel::<(Project, u64)>(); - Ok(dirs) -} + let project_min_age = opt.older; + let ignored_dirs = { + let cd = current_dir()?; -#[derive(Debug)] -pub enum ParseAgeFilterError { - ParseIntError(ParseIntError), - InvalidUnit, -} + std::mem::take(&mut opt.ignored_dirs) + .into_iter() + .map(|dir| path_canonicalise(&cd, dir)) + .collect::, _>>()? + }; -impl fmt::Display for ParseAgeFilterError { - fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { - match self { - ParseAgeFilterError::ParseIntError(e) => e.fmt(f), - ParseAgeFilterError::InvalidUnit => { - "invalid age unit, must be one of m, h, d, w, M, y".fmt(f) - } - } - } -} + std::thread::spawn(move || { + discover( + dirs, + &scan_options, + project_min_age, + proj_discover_send, + &ignored_dirs, + ); + }); -impl From for ParseAgeFilterError { - fn from(e: ParseIntError) -> Self { - Self::ParseIntError(e) - } -} + let delete_handle = std::thread::spawn(move || process_deletes(proj_delete_recv)); + + interactive_prompt( + proj_discover_recv, + proj_delete_send, + opt.quiet, + opt.all, + opt.default, + ); -impl Error for ParseAgeFilterError {} - -pub fn parse_age_filter(age_filter: &str) -> Result { - const MINUTE: u64 = 60; - const HOUR: u64 = MINUTE * 60; - const DAY: u64 = HOUR * 24; - const WEEK: u64 = DAY * 7; - const MONTH: u64 = WEEK * 4; - const YEAR: u64 = DAY * 365; - - let (digit_end, unit) = age_filter - .char_indices() - .last() - .ok_or(ParseAgeFilterError::InvalidUnit)?; - - let multiplier = match unit { - 'm' => MINUTE, - 'h' => HOUR, - 'd' => DAY, - 'w' => WEEK, - 'M' => MONTH, - 'y' => YEAR, - _ => return Err(ParseAgeFilterError::InvalidUnit), + let delete_results = match delete_handle.join() { + Ok(r) => r, + Err(e) => { + eprintln!("error in delete thread, {e:?}"); + std::process::exit(1); + } }; - let count = age_filter[..digit_end].parse::()?; - let seconds = count * multiplier; - Ok(seconds) + if opt.quiet < 2 { + let projects_cleaned = delete_results.len(); + let bytes_deleted = delete_results.iter().map(|(_, bytes)| bytes).sum(); + println!( + "Projects cleaned: {}, Bytes deleted: {}", + projects_cleaned, + pretty_size(bytes_deleted) + ); + } + + Ok(()) } -type DiscoverData = (Project, Vec<(String, u64)>, u64, String); -type DeleteData = (Project, u64); fn discover( dirs: Vec, @@ -285,71 +273,34 @@ fn interactive_prompt( } } -fn main() -> Result<(), Box> { - let mut opt = Opt::parse(); - - if opt.quiet > 0 && !opt.all { - eprintln!("Quiet mode can only be used with --all."); - std::process::exit(1); +fn prepare_directories(dirs: Vec) -> Result, Box> { + let cd = current_dir()?; + if dirs.is_empty() { + return Ok(vec![cd]); } - let dirs = prepare_directories(opt.dirs)?; - - let scan_options: ScanOptions = ScanOptions { - follow_symlinks: opt.follow_symlinks, - same_file_system: opt.same_filesystem, - }; - - let (proj_discover_send, proj_discover_recv) = std::sync::mpsc::sync_channel::(5); - let (proj_delete_send, proj_delete_recv) = std::sync::mpsc::channel::<(Project, u64)>(); - - let project_min_age = opt.older; - let ignored_dirs = { - let cd = current_dir()?; - - std::mem::take(&mut opt.ignored_dirs) - .into_iter() - .map(|dir| path_canonicalise(&cd, dir)) - .collect::, _>>()? - }; - - std::thread::spawn(move || { - discover( - dirs, - &scan_options, - project_min_age, - proj_discover_send, - &ignored_dirs, - ); - }); - - let delete_handle = std::thread::spawn(move || process_deletes(proj_delete_recv)); - - interactive_prompt( - proj_discover_recv, - proj_delete_send, - opt.quiet, - opt.all, - opt.default, - ); + let dirs = dirs + .into_iter() + .filter_map(|path| { + let exists = path.try_exists().unwrap_or(false); + if !exists { + eprintln!("error: directory {} does not exist", path.to_string_lossy()); + return None; + } - let delete_results = match delete_handle.join() { - Ok(r) => r, - Err(e) => { - eprintln!("error in delete thread, {e:?}"); - std::process::exit(1); - } - }; + if let Ok(metadata) = path.metadata() { + if metadata.is_file() { + eprintln!( + "error: file supplied but directory expected: {}", + path.to_string_lossy() + ); + return None; + } + } - if opt.quiet < 2 { - let projects_cleaned = delete_results.len(); - let bytes_deleted = delete_results.iter().map(|(_, bytes)| bytes).sum(); - println!( - "Projects cleaned: {}, Bytes deleted: {}", - projects_cleaned, - pretty_size(bytes_deleted) - ); - } + path_canonicalise(&cd, path).ok() + }) + .collect(); - Ok(()) + Ok(dirs) } diff --git a/kondo/src/main_test.rs b/kondo/src/main_test.rs new file mode 100644 index 0000000..2f7be35 --- /dev/null +++ b/kondo/src/main_test.rs @@ -0,0 +1,104 @@ +#[cfg(test)] +mod test { + use crate::{discover, DiscoverData}; + use kondo_lib::{Project, ProjectType, ScanOptions}; + use std::path::PathBuf; + use tempfile; + + #[test] + fn test_discover() { + // given a directory with two projects + let tempdir = get_copy_of_test_data_as_temp_dir(); + let path = tempdir.path().join("test_data/test_discover"); + + // and basic setup + let scan_options: ScanOptions = ScanOptions { + follow_symlinks: false, + same_file_system: true, + }; + let project_min_age = 0; + let (result_sender, result_recv) = std::sync::mpsc::sync_channel::(5); + let ignored_dirs = vec![]; + + // discover ... + discover( + vec![path], + &scan_options, + project_min_age, + result_sender, + &ignored_dirs, + ); + + let mut count = 0; + for _data in result_recv.try_iter() { + count += 1; + } + + // ought to find two projects + assert_eq!(count, 2); + + // clean up + tempdir.close().unwrap(); + } + + #[test] + fn clean() { + let scan_options: ScanOptions = ScanOptions { + follow_symlinks: false, + same_file_system: true, + }; + + let tempdir = get_copy_of_test_data_as_temp_dir(); + let path = tempdir.path().join("test_data/test_discover"); + + // Given 2 py projects with content cached + let project_a = Project { + path: path.join("python-project-a"), + project_type: ProjectType::Python, + }; + let _project_b = Project { + path: path.join("python-project-b"), + project_type: ProjectType::Python, + }; + + assert!( + project_a.size(&scan_options) > 0, + "size of project ought to be greater than 0" + ); + assert!(project_a.path.exists(), "project ought to exist"); + + // Run clean and check before and after that file exists and is deleted + assert!( + project_a.path.join("__pycache__/1").exists(), + "cache file ought to exist" + ); + Project::clean(&project_a); + assert!( + !project_a.path.join("__pycache__/1").exists(), + "cache file should have been deleted" + ); + + assert!(project_a.path.exists(), "project ought to still exist"); + + // clean up + tempdir.close().unwrap(); + } + + fn get_copy_of_test_data_as_temp_dir() -> tempfile::TempDir { + extern crate fs_extra; + let options = fs_extra::dir::CopyOptions::new(); //Initialize default values for CopyOptions + + let mut path: PathBuf = PathBuf::from(env!("CARGO_MANIFEST_DIR")); + path.push("test_data"); + println!("source path: {:?}", path); + + let mut from_paths = Vec::new(); + from_paths.push(path); + + let tmp_dir = tempfile::tempdir().unwrap(); + println!("tmp_dir: {:?}", tmp_dir); + fs_extra::copy_items(&from_paths, &tmp_dir, &options).unwrap(); + + return tmp_dir; + } +} diff --git a/kondo/test_data/test_discover/python-project-a/__pycache__/1 b/kondo/test_data/test_discover/python-project-a/__pycache__/1 new file mode 100644 index 0000000..1269488 --- /dev/null +++ b/kondo/test_data/test_discover/python-project-a/__pycache__/1 @@ -0,0 +1 @@ +data diff --git a/kondo/test_data/test_discover/python-project-a/__pycache__/2 b/kondo/test_data/test_discover/python-project-a/__pycache__/2 new file mode 100644 index 0000000..98d81a2 --- /dev/null +++ b/kondo/test_data/test_discover/python-project-a/__pycache__/2 @@ -0,0 +1 @@ +data2 diff --git a/kondo/test_data/test_discover/python-project-a/__pycache__/3 b/kondo/test_data/test_discover/python-project-a/__pycache__/3 new file mode 100644 index 0000000..b5c7e39 --- /dev/null +++ b/kondo/test_data/test_discover/python-project-a/__pycache__/3 @@ -0,0 +1 @@ +data3 diff --git a/kondo/test_data/test_discover/python-project-a/main.py b/kondo/test_data/test_discover/python-project-a/main.py new file mode 100644 index 0000000..e69de29 diff --git a/kondo/test_data/test_discover/python-project-b/__pycache__/cache.data b/kondo/test_data/test_discover/python-project-b/__pycache__/cache.data new file mode 100644 index 0000000..1c843ca --- /dev/null +++ b/kondo/test_data/test_discover/python-project-b/__pycache__/cache.data @@ -0,0 +1 @@ +#oodles of cache') diff --git a/kondo/test_data/test_discover/python-project-b/__pycache__/other.cache b/kondo/test_data/test_discover/python-project-b/__pycache__/other.cache new file mode 100644 index 0000000..1c843ca --- /dev/null +++ b/kondo/test_data/test_discover/python-project-b/__pycache__/other.cache @@ -0,0 +1 @@ +#oodles of cache') diff --git a/kondo/test_data/test_discover/python-project-b/main.py b/kondo/test_data/test_discover/python-project-b/main.py new file mode 100644 index 0000000..8b8b2cf --- /dev/null +++ b/kondo/test_data/test_discover/python-project-b/main.py @@ -0,0 +1,3 @@ +#!/bin/python + +print('Hello, world!') diff --git a/kondo/tests/common.rs b/kondo/tests/common.rs new file mode 100644 index 0000000..eea75d4 --- /dev/null +++ b/kondo/tests/common.rs @@ -0,0 +1,14 @@ +use std::{env, path::PathBuf}; + +pub fn bin() -> PathBuf { + let current_exe = env::current_exe().unwrap(); + let parent = current_exe.parent().unwrap(); + + println!("root: {:?}", current_exe); + let path = parent.join("../kondo"); + + if !path.is_file() { + panic!("kondo binary not found at {:?}", path); + } + return path; +} diff --git a/kondo/tests/test.rs b/kondo/tests/test.rs new file mode 100644 index 0000000..e1c0518 --- /dev/null +++ b/kondo/tests/test.rs @@ -0,0 +1,89 @@ +mod common; +use std::{io::Write, process::Command}; + +use kondo; + +// parse age filter is marked public. Can we use it? +#[test] +fn parse_age_filter_is_public() { + let res = kondo::parse_age_filter("10m"); + + let age_filter = res.unwrap(); + assert_eq!(age_filter, 600); +} + +#[test] +fn test_version() { + let bin = common::bin(); + + let output = Command::new(bin) + .arg("--") + .arg("--version") + .output() + .expect("failed to execute process"); + assert!(output.status.success()); +} + +#[ignore = "in progress"] +#[test] +fn test_can_run_cargo() { + let tmpdir = create_fake_python_project("testing".to_string()); + let bin = common::bin(); + + println!("tmpdr: {:?}", tmpdir.path()); + + assert!( + tmpdir + .path() + .join("testing") + .join("__pycache__") + .join("cache.data") + .exists(), + "cache ought to exist before running kondo" + ); + let output = Command::new(bin) + .arg(tmpdir.path()) + .arg("--all") + .output() + .expect("failed to execute process"); + assert!(output.status.success(), "failed to run kondo"); + assert!( + !tmpdir + .path() + .join("testing") + .join("__pycache__") + .join("cache.data") + .exists(), + "cache ought to be deleted after running kondo" + ); + // clean up + tmpdir.close().unwrap(); +} + +// Given a name, create a new simulated python project in a safe to delete directry +fn create_fake_python_project(name: String) -> tempfile::TempDir { + // Make a new project in a temporary directory + let tmp_dir = tempfile::tempdir().unwrap(); + + // make a new root in the tmp dir + let project_dir = tmp_dir.path().join(&name); + std::fs::create_dir(&project_dir).unwrap(); + + // Must have a directory to hold the project. + let cache_dir = project_dir.join("__pycache__"); + std::fs::create_dir(&cache_dir).unwrap(); + + // Must have data in the cache to delete + let mut data_file = std::fs::File::create(cache_dir.join("cache.data")).unwrap(); + data_file.write_all(b"#oodles of cache')\n").unwrap(); + let mut data_file_b = std::fs::File::create(cache_dir.join("other.cache")).unwrap(); + data_file_b.write_all(b"#oodles of cache')\n").unwrap(); + + // and a file of type .py to signal we're a python project + let mut python_file = std::fs::File::create(project_dir.join("main.py")).unwrap(); + python_file + .write_all(b"#!/bin/python\n\nprint('Hello, world!')\n") + .unwrap(); + + return tmp_dir; +} diff --git a/x/test_data/test_discover/python-project-a/main.py b/x/test_data/test_discover/python-project-a/main.py new file mode 100644 index 0000000..e69de29 diff --git a/x/test_data/test_discover/python-project-b/main.py b/x/test_data/test_discover/python-project-b/main.py new file mode 100644 index 0000000..8b8b2cf --- /dev/null +++ b/x/test_data/test_discover/python-project-b/main.py @@ -0,0 +1,3 @@ +#!/bin/python + +print('Hello, world!')