2022-11-26 23:57:50 +00:00
|
|
|
// Copyright 2022 The Jujutsu Authors
|
2022-05-02 20:03:34 +00:00
|
|
|
//
|
|
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
// you may not use this file except in compliance with the License.
|
|
|
|
// You may obtain a copy of the License at
|
|
|
|
//
|
|
|
|
// https://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
//
|
|
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
// See the License for the specific language governing permissions and
|
|
|
|
// limitations under the License.
|
|
|
|
|
2022-05-02 19:58:32 +00:00
|
|
|
use std::path::Path;
|
|
|
|
|
2023-07-03 10:04:31 +00:00
|
|
|
use crate::common::TestEnvironment;
|
2022-05-02 20:03:34 +00:00
|
|
|
|
|
|
|
pub mod common;
|
|
|
|
|
2022-05-02 19:58:32 +00:00
|
|
|
#[test]
|
|
|
|
fn test_branch_multiple_names() {
|
|
|
|
let test_env = TestEnvironment::default();
|
|
|
|
test_env.jj_cmd_success(test_env.env_root(), &["init", "repo", "--git"]);
|
|
|
|
let repo_path = test_env.env_root().join("repo");
|
|
|
|
|
2023-07-03 10:04:31 +00:00
|
|
|
let (stdout, stderr) = test_env.jj_cmd_ok(&repo_path, &["branch", "set", "foo", "bar"]);
|
|
|
|
insta::assert_snapshot!(stdout, @"");
|
|
|
|
insta::assert_snapshot!(stderr, @r###"
|
|
|
|
warning: Updating multiple branches (2).
|
|
|
|
"###);
|
2022-05-02 19:58:32 +00:00
|
|
|
|
|
|
|
insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###"
|
2023-02-09 02:53:47 +00:00
|
|
|
@ bar foo 230dd059e1b0
|
2023-03-15 03:37:56 +00:00
|
|
|
◉ 000000000000
|
2022-05-02 19:58:32 +00:00
|
|
|
"###);
|
|
|
|
|
2023-06-29 02:36:08 +00:00
|
|
|
let stdout = test_env.jj_cmd_success(&repo_path, &["branch", "delete", "foo", "bar", "foo"]);
|
|
|
|
insta::assert_snapshot!(stdout, @r###"
|
|
|
|
Deleted 2 branches.
|
|
|
|
"###);
|
2022-05-02 19:58:32 +00:00
|
|
|
insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###"
|
2023-02-09 02:53:47 +00:00
|
|
|
@ 230dd059e1b0
|
2023-03-15 03:37:56 +00:00
|
|
|
◉ 000000000000
|
2022-05-02 19:58:32 +00:00
|
|
|
"###);
|
|
|
|
}
|
|
|
|
|
2023-04-04 02:02:12 +00:00
|
|
|
#[test]
|
2023-09-03 21:07:15 +00:00
|
|
|
fn test_branch_at_root() {
|
2023-04-04 02:02:12 +00:00
|
|
|
let test_env = TestEnvironment::default();
|
|
|
|
test_env.jj_cmd_success(test_env.env_root(), &["init", "repo", "--git"]);
|
|
|
|
let repo_path = test_env.env_root().join("repo");
|
|
|
|
|
2023-09-03 21:07:15 +00:00
|
|
|
let stdout = test_env.jj_cmd_success(&repo_path, &["branch", "create", "fred", "-r=root()"]);
|
|
|
|
insta::assert_snapshot!(stdout, @"");
|
|
|
|
let (stdout, stderr) = test_env.jj_cmd_ok(&repo_path, &["git", "export"]);
|
|
|
|
insta::assert_snapshot!(stdout, @r###"
|
|
|
|
Nothing changed.
|
|
|
|
"###);
|
2023-04-04 02:02:12 +00:00
|
|
|
insta::assert_snapshot!(stderr, @r###"
|
2023-09-03 21:07:15 +00:00
|
|
|
Failed to export some branches:
|
|
|
|
fred
|
2023-04-04 02:02:12 +00:00
|
|
|
"###);
|
|
|
|
}
|
|
|
|
|
2022-11-28 14:17:33 +00:00
|
|
|
#[test]
|
|
|
|
fn test_branch_empty_name() {
|
|
|
|
let test_env = TestEnvironment::default();
|
|
|
|
test_env.jj_cmd_success(test_env.env_root(), &["init", "repo", "--git"]);
|
|
|
|
let repo_path = test_env.env_root().join("repo");
|
|
|
|
|
|
|
|
let stderr = test_env.jj_cmd_cli_error(&repo_path, &["branch", "create", ""]);
|
|
|
|
insta::assert_snapshot!(stderr, @r###"
|
2023-03-17 23:14:20 +00:00
|
|
|
error: a value is required for '<NAMES>...' but none was supplied
|
2022-11-28 14:17:33 +00:00
|
|
|
|
2023-03-17 23:14:20 +00:00
|
|
|
For more information, try '--help'.
|
2022-11-28 14:17:33 +00:00
|
|
|
"###);
|
|
|
|
}
|
2023-06-12 01:04:55 +00:00
|
|
|
|
2022-11-22 23:58:04 +00:00
|
|
|
#[test]
|
|
|
|
fn test_branch_forget_glob() {
|
|
|
|
let test_env = TestEnvironment::default();
|
|
|
|
test_env.jj_cmd_success(test_env.env_root(), &["init", "repo", "--git"]);
|
|
|
|
let repo_path = test_env.env_root().join("repo");
|
|
|
|
|
|
|
|
test_env.jj_cmd_success(&repo_path, &["branch", "set", "foo-1"]);
|
|
|
|
test_env.jj_cmd_success(&repo_path, &["branch", "set", "bar-2"]);
|
|
|
|
test_env.jj_cmd_success(&repo_path, &["branch", "set", "foo-3"]);
|
|
|
|
test_env.jj_cmd_success(&repo_path, &["branch", "set", "foo-4"]);
|
|
|
|
|
|
|
|
insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###"
|
2023-02-09 02:53:47 +00:00
|
|
|
@ bar-2 foo-1 foo-3 foo-4 230dd059e1b0
|
2023-03-15 03:37:56 +00:00
|
|
|
◉ 000000000000
|
2022-11-22 23:58:04 +00:00
|
|
|
"###);
|
|
|
|
let stdout = test_env.jj_cmd_success(&repo_path, &["branch", "forget", "--glob", "foo-[1-3]"]);
|
2023-06-29 02:36:08 +00:00
|
|
|
insta::assert_snapshot!(stdout, @r###"
|
|
|
|
Forgot 2 branches.
|
|
|
|
"###);
|
2022-11-22 23:58:04 +00:00
|
|
|
insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###"
|
2023-02-09 02:53:47 +00:00
|
|
|
@ bar-2 foo-4 230dd059e1b0
|
2023-03-15 03:37:56 +00:00
|
|
|
◉ 000000000000
|
2022-11-22 23:58:04 +00:00
|
|
|
"###);
|
|
|
|
|
|
|
|
// Forgetting a branch via both explicit name and glob pattern, or with
|
|
|
|
// multiple glob patterns, shouldn't produce an error.
|
|
|
|
let stdout = test_env.jj_cmd_success(
|
|
|
|
&repo_path,
|
|
|
|
&[
|
|
|
|
"branch", "forget", "foo-4", "--glob", "foo-*", "--glob", "foo-*",
|
|
|
|
],
|
|
|
|
);
|
|
|
|
insta::assert_snapshot!(stdout, @"");
|
|
|
|
insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###"
|
2023-02-09 02:53:47 +00:00
|
|
|
@ bar-2 230dd059e1b0
|
2023-03-15 03:37:56 +00:00
|
|
|
◉ 000000000000
|
2022-11-22 23:58:04 +00:00
|
|
|
"###);
|
|
|
|
|
|
|
|
// Malformed glob
|
|
|
|
let stderr = test_env.jj_cmd_failure(&repo_path, &["branch", "forget", "--glob", "foo-[1-3"]);
|
|
|
|
insta::assert_snapshot!(stderr, @r###"
|
|
|
|
Error: Failed to compile glob: Pattern syntax error near position 4: invalid range pattern
|
|
|
|
"###);
|
2023-06-24 05:52:00 +00:00
|
|
|
|
|
|
|
// We get an error if none of the globs match anything
|
|
|
|
let stderr = test_env.jj_cmd_failure(
|
|
|
|
&repo_path,
|
|
|
|
&[
|
|
|
|
"branch",
|
|
|
|
"forget",
|
|
|
|
"--glob=bar*",
|
|
|
|
"--glob=baz*",
|
|
|
|
"--glob=boom*",
|
|
|
|
],
|
|
|
|
);
|
|
|
|
insta::assert_snapshot!(stderr, @r###"
|
|
|
|
Error: The provided globs 'baz*', 'boom*' did not match any branches
|
|
|
|
"###);
|
2022-11-22 23:58:04 +00:00
|
|
|
}
|
2022-11-28 14:17:33 +00:00
|
|
|
|
2023-06-24 06:27:48 +00:00
|
|
|
#[test]
|
|
|
|
fn test_branch_delete_glob() {
|
|
|
|
// Set up a git repo with a branch and a jj repo that has it as a remote.
|
|
|
|
let test_env = TestEnvironment::default();
|
|
|
|
test_env.jj_cmd_success(test_env.env_root(), &["init", "repo", "--git"]);
|
|
|
|
let repo_path = test_env.env_root().join("repo");
|
|
|
|
let git_repo_path = test_env.env_root().join("git-repo");
|
|
|
|
let git_repo = git2::Repository::init_bare(git_repo_path).unwrap();
|
|
|
|
let mut tree_builder = git_repo.treebuilder(None).unwrap();
|
|
|
|
let file_oid = git_repo.blob(b"content").unwrap();
|
|
|
|
tree_builder
|
|
|
|
.insert("file", file_oid, git2::FileMode::Blob.into())
|
|
|
|
.unwrap();
|
|
|
|
test_env.jj_cmd_success(
|
|
|
|
&repo_path,
|
|
|
|
&["git", "remote", "add", "origin", "../git-repo"],
|
|
|
|
);
|
|
|
|
|
|
|
|
test_env.jj_cmd_success(&repo_path, &["describe", "-m=commit"]);
|
|
|
|
test_env.jj_cmd_success(&repo_path, &["branch", "set", "foo-1"]);
|
|
|
|
test_env.jj_cmd_success(&repo_path, &["branch", "set", "bar-2"]);
|
|
|
|
test_env.jj_cmd_success(&repo_path, &["branch", "set", "foo-3"]);
|
|
|
|
test_env.jj_cmd_success(&repo_path, &["branch", "set", "foo-4"]);
|
|
|
|
// Push to create remote-tracking branches
|
|
|
|
test_env.jj_cmd_success(&repo_path, &["git", "push", "--all"]);
|
|
|
|
|
|
|
|
insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###"
|
|
|
|
@ bar-2 foo-1 foo-3 foo-4 6fbf398c2d59
|
|
|
|
│
|
|
|
|
~
|
|
|
|
"###);
|
|
|
|
let stdout = test_env.jj_cmd_success(&repo_path, &["branch", "delete", "--glob", "foo-[1-3]"]);
|
2023-06-29 02:36:08 +00:00
|
|
|
insta::assert_snapshot!(stdout, @r###"
|
|
|
|
Deleted 2 branches.
|
|
|
|
"###);
|
2023-06-24 06:27:48 +00:00
|
|
|
insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###"
|
|
|
|
@ bar-2 foo-1@origin foo-3@origin foo-4 6fbf398c2d59
|
|
|
|
│
|
|
|
|
~
|
|
|
|
"###);
|
|
|
|
|
|
|
|
// We get an error if none of the globs match live branches. Unlike `jj branch
|
|
|
|
// forget`, it's not allowed to delete already deleted branches.
|
|
|
|
let stderr = test_env.jj_cmd_failure(&repo_path, &["branch", "delete", "--glob=foo-[1-3]"]);
|
|
|
|
insta::assert_snapshot!(stderr, @r###"
|
|
|
|
Error: The provided glob 'foo-[1-3]' did not match any branches
|
|
|
|
"###);
|
|
|
|
|
|
|
|
// Deleting a branch via both explicit name and glob pattern, or with
|
|
|
|
// multiple glob patterns, shouldn't produce an error.
|
|
|
|
let stdout = test_env.jj_cmd_success(
|
|
|
|
&repo_path,
|
|
|
|
&[
|
|
|
|
"branch", "delete", "foo-4", "--glob", "foo-*", "--glob", "foo-*",
|
|
|
|
],
|
|
|
|
);
|
|
|
|
insta::assert_snapshot!(stdout, @"");
|
|
|
|
insta::assert_snapshot!(get_log_output(&test_env, &repo_path), @r###"
|
|
|
|
@ bar-2 foo-1@origin foo-3@origin foo-4@origin 6fbf398c2d59
|
|
|
|
│
|
|
|
|
~
|
|
|
|
"###);
|
|
|
|
|
|
|
|
// The deleted branches are still there
|
|
|
|
insta::assert_snapshot!(get_branch_output(&test_env, &repo_path), @r###"
|
2023-07-11 01:46:38 +00:00
|
|
|
bar-2: qpvuntsm 6fbf398c (empty) commit
|
2023-06-24 06:27:48 +00:00
|
|
|
foo-1 (deleted)
|
2023-07-11 01:46:38 +00:00
|
|
|
@origin: qpvuntsm 6fbf398c (empty) commit
|
2023-06-24 06:27:48 +00:00
|
|
|
(this branch will be *deleted permanently* on the remote on the
|
|
|
|
next `jj git push`. Use `jj branch forget` to prevent this)
|
|
|
|
foo-3 (deleted)
|
2023-07-11 01:46:38 +00:00
|
|
|
@origin: qpvuntsm 6fbf398c (empty) commit
|
2023-06-24 06:27:48 +00:00
|
|
|
(this branch will be *deleted permanently* on the remote on the
|
|
|
|
next `jj git push`. Use `jj branch forget` to prevent this)
|
|
|
|
foo-4 (deleted)
|
2023-07-11 01:46:38 +00:00
|
|
|
@origin: qpvuntsm 6fbf398c (empty) commit
|
2023-06-24 06:27:48 +00:00
|
|
|
(this branch will be *deleted permanently* on the remote on the
|
|
|
|
next `jj git push`. Use `jj branch forget` to prevent this)
|
|
|
|
"###);
|
|
|
|
|
|
|
|
// Malformed glob
|
|
|
|
let stderr = test_env.jj_cmd_failure(&repo_path, &["branch", "delete", "--glob", "foo-[1-3"]);
|
|
|
|
insta::assert_snapshot!(stderr, @r###"
|
|
|
|
Error: Failed to compile glob: Pattern syntax error near position 4: invalid range pattern
|
|
|
|
"###);
|
|
|
|
}
|
|
|
|
|
2023-06-12 01:04:55 +00:00
|
|
|
#[test]
|
|
|
|
fn test_branch_forget_export() {
|
|
|
|
let test_env = TestEnvironment::default();
|
|
|
|
test_env.jj_cmd_success(test_env.env_root(), &["init", "repo", "--git"]);
|
|
|
|
let repo_path = test_env.env_root().join("repo");
|
|
|
|
|
|
|
|
test_env.jj_cmd_success(&repo_path, &["new"]);
|
|
|
|
test_env.jj_cmd_success(&repo_path, &["branch", "set", "foo"]);
|
|
|
|
let stdout = test_env.jj_cmd_success(&repo_path, &["branch", "list"]);
|
|
|
|
insta::assert_snapshot!(stdout, @r###"
|
2023-07-11 01:46:38 +00:00
|
|
|
foo: rlvkpnrz 65b6b74e (empty) (no description set)
|
2023-06-12 01:04:55 +00:00
|
|
|
"###);
|
|
|
|
|
|
|
|
// Exporting the branch to git creates a local-git tracking branch
|
|
|
|
let stdout = test_env.jj_cmd_success(&repo_path, &["git", "export"]);
|
|
|
|
insta::assert_snapshot!(stdout, @"");
|
|
|
|
let stdout = test_env.jj_cmd_success(&repo_path, &["branch", "forget", "foo"]);
|
|
|
|
insta::assert_snapshot!(stdout, @"");
|
2023-06-30 02:20:00 +00:00
|
|
|
// Forgetting a branch does not delete its local-git tracking branch. The
|
|
|
|
// git-tracking branch is kept.
|
2023-06-12 01:04:55 +00:00
|
|
|
let stdout = test_env.jj_cmd_success(&repo_path, &["branch", "list"]);
|
|
|
|
insta::assert_snapshot!(stdout, @r###"
|
2023-06-25 11:02:33 +00:00
|
|
|
foo (forgotten)
|
2023-07-11 01:46:38 +00:00
|
|
|
@git: rlvkpnrz 65b6b74e (empty) (no description set)
|
2023-06-25 11:15:44 +00:00
|
|
|
(this branch will be deleted from the underlying Git repo on the next `jj git export`)
|
2023-06-12 01:04:55 +00:00
|
|
|
"###);
|
2023-06-12 18:59:15 +00:00
|
|
|
let stderr = test_env.jj_cmd_failure(&repo_path, &["log", "-r=foo", "--no-graph"]);
|
|
|
|
insta::assert_snapshot!(stderr, @r###"
|
|
|
|
Error: Revision "foo" doesn't exist
|
2023-07-09 09:17:48 +00:00
|
|
|
Hint: Did you mean "foo@git"?
|
2023-06-12 18:59:15 +00:00
|
|
|
"###);
|
|
|
|
let stdout = test_env.jj_cmd_success(&repo_path, &["log", "-r=foo@git", "--no-graph"]);
|
2023-06-12 01:04:55 +00:00
|
|
|
insta::assert_snapshot!(stdout, @r###"
|
2023-07-30 05:24:34 +00:00
|
|
|
rlvkpnrz test.user@example.com 2001-02-03 04:05:08.000 +07:00 foo@git 65b6b74e
|
2023-06-12 01:04:55 +00:00
|
|
|
(empty) (no description set)
|
|
|
|
"###);
|
|
|
|
|
|
|
|
// The presence of the @git branch means that a `jj git import` is a no-op...
|
|
|
|
let stdout = test_env.jj_cmd_success(&repo_path, &["git", "import"]);
|
|
|
|
insta::assert_snapshot!(stdout, @r###"
|
|
|
|
Nothing changed.
|
|
|
|
"###);
|
|
|
|
// ... and a `jj git export` will delete the branch from git and will delete the
|
|
|
|
// git-tracking branch. In a colocated repo, this will happen automatically
|
|
|
|
// immediately after a `jj branch forget`. This is demonstrated in
|
|
|
|
// `test_git_colocated_branch_forget` in test_git_colocated.rs
|
|
|
|
let stdout = test_env.jj_cmd_success(&repo_path, &["git", "export"]);
|
|
|
|
insta::assert_snapshot!(stdout, @"");
|
|
|
|
let stdout = test_env.jj_cmd_success(&repo_path, &["branch", "list"]);
|
|
|
|
insta::assert_snapshot!(stdout, @"");
|
|
|
|
|
|
|
|
// Note that if `jj branch forget` *did* delete foo@git, a subsequent `jj
|
|
|
|
// git export` would be a no-op and a `jj git import` would resurrect
|
|
|
|
// the branch. In a normal repo, that might be OK. In a colocated repo,
|
|
|
|
// this would automatically happen before the next command, making `jj
|
|
|
|
// branch forget` useless.
|
|
|
|
}
|
|
|
|
|
2023-06-13 21:49:51 +00:00
|
|
|
#[test]
|
|
|
|
fn test_branch_forget_fetched_branch() {
|
|
|
|
// Much of this test is borrowed from `test_git_fetch_remote_only_branch` in
|
|
|
|
// test_git_fetch.rs
|
|
|
|
|
|
|
|
// Set up a git repo with a branch and a jj repo that has it as a remote.
|
|
|
|
let test_env = TestEnvironment::default();
|
|
|
|
test_env.jj_cmd_success(test_env.env_root(), &["init", "repo", "--git"]);
|
|
|
|
let repo_path = test_env.env_root().join("repo");
|
|
|
|
let git_repo_path = test_env.env_root().join("git-repo");
|
2023-06-24 06:27:48 +00:00
|
|
|
let git_repo = git2::Repository::init_bare(git_repo_path).unwrap();
|
2023-06-13 21:49:51 +00:00
|
|
|
let signature =
|
|
|
|
git2::Signature::new("Some One", "some.one@example.com", &git2::Time::new(0, 0)).unwrap();
|
|
|
|
let mut tree_builder = git_repo.treebuilder(None).unwrap();
|
|
|
|
let file_oid = git_repo.blob(b"content").unwrap();
|
|
|
|
tree_builder
|
|
|
|
.insert("file", file_oid, git2::FileMode::Blob.into())
|
|
|
|
.unwrap();
|
|
|
|
let tree_oid = tree_builder.write().unwrap();
|
|
|
|
let tree = git_repo.find_tree(tree_oid).unwrap();
|
|
|
|
test_env.jj_cmd_success(
|
|
|
|
&repo_path,
|
|
|
|
&["git", "remote", "add", "origin", "../git-repo"],
|
|
|
|
);
|
|
|
|
// Create a commit and a branch in the git repo
|
|
|
|
let first_git_repo_commit = git_repo
|
|
|
|
.commit(
|
|
|
|
Some("refs/heads/feature1"),
|
|
|
|
&signature,
|
|
|
|
&signature,
|
|
|
|
"message",
|
|
|
|
&tree,
|
|
|
|
&[],
|
|
|
|
)
|
|
|
|
.unwrap();
|
|
|
|
|
|
|
|
// Fetch normally
|
|
|
|
test_env.jj_cmd_success(&repo_path, &["git", "fetch", "--remote=origin"]);
|
|
|
|
insta::assert_snapshot!(get_branch_output(&test_env, &repo_path), @r###"
|
2023-07-11 01:46:38 +00:00
|
|
|
feature1: mzyxwzks 9f01a0e0 message
|
2023-06-13 21:49:51 +00:00
|
|
|
"###);
|
|
|
|
|
2023-06-27 22:18:15 +00:00
|
|
|
// TEST 1: with export-import
|
2023-06-13 21:49:51 +00:00
|
|
|
// Forget the branch
|
|
|
|
test_env.jj_cmd_success(&repo_path, &["branch", "forget", "feature1"]);
|
|
|
|
insta::assert_snapshot!(get_branch_output(&test_env, &repo_path), @"");
|
|
|
|
|
|
|
|
// At this point `jj git export && jj git import` does *not* recreate the
|
|
|
|
// branch. This behavior is important in colocated repos, as otherwise a
|
|
|
|
// forgotten branch would be immediately resurrected.
|
|
|
|
//
|
|
|
|
// Technically, this is because `jj branch forget` preserved
|
|
|
|
// the ref in jj view's `git_refs` tracking the local git repo's remote-tracking
|
|
|
|
// branch.
|
|
|
|
// TODO: Show that jj git push is also a no-op
|
2023-06-27 22:18:15 +00:00
|
|
|
insta::assert_snapshot!(test_env.jj_cmd_success(&repo_path, &["git", "export"]), @"");
|
2023-06-13 21:49:51 +00:00
|
|
|
insta::assert_snapshot!(test_env.jj_cmd_success(&repo_path, &["git", "import"]), @r###"
|
|
|
|
Nothing changed.
|
|
|
|
"###);
|
|
|
|
insta::assert_snapshot!(get_branch_output(&test_env, &repo_path), @"");
|
|
|
|
|
2023-06-27 22:18:15 +00:00
|
|
|
// We can fetch feature1 again.
|
|
|
|
let stdout = test_env.jj_cmd_success(&repo_path, &["git", "fetch", "--remote=origin"]);
|
|
|
|
insta::assert_snapshot!(stdout, @"");
|
|
|
|
insta::assert_snapshot!(get_branch_output(&test_env, &repo_path), @r###"
|
2023-07-11 01:46:38 +00:00
|
|
|
feature1: mzyxwzks 9f01a0e0 message
|
2023-06-27 22:18:15 +00:00
|
|
|
"###);
|
|
|
|
|
|
|
|
// TEST 2: No export/import (otherwise the same as test 1)
|
|
|
|
test_env.jj_cmd_success(&repo_path, &["branch", "forget", "feature1"]);
|
|
|
|
insta::assert_snapshot!(get_branch_output(&test_env, &repo_path), @"");
|
2023-06-30 02:20:00 +00:00
|
|
|
// Fetch works even without the export-import
|
2023-06-13 21:49:51 +00:00
|
|
|
let stdout = test_env.jj_cmd_success(&repo_path, &["git", "fetch", "--remote=origin"]);
|
2023-06-30 02:20:00 +00:00
|
|
|
insta::assert_snapshot!(stdout, @"");
|
|
|
|
insta::assert_snapshot!(get_branch_output(&test_env, &repo_path), @r###"
|
2023-07-11 01:46:38 +00:00
|
|
|
feature1: mzyxwzks 9f01a0e0 message
|
2023-06-13 21:49:51 +00:00
|
|
|
"###);
|
|
|
|
|
2023-06-27 22:18:15 +00:00
|
|
|
// TEST 3: fetch branch that was moved & forgotten
|
|
|
|
|
2023-06-13 21:49:51 +00:00
|
|
|
// Move the branch in the git repo.
|
|
|
|
git_repo
|
|
|
|
.commit(
|
|
|
|
Some("refs/heads/feature1"),
|
|
|
|
&signature,
|
|
|
|
&signature,
|
|
|
|
"another message",
|
|
|
|
&tree,
|
|
|
|
&[&git_repo.find_commit(first_git_repo_commit).unwrap()],
|
|
|
|
)
|
|
|
|
.unwrap();
|
2023-06-30 02:20:00 +00:00
|
|
|
let stdout = test_env.jj_cmd_success(&repo_path, &["branch", "forget", "feature1"]);
|
|
|
|
insta::assert_snapshot!(stdout, @"");
|
2023-06-13 21:49:51 +00:00
|
|
|
|
2023-06-30 02:20:00 +00:00
|
|
|
// Fetching a moved branch does not create a conflict
|
2023-06-13 21:49:51 +00:00
|
|
|
let stdout = test_env.jj_cmd_success(&repo_path, &["git", "fetch", "--remote=origin"]);
|
|
|
|
insta::assert_snapshot!(stdout, @"");
|
|
|
|
insta::assert_snapshot!(get_branch_output(&test_env, &repo_path), @r###"
|
2023-07-11 01:46:38 +00:00
|
|
|
feature1: ooosovrs 38aefb17 (empty) another message
|
2023-06-13 21:49:51 +00:00
|
|
|
"###);
|
|
|
|
}
|
|
|
|
|
2023-06-24 04:40:54 +00:00
|
|
|
#[test]
|
|
|
|
fn test_branch_forget_deleted_or_nonexistent_branch() {
|
|
|
|
// Much of this test is borrowed from `test_git_fetch_remote_only_branch` in
|
|
|
|
// test_git_fetch.rs
|
|
|
|
|
|
|
|
// ======== Beginning of test setup ========
|
|
|
|
// Set up a git repo with a branch and a jj repo that has it as a remote.
|
|
|
|
let test_env = TestEnvironment::default();
|
|
|
|
test_env.jj_cmd_success(test_env.env_root(), &["init", "repo", "--git"]);
|
|
|
|
let repo_path = test_env.env_root().join("repo");
|
|
|
|
let git_repo_path = test_env.env_root().join("git-repo");
|
2023-06-24 06:27:48 +00:00
|
|
|
let git_repo = git2::Repository::init_bare(git_repo_path).unwrap();
|
2023-06-24 04:40:54 +00:00
|
|
|
let signature =
|
|
|
|
git2::Signature::new("Some One", "some.one@example.com", &git2::Time::new(0, 0)).unwrap();
|
|
|
|
let mut tree_builder = git_repo.treebuilder(None).unwrap();
|
|
|
|
let file_oid = git_repo.blob(b"content").unwrap();
|
|
|
|
tree_builder
|
|
|
|
.insert("file", file_oid, git2::FileMode::Blob.into())
|
|
|
|
.unwrap();
|
|
|
|
let tree_oid = tree_builder.write().unwrap();
|
|
|
|
let tree = git_repo.find_tree(tree_oid).unwrap();
|
|
|
|
test_env.jj_cmd_success(
|
|
|
|
&repo_path,
|
|
|
|
&["git", "remote", "add", "origin", "../git-repo"],
|
|
|
|
);
|
|
|
|
// Create a commit and a branch in the git repo
|
|
|
|
git_repo
|
|
|
|
.commit(
|
|
|
|
Some("refs/heads/feature1"),
|
|
|
|
&signature,
|
|
|
|
&signature,
|
|
|
|
"message",
|
|
|
|
&tree,
|
|
|
|
&[],
|
|
|
|
)
|
|
|
|
.unwrap();
|
|
|
|
|
|
|
|
// Fetch and then delete the branch
|
|
|
|
test_env.jj_cmd_success(&repo_path, &["git", "fetch", "--remote=origin"]);
|
|
|
|
test_env.jj_cmd_success(&repo_path, &["branch", "delete", "feature1"]);
|
|
|
|
insta::assert_snapshot!(get_branch_output(&test_env, &repo_path), @r###"
|
|
|
|
feature1 (deleted)
|
2023-07-11 01:46:38 +00:00
|
|
|
@origin: mzyxwzks 9f01a0e0 message
|
2023-06-24 06:35:33 +00:00
|
|
|
(this branch will be *deleted permanently* on the remote on the
|
|
|
|
next `jj git push`. Use `jj branch forget` to prevent this)
|
2023-06-24 04:40:54 +00:00
|
|
|
"###);
|
|
|
|
|
|
|
|
// ============ End of test setup ============
|
|
|
|
|
2023-06-24 04:48:48 +00:00
|
|
|
// We can forget a deleted branch
|
|
|
|
test_env.jj_cmd_success(&repo_path, &["branch", "forget", "feature1"]);
|
|
|
|
insta::assert_snapshot!(get_branch_output(&test_env, &repo_path), @"");
|
2023-06-24 04:40:54 +00:00
|
|
|
|
|
|
|
// Can't forget a non-existent branch
|
|
|
|
let stderr = test_env.jj_cmd_failure(&repo_path, &["branch", "forget", "i_do_not_exist"]);
|
|
|
|
insta::assert_snapshot!(stderr, @r###"
|
|
|
|
Error: No such branch: i_do_not_exist
|
|
|
|
"###);
|
|
|
|
}
|
|
|
|
|
2023-06-11 04:44:48 +00:00
|
|
|
// TODO: Test `jj branch list` with a remote named `git`
|
|
|
|
|
2023-06-28 02:53:01 +00:00
|
|
|
#[test]
|
|
|
|
fn test_branch_list_filtered_by_revset() {
|
|
|
|
let test_env = TestEnvironment::default();
|
|
|
|
|
|
|
|
// Initialize remote refs
|
|
|
|
test_env.jj_cmd_success(test_env.env_root(), &["init", "remote", "--git"]);
|
|
|
|
let remote_path = test_env.env_root().join("remote");
|
|
|
|
for branch in ["remote-keep", "remote-delete", "remote-rewrite"] {
|
2023-09-03 03:47:23 +00:00
|
|
|
test_env.jj_cmd_success(&remote_path, &["new", "root()", "-m", branch]);
|
2023-06-28 02:53:01 +00:00
|
|
|
test_env.jj_cmd_success(&remote_path, &["branch", "set", branch]);
|
|
|
|
}
|
|
|
|
test_env.jj_cmd_success(&remote_path, &["new"]);
|
|
|
|
test_env.jj_cmd_success(&remote_path, &["git", "export"]);
|
|
|
|
|
|
|
|
// Initialize local refs
|
|
|
|
let mut remote_git_path = remote_path;
|
|
|
|
remote_git_path.extend([".jj", "repo", "store", "git"]);
|
|
|
|
test_env.jj_cmd_success(
|
|
|
|
test_env.env_root(),
|
|
|
|
&["git", "clone", remote_git_path.to_str().unwrap(), "local"],
|
|
|
|
);
|
|
|
|
let local_path = test_env.env_root().join("local");
|
2023-09-03 03:47:23 +00:00
|
|
|
test_env.jj_cmd_success(&local_path, &["new", "root()", "-m", "local-keep"]);
|
2023-06-28 02:53:01 +00:00
|
|
|
test_env.jj_cmd_success(&local_path, &["branch", "set", "local-keep"]);
|
|
|
|
|
|
|
|
// Mutate refs in local repository
|
|
|
|
test_env.jj_cmd_success(&local_path, &["branch", "delete", "remote-delete"]);
|
|
|
|
test_env.jj_cmd_success(&local_path, &["describe", "-mrewritten", "remote-rewrite"]);
|
|
|
|
|
|
|
|
let template = r#"separate(" ", commit_id.short(), branches, if(hidden, "(hidden)"))"#;
|
|
|
|
insta::assert_snapshot!(
|
|
|
|
test_env.jj_cmd_success(
|
|
|
|
&local_path,
|
2023-07-28 00:33:54 +00:00
|
|
|
&["log", "-r::(branches() | remote_branches())", "-T", template],
|
2023-06-28 02:53:01 +00:00
|
|
|
),
|
|
|
|
@r###"
|
|
|
|
◉ e31634b64294 remote-rewrite*
|
|
|
|
│ @ c7b4c09cd77c local-keep
|
|
|
|
├─╯
|
|
|
|
│ ◉ 3e9a5af6ef15 remote-rewrite@origin (hidden)
|
|
|
|
├─╯
|
|
|
|
│ ◉ dad5f298ca57 remote-delete@origin
|
|
|
|
├─╯
|
index: import commits in chronological order
This basically means that heads in a filtered graph appear in reverse
chronological order. Before, "jj log -r 'tags()'" in linux-stable repo would
look randomly sorted once you ran "jj debug reindex" in it.
With this change, indexing is more like breadth-first search, and BFS is
known to be bad at rendering nice graph (because branches run in parallel.)
However, we have a post process to group topological branches, so we don't
have this problem. For serialization formats like Mercurial's revlog iirc,
BFS leads to bad compression ratio, but our index isn't that kind of data.
Reindexing gets slightly slower, but I think this is negligible.
(in Git repository)
% hyperfine --warmup 3 --runs 10 "jj debug reindex --ignore-working-copy"
(original)
Time (mean ± σ): 1.521 s ± 0.027 s [User: 1.307 s, System: 0.211 s]
Range (min … max): 1.486 s … 1.573 s 10 runs
(new)
Time (mean ± σ): 1.568 s ± 0.027 s [User: 1.368 s, System: 0.197 s]
Range (min … max): 1.531 s … 1.625 s 10 runs
Another idea is to sort heads chronologically and run DFS-based topological
sorting. It's ad-hoc, but worked surprisingly well for my local repositories.
For repositories with lots of long-running branches, this commit will provide
more predictable result than DFS-based one.
2023-08-12 10:32:05 +00:00
|
|
|
│ ◉ 911e912015fb remote-keep
|
|
|
|
├─╯
|
2023-06-28 02:53:01 +00:00
|
|
|
◉ 000000000000
|
|
|
|
"###);
|
|
|
|
|
|
|
|
// All branches are listed by default.
|
|
|
|
insta::assert_snapshot!(test_env.jj_cmd_success(&local_path, &["branch", "list"]), @r###"
|
2023-07-11 01:46:38 +00:00
|
|
|
local-keep: kpqxywon c7b4c09c (empty) local-keep
|
2023-06-28 02:53:01 +00:00
|
|
|
remote-delete (deleted)
|
2023-07-11 01:46:38 +00:00
|
|
|
@origin: yxusvupt dad5f298 (empty) remote-delete
|
2023-06-28 02:53:01 +00:00
|
|
|
(this branch will be *deleted permanently* on the remote on the
|
|
|
|
next `jj git push`. Use `jj branch forget` to prevent this)
|
2023-07-11 01:46:38 +00:00
|
|
|
remote-keep: nlwprzpn 911e9120 (empty) remote-keep
|
|
|
|
remote-rewrite: xyxluytn e31634b6 (empty) rewritten
|
|
|
|
@origin (ahead by 1 commits, behind by 1 commits): xyxluytn 3e9a5af6 (empty) remote-rewrite
|
2023-06-28 02:53:01 +00:00
|
|
|
"###);
|
|
|
|
|
|
|
|
let query = |revset| test_env.jj_cmd_success(&local_path, &["branch", "list", "-r", revset]);
|
2023-07-02 00:59:34 +00:00
|
|
|
let query_error =
|
|
|
|
|revset| test_env.jj_cmd_failure(&local_path, &["branch", "list", "-r", revset]);
|
2023-06-28 02:53:01 +00:00
|
|
|
|
|
|
|
// "all()" doesn't include deleted branches since they have no local targets.
|
|
|
|
// So "all()" is identical to "branches()".
|
|
|
|
insta::assert_snapshot!(query("all()"), @r###"
|
2023-07-11 01:46:38 +00:00
|
|
|
local-keep: kpqxywon c7b4c09c (empty) local-keep
|
|
|
|
remote-keep: nlwprzpn 911e9120 (empty) remote-keep
|
|
|
|
remote-rewrite: xyxluytn e31634b6 (empty) rewritten
|
|
|
|
@origin (ahead by 1 commits, behind by 1 commits): xyxluytn 3e9a5af6 (empty) remote-rewrite
|
2023-06-28 02:53:01 +00:00
|
|
|
"###);
|
|
|
|
|
|
|
|
// Exclude remote-only branches. "remote-rewrite@origin" is included since
|
|
|
|
// local "remote-rewrite" target matches.
|
|
|
|
insta::assert_snapshot!(query("branches()"), @r###"
|
2023-07-11 01:46:38 +00:00
|
|
|
local-keep: kpqxywon c7b4c09c (empty) local-keep
|
|
|
|
remote-keep: nlwprzpn 911e9120 (empty) remote-keep
|
|
|
|
remote-rewrite: xyxluytn e31634b6 (empty) rewritten
|
|
|
|
@origin (ahead by 1 commits, behind by 1 commits): xyxluytn 3e9a5af6 (empty) remote-rewrite
|
2023-06-28 02:53:01 +00:00
|
|
|
"###);
|
|
|
|
|
|
|
|
// Select branches by name.
|
|
|
|
insta::assert_snapshot!(query("branches(remote-rewrite)"), @r###"
|
2023-07-11 01:46:38 +00:00
|
|
|
remote-rewrite: xyxluytn e31634b6 (empty) rewritten
|
|
|
|
@origin (ahead by 1 commits, behind by 1 commits): xyxluytn 3e9a5af6 (empty) remote-rewrite
|
2023-06-28 02:53:01 +00:00
|
|
|
"###);
|
|
|
|
|
|
|
|
// Can't select deleted branch.
|
2023-07-02 00:59:34 +00:00
|
|
|
insta::assert_snapshot!(query("branches(remote-delete)"), @r###"
|
|
|
|
"###);
|
|
|
|
insta::assert_snapshot!(query_error("remote-delete"), @r###"
|
|
|
|
Error: Revision "remote-delete" doesn't exist
|
2023-07-02 04:57:44 +00:00
|
|
|
Hint: Did you mean "remote-delete@origin", "remote-keep", "remote-rewrite", "remote-rewrite@origin"?
|
2023-06-28 02:53:01 +00:00
|
|
|
"###);
|
|
|
|
}
|
|
|
|
|
2022-05-02 19:58:32 +00:00
|
|
|
fn get_log_output(test_env: &TestEnvironment, cwd: &Path) -> String {
|
2023-02-28 11:30:57 +00:00
|
|
|
let template = r#"branches ++ " " ++ commit_id.short()"#;
|
2023-02-28 10:43:14 +00:00
|
|
|
test_env.jj_cmd_success(cwd, &["log", "-T", template])
|
2022-05-02 19:58:32 +00:00
|
|
|
}
|
2023-06-13 21:49:51 +00:00
|
|
|
|
|
|
|
fn get_branch_output(test_env: &TestEnvironment, repo_path: &Path) -> String {
|
|
|
|
test_env.jj_cmd_success(repo_path, &["branch", "list"])
|
|
|
|
}
|