diff --git a/cli/tests/test_rebase_command.rs b/cli/tests/test_rebase_command.rs index 888d517dc7..d5e8d74afa 100644 --- a/cli/tests/test_rebase_command.rs +++ b/cli/tests/test_rebase_command.rs @@ -181,11 +181,11 @@ fn test_rebase_branch() { create_commit(&test_env, &repo_path, "e", &["a"]); // Test the setup insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###" - @ e - │ ○ d - │ │ ○ c + @ e: a + │ ○ d: b + │ │ ○ c: b │ ├─╯ - │ ○ b + │ ○ b: a ├─╯ ○ a ◆ @@ -197,11 +197,11 @@ fn test_rebase_branch() { Rebased 3 commits "###); insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###" - ○ d - │ ○ c + ○ d: b + │ ○ c: b ├─╯ - ○ b - @ e + ○ b: e + @ e: a ○ a ◆ "###); @@ -218,12 +218,12 @@ fn test_rebase_branch() { Added 1 files, modified 0 files, removed 0 files "###); insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###" - @ e - │ ○ d + @ e: b + │ ○ d: b ├─╯ - │ ○ c + │ ○ c: b ├─╯ - ○ b + ○ b: a ○ a ◆ "###); @@ -248,12 +248,12 @@ fn test_rebase_branch() { Added 1 files, modified 0 files, removed 0 files "###); insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###" - @ e - │ ○ d + @ e: b + │ ○ d: b ├─╯ - │ ○ c + │ ○ c: b ├─╯ - ○ b + ○ b: a ○ a ◆ "###); @@ -272,11 +272,11 @@ fn test_rebase_branch_with_merge() { create_commit(&test_env, &repo_path, "e", &["a", "d"]); // Test the setup insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###" - @ e + @ e: a d ├─╮ - │ ○ d + │ ○ d: c │ ○ c - │ │ ○ b + │ │ ○ b: a ├───╯ ○ │ a ├─╯ @@ -293,11 +293,11 @@ fn test_rebase_branch_with_merge() { Added 1 files, modified 0 files, removed 0 files "###); insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###" - @ e + @ e: a d ├─╮ - │ ○ d - │ ○ c - │ ○ b + │ ○ d: c + │ ○ c: b + │ ○ b: a ├─╯ ○ a ◆ @@ -314,11 +314,11 @@ fn test_rebase_branch_with_merge() { Added 1 files, modified 0 files, removed 0 files "###); insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###" - @ e + @ e: a d ├─╮ - │ ○ d - │ ○ c - │ ○ b + │ ○ d: c + │ ○ c: b + │ ○ b: a ├─╯ ○ a ◆ @@ -338,11 +338,11 @@ fn test_rebase_single_revision() { create_commit(&test_env, &repo_path, "e", &["d"]); // Test the setup insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###" - @ e - ○ d + @ e: d + ○ d: b c ├─╮ - │ ○ c - ○ │ b + │ ○ c: a + ○ │ b: a ├─╯ ○ a ◆ @@ -360,12 +360,12 @@ fn test_rebase_single_revision() { Added 0 files, modified 0 files, removed 1 files "###); insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###" - @ e - ○ d + @ e: d + ○ d: b a ├─╮ - │ │ ○ c + │ │ ○ c: b ├───╯ - ○ │ b + ○ │ b: a ├─╯ ○ a ◆ @@ -385,12 +385,12 @@ fn test_rebase_single_revision() { Added 0 files, modified 0 files, removed 1 files "###); insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###" - @ e + @ e: b c ├─╮ - │ ○ c - ○ │ b + │ ○ c: a + ○ │ b: a ├─╯ - │ ○ d + │ ○ d: a ├─╯ ○ a ◆ @@ -409,9 +409,9 @@ fn test_rebase_single_revision_merge_parent() { create_commit(&test_env, &repo_path, "d", &["a", "c"]); // Test the setup insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###" - @ d + @ d: a c ├─╮ - │ ○ c + │ ○ c: b │ ○ b ○ │ a ├─╯ @@ -431,10 +431,10 @@ fn test_rebase_single_revision_merge_parent() { Added 0 files, modified 0 files, removed 1 files "###); insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###" - @ d + @ d: a b ├─╮ │ ○ b - │ │ ○ c + │ │ ○ c: a ├───╯ ○ │ a ├─╯ @@ -459,16 +459,16 @@ fn test_rebase_multiple_revisions() { create_commit(&test_env, &repo_path, "i", &["f"]); // Test the setup insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###" - @ i - │ ○ h - │ ○ g + @ i: f + │ ○ h: g + │ ○ g: f ├─╯ - ○ f + ○ f: c e ├─╮ - │ ○ e - │ ○ d - ○ │ c - ○ │ b + │ ○ e: d + │ ○ d: a + ○ │ c: b + ○ │ b: a ├─╯ ○ a ◆ @@ -486,18 +486,18 @@ fn test_rebase_multiple_revisions() { Added 0 files, modified 0 files, removed 2 files "###); insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###" - @ i - │ ○ h - │ ○ g + @ i: f + │ ○ h: g + │ ○ g: f ├─╯ - ○ f + ○ f: b d ├─╮ - │ ○ d - ○ │ b + │ ○ d: a + ○ │ b: a ├─╯ - │ ○ e + │ ○ e: a ├─╯ - │ ○ c + │ ○ c: a ├─╯ ○ a ◆ @@ -518,17 +518,17 @@ fn test_rebase_multiple_revisions() { Added 0 files, modified 0 files, removed 2 files "###); insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###" - @ i - │ ○ h - │ ○ g + @ i: f + │ ○ h: g + │ ○ g: f ├─╯ - ○ f + ○ f: a e ├─╮ - │ │ ○ c - │ │ ○ b + │ │ ○ c: b + │ │ ○ b: e │ ├─╯ - │ ○ e - │ ○ d + │ ○ e: d + │ ○ d: a ├─╯ ○ a ◆ @@ -538,6 +538,8 @@ fn test_rebase_multiple_revisions() { // Test with a subgraph containing a merge commit. Since the merge commit "f" // was extracted, its descendants which are not part of the subgraph will // inherit its descendants which are not in the subtree ("c" and "d"). + // "f" will retain its parent "c" since "c" is outside the target set, and not + // a descendant of any new children. let (stdout, stderr) = test_env.jj_cmd_ok(&repo_path, &["rebase", "-r", "e::g", "-d", "a"]); insta::assert_snapshot!(stdout, @""); insta::assert_snapshot!(stderr, @r###" @@ -549,17 +551,18 @@ fn test_rebase_multiple_revisions() { Added 0 files, modified 0 files, removed 2 files "###); insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###" - @ i + @ i: c d ├─╮ - │ │ ○ h + │ │ ○ h: c d ╭─┬─╯ - │ ○ d - ○ │ c - ○ │ b - ├─╯ - │ ○ g - │ ○ f - │ ○ e + │ ○ d: a + │ │ ○ g: f + │ │ ○ f: c e + ╭───┤ + │ │ ○ e: a + │ ├─╯ + ○ │ c: b + ○ │ b: a ├─╯ ○ a ◆ @@ -569,7 +572,8 @@ fn test_rebase_multiple_revisions() { // Test with commits in a disconnected subgraph. The subgraph has the // relationship d->e->f->g->h, but only "d", "f" and "h" are in the set of // rebased commits. "d" should be a new parent of "f", and "f" should be a - // new parent of "g". + // new parent of "h". "f" will retain its parent "c" since "c" is outside the + // target set, and not a descendant of any new children. let (stdout, stderr) = test_env.jj_cmd_ok( &repo_path, &["rebase", "-r", "d", "-r", "f", "-r", "h", "-d", "b"], @@ -584,17 +588,18 @@ fn test_rebase_multiple_revisions() { Added 0 files, modified 0 files, removed 2 files "###); insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###" - @ i + @ i: c e ├─╮ - │ │ ○ g + │ │ ○ g: c e ╭─┬─╯ - │ ○ e - ○ │ c - │ │ ○ h - │ │ ○ f - │ │ ○ d + │ ○ e: a + │ │ ○ h: f + │ │ ○ f: c d + ╭───┤ + │ │ ○ d: b + ○ │ │ c: b ├───╯ - ○ │ b + ○ │ b: a ├─╯ ○ a ◆ @@ -612,16 +617,16 @@ fn test_rebase_multiple_revisions() { Added 0 files, modified 0 files, removed 2 files "###); insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###" - ○ h - ○ g - │ ○ e - │ ○ d - │ @ i + ○ h: g + ○ g: f + │ ○ e: d + │ ○ d: i + │ @ i: f ├─╯ - ○ f + ○ f: c a ├─╮ - ○ │ c - ○ │ b + ○ │ c: b + ○ │ b: a ├─╯ ○ a ◆ @@ -640,10 +645,10 @@ fn test_rebase_revision_onto_descendant() { create_commit(&test_env, &repo_path, "merge", &["b", "a"]); // Test the setup insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###" - @ merge + @ merge: b a ├─╮ - │ ○ a - ○ │ b + │ ○ a: base + ○ │ b: base ├─╯ ○ base ◆ @@ -662,10 +667,10 @@ fn test_rebase_revision_onto_descendant() { Added 0 files, modified 0 files, removed 1 files "###); insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###" - @ merge + @ merge: b a ├─╮ ○ │ b - │ │ ○ base + │ │ ○ base: a │ ├─╯ │ ○ a ├─╯ @@ -692,8 +697,8 @@ fn test_rebase_revision_onto_descendant() { Added 0 files, modified 0 files, removed 1 files "###); insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###" - ○ base - @ merge + ○ base: merge + @ merge: b a ├─╮ │ ○ a ○ │ b @@ -731,7 +736,7 @@ fn test_rebase_multiple_destinations() { Rebased 1 commits onto destination "###); insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###" - ○ a + ○ a: b c ├─╮ │ @ c ○ │ b @@ -755,7 +760,7 @@ fn test_rebase_multiple_destinations() { Rebased 1 commits onto destination "###); insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###" - ○ a + ○ a: c b ├─╮ │ ○ b @ │ c @@ -775,7 +780,7 @@ fn test_rebase_multiple_destinations() { ], ); insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###" - ○ a + ○ a: c b ├─╮ │ ○ b @ │ c @@ -818,8 +823,8 @@ fn test_rebase_with_descendants() { create_commit(&test_env, &repo_path, "d", &["c"]); // Test the setup insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###" - @ d - ○ c + @ d: c + ○ c: a b ├─╮ │ ○ b ○ │ a @@ -835,10 +840,10 @@ fn test_rebase_with_descendants() { Parent commit : royxmykx 57c7246a c | c "###); insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###" - @ d - ○ c + @ d: c + ○ c: a b ├─╮ - │ ○ b + │ ○ b: a ├─╯ ○ a ◆ @@ -855,8 +860,8 @@ fn test_rebase_with_descendants() { Added 0 files, modified 0 files, removed 2 files "###); insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###" - @ d - │ ○ c + @ d: a + │ ○ c: a ├─╯ ○ a │ ○ b @@ -867,8 +872,8 @@ fn test_rebase_with_descendants() { test_env.jj_cmd_ok(&repo_path, &["undo"]); // Reminder of the setup insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###" - @ d - ○ c + @ d: c + ○ c: a b ├─╮ │ ○ b ○ │ a @@ -887,11 +892,11 @@ fn test_rebase_with_descendants() { Added 0 files, modified 0 files, removed 2 files "###); insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###" - ○ c + ○ c: a b ├─╮ - │ ○ b + │ ○ b: a ├─╯ - │ @ d + │ @ d: a ├─╯ ○ a ◆ @@ -916,11 +921,11 @@ fn test_rebase_with_descendants() { Added 0 files, modified 0 files, removed 2 files "###); insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###" - ○ c + ○ c: a b ├─╮ - │ ○ b + │ ○ b: a ├─╯ - │ @ d + │ @ d: a ├─╯ ○ a ◆ @@ -964,12 +969,12 @@ fn test_rebase_with_child_and_descendant_bug_2600() { // Test the setup insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###" - @ c - ○ b + @ c: b + ○ b: base a ├─╮ - │ ○ a + │ ○ a: base ├─╯ - ○ base + ○ base: notroot ○ notroot ◆ "###); @@ -983,12 +988,12 @@ fn test_rebase_with_child_and_descendant_bug_2600() { Skipped rebase of 1 commits that were already in place "###); insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###" - @ c - ○ b + @ c: b + ○ b: base a ├─╮ - │ ○ a + │ ○ a: base ├─╯ - ○ base + ○ base: notroot ○ notroot ◆ "###); @@ -1001,12 +1006,12 @@ fn test_rebase_with_child_and_descendant_bug_2600() { Skipped rebase of 1 commits that were already in place "###); insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###" - @ c - ○ b + @ c: b + ○ b: base a ├─╮ - │ ○ a + │ ○ a: base ├─╯ - ○ base + ○ base: notroot ○ notroot ◆ "###); @@ -1022,11 +1027,11 @@ fn test_rebase_with_child_and_descendant_bug_2600() { // Commit "a" should be rebased onto the root commit. Commit "b" should have // "base" and "a" as parents as before. insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###" - @ c - ○ b + @ c: b + ○ b: base a ├─╮ │ ○ a - ○ │ base + ○ │ base: notroot ○ │ notroot ├─╯ ◆ @@ -1036,12 +1041,12 @@ fn test_rebase_with_child_and_descendant_bug_2600() { // ====== Reminder of the setup ========= test_env.jj_cmd_ok(&repo_path, &["op", "restore", &setup_opid]); insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###" - @ c - ○ b + @ c: b + ○ b: base a ├─╮ - │ ○ a + │ ○ a: base ├─╯ - ○ base + ○ base: notroot ○ notroot ◆ "###); @@ -1054,12 +1059,12 @@ fn test_rebase_with_child_and_descendant_bug_2600() { Skipped rebase of 1 commits that were already in place "###); insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###" - @ c - ○ b + @ c: b + ○ b: base a ├─╮ - │ ○ a + │ ○ a: base ├─╯ - ○ base + ○ base: notroot ○ notroot ◆ "###); @@ -1075,10 +1080,10 @@ fn test_rebase_with_child_and_descendant_bug_2600() { // The commits in roots(a..c), i.e. commit "b" should be rebased onto "a", // which means "b" loses its "base" parent insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###" - @ c - ○ b - ○ a - ○ base + @ c: b + ○ b: a + ○ a: base + ○ base: notroot ○ notroot ◆ "###); @@ -1091,12 +1096,12 @@ fn test_rebase_with_child_and_descendant_bug_2600() { Skipped rebase of 1 commits that were already in place "###); insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###" - @ c - ○ b + @ c: b + ○ b: base a ├─╮ - │ ○ a + │ ○ a: base ├─╯ - ○ base + ○ base: notroot ○ notroot ◆ "###); @@ -1105,12 +1110,12 @@ fn test_rebase_with_child_and_descendant_bug_2600() { // ====== Reminder of the setup ========= test_env.jj_cmd_ok(&repo_path, &["op", "restore", &setup_opid]); insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###" - @ c - ○ b + @ c: b + ○ b: base a ├─╮ - │ ○ a + │ ○ a: base ├─╯ - ○ base + ○ base: notroot ○ notroot ◆ "###); @@ -1127,10 +1132,10 @@ fn test_rebase_with_child_and_descendant_bug_2600() { "###); // The user would expect unsimplified ancestry here. insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###" - @ c - ○ b + @ c: b + ○ b: notroot a ├─╮ - │ ○ a + │ ○ a: notroot ├─╯ ○ notroot │ ○ base @@ -1151,12 +1156,12 @@ fn test_rebase_with_child_and_descendant_bug_2600() { Added 0 files, modified 0 files, removed 1 files "###); insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###" - @ c - │ ○ base + @ c: b + │ ○ base: b ├─╯ - ○ b + ○ b: notroot a ├─╮ - │ ○ a + │ ○ a: notroot ├─╯ ○ notroot ◆ @@ -1175,12 +1180,12 @@ fn test_rebase_with_child_and_descendant_bug_2600() { Added 0 files, modified 0 files, removed 1 files "###); insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###" - @ c - ○ b + @ c: b + ○ b: notroot a ├─╮ - │ │ ○ base + │ │ ○ base: a │ ├─╯ - │ ○ a + │ ○ a: notroot ├─╯ ○ notroot ◆ @@ -1189,12 +1194,12 @@ fn test_rebase_with_child_and_descendant_bug_2600() { test_env.jj_cmd_ok(&repo_path, &["op", "restore", &setup_opid]); // ====== Reminder of the setup ========= insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###" - @ c - ○ b + @ c: b + ○ b: base a ├─╮ - │ ○ a + │ ○ a: base ├─╯ - ○ base + ○ base: notroot ○ notroot ◆ "###); @@ -1211,9 +1216,9 @@ fn test_rebase_with_child_and_descendant_bug_2600() { // In this case, it is unclear whether the user would always prefer unsimplified // ancestry (whether `b` should also be a direct child of the root commit). insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###" - @ c - ○ b - ○ base + @ c: b + ○ b: base + ○ base: notroot ○ notroot │ ○ a ├─╯ @@ -1233,11 +1238,11 @@ fn test_rebase_with_child_and_descendant_bug_2600() { "###); // The user would expect unsimplified ancestry here. insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###" - @ c + @ c: base a ├─╮ - │ ○ a + │ ○ a: base ├─╯ - ○ base + ○ base: notroot ○ notroot │ ○ b ├─╯ @@ -1258,12 +1263,12 @@ fn test_rebase_with_child_and_descendant_bug_2600() { Added 0 files, modified 0 files, removed 1 files "###); insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###" - ○ b - @ c + ○ b: c + @ c: base a ├─╮ - │ ○ a + │ ○ a: base ├─╯ - ○ base + ○ base: notroot ○ notroot ◆ "###); @@ -1280,12 +1285,12 @@ fn test_rebase_with_child_and_descendant_bug_2600() { Added 0 files, modified 0 files, removed 1 files "###); insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###" - @ c - │ ○ b + @ c: a + │ ○ b: base a ╭─┤ - ○ │ a + ○ │ a: base ├─╯ - ○ base + ○ base: notroot ○ notroot ◆ "###); @@ -1308,16 +1313,16 @@ fn test_rebase_revisions_after() { create_commit(&test_env, &repo_path, "f", &["e"]); // Test the setup insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###" - @ f - ○ e - │ ○ d + @ f: e + ○ e: c + │ ○ d: c ├─╯ - ○ c + ○ c: b2 b4 ├─╮ - │ ○ b4 - │ ○ b3 - ○ │ b2 - ○ │ b1 + │ ○ b4: b3 + │ ○ b3: a + ○ │ b2: b1 + ○ │ b1: a ├─╯ ○ a ◆ @@ -1335,16 +1340,16 @@ fn test_rebase_revisions_after() { Nothing changed. "###); insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###" - @ f - ○ e - │ ○ d + @ f: e + ○ e: c + │ ○ d: c ├─╯ - ○ c + ○ c: b2 b4 ├─╮ - │ ○ b4 - │ ○ b3 - ○ │ b2 - ○ │ b1 + │ ○ b4: b3 + │ ○ b3: a + ○ │ b2: b1 + ○ │ b1: a ├─╯ ○ a ◆ @@ -1358,16 +1363,16 @@ fn test_rebase_revisions_after() { Nothing changed. "###); insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###" - @ f - ○ e - │ ○ d + @ f: e + ○ e: c + │ ○ d: c ├─╯ - ○ c + ○ c: b2 b4 ├─╮ - │ ○ b4 - │ ○ b3 - ○ │ b2 - ○ │ b1 + │ ○ b4: b3 + │ ○ b3: a + ○ │ b2: b1 + ○ │ b1: a ├─╯ ○ a ◆ @@ -1384,16 +1389,16 @@ fn test_rebase_revisions_after() { Parent commit : kmkuslsw 754793f3 c | c "###); insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###" - @ f - ○ c - ○ e + @ f: c + ○ c: e + ○ e: b2 b4 ├─╮ - │ │ ○ d + │ │ ○ d: b2 b4 ╭─┬─╯ - │ ○ b4 - │ ○ b3 - ○ │ b2 - ○ │ b1 + │ ○ b4: b3 + │ ○ b3: a + ○ │ b2: b1 + ○ │ b1: a ├─╯ ○ a ◆ @@ -1411,16 +1416,16 @@ fn test_rebase_revisions_after() { Added 0 files, modified 0 files, removed 1 files "###); insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###" - ○ e - @ f - │ ○ d + ○ e: f + @ f: c + │ ○ d: c ├─╯ - ○ c + ○ c: b2 b4 ├─╮ - │ ○ b4 - │ ○ b3 - ○ │ b2 - ○ │ b1 + │ ○ b4: b3 + │ ○ b3: a + ○ │ b2: b1 + ○ │ b1: a ├─╯ ○ a ◆ @@ -1438,16 +1443,16 @@ fn test_rebase_revisions_after() { Added 0 files, modified 0 files, removed 5 files "###); insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###" - ○ e - │ ○ d + ○ e: c + │ ○ d: c ├─╯ - ○ c + ○ c: b2 b4 ├─╮ - │ ○ b4 - │ ○ b3 - ○ │ b2 - @ │ f - ○ │ b1 + │ ○ b4: b3 + │ ○ b3: a + ○ │ b2: f + @ │ f: b1 + ○ │ b1: a ├─╯ ○ a ◆ @@ -1465,16 +1470,16 @@ fn test_rebase_revisions_after() { Added 0 files, modified 0 files, removed 4 files "###); insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###" - ○ e - │ ○ d + ○ e: c + │ ○ d: c ├─╯ - ○ c + ○ c: f b4 ├─╮ - │ ○ b4 - │ ○ b3 - @ │ f - ○ │ b2 - ○ │ b1 + │ ○ b4: b3 + │ ○ b3: a + @ │ f: b2 + ○ │ b2: b1 + ○ │ b1: a ├─╯ ○ a ◆ @@ -1494,16 +1499,16 @@ fn test_rebase_revisions_after() { Added 0 files, modified 0 files, removed 1 files "###); insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###" - ○ e - │ ○ d + ○ e: f + │ ○ d: f ├─╯ - @ f - ○ c + @ f: c + ○ c: b2 b4 ├─╮ - │ ○ b4 - │ ○ b3 - ○ │ b2 - ○ │ b1 + │ ○ b4: b3 + │ ○ b3: a + ○ │ b2: b1 + ○ │ b1: a ├─╯ ○ a ◆ @@ -1524,17 +1529,17 @@ fn test_rebase_revisions_after() { Added 1 files, modified 0 files, removed 0 files "###); insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###" - @ f + @ f: e d ├─╮ - │ ○ d - ○ │ e + │ ○ d: c + ○ │ e: c ├─╯ - ○ c + ○ c: b2 b4 ├─╮ - │ ○ b4 - │ ○ b3 - ○ │ b2 - ○ │ b1 + │ ○ b4: b3 + │ ○ b3: a + ○ │ b2: b1 + ○ │ b1: a ├─╯ ○ a ◆ @@ -1555,17 +1560,17 @@ fn test_rebase_revisions_after() { Added 1 files, modified 0 files, removed 0 files "###); insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###" - @ f - ○ c + @ f: c + ○ c: b2 b4 ├─╮ - │ ○ b4 - │ ○ b3 + │ ○ b4: b3 + │ ○ b3: d e │ ├─╮ - ○ │ │ b2 - ○ │ │ b1 + ○ │ │ b2: b1 + ○ │ │ b1: d e ╰─┬─╮ - │ ○ e - ○ │ d + │ ○ e: a + ○ │ d: a ├─╯ ○ a ◆ @@ -1587,18 +1592,18 @@ fn test_rebase_revisions_after() { Added 0 files, modified 0 files, removed 3 files "###); insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###" - ○ d + ○ d: b1 b3 ├─╮ - │ │ ○ c + │ │ ○ c: b2 b4 │ │ ├─╮ - │ │ │ ○ b4 - │ │ ○ │ b2 + │ │ │ ○ b4: f + │ │ ○ │ b2: f │ │ ├─╯ - │ │ @ f - │ │ ○ e + │ │ @ f: e + │ │ ○ e: b1 b3 ╭─┬─╯ - │ ○ b3 - ○ │ b1 + │ ○ b3: a + ○ │ b1: a ├─╯ ○ a ◆ @@ -1612,26 +1617,56 @@ fn test_rebase_revisions_after() { insta::assert_snapshot!(stderr, @r###" Rebased 4 commits onto destination Rebased 2 descendant commits - Working copy now at: xznxytkn 084e0629 f | f - Parent commit : nkmrtpmo 563d78c6 e | e + Working copy now at: xznxytkn 9bc7e54c f | f + Parent commit : nkmrtpmo 0f80251b e | e Added 1 files, modified 0 files, removed 0 files "###); insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###" - @ f - ○ e - ○ d - ○ c - ○ b2 - ○ b1 + @ f: e + ○ e: d + ○ d: c + ○ c: b2 b4 ├─╮ - │ ○ b4 - │ ○ b3 + ○ │ b2: b1 + ○ │ b1: a b4 + ├─╮ + │ ○ b4: b3 + │ ○ b3: a ├─╯ ○ a ◆ "###); test_env.jj_cmd_ok(&repo_path, &["op", "restore", &setup_opid]); + // Rebase a subgraph before the parents of one of the commits in the subgraph. + // "c" had parents "b2" and "b4", but no longer has "b4" as a parent since + // "b4" would be a descendant of "c" after the rebase. + let (stdout, stderr) = + test_env.jj_cmd_ok(&repo_path, &["rebase", "-r", "b2::d", "--after", "root()"]); + insta::assert_snapshot!(stdout, @""); + insta::assert_snapshot!(stderr, @r###" + Rebased 3 commits onto destination + Rebased 6 descendant commits + Working copy now at: xznxytkn 0875aabc f | f + Parent commit : nkmrtpmo d429661b e | e + Added 1 files, modified 0 files, removed 0 files + "###); + insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###" + @ f: e + ○ e: b1 b4 + ├─╮ + │ ○ b4: b3 + │ ○ b3: a + ○ │ b1: a + ├─╯ + ○ a: d + ○ d: c + ○ c: b2 + ○ b2 + ◆ + "###); + test_env.jj_cmd_ok(&repo_path, &["op", "restore", &setup_opid]); + // Rebase a subgraph with disconnected commits. Since "b2" is an ancestor of // "e", "b2" should be a parent of "e" after the rebase. let (stdout, stderr) = test_env.jj_cmd_ok( @@ -1642,21 +1677,21 @@ fn test_rebase_revisions_after() { insta::assert_snapshot!(stderr, @r###" Rebased 2 commits onto destination Rebased 3 descendant commits - Working copy now at: xznxytkn 4fb2bb60 f | f - Parent commit : kmkuslsw cebde86a c | c + Working copy now at: xznxytkn 3238a418 f | f + Parent commit : kmkuslsw 6a51bd41 c | c Added 0 files, modified 0 files, removed 2 files "###); insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###" - @ f - │ ○ e - │ ○ b2 - │ ○ d + @ f: c + │ ○ e: b2 + │ ○ b2: d + │ ○ d: c ├─╯ - ○ c + ○ c: b1 b4 ├─╮ - │ ○ b4 - │ ○ b3 - ○ │ b1 + │ ○ b4: b3 + │ ○ b3: a + ○ │ b1: a ├─╯ ○ a ◆ @@ -1690,16 +1725,16 @@ fn test_rebase_revisions_before() { create_commit(&test_env, &repo_path, "f", &["e"]); // Test the setup insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###" - @ f - ○ e - │ ○ d + @ f: e + ○ e: c + │ ○ d: c ├─╯ - ○ c + ○ c: b2 b4 ├─╮ - │ ○ b4 - │ ○ b3 - ○ │ b2 - ○ │ b1 + │ ○ b4: b3 + │ ○ b3: a + ○ │ b2: b1 + ○ │ b1: a ├─╯ ○ a ◆ @@ -1717,16 +1752,16 @@ fn test_rebase_revisions_before() { Nothing changed. "###); insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###" - @ f - ○ e - │ ○ d + @ f: e + ○ e: c + │ ○ d: c ├─╯ - ○ c + ○ c: b2 b4 ├─╮ - │ ○ b4 - │ ○ b3 - ○ │ b2 - ○ │ b1 + │ ○ b4: b3 + │ ○ b3: a + ○ │ b2: b1 + ○ │ b1: a ├─╯ ○ a ◆ @@ -1740,16 +1775,16 @@ fn test_rebase_revisions_before() { Nothing changed. "###); insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###" - @ f - ○ e - │ ○ d + @ f: e + ○ e: c + │ ○ d: c ├─╯ - ○ c + ○ c: b2 b4 ├─╮ - │ ○ b4 - │ ○ b3 - ○ │ b2 - ○ │ b1 + │ ○ b4: b3 + │ ○ b3: a + ○ │ b2: b1 + ○ │ b1: a ├─╯ ○ a ◆ @@ -1772,17 +1807,17 @@ fn test_rebase_revisions_before() { Parent commit : nkmrtpmo e9a28d4b e | e "###); insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###" - @ f - ○ e + @ f: e + ○ e: b2 b4 ├─╮ - │ │ ○ d + │ │ ○ d: b2 b4 ╭─┬─╯ - │ ○ b4 - │ ○ b3 - ○ │ b2 - ○ │ b1 + │ ○ b4: b3 + │ ○ b3: a + ○ │ b2: b1 + ○ │ b1: a ├─╯ - ○ a + ○ a: c ○ c ◆ "###); @@ -1799,16 +1834,16 @@ fn test_rebase_revisions_before() { Added 0 files, modified 0 files, removed 1 files "###); insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###" - ○ e - @ f - │ ○ d + ○ e: f + @ f: c + │ ○ d: c ├─╯ - ○ c + ○ c: b2 b4 ├─╮ - │ ○ b4 - │ ○ b3 - ○ │ b2 - ○ │ b1 + │ ○ b4: b3 + │ ○ b3: a + ○ │ b2: b1 + ○ │ b1: a ├─╯ ○ a ◆ @@ -1826,16 +1861,16 @@ fn test_rebase_revisions_before() { Added 0 files, modified 0 files, removed 5 files "###); insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###" - ○ e - │ ○ d + ○ e: c + │ ○ d: c ├─╯ - ○ c + ○ c: b2 b4 ├─╮ - │ ○ b4 - │ ○ b3 - ○ │ b2 - @ │ f - ○ │ b1 + │ ○ b4: b3 + │ ○ b3: a + ○ │ b2: f + @ │ f: b1 + ○ │ b1: a ├─╯ ○ a ◆ @@ -1853,16 +1888,16 @@ fn test_rebase_revisions_before() { Added 0 files, modified 0 files, removed 6 files "###); insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###" - ○ e - │ ○ d + ○ e: c + │ ○ d: c ├─╯ - ○ c + ○ c: b2 b4 ├─╮ - │ ○ b4 - │ ○ b3 - ○ │ b2 - ○ │ b1 - @ │ f + │ ○ b4: b3 + │ ○ b3: a + ○ │ b2: b1 + ○ │ b1: f + @ │ f: a ├─╯ ○ a ◆ @@ -1883,16 +1918,16 @@ fn test_rebase_revisions_before() { Added 0 files, modified 0 files, removed 2 files "###); insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###" - ○ e - │ ○ d + ○ e: c + │ ○ d: c ├─╯ - ○ c - @ f + ○ c: f + @ f: b2 b4 ├─╮ - │ ○ b4 - │ ○ b3 - ○ │ b2 - ○ │ b1 + │ ○ b4: b3 + │ ○ b3: a + ○ │ b2: b1 + ○ │ b1: a ├─╯ ○ a ◆ @@ -1912,16 +1947,16 @@ fn test_rebase_revisions_before() { Parent commit : nkmrtpmo fd26fbd4 e | e "###); insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###" - @ f - ○ e - │ ○ d + @ f: e + ○ e: b1 + │ ○ d: b1 ├─╯ - ○ b1 - ○ c + ○ b1: c + ○ c: b2 b4 ├─╮ - │ ○ b4 - │ ○ b3 - ○ │ b2 + │ ○ b4: b3 + │ ○ b3: a + ○ │ b2: a ├─╯ ○ a ◆ @@ -1944,18 +1979,18 @@ fn test_rebase_revisions_before() { Added 0 files, modified 0 files, removed 4 files "###); insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###" - ○ e - │ ○ d + ○ e: c + │ ○ d: c ├─╯ - ○ c + ○ c: b2 b4 ├─╮ - │ ○ b4 - ○ │ b2 + │ ○ b4: f + ○ │ b2: f ├─╯ - @ f + @ f: b1 b3 ├─╮ - │ ○ b3 - ○ │ b1 + │ ○ b3: a + ○ │ b1: a ├─╯ ○ a ◆ @@ -1976,16 +2011,16 @@ fn test_rebase_revisions_before() { Parent commit : nkmrtpmo b5933877 e | e "###); insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###" - @ f - ○ e - │ ○ d + @ f: e + ○ e: c + │ ○ d: c ├─╯ - ○ c + ○ c: b1 b3 ├─╮ - │ ○ b3 - ○ │ b1 + │ ○ b3: a + ○ │ b1: a ├─╯ - ○ a + ○ a: b2 b4 ├─╮ │ ○ b4 ○ │ b2 @@ -2007,18 +2042,18 @@ fn test_rebase_revisions_before() { Parent commit : nkmrtpmo e31053d1 e | e "###); insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###" - @ f - ○ e - ○ c + @ f: e + ○ e: c + ○ c: b2 b4 ├─╮ - │ ○ b4 + │ ○ b4: b1 b3 │ ├─╮ - ○ │ │ b2 + ○ │ │ b2: b1 b3 ╰─┬─╮ - ○ │ │ d + ○ │ │ d: b1 b3 ╰─┬─╮ - │ ○ b3 - ○ │ b1 + │ ○ b3: a + ○ │ b1: a ├─╯ ○ a ◆ @@ -2039,22 +2074,51 @@ fn test_rebase_revisions_before() { Parent commit : kmkuslsw c0fd979a c | c "###); insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###" - @ f - │ ○ d + @ f: c + │ ○ d: c ├─╯ - ○ c + ○ c: b2 b4 ├─╮ - │ ○ b4 - │ ○ b3 - ○ │ b2 + │ ○ b4: b3 + │ ○ b3: a + ○ │ b2: a ├─╯ - ○ a - ○ e + ○ a: e + ○ e: b1 ○ b1 ◆ "###); test_env.jj_cmd_ok(&repo_path, &["op", "restore", &setup_opid]); + // Rebase a subgraph before the parents of one of the commits in the subgraph. + // "c" had parents "b2" and "b4", but no longer has "b4" as a parent since + // "b4" would be a descendant of "c" after the rebase. + let (stdout, stderr) = + test_env.jj_cmd_ok(&repo_path, &["rebase", "-r", "b2::d", "--before", "a"]); + insta::assert_snapshot!(stdout, @""); + insta::assert_snapshot!(stderr, @r###" + Rebased 3 commits onto destination + Rebased 6 descendant commits + Working copy now at: xznxytkn f5991dc7 f | f + Parent commit : nkmrtpmo 37894e3c e | e + Added 1 files, modified 0 files, removed 0 files + "###); + insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###" + @ f: e + ○ e: b1 b4 + ├─╮ + │ ○ b4: b3 + │ ○ b3: a + ○ │ b1: a + ├─╯ + ○ a: d + ○ d: c + ○ c: b2 + ○ b2 + ◆ + "###); + test_env.jj_cmd_ok(&repo_path, &["op", "restore", &setup_opid]); + // Should error if a loop will be created. let stderr = test_env.jj_cmd_failure( &repo_path, @@ -2080,14 +2144,14 @@ fn test_rebase_revisions_after_before() { create_commit(&test_env, &repo_path, "f", &["e"]); // Test the setup insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###" - @ f - ○ e - │ ○ d + @ f: e + ○ e: c + │ ○ d: c ├─╯ - ○ c + ○ c: b1 b2 ├─╮ - │ ○ b2 - ○ │ b1 + │ ○ b2: a + ○ │ b1: a ├─╯ ○ a ◆ @@ -2109,13 +2173,13 @@ fn test_rebase_revisions_after_before() { Added 1 files, modified 0 files, removed 0 files "###); insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###" - @ f - ○ d - ○ e - ○ c + @ f: d + ○ d: e + ○ e: c + ○ c: b1 b2 ├─╮ - │ ○ b2 - ○ │ b1 + │ ○ b2: a + ○ │ b1: a ├─╯ ○ a ◆ @@ -2138,15 +2202,15 @@ fn test_rebase_revisions_after_before() { Added 1 files, modified 0 files, removed 0 files "###); insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###" - @ f + @ f: e d ├─╮ - │ ○ d - ○ │ e - ○ │ c + │ ○ d: a + ○ │ e: c + ○ │ c: b1 b2 ├───╮ - │ │ ○ b2 + │ │ ○ b2: a │ ├─╯ - ○ │ b1 + ○ │ b1: a ├─╯ ○ a ◆ @@ -2170,14 +2234,14 @@ fn test_rebase_revisions_after_before() { Added 1 files, modified 0 files, removed 0 files "###); insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###" - @ f - ○ e + @ f: e + ○ e: b1 b2 c ├─┬─╮ - │ │ ○ c - │ │ ○ d + │ │ ○ c: d + │ │ ○ d: b1 b2 ╭─┬─╯ - │ ○ b2 - ○ │ b1 + │ ○ b2: a + ○ │ b1: a ├─╯ ○ a ◆ @@ -2206,15 +2270,15 @@ fn test_rebase_revisions_after_before() { Added 1 files, modified 0 files, removed 0 files "###); insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###" - @ f + @ f: b1 b2 d e ├─┬─┬─╮ - │ │ │ ○ e - │ │ ○ │ d + │ │ │ ○ e: c + │ │ ○ │ d: c │ │ ├─╯ - │ │ ○ c - │ ○ │ b2 + │ │ ○ c: a + │ ○ │ b2: a │ ├─╯ - ○ │ b1 + ○ │ b1: a ├─╯ ○ a ◆ @@ -2289,14 +2353,14 @@ fn test_rebase_skip_if_on_destination() { create_commit(&test_env, &repo_path, "f", &["e"]); // Test the setup insta::assert_snapshot!(get_long_log_output(&test_env, &repo_path), @r###" - @ f lylxulpl 88f778c5 - ○ e kmkuslsw 48dd9e3f - │ ○ d znkkpsqq 92438fc9 + @ f lylxulpl 88f778c5: e + ○ e kmkuslsw 48dd9e3f: c + │ ○ d znkkpsqq 92438fc9: c ├─╯ - ○ c vruxwmqv c41e416e + ○ c vruxwmqv c41e416e: b1 b2 ├─╮ - │ ○ b2 royxmykx 903ab0d6 - ○ │ b1 zsuskuln 072d5ae1 + │ ○ b2 royxmykx 903ab0d6: a + ○ │ b1 zsuskuln 072d5ae1: a ├─╯ ○ a rlvkpnrz 2443ea76 ◆ zzzzzzzz 00000000 @@ -2309,14 +2373,14 @@ fn test_rebase_skip_if_on_destination() { Skipped rebase of 2 commits that were already in place "###); insta::assert_snapshot!(get_long_log_output(&test_env, &repo_path), @r###" - @ f lylxulpl 88f778c5 - ○ e kmkuslsw 48dd9e3f - │ ○ d znkkpsqq 92438fc9 + @ f lylxulpl 88f778c5: e + ○ e kmkuslsw 48dd9e3f: c + │ ○ d znkkpsqq 92438fc9: c ├─╯ - ○ c vruxwmqv c41e416e + ○ c vruxwmqv c41e416e: b1 b2 ├─╮ - │ ○ b2 royxmykx 903ab0d6 - ○ │ b1 zsuskuln 072d5ae1 + │ ○ b2 royxmykx 903ab0d6: a + ○ │ b1 zsuskuln 072d5ae1: a ├─╯ ○ a rlvkpnrz 2443ea76 ◆ zzzzzzzz 00000000 @@ -2330,14 +2394,14 @@ fn test_rebase_skip_if_on_destination() { Skipped rebase of 1 commits that were already in place "###); insta::assert_snapshot!(get_long_log_output(&test_env, &repo_path), @r###" - @ f lylxulpl 88f778c5 - ○ e kmkuslsw 48dd9e3f - │ ○ d znkkpsqq 92438fc9 + @ f lylxulpl 88f778c5: e + ○ e kmkuslsw 48dd9e3f: c + │ ○ d znkkpsqq 92438fc9: c ├─╯ - ○ c vruxwmqv c41e416e + ○ c vruxwmqv c41e416e: b1 b2 ├─╮ - │ ○ b2 royxmykx 903ab0d6 - ○ │ b1 zsuskuln 072d5ae1 + │ ○ b2 royxmykx 903ab0d6: a + ○ │ b1 zsuskuln 072d5ae1: a ├─╯ ○ a rlvkpnrz 2443ea76 ◆ zzzzzzzz 00000000 @@ -2351,14 +2415,14 @@ fn test_rebase_skip_if_on_destination() { Nothing changed. "###); insta::assert_snapshot!(get_long_log_output(&test_env, &repo_path), @r###" - @ f lylxulpl 88f778c5 - ○ e kmkuslsw 48dd9e3f - │ ○ d znkkpsqq 92438fc9 + @ f lylxulpl 88f778c5: e + ○ e kmkuslsw 48dd9e3f: c + │ ○ d znkkpsqq 92438fc9: c ├─╯ - ○ c vruxwmqv c41e416e + ○ c vruxwmqv c41e416e: b1 b2 ├─╮ - │ ○ b2 royxmykx 903ab0d6 - ○ │ b1 zsuskuln 072d5ae1 + │ ○ b2 royxmykx 903ab0d6: a + ○ │ b1 zsuskuln 072d5ae1: a ├─╯ ○ a rlvkpnrz 2443ea76 ◆ zzzzzzzz 00000000 @@ -2375,15 +2439,15 @@ fn test_rebase_skip_if_on_destination() { Added 0 files, modified 0 files, removed 1 files "###); insta::assert_snapshot!(get_long_log_output(&test_env, &repo_path), @r###" - @ f lylxulpl 77cb229f - │ ○ e kmkuslsw 48dd9e3f + @ f lylxulpl 77cb229f: c + │ ○ e kmkuslsw 48dd9e3f: c ├─╯ - │ ○ d znkkpsqq 92438fc9 + │ ○ d znkkpsqq 92438fc9: c ├─╯ - ○ c vruxwmqv c41e416e + ○ c vruxwmqv c41e416e: b1 b2 ├─╮ - │ ○ b2 royxmykx 903ab0d6 - ○ │ b1 zsuskuln 072d5ae1 + │ ○ b2 royxmykx 903ab0d6: a + ○ │ b1 zsuskuln 072d5ae1: a ├─╯ ○ a rlvkpnrz 2443ea76 ◆ zzzzzzzz 00000000 @@ -2391,10 +2455,12 @@ fn test_rebase_skip_if_on_destination() { } fn get_log_output(test_env: &TestEnvironment, repo_path: &Path) -> String { - test_env.jj_cmd_success(repo_path, &["log", "-T", "branches"]) + let template = "branches ++ surround(': ', '', parents.map(|c| c.branches()))"; + test_env.jj_cmd_success(repo_path, &["log", "-T", template]) } fn get_long_log_output(test_env: &TestEnvironment, repo_path: &Path) -> String { - let template = r#"description.first_line() ++ " " ++ change_id.shortest(8) ++ " " ++ commit_id.shortest(8)"#; + let template = "branches ++ ' ' ++ change_id.shortest(8) ++ ' ' ++ commit_id.shortest(8) ++ \ + surround(': ', '', parents.map(|c| c.branches()))"; test_env.jj_cmd_success(repo_path, &["log", "-T", template]) } diff --git a/lib/src/rewrite.rs b/lib/src/rewrite.rs index dbbb971628..e750ba5ab6 100644 --- a/lib/src/rewrite.rs +++ b/lib/src/rewrite.rs @@ -487,33 +487,33 @@ pub fn move_commits( .commits(mut_repo.store()) .try_collect()?; - // Commits in the target set should only have other commits in the set as - // parents, except the roots of the set, which persist their original - // parents. - // If a commit in the set has a parent which is not in the set, but has - // an ancestor which is in the set, then the commit will have that ancestor - // as a parent. - let mut target_commits_internal_parents: HashMap> = HashMap::new(); + // Compute the parents of all commits in the connected target set, allowing only + // commits in the target set as parents. The parents of each commit are + // identical to the ones found using a preorder DFS of the node's ancestors, + // starting from the node itself, and avoiding traversing an edge if the + // parent is in the target set. + let mut connected_target_commits_internal_parents: HashMap> = + HashMap::new(); for commit in connected_target_commits.iter().rev() { // The roots of the set will not have any parents found in - // `target_commits_internal_parents`, and will be stored in - // `target_commits_internal_parents` as an empty vector. + // `connected_target_commits_internal_parents`, and will be stored as an empty + // vector. let mut new_parents = vec![]; for old_parent in commit.parent_ids() { if target_commit_ids.contains(old_parent) { new_parents.push(old_parent.clone()); - } else if let Some(parents) = target_commits_internal_parents.get(old_parent) { + } else if let Some(parents) = connected_target_commits_internal_parents.get(old_parent) + { new_parents.extend(parents.iter().cloned()); } } - target_commits_internal_parents.insert(commit.id().clone(), new_parents); + connected_target_commits_internal_parents.insert(commit.id().clone(), new_parents); } - target_commits_internal_parents.retain(|id, _| target_commit_ids.contains(id)); // Compute the roots of `target_commits`. - let target_roots: HashSet<_> = target_commits_internal_parents + let target_roots: HashSet<_> = connected_target_commits_internal_parents .iter() - .filter(|(_, parents)| parents.is_empty()) + .filter(|(commit_id, parents)| target_commit_ids.contains(commit_id) && parents.is_empty()) .map(|(commit_id, _)| commit_id.clone()) .collect(); @@ -710,17 +710,33 @@ pub fn move_commits( if let Some(new_child_parents) = new_children_parents.get(commit_id) { new_child_parents.clone() } - // Commits in the target set should persist only rebased parents from the target - // sets. - else if let Some(target_commit_parents) = - target_commits_internal_parents.get(commit_id) - { - // If the commit does not have any parents in the target set, it is one of the - // commits in the root set, and should be rebased onto the new destination. - if target_commit_parents.is_empty() { + // Commit is in the target set. + else if target_commit_ids.contains(commit_id) { + // If the commit is a root of the target set, it should be rebased onto the new destination. + if target_roots.contains(commit_id) { new_parent_ids.clone() - } else { - target_commit_parents.clone() + } + // Otherwise: + // 1. Keep parents which are within the target set. + // 2. Replace parents which are outside the target set but are part of the + // connected target set with their ancestor commits which are in the target + // set. + // 3. Keep other parents outside the target set if they are not descendants of the + // new children of the target set. + else { + let mut new_parents = vec![]; + for parent_id in commit.parent_ids() { + if target_commit_ids.contains(parent_id) { + new_parents.push(parent_id.clone()); + } else if let Some(parents) = + connected_target_commits_internal_parents.get(parent_id) { + new_parents.extend(parents.iter().cloned()); + } else if !new_children.iter().any(|new_child| { + mut_repo.index().is_ancestor(new_child.id(), parent_id) }) { + new_parents.push(parent_id.clone()); + } + } + new_parents } } // Commits outside the target set should have references to commits inside the set