From 9861061d7b05090b0ebae785eb3fb5690353ad77 Mon Sep 17 00:00:00 2001 From: Waleed Khan Date: Fri, 27 May 2022 19:09:08 -0700 Subject: [PATCH] cli: make `jj branch` take subcommands, not flags As per https://github.com/martinvonz/jj/issues/330. --- .vscode/settings.json | 3 + src/commands.rs | 286 ++++++++++++++++++++++------------- tests/test_alias.rs | 4 +- tests/test_branch_command.rs | 19 +-- tests/test_edit_command.rs | 2 +- tests/test_git_colocated.rs | 2 +- tests/test_git_push.rs | 4 +- tests/test_move_command.rs | 20 +-- tests/test_rebase_command.rs | 2 +- tests/test_squash_command.rs | 16 +- 10 files changed, 215 insertions(+), 143 deletions(-) create mode 100644 .vscode/settings.json diff --git a/.vscode/settings.json b/.vscode/settings.json new file mode 100644 index 00000000000..ab9cb8a875c --- /dev/null +++ b/.vscode/settings.json @@ -0,0 +1,3 @@ +{ + "files.trimTrailingWhitespace": false +} diff --git a/src/commands.rs b/src/commands.rs index 9676d9b4273..3291f91cbdc 100644 --- a/src/commands.rs +++ b/src/commands.rs @@ -57,6 +57,7 @@ use jujutsu_lib::settings::UserSettings; use jujutsu_lib::store::Store; use jujutsu_lib::transaction::Transaction; use jujutsu_lib::tree::{merge_trees, Tree, TreeDiffIterator, TreeMergeError}; +use jujutsu_lib::view::View; use jujutsu_lib::working_copy::{ CheckoutStats, LockedWorkingCopy, ResetError, SnapshotError, WorkingCopy, }; @@ -1129,8 +1130,8 @@ enum Commands { Merge(MergeArgs), Rebase(RebaseArgs), Backout(BackoutArgs), - Branch(BranchArgs), - Branches(BranchesArgs), + #[clap(subcommand)] + Branch(BranchSubcommand), /// Undo an operation (shortcut for `jj op undo`) Undo(OperationUndoArgs), Operation(OperationArgs), @@ -1607,44 +1608,69 @@ struct BackoutArgs { destination: Vec, } -/// Create, update, or delete a branch +/// Manage branches. /// /// For information about branches, see /// https://github.com/martinvonz/jj/blob/main/docs/branches.md. -#[derive(clap::Args, Clone, Debug)] -struct BranchArgs { - /// The branch's target revision - #[clap(long, short, group = "action")] - revision: Option, - - /// Allow moving the branch backwards or sideways - #[clap(long, requires = "revision")] - allow_backwards: bool, - - /// Delete the branch locally +#[derive(clap::Subcommand, Clone, Debug)] +enum BranchSubcommand { + /// Create a new branch. + #[clap(visible_alias("c"))] + Create { + /// The branch's target revision. + #[clap(long, short)] + revision: Option, + + /// The branches to create. + #[clap(required = true)] + names: Vec, + }, + + /// Delete an existing branch and propagate the deletion to remotes on the + /// next push. + #[clap(visible_alias("d"))] + Delete { + /// The branches to delete. + #[clap(required = true)] + names: Vec, + }, + + /// Delete the local version of an existing branch, without propagating the + /// deletion to remotes. + #[clap(visible_alias("f"))] + Forget { + /// The branches to delete. + #[clap(required = true)] + names: Vec, + }, + + /// List branches and their targets /// - /// The deletion will be propagated to remotes on push. - #[clap(long, group = "action")] - delete: bool, - - /// The name of the branch to move or delete - #[clap(long, group = "action")] - forget: bool, - - /// The branches to update. - names: Vec, + /// A remote branch will be included only if its target is different from + /// the local target. For a conflicted branch (both local and remote), old + /// target revisions are preceded by a "-" and new target revisions are + /// preceded by a "+". For information about branches, see + /// https://github.com/martinvonz/jj/blob/main/docs/branches.md. + #[clap(visible_alias("l"))] + List, + + /// Update a given branch to point to a certain commit. + #[clap(visible_alias("s"))] + Set { + /// The branch's target revision. + #[clap(long, short)] + revision: Option, + + /// Allow moving the branch backwards or sideways. + #[clap(long)] + allow_backwards: bool, + + /// The branches to update. + #[clap(required = true)] + names: Vec, + }, } -/// List branches and their targets -/// -/// A remote branch will be included only if its target is different from the -/// local target. For a conflicted branch (both local and remote), old target -/// revisions are preceded by a "-" and new target revisions are preceded by a -/// "+". For information about branches, see -/// https://github.com/martinvonz/jj/blob/main/docs/branches.md. -#[derive(clap::Args, Clone, Debug)] -struct BranchesArgs {} - /// Commands for working with the operation log /// /// Commands for working with the operation log. For information about the @@ -3998,91 +4024,150 @@ fn is_fast_forward(repo: RepoRef, branch_name: &str, new_target_id: &CommitId) - } } -fn cmd_branch(ui: &mut Ui, command: &CommandHelper, args: &BranchArgs) -> Result<(), CommandError> { +fn cmd_branch( + ui: &mut Ui, + command: &CommandHelper, + subcommand: &BranchSubcommand, +) -> Result<(), CommandError> { let mut workspace_command = command.workspace_helper(ui)?; - let branch_names: Vec<&str> = if args.delete || args.forget { - let view = workspace_command.repo().view(); - args.names - .iter() - .map(|branch_name| match view.get_local_branch(branch_name) { - Some(_) => Ok(branch_name.as_str()), - None => Err(CommandError::UserError(format!( + let view = workspace_command.repo().view(); + fn validate_branch_names_exist<'a>( + view: &'a View, + names: &'a [String], + ) -> Result<(), CommandError> { + for branch_name in names { + if view.get_local_branch(branch_name).is_none() { + return Err(CommandError::UserError(format!( "No such branch: {}", branch_name - ))), - }) - .try_collect()? - } else { - args.names.iter().map(|name| name.as_str()).collect() - }; + ))); + } + } + Ok(()) + } - if branch_names.is_empty() { - ui.write_warn("warning: No branches provided.\n")?; + fn make_branch_term(branch_names: &[impl AsRef]) -> String { + match branch_names { + [branch_name] => format!("branch {}", branch_name.as_ref()), + branch_names => { + format!( + "branches {}", + branch_names.iter().map(AsRef::as_ref).join(", ") + ) + } + } } - let branch_term = if branch_names.len() == 1 { - "branch" - } else { - "branches" - }; - let branch_term = format!("{branch_term} {}", branch_names.join(", ")); - if args.delete { - let mut tx = workspace_command.start_transaction(&format!("delete {branch_term}")); - for branch_name in branch_names { - tx.mut_repo().remove_local_branch(branch_name); + match subcommand { + BranchSubcommand::Create { revision, names } => { + let branch_names: Vec<&str> = names + .iter() + .map(|branch_name| match view.get_local_branch(branch_name) { + Some(_) => Err(CommandError::UserError(format!( + "Branch already exists: {} (use `jj branch set` to update it)", + branch_name + ))), + None => Ok(branch_name.as_str()), + }) + .try_collect()?; + + if branch_names.len() > 1 { + ui.write_warn(format!( + "warning: Creating multiple branches ({}).\n", + branch_names.len() + ))?; + } + + let target_commit = + workspace_command.resolve_single_rev(ui, revision.as_deref().unwrap_or("@"))?; + let mut tx = workspace_command.start_transaction(&format!( + "create {} pointing to commit {}", + make_branch_term(&branch_names), + target_commit.id().hex() + )); + for branch_name in branch_names { + tx.mut_repo().set_local_branch( + branch_name.to_string(), + RefTarget::Normal(target_commit.id().clone()), + ); + } + workspace_command.finish_transaction(ui, tx)?; } - workspace_command.finish_transaction(ui, tx)?; - } else if args.forget { - let mut tx = workspace_command.start_transaction(&format!("forget {branch_term}")); - for branch_name in branch_names { - tx.mut_repo().remove_branch(branch_name); + + BranchSubcommand::Set { + revision, + allow_backwards, + names: branch_names, + } => { + if branch_names.len() > 1 { + ui.write_warn(format!( + "warning: Updating multiple branches ({}).\n", + branch_names.len() + ))?; + } + + let target_commit = + workspace_command.resolve_single_rev(ui, revision.as_deref().unwrap_or("@"))?; + if !allow_backwards + && !branch_names.iter().all(|branch_name| { + is_fast_forward( + workspace_command.repo().as_repo_ref(), + branch_name, + target_commit.id(), + ) + }) + { + return Err(CommandError::UserError( + "Use --allow-backwards to allow moving a branch backwards or sideways" + .to_string(), + )); + } + let mut tx = workspace_command.start_transaction(&format!( + "point {} to commit {}", + make_branch_term(branch_names), + target_commit.id().hex() + )); + for branch_name in branch_names { + tx.mut_repo().set_local_branch( + branch_name.to_string(), + RefTarget::Normal(target_commit.id().clone()), + ); + } + workspace_command.finish_transaction(ui, tx)?; } - workspace_command.finish_transaction(ui, tx)?; - } else { - if branch_names.len() > 1 { - ui.write_warn(format!( - "warning: Updating multiple branches ({}).\n", - branch_names.len() - ))?; + + BranchSubcommand::Delete { names } => { + validate_branch_names_exist(view, names)?; + let mut tx = + workspace_command.start_transaction(&format!("delete {}", make_branch_term(names))); + for branch_name in names { + tx.mut_repo().remove_local_branch(branch_name); + } + workspace_command.finish_transaction(ui, tx)?; } - let target_commit = - workspace_command.resolve_single_rev(ui, args.revision.as_deref().unwrap_or("@"))?; - if !args.allow_backwards - && !branch_names.iter().all(|branch_name| { - is_fast_forward( - workspace_command.repo().as_repo_ref(), - branch_name, - target_commit.id(), - ) - }) - { - return Err(CommandError::UserError( - "Use --allow-backwards to allow moving a branch backwards or sideways".to_string(), - )); + BranchSubcommand::Forget { names } => { + validate_branch_names_exist(view, names)?; + let mut tx = + workspace_command.start_transaction(&format!("forget {}", make_branch_term(names))); + for branch_name in names { + tx.mut_repo().remove_branch(branch_name); + } + workspace_command.finish_transaction(ui, tx)?; } - let mut tx = workspace_command.start_transaction(&format!( - "point {branch_term} to commit {}", - target_commit.id().hex() - )); - for branch_name in branch_names { - tx.mut_repo().set_local_branch( - branch_name.to_string(), - RefTarget::Normal(target_commit.id().clone()), - ); + + BranchSubcommand::List => { + list_branches(ui, &workspace_command)?; } - workspace_command.finish_transaction(ui, tx)?; } Ok(()) } -fn cmd_branches( +fn list_branches( ui: &mut Ui, - command: &CommandHelper, - _args: &BranchesArgs, + workspace_command: &WorkspaceCommandHelper, ) -> Result<(), CommandError> { - let workspace_command = command.workspace_helper(ui)?; let repo = workspace_command.repo(); let workspace_id = workspace_command.workspace_id(); @@ -5145,7 +5230,6 @@ where Commands::Rebase(sub_args) => cmd_rebase(ui, &command_helper, sub_args), Commands::Backout(sub_args) => cmd_backout(ui, &command_helper, sub_args), Commands::Branch(sub_args) => cmd_branch(ui, &command_helper, sub_args), - Commands::Branches(sub_args) => cmd_branches(ui, &command_helper, sub_args), Commands::Undo(sub_args) => cmd_op_undo(ui, &command_helper, sub_args), Commands::Operation(sub_args) => cmd_operation(ui, &command_helper, sub_args), Commands::Workspace(sub_args) => cmd_workspace(ui, &command_helper, sub_args), diff --git a/tests/test_alias.rs b/tests/test_alias.rs index c34c058e395..fe5c92129c6 100644 --- a/tests/test_alias.rs +++ b/tests/test_alias.rs @@ -27,11 +27,11 @@ fn test_alias_basic() { b = ["log", "-r", "@", "-T", "branches"] "#, ); - test_env.jj_cmd_success(&repo_path, &["branch", "my-branch"]); + test_env.jj_cmd_success(&repo_path, &["branch", "create", "my-branch"]); let stdout = test_env.jj_cmd_success(&repo_path, &["b"]); insta::assert_snapshot!(stdout, @r###" @ my-branch - ~ + ~ "###); } diff --git a/tests/test_branch_command.rs b/tests/test_branch_command.rs index 389159fbc48..8da98ccbabb 100644 --- a/tests/test_branch_command.rs +++ b/tests/test_branch_command.rs @@ -35,7 +35,7 @@ fn test_branch_multiple_names() { let repo_path = test_env.env_root().join("repo"); let assert = test_env - .jj_cmd(&repo_path, &["branch", "foo", "bar"]) + .jj_cmd(&repo_path, &["branch", "set", "foo", "bar"]) .assert() .success(); insta::assert_snapshot!(get_stdout_string(&assert), @""); @@ -47,7 +47,7 @@ fn test_branch_multiple_names() { o 000000000000 "###); - let stdout = test_env.jj_cmd_success(&repo_path, &["branch", "--delete", "foo", "bar"]); + let stdout = test_env.jj_cmd_success(&repo_path, &["branch", "delete", "foo", "bar"]); insta::assert_snapshot!(stdout, @""); insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###" @ 230dd059e1b0 @@ -55,21 +55,6 @@ fn test_branch_multiple_names() { "###); } -#[test] -fn test_branch_hint_no_branches() { - let test_env = TestEnvironment::default(); - test_env.jj_cmd_success(test_env.env_root(), &["init", "repo", "--git"]); - let repo_path = test_env.env_root().join("repo"); - - let assert = test_env - .jj_cmd(&repo_path, &["branch", "--delete"]) - .assert() - .success(); - let stderr = get_stderr_string(&assert); - insta::assert_snapshot!(stderr, @"warning: No branches provided. -"); -} - fn get_log_output(test_env: &TestEnvironment, cwd: &Path) -> String { test_env.jj_cmd_success(cwd, &["log", "-T", r#"branches " " commit_id.short()"#]) } diff --git a/tests/test_edit_command.rs b/tests/test_edit_command.rs index 295bdf75da1..8e7b39f86e6 100644 --- a/tests/test_edit_command.rs +++ b/tests/test_edit_command.rs @@ -97,7 +97,7 @@ fn test_edit_merge() { std::fs::write(repo_path.join("file1"), "a\n").unwrap(); std::fs::write(repo_path.join("file2"), "a\n").unwrap(); test_env.jj_cmd_success(&repo_path, &["new"]); - test_env.jj_cmd_success(&repo_path, &["branch", "b"]); + test_env.jj_cmd_success(&repo_path, &["branch", "create", "b"]); std::fs::write(repo_path.join("file1"), "b\n").unwrap(); std::fs::write(repo_path.join("file2"), "b\n").unwrap(); test_env.jj_cmd_success(&repo_path, &["co", "@-"]); diff --git a/tests/test_git_colocated.rs b/tests/test_git_colocated.rs index ba57aa52b1d..06710efe54a 100644 --- a/tests/test_git_colocated.rs +++ b/tests/test_git_colocated.rs @@ -51,7 +51,7 @@ fn test_git_colocated_rebase_on_import() { std::fs::write(workspace_root.join("file"), "contents").unwrap(); test_env.jj_cmd_success(&workspace_root, &["close", "-m", "add a file"]); std::fs::write(workspace_root.join("file"), "modified").unwrap(); - test_env.jj_cmd_success(&workspace_root, &["branch", "master"]); + test_env.jj_cmd_success(&workspace_root, &["branch", "set", "master"]); test_env.jj_cmd_success(&workspace_root, &["close", "-m", "modify a file"]); // TODO: We shouldn't need this command here to trigger an import of the // refs/heads/master we just exported diff --git a/tests/test_git_push.rs b/tests/test_git_push.rs index 7ac3cbd2f71..e7b9b6abec5 100644 --- a/tests/test_git_push.rs +++ b/tests/test_git_push.rs @@ -36,7 +36,7 @@ fn test_git_push() { // When pushing everything, won't push an open commit even if there's a branch // on it - test_env.jj_cmd_success(&workspace_root, &["branch", "my-branch"]); + test_env.jj_cmd_success(&workspace_root, &["branch", "create", "my-branch"]); let stdout = test_env.jj_cmd_success(&workspace_root, &["git", "push"]); insta::assert_snapshot!(stdout, @r###" Skipping branch 'my-branch' since it points to an open commit. @@ -63,7 +63,7 @@ fn test_git_push() { test_env.jj_cmd_success(&workspace_root, &["close", "-m", "second"]); std::fs::write(workspace_root.join("file"), "third").unwrap(); test_env.jj_cmd_success(&workspace_root, &["rebase", "-r", "@", "-d", "@--"]); - test_env.jj_cmd_success(&workspace_root, &["branch", "my-branch"]); + test_env.jj_cmd_success(&workspace_root, &["branch", "set", "my-branch"]); test_env.jj_cmd_success(&workspace_root, &["close", "-m", "third"]); let stderr = test_env.jj_cmd_failure(&workspace_root, &["git", "push"]); insta::assert_snapshot!(stderr, @r###" diff --git a/tests/test_move_command.rs b/tests/test_move_command.rs index 528293279c3..bca282c3fb8 100644 --- a/tests/test_move_command.rs +++ b/tests/test_move_command.rs @@ -35,25 +35,25 @@ fn test_move() { // // When moving changes between e.g. C and F, we should not get unrelated changes // from B and D. - test_env.jj_cmd_success(&repo_path, &["branch", "a"]); + test_env.jj_cmd_success(&repo_path, &["branch", "create", "a"]); std::fs::write(repo_path.join("file1"), "a\n").unwrap(); std::fs::write(repo_path.join("file2"), "a\n").unwrap(); std::fs::write(repo_path.join("file3"), "a\n").unwrap(); test_env.jj_cmd_success(&repo_path, &["new"]); - test_env.jj_cmd_success(&repo_path, &["branch", "b"]); + test_env.jj_cmd_success(&repo_path, &["branch", "create", "b"]); std::fs::write(repo_path.join("file3"), "b\n").unwrap(); test_env.jj_cmd_success(&repo_path, &["new"]); - test_env.jj_cmd_success(&repo_path, &["branch", "c"]); + test_env.jj_cmd_success(&repo_path, &["branch", "create", "c"]); std::fs::write(repo_path.join("file1"), "c\n").unwrap(); test_env.jj_cmd_success(&repo_path, &["co", "a"]); test_env.jj_cmd_success(&repo_path, &["new"]); - test_env.jj_cmd_success(&repo_path, &["branch", "d"]); + test_env.jj_cmd_success(&repo_path, &["branch", "create", "d"]); std::fs::write(repo_path.join("file3"), "d\n").unwrap(); test_env.jj_cmd_success(&repo_path, &["new"]); - test_env.jj_cmd_success(&repo_path, &["branch", "e"]); + test_env.jj_cmd_success(&repo_path, &["branch", "create", "e"]); std::fs::write(repo_path.join("file2"), "e\n").unwrap(); test_env.jj_cmd_success(&repo_path, &["new"]); - test_env.jj_cmd_success(&repo_path, &["branch", "f"]); + test_env.jj_cmd_success(&repo_path, &["branch", "create", "f"]); std::fs::write(repo_path.join("file2"), "f\n").unwrap(); // Test the setup insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###" @@ -162,20 +162,20 @@ fn test_move_partial() { // D B // |/ // A - test_env.jj_cmd_success(&repo_path, &["branch", "a"]); + test_env.jj_cmd_success(&repo_path, &["branch", "create", "a"]); std::fs::write(repo_path.join("file1"), "a\n").unwrap(); std::fs::write(repo_path.join("file2"), "a\n").unwrap(); std::fs::write(repo_path.join("file3"), "a\n").unwrap(); test_env.jj_cmd_success(&repo_path, &["new"]); - test_env.jj_cmd_success(&repo_path, &["branch", "b"]); + test_env.jj_cmd_success(&repo_path, &["branch", "create", "b"]); std::fs::write(repo_path.join("file3"), "b\n").unwrap(); test_env.jj_cmd_success(&repo_path, &["new"]); - test_env.jj_cmd_success(&repo_path, &["branch", "c"]); + test_env.jj_cmd_success(&repo_path, &["branch", "create", "c"]); std::fs::write(repo_path.join("file1"), "c\n").unwrap(); std::fs::write(repo_path.join("file2"), "c\n").unwrap(); test_env.jj_cmd_success(&repo_path, &["co", "a"]); test_env.jj_cmd_success(&repo_path, &["new"]); - test_env.jj_cmd_success(&repo_path, &["branch", "d"]); + test_env.jj_cmd_success(&repo_path, &["branch", "create", "d"]); std::fs::write(repo_path.join("file3"), "d\n").unwrap(); // Test the setup insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###" diff --git a/tests/test_rebase_command.rs b/tests/test_rebase_command.rs index a09a440a787..5f328e8df6e 100644 --- a/tests/test_rebase_command.rs +++ b/tests/test_rebase_command.rs @@ -32,7 +32,7 @@ fn create_commit(test_env: &TestEnvironment, repo_path: &Path, name: &str, paren test_env.jj_cmd_success(repo_path, &["co", "@-"]); } std::fs::write(repo_path.join(name), &format!("{name}\n")).unwrap(); - test_env.jj_cmd_success(repo_path, &["branch", name]); + test_env.jj_cmd_success(repo_path, &["branch", "create", name]); test_env.jj_cmd_success(repo_path, &["close", "-m", name]); } diff --git a/tests/test_squash_command.rs b/tests/test_squash_command.rs index fd09a55137d..dfac6a3a378 100644 --- a/tests/test_squash_command.rs +++ b/tests/test_squash_command.rs @@ -22,13 +22,13 @@ fn test_squash() { test_env.jj_cmd_success(test_env.env_root(), &["init", "repo", "--git"]); let repo_path = test_env.env_root().join("repo"); - test_env.jj_cmd_success(&repo_path, &["branch", "a"]); + test_env.jj_cmd_success(&repo_path, &["branch", "create", "a"]); std::fs::write(repo_path.join("file1"), "a\n").unwrap(); test_env.jj_cmd_success(&repo_path, &["new"]); - test_env.jj_cmd_success(&repo_path, &["branch", "b"]); + test_env.jj_cmd_success(&repo_path, &["branch", "create", "b"]); std::fs::write(repo_path.join("file1"), "b\n").unwrap(); test_env.jj_cmd_success(&repo_path, &["new"]); - test_env.jj_cmd_success(&repo_path, &["branch", "c"]); + test_env.jj_cmd_success(&repo_path, &["branch", "create", "c"]); std::fs::write(repo_path.join("file1"), "c\n").unwrap(); // Test the setup let template = r#"commit_id.short() " " branches"#; @@ -83,10 +83,10 @@ fn test_squash() { test_env.jj_cmd_success(&repo_path, &["undo"]); test_env.jj_cmd_success(&repo_path, &["co", "b"]); test_env.jj_cmd_success(&repo_path, &["new"]); - test_env.jj_cmd_success(&repo_path, &["branch", "d"]); + test_env.jj_cmd_success(&repo_path, &["branch", "create", "d"]); std::fs::write(repo_path.join("file2"), "d\n").unwrap(); test_env.jj_cmd_success(&repo_path, &["merge", "-m", "merge", "c", "d"]); - test_env.jj_cmd_success(&repo_path, &["branch", "e", "-r", "@+"]); + test_env.jj_cmd_success(&repo_path, &["branch", "create", "e", "-r", "@+"]); let stdout = test_env.jj_cmd_success(&repo_path, &["log", "-T", template]); insta::assert_snapshot!(stdout, @r###" o b9ad3fdfc2c4 e @@ -134,15 +134,15 @@ fn test_squash_partial() { test_env.jj_cmd_success(test_env.env_root(), &["init", "repo", "--git"]); let repo_path = test_env.env_root().join("repo"); - test_env.jj_cmd_success(&repo_path, &["branch", "a"]); + test_env.jj_cmd_success(&repo_path, &["branch", "create", "a"]); std::fs::write(repo_path.join("file1"), "a\n").unwrap(); std::fs::write(repo_path.join("file2"), "a\n").unwrap(); test_env.jj_cmd_success(&repo_path, &["new"]); - test_env.jj_cmd_success(&repo_path, &["branch", "b"]); + test_env.jj_cmd_success(&repo_path, &["branch", "create", "b"]); std::fs::write(repo_path.join("file1"), "b\n").unwrap(); std::fs::write(repo_path.join("file2"), "b\n").unwrap(); test_env.jj_cmd_success(&repo_path, &["new"]); - test_env.jj_cmd_success(&repo_path, &["branch", "c"]); + test_env.jj_cmd_success(&repo_path, &["branch", "create", "c"]); std::fs::write(repo_path.join("file1"), "c\n").unwrap(); std::fs::write(repo_path.join("file2"), "c\n").unwrap(); // Test the setup