From 5cbf8445c10fcce3299d06ff5a7acdd89f5e993b Mon Sep 17 00:00:00 2001 From: Casey Rodarmor Date: Sun, 7 Jul 2024 19:06:30 -0700 Subject: [PATCH] Lexiclean search directory so `..` does not check the current directory If the search directory was `..`, for example in the invocation `just ../foo`, we would wind up checking the justfile in the current directory since we did `INVOCATION_DIRECTORY/..`.ancestors(), which would first return `INVOCATION_DIRECTORY`. Instead, lexiclean the result of joining th invocation directory with the search directory, so `..` is removed, and `ancestors()` doesn't return the invocation directory. --- src/search.rs | 1 + src/subcommand.rs | 7 ++++--- tests/search.rs | 16 ++++++++++++++++ 3 files changed, 21 insertions(+), 3 deletions(-) diff --git a/src/search.rs b/src/search.rs index 5085419529..4d061e0f55 100644 --- a/src/search.rs +++ b/src/search.rs @@ -4,6 +4,7 @@ const DEFAULT_JUSTFILE_NAME: &str = JUSTFILE_NAMES[0]; pub(crate) const JUSTFILE_NAMES: [&str; 2] = ["justfile", ".justfile"]; const PROJECT_ROOT_CHILDREN: &[&str] = &[".bzr", ".git", ".hg", ".svn", "_darcs"]; +#[derive(Debug)] pub(crate) struct Search { pub(crate) justfile: PathBuf, pub(crate) working_directory: PathBuf, diff --git a/src/subcommand.rs b/src/subcommand.rs index 6173bfdf6e..65c80a8320 100644 --- a/src/subcommand.rs +++ b/src/subcommand.rs @@ -110,9 +110,10 @@ impl Subcommand { ) { let starting_path = match &config.search_config { SearchConfig::FromInvocationDirectory => config.invocation_directory.clone(), - SearchConfig::FromSearchDirectory { search_directory } => { - env::current_dir().unwrap().join(search_directory) - } + SearchConfig::FromSearchDirectory { search_directory } => config + .invocation_directory + .join(search_directory) + .lexiclean(), _ => unreachable!(), }; diff --git a/tests/search.rs b/tests/search.rs index daac5bf554..b7ebe47f5f 100644 --- a/tests/search.rs +++ b/tests/search.rs @@ -143,6 +143,22 @@ fn single_upwards() { search_test(path, &["../"]); } +#[test] +fn double_upwards() { + let tmp = temptree! { + justfile: "default:\n\techo ok", + foo: { + bar: { + justfile: "default:\n\techo foo", + }, + }, + }; + + let path = tmp.path().join("foo/bar"); + + search_test(path, &["../default"]); +} + #[test] fn find_dot_justfile() { Test::new()