cli: do not search ancestor paths specified by -R/--repository

If a workspace path is explicitly specified, it must point to the exact
workspace directory. This is the same behavior as 'hg -R'. OTOH, 'git -C'
is the option to chdir, so it makes sense to search .git from that directory.

This also fixes 'jj -R ../..' which would previously look up '../..', '..',
'.', ...
This commit is contained in:
Yuya Nishihara 2023-01-10 12:52:06 +09:00
parent 40a9f75441
commit cb0b9e590a
3 changed files with 33 additions and 7 deletions

View file

@ -50,6 +50,9 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
* Commit description set by `-m`/`--message` is now terminated with a newline * Commit description set by `-m`/`--message` is now terminated with a newline
character, just like descriptions set by editor are. character, just like descriptions set by editor are.
* The `-R`/`--repository` path must be a valid workspace directory. Its
ancestor directories are no longer searched.
### Contributors ### Contributors
Thanks to the people who made this release happen! Thanks to the people who made this release happen!

View file

@ -977,15 +977,18 @@ fn init_workspace_loader(
cwd: &Path, cwd: &Path,
global_args: &GlobalArgs, global_args: &GlobalArgs,
) -> Result<WorkspaceLoader, CommandError> { ) -> Result<WorkspaceLoader, CommandError> {
let workspace_path_str = global_args.repository.as_deref().unwrap_or("."); let workspace_root = if let Some(path) = global_args.repository.as_ref() {
let workspace_path = cwd.join(workspace_path_str); cwd.join(path)
let workspace_root = workspace_path } else {
.ancestors() cwd.ancestors()
.find(|path| path.join(".jj").is_dir()) .find(|path| path.join(".jj").is_dir())
.unwrap_or(&workspace_path); .unwrap_or(cwd)
WorkspaceLoader::init(workspace_root).map_err(|err| match err { .to_owned()
};
WorkspaceLoader::init(&workspace_root).map_err(|err| match err {
WorkspaceLoadError::NoWorkspaceHere(wc_path) => { WorkspaceLoadError::NoWorkspaceHere(wc_path) => {
// Prefer user-specified workspace_path_str instead of absolute wc_path. // Prefer user-specified workspace_path_str instead of absolute wc_path.
let workspace_path_str = global_args.repository.as_deref().unwrap_or(".");
let message = format!(r#"There is no jj repo in "{workspace_path_str}""#); let message = format!(r#"There is no jj repo in "{workspace_path_str}""#);
let git_dir = wc_path.join(".git"); let git_dir = wc_path.join(".git");
if git_dir.is_dir() { if git_dir.is_dir() {

View file

@ -125,6 +125,26 @@ fn test_resolve_workspace_directory() {
Working copy : 230dd059e1b0 (no description set) Working copy : 230dd059e1b0 (no description set)
The working copy is clean The working copy is clean
"###); "###);
// Explicit subdirectory path
let stderr = test_env.jj_cmd_failure(&subdir, &["status", "-R", "."]);
insta::assert_snapshot!(stderr, @r###"
Error: There is no jj repo in "."
"###);
// Valid explicit path
let stdout = test_env.jj_cmd_success(&subdir, &["status", "-R", "../.."]);
insta::assert_snapshot!(stdout, @r###"
Parent commit: 000000000000 (no description set)
Working copy : 230dd059e1b0 (no description set)
The working copy is clean
"###);
// "../../..".ancestors() contains "../..", but it should never be looked up.
let stderr = test_env.jj_cmd_failure(&subdir, &["status", "-R", "../../.."]);
insta::assert_snapshot!(stderr, @r###"
Error: There is no jj repo in "../../.."
"###);
} }
#[test] #[test]