forked from mirrors/jj
cli: run "op abandon" without loading repo, reject --at-op
If indexing failed due to missing commit objects, the repo won't be loadable without --ignore-working-copy (at least in colocated environment.) In that case, we can use "op abandon" to recover, but we had to work around the failed index loading by --ignore-working-copy. Since "op abandon" isn't a repo-level command, it's better to bypass working-copy snapshot and import of git refs at all. --at-op is rejected because it's useless and we'll need extra care for "@" expression resolution and working-copy updates.
This commit is contained in:
parent
f70107dad9
commit
faa9b8d77f
2 changed files with 98 additions and 17 deletions
|
@ -23,7 +23,8 @@ use jj_lib::op_walk;
|
||||||
use jj_lib::repo::Repo;
|
use jj_lib::repo::Repo;
|
||||||
|
|
||||||
use crate::cli_util::{
|
use crate::cli_util::{
|
||||||
user_error, user_error_with_hint, CommandError, CommandHelper, LogContentFormat,
|
short_operation_hash, user_error, user_error_with_hint, CommandError, CommandHelper,
|
||||||
|
LogContentFormat,
|
||||||
};
|
};
|
||||||
use crate::graphlog::{get_graphlog, Edge};
|
use crate::graphlog::{get_graphlog, Edge};
|
||||||
use crate::operation_templater;
|
use crate::operation_templater;
|
||||||
|
@ -279,27 +280,37 @@ fn cmd_op_abandon(
|
||||||
command: &CommandHelper,
|
command: &CommandHelper,
|
||||||
args: &OperationAbandonArgs,
|
args: &OperationAbandonArgs,
|
||||||
) -> Result<(), CommandError> {
|
) -> Result<(), CommandError> {
|
||||||
let mut workspace_command = command.workspace_helper(ui)?;
|
// Don't load the repo so that this command can be used to recover from
|
||||||
let repo = workspace_command.repo();
|
// corrupted repo state.
|
||||||
let current_head_op = repo.operation();
|
let mut workspace = command.load_workspace()?;
|
||||||
|
let repo_loader = workspace.repo_loader();
|
||||||
|
let op_store = repo_loader.op_store();
|
||||||
|
// It doesn't make sense to create concurrent operations that will be merged
|
||||||
|
// with the current head.
|
||||||
|
let head_op_str = &command.global_args().at_operation;
|
||||||
|
if head_op_str != "@" {
|
||||||
|
return Err(user_error("--at-op is not respected"));
|
||||||
|
}
|
||||||
|
let current_head_op = op_walk::resolve_op_for_load(repo_loader, head_op_str)?;
|
||||||
|
let resolve_op = |op_str| op_walk::resolve_op_at(op_store, ¤t_head_op, op_str);
|
||||||
let (abandon_root_op, abandon_head_op) =
|
let (abandon_root_op, abandon_head_op) =
|
||||||
if let Some((root_op_str, head_op_str)) = args.operation.split_once("..") {
|
if let Some((root_op_str, head_op_str)) = args.operation.split_once("..") {
|
||||||
let root_op = if root_op_str.is_empty() {
|
let root_op = if root_op_str.is_empty() {
|
||||||
// TODO: Introduce a virtual root operation and use it instead.
|
// TODO: Introduce a virtual root operation and use it instead.
|
||||||
op_walk::walk_ancestors(slice::from_ref(current_head_op))
|
op_walk::walk_ancestors(slice::from_ref(¤t_head_op))
|
||||||
.last()
|
.last()
|
||||||
.unwrap()?
|
.unwrap()?
|
||||||
} else {
|
} else {
|
||||||
workspace_command.resolve_single_op(root_op_str)?
|
resolve_op(root_op_str)?
|
||||||
};
|
};
|
||||||
let head_op = if head_op_str.is_empty() {
|
let head_op = if head_op_str.is_empty() {
|
||||||
current_head_op.clone()
|
current_head_op.clone()
|
||||||
} else {
|
} else {
|
||||||
workspace_command.resolve_single_op(head_op_str)?
|
resolve_op(head_op_str)?
|
||||||
};
|
};
|
||||||
(root_op, head_op)
|
(root_op, head_op)
|
||||||
} else {
|
} else {
|
||||||
let op = workspace_command.resolve_single_op(&args.operation)?;
|
let op = resolve_op(&args.operation)?;
|
||||||
let parent_ops: Vec<_> = op.parents().try_collect()?;
|
let parent_ops: Vec<_> = op.parents().try_collect()?;
|
||||||
let parent_op = match parent_ops.len() {
|
let parent_op = match parent_ops.len() {
|
||||||
0 => return Err(user_error("Cannot abandon the root operation")),
|
0 => return Err(user_error("Cannot abandon the root operation")),
|
||||||
|
@ -309,7 +320,7 @@ fn cmd_op_abandon(
|
||||||
(parent_op, op)
|
(parent_op, op)
|
||||||
};
|
};
|
||||||
|
|
||||||
if abandon_head_op == *current_head_op {
|
if abandon_head_op == current_head_op {
|
||||||
return Err(user_error_with_hint(
|
return Err(user_error_with_hint(
|
||||||
"Cannot abandon the current operation",
|
"Cannot abandon the current operation",
|
||||||
"Run `jj undo` to revert the current operation, then use `jj op abandon`",
|
"Run `jj undo` to revert the current operation, then use `jj op abandon`",
|
||||||
|
@ -318,9 +329,9 @@ fn cmd_op_abandon(
|
||||||
|
|
||||||
// Reparent descendants, count the number of abandoned operations.
|
// Reparent descendants, count the number of abandoned operations.
|
||||||
let stats = op_walk::reparent_range(
|
let stats = op_walk::reparent_range(
|
||||||
repo.op_store().as_ref(),
|
op_store.as_ref(),
|
||||||
slice::from_ref(&abandon_head_op),
|
slice::from_ref(&abandon_head_op),
|
||||||
slice::from_ref(current_head_op),
|
slice::from_ref(¤t_head_op),
|
||||||
&abandon_root_op,
|
&abandon_root_op,
|
||||||
)?;
|
)?;
|
||||||
let [new_head_id]: [OperationId; 1] = stats.new_head_ids.try_into().unwrap();
|
let [new_head_id]: [OperationId; 1] = stats.new_head_ids.try_into().unwrap();
|
||||||
|
@ -334,12 +345,25 @@ fn cmd_op_abandon(
|
||||||
stats.unreachable_count,
|
stats.unreachable_count,
|
||||||
stats.rewritten_count,
|
stats.rewritten_count,
|
||||||
)?;
|
)?;
|
||||||
repo.op_heads_store()
|
repo_loader
|
||||||
|
.op_heads_store()
|
||||||
.update_op_heads(slice::from_ref(current_head_op.id()), &new_head_id);
|
.update_op_heads(slice::from_ref(current_head_op.id()), &new_head_id);
|
||||||
// Remap the operation id of the current workspace. If there were any
|
// Remap the operation id of the current workspace. If there were any
|
||||||
// concurrent operations, user will need to re-abandon their ancestors.
|
// concurrent operations, user will need to re-abandon their ancestors.
|
||||||
let (locked_ws, _) = workspace_command.start_working_copy_mutation()?;
|
if !command.global_args().ignore_working_copy {
|
||||||
locked_ws.finish(new_head_id)?;
|
let mut locked_ws = workspace.start_working_copy_mutation()?;
|
||||||
|
let old_op_id = locked_ws.locked_wc().old_operation_id();
|
||||||
|
if old_op_id != current_head_op.id() {
|
||||||
|
writeln!(
|
||||||
|
ui.warning(),
|
||||||
|
"The working copy operation {} is not updated because it differs from the repo {}.",
|
||||||
|
short_operation_hash(old_op_id),
|
||||||
|
short_operation_hash(current_head_op.id()),
|
||||||
|
)?;
|
||||||
|
} else {
|
||||||
|
locked_ws.finish(new_head_id)?
|
||||||
|
}
|
||||||
|
}
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -344,6 +344,12 @@ fn test_op_abandon_ancestors() {
|
||||||
Hint: Run `jj undo` to revert the current operation, then use `jj op abandon`
|
Hint: Run `jj undo` to revert the current operation, then use `jj op abandon`
|
||||||
"###);
|
"###);
|
||||||
|
|
||||||
|
// Can't create concurrent abandoned operations explicitly.
|
||||||
|
let stderr = test_env.jj_cmd_failure(&repo_path, &["op", "abandon", "--at-op=@-", "@"]);
|
||||||
|
insta::assert_snapshot!(stderr, @r###"
|
||||||
|
Error: --at-op is not respected
|
||||||
|
"###);
|
||||||
|
|
||||||
// Abandon the current operation by undoing it first.
|
// Abandon the current operation by undoing it first.
|
||||||
test_env.jj_cmd_ok(&repo_path, &["undo"]);
|
test_env.jj_cmd_ok(&repo_path, &["undo"]);
|
||||||
let (_stdout, stderr) = test_env.jj_cmd_ok(&repo_path, &["op", "abandon", "@-"]);
|
let (_stdout, stderr) = test_env.jj_cmd_ok(&repo_path, &["op", "abandon", "@-"]);
|
||||||
|
@ -352,11 +358,11 @@ fn test_op_abandon_ancestors() {
|
||||||
"###);
|
"###);
|
||||||
insta::assert_snapshot!(
|
insta::assert_snapshot!(
|
||||||
test_env.jj_cmd_success(&repo_path, &["debug", "workingcopy", "--ignore-working-copy"]), @r###"
|
test_env.jj_cmd_success(&repo_path, &["debug", "workingcopy", "--ignore-working-copy"]), @r###"
|
||||||
Current operation: OperationId("05aebafee59813d56c0ea1576520b3074f5ba3e128f2b31df7370284cee593bed5043475dc2cdd30a6f22662c1dfb6aba92b83806147e77c17ad14356c07079d")
|
Current operation: OperationId("571221174898ef510c580f96bfb54f720bcfe0cd457e2ac5531d511fd10762b883f89b06c4ce5a8924db74406ddb59adbc2cbe0204540c7749ca24ded3fce94b")
|
||||||
Current tree: Legacy(TreeId("4b825dc642cb6eb9a060e54bf8d69288fbee4904"))
|
Current tree: Legacy(TreeId("4b825dc642cb6eb9a060e54bf8d69288fbee4904"))
|
||||||
"###);
|
"###);
|
||||||
insta::assert_snapshot!(test_env.jj_cmd_success(&repo_path, &["op", "log"]), @r###"
|
insta::assert_snapshot!(test_env.jj_cmd_success(&repo_path, &["op", "log"]), @r###"
|
||||||
@ 05aebafee598 test-username@host.example.com 2001-02-03 04:05:20.000 +07:00 - 2001-02-03 04:05:20.000 +07:00
|
@ 571221174898 test-username@host.example.com 2001-02-03 04:05:21.000 +07:00 - 2001-02-03 04:05:21.000 +07:00
|
||||||
│ undo operation ee40c9ad806a7d42f351beab5aa81a8ac38d926d02711c059229bf6a7388b7b4a7c04c004067ee6c5b6253e8398fa82bc74d0d621f8bc2c8c11f33d445f90b77
|
│ undo operation ee40c9ad806a7d42f351beab5aa81a8ac38d926d02711c059229bf6a7388b7b4a7c04c004067ee6c5b6253e8398fa82bc74d0d621f8bc2c8c11f33d445f90b77
|
||||||
│ args: jj undo
|
│ args: jj undo
|
||||||
◉ fb5252a68411 test-username@host.example.com 2001-02-03 04:05:09.000 +07:00 - 2001-02-03 04:05:09.000 +07:00
|
◉ fb5252a68411 test-username@host.example.com 2001-02-03 04:05:09.000 +07:00 - 2001-02-03 04:05:09.000 +07:00
|
||||||
|
@ -372,12 +378,63 @@ fn test_op_abandon_ancestors() {
|
||||||
Nothing changed.
|
Nothing changed.
|
||||||
"###);
|
"###);
|
||||||
insta::assert_snapshot!(test_env.jj_cmd_success(&repo_path, &["op", "log", "-l1"]), @r###"
|
insta::assert_snapshot!(test_env.jj_cmd_success(&repo_path, &["op", "log", "-l1"]), @r###"
|
||||||
@ 05aebafee598 test-username@host.example.com 2001-02-03 04:05:20.000 +07:00 - 2001-02-03 04:05:20.000 +07:00
|
@ 571221174898 test-username@host.example.com 2001-02-03 04:05:21.000 +07:00 - 2001-02-03 04:05:21.000 +07:00
|
||||||
│ undo operation ee40c9ad806a7d42f351beab5aa81a8ac38d926d02711c059229bf6a7388b7b4a7c04c004067ee6c5b6253e8398fa82bc74d0d621f8bc2c8c11f33d445f90b77
|
│ undo operation ee40c9ad806a7d42f351beab5aa81a8ac38d926d02711c059229bf6a7388b7b4a7c04c004067ee6c5b6253e8398fa82bc74d0d621f8bc2c8c11f33d445f90b77
|
||||||
│ args: jj undo
|
│ args: jj undo
|
||||||
"###);
|
"###);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_op_abandon_without_updating_working_copy() {
|
||||||
|
let test_env = TestEnvironment::default();
|
||||||
|
test_env.jj_cmd_ok(test_env.env_root(), &["init", "repo", "--git"]);
|
||||||
|
let repo_path = test_env.env_root().join("repo");
|
||||||
|
|
||||||
|
test_env.jj_cmd_ok(&repo_path, &["commit", "-m", "commit 1"]);
|
||||||
|
test_env.jj_cmd_ok(&repo_path, &["commit", "-m", "commit 2"]);
|
||||||
|
test_env.jj_cmd_ok(&repo_path, &["commit", "-m", "commit 3"]);
|
||||||
|
|
||||||
|
// Abandon without updating the working copy.
|
||||||
|
let (_stdout, stderr) = test_env.jj_cmd_ok(
|
||||||
|
&repo_path,
|
||||||
|
&["op", "abandon", "@-", "--ignore-working-copy"],
|
||||||
|
);
|
||||||
|
insta::assert_snapshot!(stderr, @r###"
|
||||||
|
Abandoned 1 operations and reparented 1 descendant operations.
|
||||||
|
"###);
|
||||||
|
insta::assert_snapshot!(
|
||||||
|
test_env.jj_cmd_success(&repo_path, &["debug", "workingcopy", "--ignore-working-copy"]), @r###"
|
||||||
|
Current operation: OperationId("a6a87becb46cb138b33b8bc238ff066e1141da3e988574260ab54db676a68a070a592cacfd37e06177f604882f7de01189e2efb75148bc00ee5f9f55513feb26")
|
||||||
|
Current tree: Legacy(TreeId("4b825dc642cb6eb9a060e54bf8d69288fbee4904"))
|
||||||
|
"###);
|
||||||
|
insta::assert_snapshot!(
|
||||||
|
test_env.jj_cmd_success(&repo_path, &["op", "log", "-l1", "--ignore-working-copy"]), @r###"
|
||||||
|
@ 5dca1e30e810 test-username@host.example.com 2001-02-03 04:05:10.000 +07:00 - 2001-02-03 04:05:10.000 +07:00
|
||||||
|
│ commit 268f5f16139313ff25bef31280b2ec2e675200f3
|
||||||
|
│ args: jj commit -m 'commit 3'
|
||||||
|
"###);
|
||||||
|
|
||||||
|
// The working-copy operation id isn't updated if it differs from the repo.
|
||||||
|
// It could be updated if the tree matches, but there's no extra logic for
|
||||||
|
// that.
|
||||||
|
let (_stdout, stderr) = test_env.jj_cmd_ok(&repo_path, &["op", "abandon", "@-"]);
|
||||||
|
insta::assert_snapshot!(stderr, @r###"
|
||||||
|
Abandoned 1 operations and reparented 1 descendant operations.
|
||||||
|
The working copy operation a6a87becb46c is not updated because it differs from the repo 5dca1e30e810.
|
||||||
|
"###);
|
||||||
|
insta::assert_snapshot!(
|
||||||
|
test_env.jj_cmd_success(&repo_path, &["debug", "workingcopy", "--ignore-working-copy"]), @r###"
|
||||||
|
Current operation: OperationId("a6a87becb46cb138b33b8bc238ff066e1141da3e988574260ab54db676a68a070a592cacfd37e06177f604882f7de01189e2efb75148bc00ee5f9f55513feb26")
|
||||||
|
Current tree: Legacy(TreeId("4b825dc642cb6eb9a060e54bf8d69288fbee4904"))
|
||||||
|
"###);
|
||||||
|
insta::assert_snapshot!(
|
||||||
|
test_env.jj_cmd_success(&repo_path, &["op", "log", "-l1", "--ignore-working-copy"]), @r###"
|
||||||
|
@ e3b64811d26b test-username@host.example.com 2001-02-03 04:05:10.000 +07:00 - 2001-02-03 04:05:10.000 +07:00
|
||||||
|
│ commit 268f5f16139313ff25bef31280b2ec2e675200f3
|
||||||
|
│ args: jj commit -m 'commit 3'
|
||||||
|
"###);
|
||||||
|
}
|
||||||
|
|
||||||
fn get_log_output(test_env: &TestEnvironment, repo_path: &Path, op_id: &str) -> String {
|
fn get_log_output(test_env: &TestEnvironment, repo_path: &Path, op_id: &str) -> String {
|
||||||
test_env.jj_cmd_success(
|
test_env.jj_cmd_success(
|
||||||
repo_path,
|
repo_path,
|
||||||
|
|
Loading…
Reference in a new issue