diff --git a/Cargo.lock b/Cargo.lock index 2b07598..1ae0ca3 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" @@ -174,12 +180,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" @@ -239,14 +257,18 @@ version = "0.7.0" dependencies = [ "clap", "clap_complete", + "fs_extra", "kondo-lib", + "tempfile", ] [[package]] name = "kondo-lib" version = "0.7.0" dependencies = [ + "fs_extra", "ignore", + "tempfile", "walkdir", ] @@ -304,6 +326,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" @@ -339,7 +370,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", @@ -378,6 +409,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-lib/Cargo.toml b/kondo-lib/Cargo.toml index e0c92f3..c3cd899 100644 --- a/kondo-lib/Cargo.toml +++ b/kondo-lib/Cargo.toml @@ -11,3 +11,10 @@ license = "MIT" [dependencies] ignore = "0.4.18" walkdir = "2" + +[dev-dependencies] +# need recursive copy for testing +fs_extra = "1.3.0" + +# need temporary test dirs +tempfile = "3.8.0" diff --git a/kondo-lib/src/lib.rs b/kondo-lib/src/lib.rs index 744980f..3db4aef 100644 --- a/kondo-lib/src/lib.rs +++ b/kondo-lib/src/lib.rs @@ -1,3 +1,5 @@ +mod lib_test; + use std::{ borrow::Cow, error::{self, Error}, @@ -482,6 +484,8 @@ pub fn path_canonicalise( #[cfg(test)] mod tests { + use std::fs::create_dir_all; + use super::print_elapsed; #[test] @@ -522,4 +526,45 @@ mod tests { assert_eq!(print_elapsed(2419200 * 25), "25 months ago"); assert_eq!(print_elapsed(2419200 * 48), "4 years ago"); } + + #[test] + fn test_path_canonicalise_works_on_extant_tail() { + let str_tail = "idoexist".to_string(); + + let base = tempfile::tempdir().unwrap().into_path(); + let expected = base.join(str_tail.clone()); + let tail = std::path::PathBuf::from(str_tail.clone()); + + // must make sure the tail exists in base dir, or we'll error that it doesn't exist + create_dir_all(base.join(std::path::PathBuf::from(str_tail))).unwrap(); + + let actual = super::path_canonicalise(&base, tail).unwrap(); + assert_eq!(actual, expected); + } + + #[test] + #[should_panic] + fn path_canonicalise_fails_on_nonextant_tail() { + let str_tail = "idontexist".to_string(); + + let base = tempfile::tempdir().unwrap().into_path(); + let tail = std::path::PathBuf::from(str_tail.clone()); + + // here tail has not been created + + super::path_canonicalise(&base, tail).unwrap(); + } + #[test] + fn path_canonicalise_works_on_absolute_dirs_that_dont_exist() { + let str_tail = "/idontexist".to_string(); + + let base = tempfile::tempdir().unwrap().into_path(); + let expected = base.join(str_tail.clone()); + let tail = std::path::PathBuf::from(str_tail.clone()); + + assert!(!tail.exists()); + + let actual = super::path_canonicalise(&base, tail).unwrap(); + assert_eq!(actual, expected); + } } diff --git a/kondo-lib/src/lib_test.rs b/kondo-lib/src/lib_test.rs new file mode 100644 index 0000000..b9ca6cf --- /dev/null +++ b/kondo-lib/src/lib_test.rs @@ -0,0 +1,126 @@ +#[cfg(test)] +mod test { + + use crate::{Project, ProjectType, ScanOptions}; + use std::{io::Write, path::PathBuf}; + + // Given test data, clean should remove some files + #[test] + fn test_clean() { + let scan_options: ScanOptions = ScanOptions { + follow_symlinks: false, + same_file_system: true, + }; + + let tempdir = create_fake_python_project("test_data".to_string()); + let path = tempdir.path().join("test_data"); + + println!("path: {:?}", path); + println!("tempdir: {:?}", tempdir.path()); + + let project_a = Project { + path, + 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__/cache.data").exists(), + "cache file ought to exist" + ); + Project::clean(&project_a); + assert!( + !project_a.path.join("__pycache__/cache.data").exists(), + "cache file should have been deleted" + ); + + assert!(project_a.path.exists(), "project ought to still exist"); + + // clean up + tempdir.close().unwrap(); + } + + // #[ignore = "this is probably "] + #[test] + fn test_clean_nested_python_projects() { + // make alpha project + let alpha_tmp_dir = create_fake_python_project("alpha".to_string()); + + // inside of alpha, make nested project + let project_nested_dir = create_fake_python_project_in_dir( + alpha_tmp_dir.path().clone().to_path_buf(), + "nested".to_string(), + ); + + // Given alpha project + let project_alpha = Project { + path: alpha_tmp_dir.into_path(), + project_type: ProjectType::Python, + }; + // and nested project + let project_nested = Project { + path: project_nested_dir.clone(), + project_type: ProjectType::Python, + }; + + // Clean! + Project::clean(&project_alpha); + Project::clean(&project_nested); + // Both project dirs exist + assert!( + project_alpha.path.exists(), + "project alpha ought to still exist" + ); + assert!( + project_nested_dir.exists(), + "nested project ought to still exist" + ); + + // Both cache files are gone + assert!( + !project_alpha.path.join("__pycache__/cache.data").exists(), + "cache file of alpha should have been deleted" + ); + assert!( + !project_nested_dir.join("__pycache__/cache.data").exists(), + "cache file of nested project should have been deleted" + ); + } + // TODO: this code is duplicated at konod/src/main.rs + // Given a name, create a new simulated python project in a safe to delete directry + pub fn create_fake_python_project(name: String) -> tempfile::TempDir { + // Make a new project in a temporary directory + let tmp_dir = tempfile::tempdir().unwrap(); + create_fake_python_project_in_dir(tmp_dir.path().to_path_buf(), name); + tmp_dir + } + + pub fn create_fake_python_project_in_dir(dir: PathBuf, name: String) -> PathBuf { + // make a new root in the dir + let project_dir = dir.join(name); + std::fs::create_dir_all(&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(); + project_dir.to_path_buf() + } +} diff --git a/kondo/Cargo.toml b/kondo/Cargo.toml index 1d8d6bb..163c1e0 100644 --- a/kondo/Cargo.toml +++ b/kondo/Cargo.toml @@ -17,11 +17,28 @@ keywords = ["clean", "cleanup", "delete", "free"] exclude = ["test_dir"] edition = "2021" - [dependencies] clap = { version = "4", features = ["derive"] } clap_complete = "4" +[dev-dependencies] +# 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..386ccce --- /dev/null +++ b/kondo/src/lib.rs @@ -0,0 +1,77 @@ +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. BUG?"] +#[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 aac3c9e..232d063 100644 --- a/kondo/src/main.rs +++ b/kondo/src/main.rs @@ -1,9 +1,9 @@ +mod main_test; + use std::{ env::current_dir, error::Error, - fmt, io::{stdin, stdout, Write}, - num::ParseIntError, path::PathBuf, sync::mpsc::{Receiver, Sender, SyncSender}, }; @@ -15,6 +15,11 @@ use kondo_lib::{ dir_size, path_canonicalise, pretty_size, print_elapsed, scan, Project, ScanOptions, }; +use kondo::parse_age_filter; + +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")] @@ -59,93 +64,81 @@ 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 let Some(generator) = opt.generator { + let mut cmd = Opt::command(); + eprintln!("Generating completion file for {generator:?}..."); + print_completions(generator, &mut cmd); + return Ok(()); } - 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; - } + if opt.quiet > 0 && !opt.all { + eprintln!("Quiet mode can only be used with --all."); + 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; - } - } + let dirs = prepare_directories(opt.dirs)?; - path_canonicalise(&cd, path).ok() - }) - .collect(); + let scan_options: ScanOptions = ScanOptions { + follow_symlinks: opt.follow_symlinks, + same_file_system: opt.same_filesystem, + }; - Ok(dirs) -} + 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)>(); -#[derive(Debug)] -pub enum ParseAgeFilterError { - ParseIntError(ParseIntError), - InvalidUnit, -} + let project_min_age = opt.older; + let ignored_dirs = { + let cd = current_dir()?; -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::mem::take(&mut opt.ignored_dirs) + .into_iter() + .map(|dir| path_canonicalise(&cd, dir)) + .collect::, _>>()? + }; -impl From for ParseAgeFilterError { - fn from(e: ParseIntError) -> Self { - Self::ParseIntError(e) - } -} + std::thread::spawn(move || { + discover( + dirs, + &scan_options, + project_min_age, + proj_discover_send, + &ignored_dirs, + ); + }); -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_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 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) + ); + } -type DiscoverData = (Project, Vec<(String, u64)>, u64, String); -type DeleteData = (Project, u64); + Ok(()) +} fn discover( dirs: Vec, @@ -294,78 +287,47 @@ fn print_completions(gen: G, cmd: &mut Command) { generate(gen, cmd, cmd.get_name().to_string(), &mut stdout()); } -fn main() -> Result<(), Box> { - let mut opt = Opt::parse(); - - if let Some(generator) = opt.generator { - let mut cmd = Opt::command(); - eprintln!("Generating completion file for {generator:?}..."); - print_completions(generator, &mut cmd); - return Ok(()); - } - - 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()?; + 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; + } - std::mem::take(&mut opt.ignored_dirs) - .into_iter() - .map(|dir| path_canonicalise(&cd, dir)) - .collect::, _>>()? - }; + if let Ok(metadata) = path.metadata() { + if metadata.is_file() { + eprintln!( + "error: file supplied but directory expected: {}", + path.to_string_lossy() + ); + return None; + } + } - std::thread::spawn(move || { - discover( - dirs, - &scan_options, - project_min_age, - proj_discover_send, - &ignored_dirs, - ); - }); + path_canonicalise(&cd, path).ok() + }) + .collect(); - let delete_handle = std::thread::spawn(move || process_deletes(proj_delete_recv)); + Ok(dirs) +} - interactive_prompt( - proj_discover_recv, - proj_delete_send, - opt.quiet, - opt.all, - opt.default, - ); +#[cfg(test)] +mod test { - let delete_results = match delete_handle.join() { - Ok(r) => r, - Err(e) => { - eprintln!("error in delete thread, {e:?}"); - std::process::exit(1); - } - }; + // parse age filter is marked public. Can we use it? + #[test] + fn parse_age_filter_is_public() { + let res = kondo::parse_age_filter("10m"); - 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) - ); + let age_filter = res.unwrap(); + assert_eq!(age_filter, 600); } - - Ok(()) } diff --git a/kondo/src/main_test.rs b/kondo/src/main_test.rs new file mode 100644 index 0000000..46d11a5 --- /dev/null +++ b/kondo/src/main_test.rs @@ -0,0 +1,132 @@ +#[cfg(test)] +mod test { + use crate::{discover, prepare_directories, DiscoverData}; + use kondo_lib::ScanOptions; + use std::io::Write; + + #[test] + fn test_prepare_directories_that_exist() { + let tempdir = tempfile::tempdir().unwrap(); + let path = tempdir.path().join("test_directory"); + + // Given a directory that exists + std::fs::create_dir(&path).unwrap(); + + // When we prepare the directories + let dirs = prepare_directories(vec![path.clone()]).unwrap(); + + // Then we ought to get back the same directory + assert_eq!(dirs[0], path); + + // clean up + tempdir.close().unwrap(); + } + #[test] + fn test_prepare_directories_that_do_not_exist() { + let tempdir = tempfile::tempdir().unwrap(); // is created + let path = tempdir.path().join("test_directory"); // is not created + + // Given a directory that DOES NOT exist + // When we prepare the directories + let dirs = prepare_directories(vec![path.clone()]).unwrap(); + + // Then there is not directory + assert_eq!(dirs.len(), 0); + + // clean up + tempdir.close().unwrap(); + } + + #[test] + fn test_discover() { + let tempdir = create_fake_python_project("test_data".to_string()); + + // 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![tempdir.path().join("test_data").to_path_buf()], + &scan_options, + project_min_age, + result_sender, + &ignored_dirs, + ); + + let count = result_recv.try_iter().count(); + + // ought to find the right number of projects + assert_eq!(count, 1); + + // clean up + tempdir.close().unwrap(); + } + + #[ignore = "does discover not recurse? this test is running discover a level above the project. it doesn't work. above does, at the same level."] + #[test] + fn test_discover_broken() { + let tempdir = create_fake_python_project("test_data".to_string()); + + // 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 ... + println!("discover in {:?}", tempdir.path()); + discover( + vec![tempdir.path().to_path_buf()], + &scan_options, + project_min_age, + result_sender, + &ignored_dirs, + ); + + let count = result_recv.try_iter().count(); + + // ought to find the right number of projects + assert_eq!(count, 1); + + // clean up + tempdir.close().unwrap(); + } + + // TODO: this code is duplicated at kondo-lib/src/lib_test.rs + // Given a name, create a new simulated python project in a safe to delete directry + pub 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(); + + tmp_dir + } +} diff --git a/kondo/tests/common.rs b/kondo/tests/common.rs new file mode 100644 index 0000000..8c00eda --- /dev/null +++ b/kondo/tests/common.rs @@ -0,0 +1,63 @@ +use std::{env, path::PathBuf}; + +// Enables us to get a handle on the recently build kondo binary. +// When running tests, our executable is in the target directory. +// This returns the PathBuf to the kondo binary in the same target directory. +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); + } + path +} + +// Enables us to get a temporary directory that contains a copy of test data +// specified by `scenario`. +pub fn with_temp_dir_from(scenario: String, f: F) +where + F: FnOnce(PathBuf), +{ + let tmp_dir = get_copy_of_test_data_as_temp_dir(scenario); + f(tmp_dir.path().to_path_buf()); + tmp_dir.close().unwrap(); +} + +pub fn with_cache_at(cache: PathBuf, f: F) +where + F: FnOnce(PathBuf), +{ + assert!(cache.exists(), "cache must exist before running kondo"); + f(cache.clone()); + assert!(!cache.exists(), "cache must be deleted after running kondo"); +} + +// Given `scenario` in the `test_data` directory, this returns a temporary +// directory that contains a copy of the scenario's data. +pub fn get_copy_of_test_data_as_temp_dir(scenario: String) -> tempfile::TempDir { + extern crate fs_extra; + let options = fs_extra::dir::CopyOptions::new(); + + let project_directory: PathBuf = PathBuf::from(env!("CARGO_MANIFEST_DIR")); + let data_directory = project_directory.join("../test_data"); + let scenario_directory = data_directory.join(scenario.clone()); + + if !scenario_directory.exists() { + panic!( + "scenario {:?} does not exist: {:?}. Create it in src or you have a typo", + scenario.clone().to_string(), + scenario_directory + ); + } + let from_paths = vec![scenario_directory]; + + let tmp_dir = tempfile::tempdir().unwrap(); + println!("tmp_dir: {:?}", tmp_dir); + fs_extra::copy_items(&from_paths, &tmp_dir, &options).unwrap(); + + tmp_dir +} diff --git a/kondo/tests/test.rs b/kondo/tests/test.rs new file mode 100644 index 0000000..7f77897 --- /dev/null +++ b/kondo/tests/test.rs @@ -0,0 +1,178 @@ +mod common; +use common::with_temp_dir_from; +use std::{fs, process::Command}; + +use crate::common::with_cache_at; + +#[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()); +} + +#[test] +fn test_cli_run_kondo_all_in_python_project() { + let scenario = "scenario_a".to_string(); + with_temp_dir_from(scenario.clone(), |tmpdir| { + let bin = common::bin(); + println!("tmpdr: {:?}", tmpdir); + + with_cache_at( + tmpdir + .clone() + .join(scenario.clone()) + .join("python-project-a") + .join("__pycache__") + .join("1"), + |cache| { + // run kondo --all in the temp dir + let mut cmd = Command::new(bin); + + let cmd_w_args = cmd + .arg(tmpdir.join(scenario.clone())) // note the path tmp/scenario-name + .arg("--all"); + print!("cmd_w_args: {:?}", cmd_w_args); + let output = cmd_w_args.output().unwrap(); + + assert!(output.status.success(), "failed to run kondo"); + println!("stdout: {}", String::from_utf8_lossy(&output.stdout)); + }, + ); + }); +} + +#[ignore = "failing unexpectedly. should work, but doesn't see the project unless we specify the right top dir as in the test above. I'd expect kondo to recurse into the scenario directory. "] +#[test] +fn test_cli_run_kondo_all_above_project_fails() { + let scenario = "scenario_a".to_string(); + with_temp_dir_from(scenario.clone(), |tmpdir| { + let bin = common::bin(); + println!("tmpdr: {:?}", tmpdir.clone()); + + with_cache_at( + tmpdir + .clone() + .join(scenario.clone()) + .join("python-project-a") + .join("__pycache__") + .join("1"), + |cache| { + // run kondo --all in the temp dir + let mut cmd = Command::new(bin); + + let cmd_w_args = cmd + .arg(tmpdir.clone()) // here note the path, just tmpdir + .arg("--all"); + + print!("cmd_w_args: {:?}", cmd_w_args); + let output = cmd_w_args.output().unwrap(); + + assert!(output.status.success(), "failed to run kondo"); + println!("stdout: {}", String::from_utf8_lossy(&output.stdout)); + }, + ); + }); +} + +#[ignore = "nested projects not working yet"] +#[test] +fn test_cli_run_kondo_scenario_nested_a() { + let scenario = "scenario_nested_a".to_string(); + with_temp_dir_from(scenario.clone(), |tmpdir| { + let bin = common::bin(); + println!("tmpdr: {:?}", tmpdir.clone()); + + assert!( + tmpdir + .clone() + .join(scenario.clone()) + .join("python-project-a") + .join("__pycache__") + .join("1") + .exists(), + "cache must exist before running kondo" + ); + + // run kondo --all in the temp dir + let mut cmd = Command::new(bin); + + let cmd_w_args = cmd.arg(tmpdir.join(scenario.clone())).arg("--all"); + let output = cmd_w_args.output().unwrap(); + + assert!(output.status.success(), "failed to run kondo"); + println!("stdout: {}", String::from_utf8_lossy(&output.stdout)); + + assert!( + !tmpdir + .clone() + .join(scenario.clone()) + .join("python-project-a") + .join("__pycache__") + .join("1") + .exists(), + "cache ought to be deleted after running kondo" + ); + + assert!( + !tmpdir + .clone() + .join(scenario.clone()) + .join("python-project-a") + .join("sub-project") + .join("python-project-b") + .join("__pycache__") + .join("cache.data") + .exists(), + "cache in nested project ought to be deleted after running kondo" + ); + }); +} + +#[test] +fn play_a() { + let scenario = "scenario_a".to_string(); + with_temp_dir_from(scenario.clone(), |path| { + println!("path: {:?}", path); + + let paths = fs::read_dir(path.clone()).unwrap(); + for path in paths { + println!("Name: {}", path.unwrap().path().display()) + } + + assert!(path.exists(), "dir ought to exist"); + + let paths = fs::read_dir(path.clone().join(scenario.clone())).unwrap(); + for path in paths { + println!("Name: {}", path.unwrap().path().display()) + } + assert!(path.join(scenario.clone()).exists(), "dir ought to exist"); + }); +} + +#[test] +#[ignore = "bug: --ignored-dirs that don't exist cause a failure"] +fn non_extant_ignore_dirs_work() { + let scenario = "scenario_nested_a".to_string(); + with_temp_dir_from(scenario.clone(), |tmpdir| { + let bin = common::bin(); + + // run kondo + let mut cmd = Command::new(bin); + let cmd_w_args = cmd + .arg(tmpdir.join(scenario.clone())) + .arg(tmpdir.clone()) + .arg("--ignored-dirs=doesnotexist") + .arg("--all"); + let output = cmd_w_args.output().unwrap(); + println!("stdout: {}", String::from_utf8_lossy(&output.stdout)); + println!("stderr: {}", String::from_utf8_lossy(&output.stderr)); + + assert!(output.status.success(), "failed to run kondo"); + }); +} diff --git a/test_data/scenario_a/python-project-a/__pycache__/1 b/test_data/scenario_a/python-project-a/__pycache__/1 new file mode 100644 index 0000000..1269488 --- /dev/null +++ b/test_data/scenario_a/python-project-a/__pycache__/1 @@ -0,0 +1 @@ +data diff --git a/test_data/scenario_a/python-project-a/__pycache__/2 b/test_data/scenario_a/python-project-a/__pycache__/2 new file mode 100644 index 0000000..98d81a2 --- /dev/null +++ b/test_data/scenario_a/python-project-a/__pycache__/2 @@ -0,0 +1 @@ +data2 diff --git a/test_data/scenario_a/python-project-a/__pycache__/3 b/test_data/scenario_a/python-project-a/__pycache__/3 new file mode 100644 index 0000000..b5c7e39 --- /dev/null +++ b/test_data/scenario_a/python-project-a/__pycache__/3 @@ -0,0 +1 @@ +data3 diff --git a/test_data/scenario_a/python-project-a/main.py b/test_data/scenario_a/python-project-a/main.py new file mode 100644 index 0000000..e69de29 diff --git a/test_data/scenario_a/python-project-b/__pycache__/cache.data b/test_data/scenario_a/python-project-b/__pycache__/cache.data new file mode 100644 index 0000000..1c843ca --- /dev/null +++ b/test_data/scenario_a/python-project-b/__pycache__/cache.data @@ -0,0 +1 @@ +#oodles of cache') diff --git a/test_data/scenario_a/python-project-b/__pycache__/other.cache b/test_data/scenario_a/python-project-b/__pycache__/other.cache new file mode 100644 index 0000000..1c843ca --- /dev/null +++ b/test_data/scenario_a/python-project-b/__pycache__/other.cache @@ -0,0 +1 @@ +#oodles of cache') diff --git a/test_data/scenario_a/python-project-b/main.py b/test_data/scenario_a/python-project-b/main.py new file mode 100644 index 0000000..8b8b2cf --- /dev/null +++ b/test_data/scenario_a/python-project-b/main.py @@ -0,0 +1,3 @@ +#!/bin/python + +print('Hello, world!') diff --git a/test_data/scenario_b/cargo-project-a/Cargo.toml b/test_data/scenario_b/cargo-project-a/Cargo.toml new file mode 100644 index 0000000..026a1fb --- /dev/null +++ b/test_data/scenario_b/cargo-project-a/Cargo.toml @@ -0,0 +1 @@ +# empty example is enough to signal this is a cargo project \ No newline at end of file diff --git a/test_data/scenario_b/cargo-project-a/target/binaries-and-such b/test_data/scenario_b/cargo-project-a/target/binaries-and-such new file mode 100644 index 0000000..feec270 --- /dev/null +++ b/test_data/scenario_b/cargo-project-a/target/binaries-and-such @@ -0,0 +1 @@ +// This file represents cargo build artifacts \ No newline at end of file diff --git a/test_data/scenario_nested_a/python-project-a/__pycache__/1 b/test_data/scenario_nested_a/python-project-a/__pycache__/1 new file mode 100644 index 0000000..1269488 --- /dev/null +++ b/test_data/scenario_nested_a/python-project-a/__pycache__/1 @@ -0,0 +1 @@ +data diff --git a/test_data/scenario_nested_a/python-project-a/__pycache__/2 b/test_data/scenario_nested_a/python-project-a/__pycache__/2 new file mode 100644 index 0000000..98d81a2 --- /dev/null +++ b/test_data/scenario_nested_a/python-project-a/__pycache__/2 @@ -0,0 +1 @@ +data2 diff --git a/test_data/scenario_nested_a/python-project-a/__pycache__/3 b/test_data/scenario_nested_a/python-project-a/__pycache__/3 new file mode 100644 index 0000000..b5c7e39 --- /dev/null +++ b/test_data/scenario_nested_a/python-project-a/__pycache__/3 @@ -0,0 +1 @@ +data3 diff --git a/test_data/scenario_nested_a/python-project-a/main.py b/test_data/scenario_nested_a/python-project-a/main.py new file mode 100644 index 0000000..e69de29 diff --git a/test_data/scenario_nested_a/python-project-a/sub-project/python-project-b/__pycache__/cache.data b/test_data/scenario_nested_a/python-project-a/sub-project/python-project-b/__pycache__/cache.data new file mode 100644 index 0000000..1c843ca --- /dev/null +++ b/test_data/scenario_nested_a/python-project-a/sub-project/python-project-b/__pycache__/cache.data @@ -0,0 +1 @@ +#oodles of cache') diff --git a/test_data/scenario_nested_a/python-project-a/sub-project/python-project-b/__pycache__/other.cache b/test_data/scenario_nested_a/python-project-a/sub-project/python-project-b/__pycache__/other.cache new file mode 100644 index 0000000..1c843ca --- /dev/null +++ b/test_data/scenario_nested_a/python-project-a/sub-project/python-project-b/__pycache__/other.cache @@ -0,0 +1 @@ +#oodles of cache') diff --git a/test_data/scenario_nested_a/python-project-a/sub-project/python-project-b/main.py b/test_data/scenario_nested_a/python-project-a/sub-project/python-project-b/main.py new file mode 100644 index 0000000..8b8b2cf --- /dev/null +++ b/test_data/scenario_nested_a/python-project-a/sub-project/python-project-b/main.py @@ -0,0 +1,3 @@ +#!/bin/python + +print('Hello, world!')