use std::path::PathBuf;
use test_case::test_case;
use testutils::TestResult;
use crate::common::CommandOutput;
use crate::common::TestEnvironment;
use crate::common::TestWorkDir;
#[must_use]
fn get_log_output(work_dir: &TestWorkDir) -> CommandOutput {
let template = r#"separate(" ", change_id.short(), empty, local_bookmarks, description)"#;
work_dir.run_jj(["log", "-T", template])
}
#[must_use]
fn get_log_with_summary(work_dir: &TestWorkDir) -> CommandOutput {
let template = r#"separate(" ", change_id.short(), local_bookmarks, description)"#;
work_dir.run_jj(["log", "-T", template, "--summary"])
}
#[must_use]
fn get_workspace_log_output(work_dir: &TestWorkDir) -> CommandOutput {
let template = r#"separate(" ", change_id.short(), working_copies, description)"#;
work_dir.run_jj(["log", "-T", template, "-r", "all()"])
}
#[must_use]
fn get_recorded_dates(work_dir: &TestWorkDir, revset: &str) -> CommandOutput {
let template = r#"separate("\n", "Author date: " ++ author.timestamp(), "Committer date: " ++ committer.timestamp())"#;
work_dir.run_jj(["log", "--no-graph", "-T", template, "-r", revset])
}
#[test]
fn test_split_by_paths() -> TestResult {
let mut test_env = TestEnvironment::default();
let edit_script = test_env.set_up_fake_editor();
test_env.run_jj_in(".", ["git", "init", "repo"]).success();
let work_dir = test_env.work_dir("repo");
work_dir.write_file("file1", "foo");
work_dir.write_file("file2", "foo");
work_dir.write_file("file3", "foo");
insta::assert_snapshot!(get_log_output(&work_dir), @"
@ qpvuntsmwlqt false
◆ zzzzzzzzzzzz true
[EOF]
");
insta::assert_snapshot!(get_recorded_dates(&work_dir, "@"), @"
Author date: 2001-02-03 04:05:08.000 +07:00
Committer date: 2001-02-03 04:05:08.000 +07:00[EOF]
");
std::fs::write(
&edit_script,
[
"dump editor0",
"write\n",
"next invocation\n",
"dump editor1",
]
.join("\0"),
)?;
let output = work_dir.run_jj([
"split",
"file2",
r#"--config=templates.commit_trailers='"Trailer: value"'"#,
]);
insta::assert_snapshot!(output, @"
------- stderr -------
Selected changes : qpvuntsm 6dbc7747 (no description set)
Remaining changes: zsuskuln 42cbbc02 (no description set)
Working copy (@) now at: zsuskuln 42cbbc02 (no description set)
Parent commit (@-) : qpvuntsm 6dbc7747 (no description set)
[EOF]
");
insta::assert_snapshot!(
std::fs::read_to_string(test_env.env_root().join("editor0"))?, @r#"
JJ: Enter a description for the selected changes.
Trailer: value
JJ: Change ID: qpvuntsm
JJ: This commit contains the following changes:
JJ: A file2
JJ:
JJ: Lines starting with "JJ:" (like this one) will be removed.
"#);
assert!(!test_env.env_root().join("editor1").exists());
insta::assert_snapshot!(get_log_output(&work_dir), @"
@ zsuskulnrvyr false
○ qpvuntsmwlqt false
◆ zzzzzzzzzzzz true
[EOF]
");
insta::assert_snapshot!(get_recorded_dates(&work_dir, "@"), @"
Author date: 2001-02-03 04:05:08.000 +07:00
Committer date: 2001-02-03 04:05:10.000 +07:00[EOF]
");
insta::assert_snapshot!(get_recorded_dates(&work_dir, "@-"), @"
Author date: 2001-02-03 04:05:08.000 +07:00
Committer date: 2001-02-03 04:05:10.000 +07:00[EOF]
");
let output = work_dir.run_jj(["diff", "-s", "-r", "@-"]);
insta::assert_snapshot!(output, @"
A file2
[EOF]
");
let output = work_dir.run_jj(["diff", "-s"]);
insta::assert_snapshot!(output, @"
A file1
A file3
[EOF]
");
std::fs::write(&edit_script, "")?;
let output = work_dir.run_jj(["split", "-r", "@-", "."]);
insta::assert_snapshot!(output, @"
------- stderr -------
Warning: All changes have been selected, so the original revision will become empty
Rebased 1 descendant commits
Selected changes : qpvuntsm 9fd1c9e1 (no description set)
Remaining changes: znkkpsqq 41e0da21 (empty) (no description set)
Working copy (@) now at: zsuskuln a06e40b8 (no description set)
Parent commit (@-) : znkkpsqq 41e0da21 (empty) (no description set)
[EOF]
");
insta::assert_snapshot!(get_log_output(&work_dir), @"
@ zsuskulnrvyr false
○ znkkpsqqskkl true
○ qpvuntsmwlqt false
◆ zzzzzzzzzzzz true
[EOF]
");
let output = work_dir.run_jj(["diff", "-s", "-r", "@--"]);
insta::assert_snapshot!(output, @"
A file2
[EOF]
");
work_dir.run_jj(["abandon", "@-"]).success();
std::fs::write(&edit_script, "")?;
let output = work_dir.run_jj(["split", "-r", "@-", "nonexistent"]);
insta::assert_snapshot!(output, @"
------- stderr -------
Warning: No matching entries for paths: nonexistent
Warning: No changes have been selected, so the new revision will be empty
Rebased 1 descendant commits
Selected changes : qpvuntsm 49416632 (empty) (no description set)
Remaining changes: lylxulpl 718afbf5 (no description set)
Working copy (@) now at: zsuskuln 0ed53ee6 (no description set)
Parent commit (@-) : lylxulpl 718afbf5 (no description set)
[EOF]
");
insta::assert_snapshot!(get_log_output(&work_dir), @"
@ zsuskulnrvyr false
○ lylxulplsnyw false
○ qpvuntsmwlqt true
◆ zzzzzzzzzzzz true
[EOF]
");
let output = work_dir.run_jj(["diff", "-s", "-r", "@-"]);
insta::assert_snapshot!(output, @"
A file2
[EOF]
");
work_dir.run_jj(["new"]).success();
work_dir.remove_file("file1");
let output = work_dir.run_jj(["split", "file1"]);
insta::assert_snapshot!(output, @"
------- stderr -------
Warning: All changes have been selected, so the original revision will become empty
Selected changes : uyznsvlq 971ccc0b (no description set)
Remaining changes: xznxytkn a267cd96 (empty) (no description set)
Working copy (@) now at: smwtzssm 6715dc2c (empty) (no description set)
Parent commit (@-) : uyznsvlq 971ccc0b (no description set)
[EOF]
");
Ok(())
}
#[test]
fn test_split_with_non_empty_description() -> TestResult {
let mut test_env = TestEnvironment::default();
let edit_script = test_env.set_up_fake_editor();
test_env.run_jj_in(".", ["git", "init", "repo"]).success();
test_env.add_config(r#"template-aliases.default_commit_description = '"\n\nTESTED=TODO\n"'"#);
let work_dir = test_env.work_dir("repo");
work_dir.write_file("file1", "foo\n");
work_dir.write_file("file2", "bar\n");
work_dir.run_jj(["describe", "-m", "test"]).success();
std::fs::write(
edit_script,
[
"dump editor1",
"write\npart 1",
"next invocation\n",
"dump editor2",
"write\npart 2",
]
.join("\0"),
)?;
let output = work_dir.run_jj(["split", "file1"]);
insta::assert_snapshot!(output, @"
------- stderr -------
Selected changes : qpvuntsm c7f7b14b part 1
Remaining changes: kkmpptxz ac33a5a9 part 2
Working copy (@) now at: kkmpptxz ac33a5a9 part 2
Parent commit (@-) : qpvuntsm c7f7b14b part 1
[EOF]
");
insta::assert_snapshot!(
std::fs::read_to_string(test_env.env_root().join("editor1"))?, @r#"
JJ: Enter a description for the selected changes.
test
JJ: Change ID: qpvuntsm
JJ: This commit contains the following changes:
JJ: A file1
JJ:
JJ: Lines starting with "JJ:" (like this one) will be removed.
"#);
insta::assert_snapshot!(
std::fs::read_to_string(test_env.env_root().join("editor2"))?, @r#"
JJ: Enter a description for the remaining changes.
test
JJ: Change ID: kkmpptxz
JJ: This commit contains the following changes:
JJ: A file2
JJ:
JJ: Lines starting with "JJ:" (like this one) will be removed.
"#);
insta::assert_snapshot!(get_log_output(&work_dir), @"
@ kkmpptxzrspx false part 2
○ qpvuntsmwlqt false part 1
◆ zzzzzzzzzzzz true
[EOF]
");
Ok(())
}
#[test]
fn test_split_with_default_description() -> TestResult {
let mut test_env = TestEnvironment::default();
let edit_script = test_env.set_up_fake_editor();
test_env.run_jj_in(".", ["git", "init", "repo"]).success();
test_env.add_config(r#"template-aliases.default_commit_description = '"\n\nTESTED=TODO\n"'"#);
let work_dir = test_env.work_dir("repo");
work_dir.write_file("file1", "foo\n");
work_dir.write_file("file2", "bar\n");
std::fs::write(
edit_script,
["dump editor1", "next invocation\n", "dump editor2"].join("\0"),
)?;
let output = work_dir.run_jj(["split", "file1"]);
insta::assert_snapshot!(output, @"
------- stderr -------
Selected changes : qpvuntsm ff633dcc TESTED=TODO
Remaining changes: rlvkpnrz b1d20b7e (no description set)
Working copy (@) now at: rlvkpnrz b1d20b7e (no description set)
Parent commit (@-) : qpvuntsm ff633dcc TESTED=TODO
[EOF]
");
insta::assert_snapshot!(
std::fs::read_to_string(test_env.env_root().join("editor1"))?, @r#"
JJ: Enter a description for the selected changes.
TESTED=TODO
JJ: Change ID: qpvuntsm
JJ: This commit contains the following changes:
JJ: A file1
JJ:
JJ: Lines starting with "JJ:" (like this one) will be removed.
"#);
assert!(!test_env.env_root().join("editor2").exists());
insta::assert_snapshot!(get_log_output(&work_dir), @"
@ rlvkpnrzqnoo false
○ qpvuntsmwlqt false TESTED=TODO
◆ zzzzzzzzzzzz true
[EOF]
");
Ok(())
}
#[test]
fn test_split_with_descendants() -> TestResult {
let mut test_env = TestEnvironment::default();
let edit_script = test_env.set_up_fake_editor();
test_env.run_jj_in(".", ["git", "init", "repo"]).success();
let work_dir = test_env.work_dir("repo");
work_dir.write_file("file1", "foo\n");
work_dir.write_file("file2", "bar\n");
work_dir
.run_jj(["commit", "-m", "Add file1 & file2"])
.success();
work_dir.write_file("file3", "baz\n");
work_dir.run_jj(["commit", "-m", "Add file3"]).success();
work_dir.write_file("file4", "foobarbaz\n");
work_dir.run_jj(["describe", "-m", "Add file4"]).success();
insta::assert_snapshot!(get_log_output(&work_dir), @"
@ kkmpptxzrspx false Add file4
○ rlvkpnrzqnoo false Add file3
○ qpvuntsmwlqt false Add file1 & file2
◆ zzzzzzzzzzzz true
[EOF]
");
std::fs::write(
edit_script,
[
"dump editor1",
"write\nAdd file1",
"next invocation\n",
"dump editor2",
"write\nAdd file2",
]
.join("\0"),
)?;
let output = work_dir.run_jj(["split", "file1", "-r", "qpvu"]);
insta::assert_snapshot!(output, @"
------- stderr -------
Rebased 2 descendant commits
Selected changes : qpvuntsm 74306e35 Add file1
Remaining changes: royxmykx 0a37745e Add file2
Working copy (@) now at: kkmpptxz 7ee84812 Add file4
Parent commit (@-) : rlvkpnrz d335bd94 Add file3
[EOF]
");
insta::assert_snapshot!(get_log_output(&work_dir), @"
@ kkmpptxzrspx false Add file4
○ rlvkpnrzqnoo false Add file3
○ royxmykxtrkr false Add file2
○ qpvuntsmwlqt false Add file1
◆ zzzzzzzzzzzz true
[EOF]
");
insta::assert_snapshot!(
std::fs::read_to_string(test_env.env_root().join("editor1"))?, @r#"
JJ: Enter a description for the selected changes.
Add file1 & file2
JJ: Change ID: qpvuntsm
JJ: This commit contains the following changes:
JJ: A file1
JJ:
JJ: Lines starting with "JJ:" (like this one) will be removed.
"#);
insta::assert_snapshot!(
std::fs::read_to_string(test_env.env_root().join("editor2"))?, @r#"
JJ: Enter a description for the remaining changes.
Add file1 & file2
JJ: Change ID: royxmykx
JJ: This commit contains the following changes:
JJ: A file2
JJ:
JJ: Lines starting with "JJ:" (like this one) will be removed.
"#);
let evolog_1 = work_dir.run_jj(["evolog", "-r", "qpvun"]);
insta::assert_snapshot!(evolog_1, @"
○ qpvuntsm test.user@example.com 2001-02-03 08:05:12 74306e35
│ Add file1
│ -- operation ec53c39a72a8 split commit 1d2499e72cefc8a2b87ebb47569140857b96189f
○ qpvuntsm/1 test.user@example.com 2001-02-03 08:05:08 1d2499e7 (hidden)
│ Add file1 & file2
│ -- operation 8185a834cefe commit f5700f8ef89e290e4e90ae6adc0908707e0d8c85
○ qpvuntsm/2 test.user@example.com 2001-02-03 08:05:08 f5700f8e (hidden)
│ (no description set)
│ -- operation d36e968aa414 snapshot working copy
○ qpvuntsm/3 test.user@example.com 2001-02-03 08:05:07 e8849ae1 (hidden)
(empty) (no description set)
-- operation 90267f31f904 add workspace 'default'
[EOF]
");
let evolog_2 = work_dir.run_jj(["evolog", "-r", "royxm"]);
insta::assert_snapshot!(evolog_2, @"
○ royxmykx test.user@example.com 2001-02-03 08:05:12 0a37745e
│ Add file2
│ -- operation ec53c39a72a8 split commit 1d2499e72cefc8a2b87ebb47569140857b96189f
○ qpvuntsm/1 test.user@example.com 2001-02-03 08:05:08 1d2499e7 (hidden)
│ Add file1 & file2
│ -- operation 8185a834cefe commit f5700f8ef89e290e4e90ae6adc0908707e0d8c85
○ qpvuntsm/2 test.user@example.com 2001-02-03 08:05:08 f5700f8e (hidden)
│ (no description set)
│ -- operation d36e968aa414 snapshot working copy
○ qpvuntsm/3 test.user@example.com 2001-02-03 08:05:07 e8849ae1 (hidden)
(empty) (no description set)
-- operation 90267f31f904 add workspace 'default'
[EOF]
");
Ok(())
}
#[test]
fn test_split_with_merge_child() -> TestResult {
let mut test_env = TestEnvironment::default();
let edit_script = test_env.set_up_fake_editor();
test_env.run_jj_in(".", ["git", "init", "repo"]).success();
let work_dir = test_env.work_dir("repo");
work_dir.run_jj(["describe", "-m=1"]).success();
work_dir.run_jj(["new", "root()", "-m=a"]).success();
work_dir.write_file("file1", "foo\n");
work_dir.write_file("file2", "bar\n");
work_dir
.run_jj(["new", "subject(1)", "subject(a)", "-m=2"])
.success();
insta::assert_snapshot!(get_log_output(&work_dir), @"
@ zsuskulnrvyr true 2
├─╮
│ ○ kkmpptxzrspx false a
○ │ qpvuntsmwlqt true 1
├─╯
◆ zzzzzzzzzzzz true
[EOF]
");
std::fs::write(
edit_script,
["write\nAdd file1", "next invocation\n", "write\nAdd file2"].join("\0"),
)?;
let output = work_dir.run_jj(["split", "-rsubject(a)", "file1"]);
insta::assert_snapshot!(output, @"
------- stderr -------
Rebased 1 descendant commits
Selected changes : kkmpptxz cc199567 Add file1
Remaining changes: royxmykx e488409f Add file2
Working copy (@) now at: zsuskuln ace61421 (empty) 2
Parent commit (@-) : qpvuntsm 884fe9b9 (empty) 1
Parent commit (@-) : royxmykx e488409f Add file2
[EOF]
");
insta::assert_snapshot!(get_log_output(&work_dir), @"
@ zsuskulnrvyr true 2
├─╮
│ ○ royxmykxtrkr false Add file2
│ ○ kkmpptxzrspx false Add file1
○ │ qpvuntsmwlqt true 1
├─╯
◆ zzzzzzzzzzzz true
[EOF]
");
Ok(())
}
#[test]
fn test_split_parallel_no_descendants() -> TestResult {
let mut test_env = TestEnvironment::default();
let edit_script = test_env.set_up_fake_editor();
test_env.run_jj_in(".", ["git", "init", "repo"]).success();
test_env.add_config(r#"template-aliases.default_commit_description = '"\n\nTESTED=TODO\n"'"#);
let work_dir = test_env.work_dir("repo");
work_dir.write_file("file1", "foo\n");
work_dir.write_file("file2", "bar\n");
insta::assert_snapshot!(get_log_output(&work_dir), @"
@ qpvuntsmwlqt false
◆ zzzzzzzzzzzz true
[EOF]
");
std::fs::write(
edit_script,
["dump editor1", "next invocation\n", "dump editor2"].join("\0"),
)?;
let output = work_dir.run_jj(["split", "--parallel", "file1"]);
insta::assert_snapshot!(output, @"
------- stderr -------
Selected changes : qpvuntsm 7bcd474c TESTED=TODO
Remaining changes: kkmpptxz 431886f6 (no description set)
Working copy (@) now at: kkmpptxz 431886f6 (no description set)
Parent commit (@-) : zzzzzzzz 00000000 (empty) (no description set)
Added 0 files, modified 0 files, removed 1 files
[EOF]
");
insta::assert_snapshot!(get_log_output(&work_dir), @"
@ kkmpptxzrspx false
│ ○ qpvuntsmwlqt false TESTED=TODO
├─╯
◆ zzzzzzzzzzzz true
[EOF]
");
insta::assert_snapshot!(
std::fs::read_to_string(test_env.env_root().join("editor1"))?, @r#"
JJ: Enter a description for the selected changes.
TESTED=TODO
JJ: Change ID: qpvuntsm
JJ: This commit contains the following changes:
JJ: A file1
JJ:
JJ: Lines starting with "JJ:" (like this one) will be removed.
"#);
assert!(!test_env.env_root().join("editor2").exists());
let evolog_1 = work_dir.run_jj(["evolog", "-r", "qpvun"]);
insta::assert_snapshot!(evolog_1, @"
○ qpvuntsm test.user@example.com 2001-02-03 08:05:09 7bcd474c
│ TESTED=TODO
│ -- operation 90307a87dc57 split commit f5700f8ef89e290e4e90ae6adc0908707e0d8c85
○ qpvuntsm/1 test.user@example.com 2001-02-03 08:05:08 f5700f8e (hidden)
│ (no description set)
│ -- operation f07456070284 snapshot working copy
○ qpvuntsm/2 test.user@example.com 2001-02-03 08:05:07 e8849ae1 (hidden)
(empty) (no description set)
-- operation 90267f31f904 add workspace 'default'
[EOF]
");
let evolog_2 = work_dir.run_jj(["evolog", "-r", "kkmpp"]);
insta::assert_snapshot!(evolog_2, @"
@ kkmpptxz test.user@example.com 2001-02-03 08:05:09 431886f6
│ (no description set)
│ -- operation 90307a87dc57 split commit f5700f8ef89e290e4e90ae6adc0908707e0d8c85
○ qpvuntsm/1 test.user@example.com 2001-02-03 08:05:08 f5700f8e (hidden)
│ (no description set)
│ -- operation f07456070284 snapshot working copy
○ qpvuntsm/2 test.user@example.com 2001-02-03 08:05:07 e8849ae1 (hidden)
(empty) (no description set)
-- operation 90267f31f904 add workspace 'default'
[EOF]
");
Ok(())
}
#[test]
fn test_split_parallel_with_descendants() -> TestResult {
let mut test_env = TestEnvironment::default();
let edit_script = test_env.set_up_fake_editor();
test_env.run_jj_in(".", ["git", "init", "repo"]).success();
let work_dir = test_env.work_dir("repo");
work_dir.write_file("file1", "foo\n");
work_dir.write_file("file2", "bar\n");
work_dir
.run_jj(["commit", "-m", "Add file1 & file2"])
.success();
work_dir.write_file("file3", "baz\n");
work_dir.run_jj(["commit", "-m", "Add file3"]).success();
work_dir.write_file("file4", "foobarbaz\n");
work_dir.run_jj(["describe", "-m", "Add file4"]).success();
work_dir.run_jj(["prev", "--edit"]).success();
work_dir.run_jj(["prev", "--edit"]).success();
insta::assert_snapshot!(get_log_output(&work_dir), @"
○ kkmpptxzrspx false Add file4
○ rlvkpnrzqnoo false Add file3
@ qpvuntsmwlqt false Add file1 & file2
◆ zzzzzzzzzzzz true
[EOF]
");
std::fs::write(
edit_script,
[
"dump editor1",
"write\nAdd file1",
"next invocation\n",
"dump editor2",
"write\nAdd file2",
]
.join("\0"),
)?;
let output = work_dir.run_jj(["split", "--parallel", "file1"]);
insta::assert_snapshot!(output, @"
------- stderr -------
Rebased 2 descendant commits
Selected changes : qpvuntsm 18c85f56 Add file1
Remaining changes: vruxwmqv cbdfd9cf Add file2
Working copy (@) now at: vruxwmqv cbdfd9cf Add file2
Parent commit (@-) : zzzzzzzz 00000000 (empty) (no description set)
Added 0 files, modified 0 files, removed 1 files
[EOF]
");
insta::assert_snapshot!(get_log_output(&work_dir), @"
○ kkmpptxzrspx false Add file4
○ rlvkpnrzqnoo false Add file3
├─╮
│ @ vruxwmqvtpmx false Add file2
○ │ qpvuntsmwlqt false Add file1
├─╯
◆ zzzzzzzzzzzz true
[EOF]
");
insta::assert_snapshot!(
std::fs::read_to_string(test_env.env_root().join("editor1"))?, @r#"
JJ: Enter a description for the selected changes.
Add file1 & file2
JJ: Change ID: qpvuntsm
JJ: This commit contains the following changes:
JJ: A file1
JJ:
JJ: Lines starting with "JJ:" (like this one) will be removed.
"#);
insta::assert_snapshot!(
std::fs::read_to_string(test_env.env_root().join("editor2"))?, @r#"
JJ: Enter a description for the remaining changes.
Add file1 & file2
JJ: Change ID: vruxwmqv
JJ: This commit contains the following changes:
JJ: A file2
JJ:
JJ: Lines starting with "JJ:" (like this one) will be removed.
"#);
Ok(())
}
#[test]
fn test_split_parallel_with_merge_child() -> TestResult {
let mut test_env = TestEnvironment::default();
let edit_script = test_env.set_up_fake_editor();
test_env.run_jj_in(".", ["git", "init", "repo"]).success();
let work_dir = test_env.work_dir("repo");
work_dir.run_jj(["describe", "-m=1"]).success();
work_dir.run_jj(["new", "root()", "-m=a"]).success();
work_dir.write_file("file1", "foo\n");
work_dir.write_file("file2", "bar\n");
work_dir
.run_jj(["new", "subject(1)", "subject(a)", "-m=2"])
.success();
insta::assert_snapshot!(get_log_output(&work_dir), @"
@ zsuskulnrvyr true 2
├─╮
│ ○ kkmpptxzrspx false a
○ │ qpvuntsmwlqt true 1
├─╯
◆ zzzzzzzzzzzz true
[EOF]
");
std::fs::write(
edit_script,
["write\nAdd file1", "next invocation\n", "write\nAdd file2"].join("\0"),
)?;
let output = work_dir.run_jj(["split", "-rsubject(a)", "--parallel", "file1"]);
insta::assert_snapshot!(output, @"
------- stderr -------
Rebased 1 descendant commits
Selected changes : kkmpptxz cc199567 Add file1
Remaining changes: royxmykx 82a5c527 Add file2
Working copy (@) now at: zsuskuln b7cdcdec (empty) 2
Parent commit (@-) : qpvuntsm 884fe9b9 (empty) 1
Parent commit (@-) : kkmpptxz cc199567 Add file1
Parent commit (@-) : royxmykx 82a5c527 Add file2
[EOF]
");
insta::assert_snapshot!(get_log_output(&work_dir), @"
@ zsuskulnrvyr true 2
├─┬─╮
│ │ ○ royxmykxtrkr false Add file2
│ ○ │ kkmpptxzrspx false Add file1
│ ├─╯
○ │ qpvuntsmwlqt true 1
├─╯
◆ zzzzzzzzzzzz true
[EOF]
");
Ok(())
}
#[test]
fn test_split_parallel_with_conflict() -> TestResult {
let mut test_env = TestEnvironment::default();
let diff_editor = test_env.set_up_fake_diff_editor();
test_env.run_jj_in(".", ["git", "init", "repo"]).success();
let work_dir = test_env.work_dir("repo");
work_dir.write_file("file", "line 1\nline 2\nline 3\n");
work_dir.run_jj(["new"]).success();
work_dir.write_file("file", "line 1\nline 2.1\nline 2.2\nline 3\n");
work_dir.run_jj(["new"]).success();
work_dir.write_file("file", "line 1\nline 3\n");
work_dir.run_jj(["prev", "--edit"]).success();
insta::assert_snapshot!(get_log_output(&work_dir), @"
○ kkmpptxzrspx false
@ rlvkpnrzqnoo false
○ qpvuntsmwlqt false
◆ zzzzzzzzzzzz true
[EOF]
");
std::fs::write(
diff_editor,
["write file\nline 1\nline 2.1\nline 3\n"].join("\0"),
)?;
let output = work_dir.run_jj(["split", "--parallel", "-i", "-m="]);
insta::assert_snapshot!(output, @"
------- stderr -------
Rebased 1 descendant commits
Selected changes : rlvkpnrz abe15fea (no description set)
Remaining changes: royxmykx 4bbc5826 (conflict) (no description set)
Working copy (@) now at: royxmykx 4bbc5826 (conflict) (no description set)
Parent commit (@-) : qpvuntsm ee8e9376 (no description set)
Added 0 files, modified 1 files, removed 0 files
Warning: There are unresolved conflicts at these paths:
file 2-sided conflict
New conflicts appeared in 1 commits:
royxmykx 4bbc5826 (conflict) (no description set)
Hint: To resolve the conflicts, start by creating a commit on top of
the conflicted commit:
jj new royxmykx
Then use `jj resolve`, or edit the conflict markers in the file directly.
Once the conflicts are resolved, you can inspect the result with `jj diff`.
Then run `jj squash` to move the resolution into the conflicted commit.
[EOF]
");
insta::assert_snapshot!(get_log_output(&work_dir), @"
○ kkmpptxzrspx false
├─╮
│ @ royxmykxtrkr false
○ │ rlvkpnrzqnoo false
├─╯
○ qpvuntsmwlqt false
◆ zzzzzzzzzzzz true
[EOF]
");
insta::assert_snapshot!(work_dir.read_file("file"), @r"
line 1
<<<<<<< conflict 1 of 1
%%%%%%% diff from: selected changes for split (from rlvkpnrz 35382813)
\\\\\\\ to: split revision (rlvkpnrz 35382813)
line 2.1
+line 2.2
+++++++ qpvuntsm ee8e9376 (parents of split revision)
line 2
>>>>>>> conflict 1 of 1 ends
line 3
");
insta::assert_snapshot!(work_dir.run_jj(["file", "show", "-r=@+-~@", "file"]), @"
line 1
line 2.1
line 3
[EOF]
");
insta::assert_snapshot!(work_dir.run_jj(["file", "show", "-r=@+", "file"]), @"
line 1
line 3
[EOF]
");
Ok(())
}
#[test]
fn test_split_empty() -> TestResult {
let mut test_env = TestEnvironment::default();
test_env.set_up_fake_editor();
test_env.run_jj_in(".", ["git", "init", "repo"]).success();
let work_dir = test_env.work_dir("repo");
work_dir.run_jj(["describe", "--message", "abc"]).success();
let output = work_dir.run_jj(["split"]);
insta::assert_snapshot!(output, @"
------- stderr -------
Hint: Using default editor ':builtin'; run `jj config set --user ui.diff-editor :builtin` to disable this message.
Warning: Empty diff - won't run diff editor.
Warning: All changes have been selected, so the original revision will become empty
Selected changes : qpvuntsm a8bcd860 (empty) abc
Remaining changes: kkmpptxz 304fb14c (empty) abc
Working copy (@) now at: kkmpptxz 304fb14c (empty) abc
Parent commit (@-) : qpvuntsm a8bcd860 (empty) abc
[EOF]
");
let output = work_dir.run_jj(["split", "@"]);
insta::assert_snapshot!(output, @r"
------- stderr -------
Warning: No matching entries for paths: @
Warning: All changes have been selected, so the original revision will become empty
Selected changes : kkmpptxz cd55fd14 (empty) abc
Remaining changes: zsuskuln 49a292cc (empty) abc
Working copy (@) now at: zsuskuln 49a292cc (empty) abc
Parent commit (@-) : kkmpptxz cd55fd14 (empty) abc
[EOF]
");
Ok(())
}
#[test]
fn test_split_message_editor_avoids_unc() -> TestResult {
let mut test_env = TestEnvironment::default();
let edit_script = test_env.set_up_fake_editor();
test_env.run_jj_in(".", ["git", "init", "repo"]).success();
let work_dir = test_env.work_dir("repo");
work_dir.write_file("file1", "foo");
work_dir.write_file("file2", "foo");
std::fs::write(edit_script, "dump-path path")?;
work_dir.run_jj(["split", "file2"]).success();
let edited_path = PathBuf::from(std::fs::read_to_string(test_env.env_root().join("path"))?);
assert_eq!(edited_path, dunce::simplified(&edited_path));
Ok(())
}
#[test]
fn test_split_interactive() -> TestResult {
let mut test_env = TestEnvironment::default();
let edit_script = test_env.set_up_fake_editor();
let diff_editor = test_env.set_up_fake_diff_editor();
test_env.run_jj_in(".", ["git", "init", "repo"]).success();
let work_dir = test_env.work_dir("repo");
work_dir.write_file("file1", "foo\n");
work_dir.write_file("file2", "bar\n");
std::fs::write(edit_script, ["dump editor"].join("\0"))?;
let diff_script = ["rm file2", "dump JJ-INSTRUCTIONS instrs"].join("\0");
std::fs::write(diff_editor, diff_script)?;
let output = work_dir.run_jj(["split"]);
insta::assert_snapshot!(output, @"
------- stderr -------
Selected changes : qpvuntsm c664a51b (no description set)
Remaining changes: rlvkpnrz 7e5d65b1 (no description set)
Working copy (@) now at: rlvkpnrz 7e5d65b1 (no description set)
Parent commit (@-) : qpvuntsm c664a51b (no description set)
[EOF]
");
insta::assert_snapshot!(
std::fs::read_to_string(test_env.env_root().join("instrs"))?, @"
You are splitting a commit into two: qpvuntsm f5700f8e (no description set)
The diff initially shows the changes in the commit you're splitting.
Adjust the right side until it shows the contents you want to split into the
new commit.
The changes that are not selected will replace the original commit.
");
insta::assert_snapshot!(
std::fs::read_to_string(test_env.env_root().join("editor"))?, @r#"
JJ: Enter a description for the selected changes.
JJ: Change ID: qpvuntsm
JJ: This commit contains the following changes:
JJ: A file1
JJ:
JJ: Lines starting with "JJ:" (like this one) will be removed.
"#);
let output = work_dir.run_jj(["log", "--summary"]);
insta::assert_snapshot!(output, @"
@ rlvkpnrz test.user@example.com 2001-02-03 08:05:08 7e5d65b1
│ (no description set)
│ A file2
○ qpvuntsm test.user@example.com 2001-02-03 08:05:08 c664a51b
│ (no description set)
│ A file1
◆ zzzzzzzz root() 00000000
[EOF]
");
Ok(())
}
#[test]
fn test_split_interactive_with_paths() -> TestResult {
let mut test_env = TestEnvironment::default();
let edit_script = test_env.set_up_fake_editor();
let diff_editor = test_env.set_up_fake_diff_editor();
test_env.run_jj_in(".", ["git", "init", "repo"]).success();
let work_dir = test_env.work_dir("repo");
work_dir.write_file("file2", "");
work_dir.write_file("file3", "");
work_dir.run_jj(["new"]).success();
work_dir.write_file("file1", "foo\n");
work_dir.write_file("file2", "bar\n");
work_dir.write_file("file3", "baz\n");
std::fs::write(edit_script, ["dump editor"].join("\0"))?;
let diff_script = [
"files-before file2",
"files-after JJ-INSTRUCTIONS file1 file2",
"reset file2",
]
.join("\0");
std::fs::write(diff_editor, diff_script)?;
let output = work_dir.run_jj(["split", "-i", "file1", "file2"]);
insta::assert_snapshot!(output, @"
------- stderr -------
Selected changes : rlvkpnrz cdc9960a (no description set)
Remaining changes: kkmpptxz 7255f070 (no description set)
Working copy (@) now at: kkmpptxz 7255f070 (no description set)
Parent commit (@-) : rlvkpnrz cdc9960a (no description set)
[EOF]
");
insta::assert_snapshot!(
std::fs::read_to_string(test_env.env_root().join("editor"))?, @r#"
JJ: Enter a description for the selected changes.
JJ: Change ID: rlvkpnrz
JJ: This commit contains the following changes:
JJ: A file1
JJ:
JJ: Lines starting with "JJ:" (like this one) will be removed.
"#);
let output = work_dir.run_jj(["log", "--summary"]);
insta::assert_snapshot!(output, @"
@ kkmpptxz test.user@example.com 2001-02-03 08:05:09 7255f070
│ (no description set)
│ M file2
│ M file3
○ rlvkpnrz test.user@example.com 2001-02-03 08:05:09 cdc9960a
│ (no description set)
│ A file1
○ qpvuntsm test.user@example.com 2001-02-03 08:05:08 ff687a2f
│ (no description set)
│ A file2
│ A file3
◆ zzzzzzzz root() 00000000
[EOF]
");
Ok(())
}
#[test]
fn test_split_with_multiple_workspaces_same_working_copy() -> TestResult {
let mut test_env = TestEnvironment::default();
let edit_script = test_env.set_up_fake_editor();
test_env.run_jj_in(".", ["git", "init", "main"]).success();
let main_dir = test_env.work_dir("main");
let secondary_dir = test_env.work_dir("secondary");
main_dir.run_jj(["desc", "-m", "first-commit"]).success();
main_dir.write_file("file1", "foo");
main_dir.write_file("file2", "foo");
main_dir
.run_jj(["workspace", "add", "--name", "second", "../secondary"])
.success();
secondary_dir
.run_jj(["edit", "-r", "subject(first-commit)"])
.success();
insta::assert_snapshot!(get_workspace_log_output(&main_dir), @"
@ qpvuntsmwlqt default@ second@ first-commit
◆ zzzzzzzzzzzz
[EOF]
");
let setup_opid = main_dir.current_operation_id();
std::fs::write(
&edit_script,
["", "next invocation\n", "write\nsecond-commit"].join("\0"),
)?;
main_dir.run_jj(["split", "file2"]).success();
insta::assert_snapshot!(get_workspace_log_output(&main_dir), @"
@ royxmykxtrkr default@ second@ second-commit
○ qpvuntsmwlqt first-commit
◆ zzzzzzzzzzzz
[EOF]
");
main_dir.run_jj(["op", "restore", &setup_opid]).success();
std::fs::write(
&edit_script,
["", "next invocation\n", "write\nsecond-commit"].join("\0"),
)?;
main_dir.run_jj(["split", "file2", "--parallel"]).success();
insta::assert_snapshot!(get_workspace_log_output(&main_dir), @"
@ yostqsxwqrlt default@ second@ second-commit
│ ○ qpvuntsmwlqt first-commit
├─╯
◆ zzzzzzzzzzzz
[EOF]
");
Ok(())
}
#[test]
fn test_split_with_multiple_workspaces_different_working_copy() -> TestResult {
let mut test_env = TestEnvironment::default();
let edit_script = test_env.set_up_fake_editor();
test_env.run_jj_in(".", ["git", "init", "main"]).success();
let main_dir = test_env.work_dir("main");
main_dir.run_jj(["desc", "-m", "first-commit"]).success();
main_dir.write_file("file1", "foo");
main_dir.write_file("file2", "foo");
main_dir
.run_jj(["workspace", "add", "--name", "second", "../secondary"])
.success();
insta::assert_snapshot!(get_workspace_log_output(&main_dir), @"
@ qpvuntsmwlqt default@ first-commit
│ ○ pmmvwywvzvvn second@
├─╯
◆ zzzzzzzzzzzz
[EOF]
");
let setup_opid = main_dir.current_operation_id();
std::fs::write(
&edit_script,
["", "next invocation\n", "write\nsecond-commit"].join("\0"),
)?;
main_dir.run_jj(["split", "file2"]).success();
insta::assert_snapshot!(get_workspace_log_output(&main_dir), @"
@ mzvwutvlkqwt default@ second-commit
○ qpvuntsmwlqt first-commit
│ ○ pmmvwywvzvvn second@
├─╯
◆ zzzzzzzzzzzz
[EOF]
");
main_dir.run_jj(["op", "restore", &setup_opid]).success();
std::fs::write(
&edit_script,
["", "next invocation\n", "write\nsecond-commit"].join("\0"),
)?;
main_dir.run_jj(["split", "file2", "--parallel"]).success();
insta::assert_snapshot!(get_workspace_log_output(&main_dir), @"
@ vruxwmqvtpmx default@ second-commit
│ ○ qpvuntsmwlqt first-commit
├─╯
│ ○ pmmvwywvzvvn second@
├─╯
◆ zzzzzzzzzzzz
[EOF]
");
Ok(())
}
#[test]
fn test_split_with_non_empty_description_and_trailers() -> TestResult {
let mut test_env = TestEnvironment::default();
let edit_script = test_env.set_up_fake_editor();
test_env.run_jj_in(".", ["git", "init", "repo"]).success();
test_env.add_config(r#"template-aliases.default_commit_description = '"\n\nTESTED=TODO\n"'"#);
let work_dir = test_env.work_dir("repo");
work_dir.write_file("file1", "foo\n");
work_dir.write_file("file2", "bar\n");
work_dir.run_jj(["describe", "-m", "test"]).success();
std::fs::write(
edit_script,
[
"dump editor1",
"write\npart 1",
"next invocation\n",
"dump editor2",
"write\npart 2",
]
.join("\0"),
)?;
test_env.add_config(
r#"[templates]
commit_trailers = '''"Signed-off-by: " ++ committer.email()'''"#,
);
let output = work_dir.run_jj(["split", "file1"]);
insta::assert_snapshot!(output, @"
------- stderr -------
Selected changes : qpvuntsm c7f7b14b part 1
Remaining changes: kkmpptxz ac33a5a9 part 2
Working copy (@) now at: kkmpptxz ac33a5a9 part 2
Parent commit (@-) : qpvuntsm c7f7b14b part 1
[EOF]
");
insta::assert_snapshot!(
std::fs::read_to_string(test_env.env_root().join("editor1"))?, @r#"
JJ: Enter a description for the selected changes.
test
Signed-off-by: test.user@example.com
JJ: Change ID: qpvuntsm
JJ: This commit contains the following changes:
JJ: A file1
JJ:
JJ: Lines starting with "JJ:" (like this one) will be removed.
"#);
insta::assert_snapshot!(
std::fs::read_to_string(test_env.env_root().join("editor2"))?, @r#"
JJ: Enter a description for the remaining changes.
test
Signed-off-by: test.user@example.com
JJ: Change ID: kkmpptxz
JJ: This commit contains the following changes:
JJ: A file2
JJ:
JJ: Lines starting with "JJ:" (like this one) will be removed.
"#);
insta::assert_snapshot!(get_log_output(&work_dir), @"
@ kkmpptxzrspx false part 2
○ qpvuntsmwlqt false part 1
◆ zzzzzzzzzzzz true
[EOF]
");
Ok(())
}
#[test]
fn test_split_with_message() -> TestResult {
let test_env = TestEnvironment::default();
test_env.run_jj_in(".", ["git", "init", "repo"]).success();
let work_dir = test_env.work_dir("repo");
work_dir.write_file("file1", "foo\n");
work_dir.write_file("file2", "bar\n");
work_dir.run_jj(["describe", "-m", "my feature"]).success();
let setup_opid = work_dir.current_operation_id();
let output = work_dir.run_jj(["split", "-m", "fix in file1", "file1"]);
insta::assert_snapshot!(output, @"
------- stderr -------
Selected changes : qpvuntsm f2a70519 fix in file1
Remaining changes: kkmpptxz cac11766 my feature
Working copy (@) now at: kkmpptxz cac11766 my feature
Parent commit (@-) : qpvuntsm f2a70519 fix in file1
[EOF]
");
insta::assert_snapshot!(get_log_output(&work_dir), @"
@ kkmpptxzrspx false my feature
○ qpvuntsmwlqt false fix in file1
◆ zzzzzzzzzzzz true
[EOF]
");
work_dir.run_jj(["op", "restore", &setup_opid]).success();
let output = work_dir.run_jj([
"split",
"--config",
r#"templates.commit_trailers='"CC: " ++ committer.email()'"#,
"-m",
"fix in file1",
"file1",
]);
insta::assert_snapshot!(output, @"
------- stderr -------
Selected changes : qpvuntsm d01cf12d fix in file1
Remaining changes: royxmykx b1556ed9 my feature
Working copy (@) now at: royxmykx b1556ed9 my feature
Parent commit (@-) : qpvuntsm d01cf12d fix in file1
[EOF]
");
insta::assert_snapshot!(get_log_output(&work_dir), @"
@ royxmykxtrkr false my feature
○ qpvuntsmwlqt false fix in file1
│
│ CC: test.user@example.com
◆ zzzzzzzzzzzz true
[EOF]
");
Ok(())
}
#[test]
fn test_split_move_first_commit() -> TestResult {
let test_env = TestEnvironment::default();
test_env.run_jj_in(".", ["git", "init", "repo"]).success();
let work_dir = test_env.work_dir("repo");
work_dir.write_file("file1", "foo\n");
work_dir.write_file("file2", "bar\n");
work_dir.run_jj(["commit", "-m", "file2"]).success();
work_dir.write_file("file3", "bar\n");
work_dir.run_jj(["commit", "-m", "file3"]).success();
work_dir.write_file("file4", "bar\n");
work_dir.run_jj(["commit", "-m", "file4"]).success();
work_dir.run_jj(["new", "root()"]).success();
work_dir.write_file("file5", "bar\n");
work_dir.run_jj(["commit", "-m", "file5"]).success();
insta::assert_snapshot!(get_log_with_summary(&work_dir), @"
@ royxmykxtrkr
○ mzvwutvlkqwt file5
│ A file5
│ ○ kkmpptxzrspx file4
│ │ A file4
│ ○ rlvkpnrzqnoo file3
│ │ A file3
│ ○ qpvuntsmwlqt file2
├─╯ A file1
│ A file2
◆ zzzzzzzzzzzz
[EOF]
");
let setup_opid = work_dir.current_operation_id();
let output = work_dir.run_jj([
"split",
"-m",
"file1",
"-r",
"qpvuntsmwlqt",
"--insert-before",
"qpvuntsmwlqt",
"file1",
]);
insta::assert_snapshot!(output, @"
------- stderr -------
Rebased 2 descendant commits
Selected changes : vruxwmqv bf94c29a file1
Remaining changes: qpvuntsm 66b1d4f1 file2
[EOF]
");
insta::assert_snapshot!(get_log_with_summary(&work_dir), @"
@ royxmykxtrkr
○ mzvwutvlkqwt file5
│ A file5
│ ○ kkmpptxzrspx file4
│ │ A file4
│ ○ rlvkpnrzqnoo file3
│ │ A file3
│ ○ qpvuntsmwlqt file2
│ │ A file2
│ ○ vruxwmqvtpmx file1
├─╯ A file1
◆ zzzzzzzzzzzz
[EOF]
");
work_dir.run_jj(["op", "restore", &setup_opid]).success();
let output = work_dir.run_jj([
"split",
"-m",
"file1",
"-r",
"qpvuntsmwlqt",
"--insert-after",
"qpvuntsmwlqt",
"file1",
]);
insta::assert_snapshot!(output, @"
------- stderr -------
Rebased 2 descendant commits
Selected changes : kpqxywon 08294e90 file1
Remaining changes: qpvuntsm 76ebcbb8 file2
[EOF]
");
insta::assert_snapshot!(get_log_with_summary(&work_dir), @"
@ royxmykxtrkr
○ mzvwutvlkqwt file5
│ A file5
│ ○ kkmpptxzrspx file4
│ │ A file4
│ ○ rlvkpnrzqnoo file3
│ │ A file3
│ ○ kpqxywonksrl file1
│ │ A file1
│ ○ qpvuntsmwlqt file2
├─╯ A file2
◆ zzzzzzzzzzzz
[EOF]
");
work_dir.run_jj(["op", "restore", &setup_opid]).success();
let output = work_dir.run_jj([
"split",
"-m",
"file1",
"-r",
"qpvuntsmwlqt",
"--onto",
"rlvkpnrzqnoo",
"file1",
]);
insta::assert_snapshot!(output, @"
------- stderr -------
Rebased 2 descendant commits
Selected changes : lylxulpl b42b2604 file1
Remaining changes: qpvuntsm 0f76cbf0 file2
[EOF]
");
insta::assert_snapshot!(get_log_with_summary(&work_dir), @"
@ royxmykxtrkr
○ mzvwutvlkqwt file5
│ A file5
│ ○ kkmpptxzrspx file4
│ │ A file4
│ │ ○ lylxulplsnyw file1
│ ├─╯ A file1
│ ○ rlvkpnrzqnoo file3
│ │ A file3
│ ○ qpvuntsmwlqt file2
├─╯ A file2
◆ zzzzzzzzzzzz
[EOF]
");
work_dir.run_jj(["op", "restore", &setup_opid]).success();
let output = work_dir.run_jj([
"split",
"-m",
"file1",
"-r",
"qpvuntsmwlqt",
"--insert-after",
"qpvuntsmwlqt",
"--insert-before",
"kkmpptxzrspx",
"file1",
]);
insta::assert_snapshot!(output, @"
------- stderr -------
Rebased 2 descendant commits
Selected changes : uyznsvlq d0338445 file1
Remaining changes: qpvuntsm 16d41320 file2
[EOF]
");
insta::assert_snapshot!(get_log_with_summary(&work_dir), @"
@ royxmykxtrkr
○ mzvwutvlkqwt file5
│ A file5
│ ○ kkmpptxzrspx file4
│ ├─╮ A file4
│ │ ○ uyznsvlquzzm file1
│ │ │ A file1
│ ○ │ rlvkpnrzqnoo file3
│ ├─╯ A file3
│ ○ qpvuntsmwlqt file2
├─╯ A file2
◆ zzzzzzzzzzzz
[EOF]
");
work_dir.run_jj(["op", "restore", &setup_opid]).success();
let output = work_dir.run_jj([
"split",
"-m",
"file1",
"-r",
"qpvuntsmwlqt",
"--before",
"@",
"file1",
]);
insta::assert_snapshot!(output, @"
------- stderr -------
Rebased 3 descendant commits
Selected changes : nmzmmopx 72225233 file1
Remaining changes: qpvuntsm 98b70782 file2
Working copy (@) now at: royxmykx c3dd10b0 (empty) (no description set)
Parent commit (@-) : nmzmmopx 72225233 file1
Added 1 files, modified 0 files, removed 0 files
[EOF]
");
insta::assert_snapshot!(get_log_with_summary(&work_dir), @"
@ royxmykxtrkr
○ nmzmmopxokps file1
│ A file1
○ mzvwutvlkqwt file5
│ A file5
│ ○ kkmpptxzrspx file4
│ │ A file4
│ ○ rlvkpnrzqnoo file3
│ │ A file3
│ ○ qpvuntsmwlqt file2
├─╯ A file2
◆ zzzzzzzzzzzz
[EOF]
");
work_dir.run_jj(["op", "restore", &setup_opid]).success();
let output = work_dir.run_jj([
"split",
"-m",
"file1",
"-r",
"qpvuntsmwlqt",
"--after",
"mzvwutvlkqwt",
"--after",
"kkmpptxzrspx",
"file1",
]);
insta::assert_snapshot!(output, @"
------- stderr -------
Rebased 3 descendant commits
Selected changes : nlrtlrxv 1b6975b0 file1
Remaining changes: qpvuntsm 905586dd file2
Working copy (@) now at: royxmykx 85be9860 (empty) (no description set)
Parent commit (@-) : nlrtlrxv 1b6975b0 file1
Added 4 files, modified 0 files, removed 0 files
[EOF]
");
insta::assert_snapshot!(get_log_with_summary(&work_dir), @"
@ royxmykxtrkr
○ nlrtlrxvuusk file1
├─╮ A file1
│ ○ kkmpptxzrspx file4
│ │ A file4
│ ○ rlvkpnrzqnoo file3
│ │ A file3
│ ○ qpvuntsmwlqt file2
│ │ A file2
○ │ mzvwutvlkqwt file5
├─╯ A file5
◆ zzzzzzzzzzzz
[EOF]
");
Ok(())
}
enum BookmarkBehavior {
Default,
MoveBookmarkToChild,
LeaveBookmarkWithTarget,
}
#[test_case(BookmarkBehavior::Default; "default_behavior")]
#[test_case(BookmarkBehavior::MoveBookmarkToChild; "move_bookmark_to_child")]
#[test_case(BookmarkBehavior::LeaveBookmarkWithTarget; "leave_bookmark_with_target")]
fn test_split_with_bookmarks(bookmark_behavior: BookmarkBehavior) -> TestResult {
let mut test_env = TestEnvironment::default();
let edit_script = test_env.set_up_fake_editor();
test_env.run_jj_in(".", ["git", "init", "main"]).success();
let main_dir = test_env.work_dir("main");
match bookmark_behavior {
BookmarkBehavior::LeaveBookmarkWithTarget => {
test_env.add_config("split.legacy-bookmark-behavior=false");
}
BookmarkBehavior::MoveBookmarkToChild => {
test_env.add_config("split.legacy-bookmark-behavior=true");
}
BookmarkBehavior::Default => (),
}
main_dir.run_jj(["desc", "-m", "first-commit"]).success();
main_dir.write_file("file1", "foo");
main_dir.write_file("file2", "foo");
main_dir
.run_jj(["bookmark", "set", "*le-signet*", "-r", "@"])
.success();
insta::allow_duplicates! {
insta::assert_snapshot!(get_log_output(&main_dir), @"
@ qpvuntsmwlqt false *le-signet* first-commit
◆ zzzzzzzzzzzz true
[EOF]
");
}
let setup_opid = main_dir.current_operation_id();
std::fs::write(
&edit_script,
["", "next invocation\n", "write\nsecond-commit"].join("\0"),
)?;
let output = main_dir.run_jj(["split", "file2"]);
match bookmark_behavior {
BookmarkBehavior::LeaveBookmarkWithTarget => {
insta::allow_duplicates! {
insta::assert_snapshot!(output, @"
------- stderr -------
Selected changes : qpvuntsm a481fe8a *le-signet* | first-commit
Remaining changes: mzvwutvl 5f597a6e second-commit
Working copy (@) now at: mzvwutvl 5f597a6e second-commit
Parent commit (@-) : qpvuntsm a481fe8a *le-signet* | first-commit
[EOF]
");
}
insta::allow_duplicates! {
insta::assert_snapshot!(get_log_output(&main_dir), @"
@ mzvwutvlkqwt false second-commit
○ qpvuntsmwlqt false *le-signet* first-commit
◆ zzzzzzzzzzzz true
[EOF]
");
}
}
BookmarkBehavior::Default | BookmarkBehavior::MoveBookmarkToChild => {
insta::allow_duplicates! {
insta::assert_snapshot!(output, @"
------- stderr -------
Selected changes : qpvuntsm a481fe8a first-commit
Remaining changes: mzvwutvl 5f597a6e *le-signet* | second-commit
Working copy (@) now at: mzvwutvl 5f597a6e *le-signet* | second-commit
Parent commit (@-) : qpvuntsm a481fe8a first-commit
[EOF]
");
}
insta::allow_duplicates! {
insta::assert_snapshot!(get_log_output(&main_dir), @"
@ mzvwutvlkqwt false *le-signet* second-commit
○ qpvuntsmwlqt false first-commit
◆ zzzzzzzzzzzz true
[EOF]
");
}
}
}
main_dir.run_jj(["op", "restore", &setup_opid]).success();
std::fs::write(
&edit_script,
["", "next invocation\n", "write\nsecond-commit"].join("\0"),
)?;
main_dir.run_jj(["split", "file2", "--parallel"]).success();
match bookmark_behavior {
BookmarkBehavior::LeaveBookmarkWithTarget => {
insta::allow_duplicates! {
insta::assert_snapshot!(get_log_output(&main_dir), @"
@ vruxwmqvtpmx false second-commit
│ ○ qpvuntsmwlqt false *le-signet* first-commit
├─╯
◆ zzzzzzzzzzzz true
[EOF]
");
}
}
BookmarkBehavior::Default | BookmarkBehavior::MoveBookmarkToChild => {
insta::allow_duplicates! {
insta::assert_snapshot!(get_log_output(&main_dir), @"
@ vruxwmqvtpmx false *le-signet* second-commit
│ ○ qpvuntsmwlqt false first-commit
├─╯
◆ zzzzzzzzzzzz true
[EOF]
");
}
}
}
Ok(())
}
#[test]
fn test_split_with_editor_and_message_args() -> TestResult {
let mut test_env = TestEnvironment::default();
let edit_script = test_env.set_up_fake_editor();
test_env.run_jj_in(".", ["git", "init", "repo"]).success();
let work_dir = test_env.work_dir("repo");
work_dir.write_file("file1", "foo\n");
work_dir.write_file("file2", "bar\n");
work_dir
.run_jj(["describe", "-m", "original description"])
.success();
std::fs::write(
&edit_script,
[
"dump editor1",
"write\nedited message 1",
"next invocation\n",
"dump editor2",
"write\nedited message 2",
]
.join("\0"),
)?;
work_dir
.run_jj([
"split",
"-m",
"message from command line",
"--editor",
"file1",
])
.success();
insta::assert_snapshot!(
std::fs::read_to_string(test_env.env_root().join("editor1"))?, @r#"
JJ: Enter a description for the selected changes.
message from command line
JJ: Change ID: qpvuntsm
JJ: This commit contains the following changes:
JJ: A file1
JJ:
JJ: Lines starting with "JJ:" (like this one) will be removed.
"#);
insta::assert_snapshot!(
std::fs::read_to_string(test_env.env_root().join("editor2"))?, @r#"
JJ: Enter a description for the remaining changes.
original description
JJ: Change ID: kkmpptxz
JJ: This commit contains the following changes:
JJ: A file2
JJ:
JJ: Lines starting with "JJ:" (like this one) will be removed.
"#);
insta::assert_snapshot!(get_log_output(&work_dir), @"
@ kkmpptxzrspx false edited message 2
○ qpvuntsmwlqt false edited message 1
◆ zzzzzzzzzzzz true
[EOF]
");
Ok(())
}
#[test]
fn test_split_with_editor_and_empty_message() -> TestResult {
let mut test_env = TestEnvironment::default();
let edit_script = test_env.set_up_fake_editor();
test_env.run_jj_in(".", ["git", "init", "repo"]).success();
let work_dir = test_env.work_dir("repo");
work_dir.write_file("file1", "foo\n");
work_dir.write_file("file2", "bar\n");
work_dir
.run_jj(["describe", "-m", "original description"])
.success();
std::fs::write(
&edit_script,
[
"dump editor1",
"write\nfirst commit",
"next invocation\n",
"dump editor2",
"write\nsecond commit",
]
.join("\0"),
)?;
work_dir
.run_jj([
"split",
"-m",
"",
"--editor",
"--config",
r#"templates.commit_trailers='"Trailer: value"'"#,
"file1",
])
.success();
insta::assert_snapshot!(
std::fs::read_to_string(test_env.env_root().join("editor1"))?, @r#"
JJ: Enter a description for the selected changes.
Trailer: value
JJ: Change ID: qpvuntsm
JJ: This commit contains the following changes:
JJ: A file1
JJ:
JJ: Lines starting with "JJ:" (like this one) will be removed.
"#);
insta::assert_snapshot!(
std::fs::read_to_string(test_env.env_root().join("editor2"))?, @r#"
JJ: Enter a description for the remaining changes.
original description
Trailer: value
JJ: Change ID: kkmpptxz
JJ: This commit contains the following changes:
JJ: A file2
JJ:
JJ: Lines starting with "JJ:" (like this one) will be removed.
"#);
insta::assert_snapshot!(get_log_output(&work_dir), @"
@ kkmpptxzrspx false second commit
○ qpvuntsmwlqt false first commit
◆ zzzzzzzzzzzz true
[EOF]
");
Ok(())
}
#[test]
fn test_split_with_editor_without_message() -> TestResult {
let mut test_env = TestEnvironment::default();
let edit_script = test_env.set_up_fake_editor();
test_env.run_jj_in(".", ["git", "init", "repo"]).success();
let work_dir = test_env.work_dir("repo");
work_dir.write_file("file1", "foo\n");
work_dir.write_file("file2", "bar\n");
work_dir
.run_jj(["describe", "-m", "original description"])
.success();
std::fs::write(
&edit_script,
[
"dump editor0",
"write\nfrom editor1",
"next invocation\n",
"dump editor1",
"write\nfrom editor2",
]
.join("\0"),
)?;
work_dir.run_jj(["split", "--editor", "file1"]).success();
insta::assert_snapshot!(
std::fs::read_to_string(test_env.env_root().join("editor0"))?, @r#"
JJ: Enter a description for the selected changes.
original description
JJ: Change ID: qpvuntsm
JJ: This commit contains the following changes:
JJ: A file1
JJ:
JJ: Lines starting with "JJ:" (like this one) will be removed.
"#);
insta::assert_snapshot!(
std::fs::read_to_string(test_env.env_root().join("editor1"))?, @r#"
JJ: Enter a description for the remaining changes.
original description
JJ: Change ID: kkmpptxz
JJ: This commit contains the following changes:
JJ: A file2
JJ:
JJ: Lines starting with "JJ:" (like this one) will be removed.
"#);
insta::assert_snapshot!(get_log_output(&work_dir), @"
@ kkmpptxzrspx false from editor2
○ qpvuntsmwlqt false from editor1
◆ zzzzzzzzzzzz true
[EOF]
");
Ok(())
}