2022-11-26 23:57:50 +00:00
|
|
|
// Copyright 2020 The Jujutsu Authors
|
2020-12-12 08:00:42 +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.
|
|
|
|
|
2023-10-13 14:37:17 +00:00
|
|
|
use jj_lib::backend::CommitId;
|
2023-10-11 17:15:17 +00:00
|
|
|
use jj_lib::op_store::{RefTarget, RemoteRef, RemoteRefState, WorkspaceId};
|
2023-06-28 14:12:40 +00:00
|
|
|
use jj_lib::repo::Repo;
|
2021-12-05 20:44:08 +00:00
|
|
|
use maplit::hashset;
|
2022-12-24 15:38:20 +00:00
|
|
|
use testutils::{
|
2023-12-22 05:33:00 +00:00
|
|
|
assert_rebased_onto, create_random_commit, write_random_commit, CommitGraphBuilder, TestRepo,
|
2022-12-24 15:38:20 +00:00
|
|
|
};
|
2020-12-12 08:00:42 +00:00
|
|
|
|
2023-09-19 12:36:45 +00:00
|
|
|
#[test]
|
|
|
|
fn test_edit() {
|
2022-11-05 05:34:24 +00:00
|
|
|
// Test that MutableRepo::edit() uses the requested commit (not a new child)
|
2020-12-12 08:00:42 +00:00
|
|
|
let settings = testutils::user_settings();
|
2023-09-19 12:36:45 +00:00
|
|
|
let test_repo = TestRepo::init();
|
2022-02-05 23:13:07 +00:00
|
|
|
let repo = &test_repo.repo;
|
2020-12-12 08:00:42 +00:00
|
|
|
|
2023-12-12 05:42:05 +00:00
|
|
|
let mut tx = repo.start_transaction(&settings);
|
2022-12-24 15:38:20 +00:00
|
|
|
let wc_commit = write_random_commit(tx.mut_repo(), &settings);
|
2023-12-12 05:42:05 +00:00
|
|
|
let repo = tx.commit("test");
|
2020-12-12 08:00:42 +00:00
|
|
|
|
2023-12-12 05:42:05 +00:00
|
|
|
let mut tx = repo.start_transaction(&settings);
|
2022-02-05 23:13:07 +00:00
|
|
|
let ws_id = WorkspaceId::default();
|
2022-10-20 23:33:14 +00:00
|
|
|
tx.mut_repo().edit(ws_id.clone(), &wc_commit).unwrap();
|
2023-12-12 05:42:05 +00:00
|
|
|
let repo = tx.commit("test");
|
2022-09-18 21:46:12 +00:00
|
|
|
assert_eq!(repo.view().get_wc_commit_id(&ws_id), Some(wc_commit.id()));
|
2020-12-12 08:00:42 +00:00
|
|
|
}
|
|
|
|
|
2023-09-19 12:36:45 +00:00
|
|
|
#[test]
|
|
|
|
fn test_checkout() {
|
2022-11-05 05:34:24 +00:00
|
|
|
// Test that MutableRepo::check_out() creates a child
|
2020-12-12 08:00:42 +00:00
|
|
|
let settings = testutils::user_settings();
|
2023-09-19 12:36:45 +00:00
|
|
|
let test_repo = TestRepo::init();
|
2022-02-05 23:13:07 +00:00
|
|
|
let repo = &test_repo.repo;
|
2020-12-12 08:00:42 +00:00
|
|
|
|
2023-12-12 05:42:05 +00:00
|
|
|
let mut tx = repo.start_transaction(&settings);
|
2023-01-25 16:29:15 +00:00
|
|
|
let wc_commit_parent = write_random_commit(tx.mut_repo(), &settings);
|
2023-12-12 05:42:05 +00:00
|
|
|
let repo = tx.commit("test");
|
2020-12-12 08:00:42 +00:00
|
|
|
|
2023-12-12 05:42:05 +00:00
|
|
|
let mut tx = repo.start_transaction(&settings);
|
2022-02-05 23:13:07 +00:00
|
|
|
let ws_id = WorkspaceId::default();
|
2023-01-25 16:29:15 +00:00
|
|
|
let wc_commit = tx
|
2022-02-05 23:13:07 +00:00
|
|
|
.mut_repo()
|
2023-01-25 16:29:15 +00:00
|
|
|
.check_out(ws_id.clone(), &settings, &wc_commit_parent)
|
2022-12-24 05:09:19 +00:00
|
|
|
.unwrap();
|
2023-08-28 20:09:19 +00:00
|
|
|
assert_eq!(wc_commit.tree_id(), wc_commit_parent.tree_id());
|
2023-01-25 16:29:15 +00:00
|
|
|
assert_eq!(wc_commit.parents().len(), 1);
|
|
|
|
assert_eq!(wc_commit.parents()[0].id(), wc_commit_parent.id());
|
2023-12-12 05:42:05 +00:00
|
|
|
let repo = tx.commit("test");
|
2023-01-25 16:29:15 +00:00
|
|
|
assert_eq!(repo.view().get_wc_commit_id(&ws_id), Some(wc_commit.id()));
|
2020-12-12 08:00:42 +00:00
|
|
|
}
|
|
|
|
|
2023-09-19 12:36:45 +00:00
|
|
|
#[test]
|
|
|
|
fn test_checkout_previous_not_empty() {
|
2021-10-07 06:28:30 +00:00
|
|
|
// Test that MutableRepo::check_out() does not usually abandon the previous
|
2020-12-12 08:00:42 +00:00
|
|
|
// commit.
|
|
|
|
let settings = testutils::user_settings();
|
2023-09-19 12:36:45 +00:00
|
|
|
let test_repo = TestRepo::init();
|
2022-02-05 23:13:07 +00:00
|
|
|
let repo = &test_repo.repo;
|
2020-12-12 08:00:42 +00:00
|
|
|
|
2023-12-12 05:42:05 +00:00
|
|
|
let mut tx = repo.start_transaction(&settings);
|
2021-03-16 23:32:51 +00:00
|
|
|
let mut_repo = tx.mut_repo();
|
2023-01-25 16:29:15 +00:00
|
|
|
let old_wc_commit = write_random_commit(mut_repo, &settings);
|
2022-02-05 23:13:07 +00:00
|
|
|
let ws_id = WorkspaceId::default();
|
2023-01-25 16:29:15 +00:00
|
|
|
mut_repo.edit(ws_id.clone(), &old_wc_commit).unwrap();
|
2023-12-12 05:42:05 +00:00
|
|
|
let repo = tx.commit("test");
|
2020-12-12 08:00:42 +00:00
|
|
|
|
2023-12-12 05:42:05 +00:00
|
|
|
let mut tx = repo.start_transaction(&settings);
|
2021-03-16 23:32:51 +00:00
|
|
|
let mut_repo = tx.mut_repo();
|
2023-01-25 16:29:15 +00:00
|
|
|
let new_wc_commit = write_random_commit(mut_repo, &settings);
|
|
|
|
mut_repo.edit(ws_id, &new_wc_commit).unwrap();
|
2022-04-28 05:30:03 +00:00
|
|
|
mut_repo.rebase_descendants(&settings).unwrap();
|
2023-01-25 16:29:15 +00:00
|
|
|
assert!(mut_repo.view().heads().contains(old_wc_commit.id()));
|
2020-12-12 08:00:42 +00:00
|
|
|
}
|
|
|
|
|
2023-09-19 12:36:45 +00:00
|
|
|
#[test]
|
|
|
|
fn test_checkout_previous_empty() {
|
2021-10-07 06:28:30 +00:00
|
|
|
// Test that MutableRepo::check_out() abandons the previous commit if it was
|
2020-12-12 08:00:42 +00:00
|
|
|
// empty.
|
|
|
|
let settings = testutils::user_settings();
|
2023-09-19 12:36:45 +00:00
|
|
|
let test_repo = TestRepo::init();
|
2022-02-05 23:13:07 +00:00
|
|
|
let repo = &test_repo.repo;
|
2020-12-12 08:00:42 +00:00
|
|
|
|
2023-12-12 05:42:05 +00:00
|
|
|
let mut tx = repo.start_transaction(&settings);
|
2021-03-16 23:32:51 +00:00
|
|
|
let mut_repo = tx.mut_repo();
|
2023-01-25 16:29:15 +00:00
|
|
|
let old_wc_commit = mut_repo
|
2022-12-25 16:36:13 +00:00
|
|
|
.new_commit(
|
|
|
|
&settings,
|
|
|
|
vec![repo.store().root_commit_id().clone()],
|
2023-08-30 14:57:13 +00:00
|
|
|
repo.store().empty_merged_tree_id(),
|
2022-12-25 16:36:13 +00:00
|
|
|
)
|
|
|
|
.write()
|
|
|
|
.unwrap();
|
2022-02-05 23:13:07 +00:00
|
|
|
let ws_id = WorkspaceId::default();
|
2023-01-25 16:29:15 +00:00
|
|
|
mut_repo.edit(ws_id.clone(), &old_wc_commit).unwrap();
|
2023-12-12 05:42:05 +00:00
|
|
|
let repo = tx.commit("test");
|
2020-12-12 08:00:42 +00:00
|
|
|
|
2023-12-12 05:42:05 +00:00
|
|
|
let mut tx = repo.start_transaction(&settings);
|
2021-03-16 23:32:51 +00:00
|
|
|
let mut_repo = tx.mut_repo();
|
2022-12-24 15:38:20 +00:00
|
|
|
let new_wc_commit = write_random_commit(mut_repo, &settings);
|
2022-10-20 23:33:14 +00:00
|
|
|
mut_repo.edit(ws_id, &new_wc_commit).unwrap();
|
2022-04-28 05:30:03 +00:00
|
|
|
mut_repo.rebase_descendants(&settings).unwrap();
|
2023-01-25 16:29:15 +00:00
|
|
|
assert!(!mut_repo.view().heads().contains(old_wc_commit.id()));
|
2020-12-12 08:00:42 +00:00
|
|
|
}
|
2020-12-30 04:00:28 +00:00
|
|
|
|
2023-09-19 12:36:45 +00:00
|
|
|
#[test]
|
|
|
|
fn test_checkout_previous_empty_with_description() {
|
2022-05-21 17:23:46 +00:00
|
|
|
// Test that MutableRepo::check_out() does not abandon the previous commit if it
|
|
|
|
// has a non-empty description.
|
|
|
|
let settings = testutils::user_settings();
|
2023-09-19 12:36:45 +00:00
|
|
|
let test_repo = TestRepo::init();
|
2022-05-21 17:23:46 +00:00
|
|
|
let repo = &test_repo.repo;
|
|
|
|
|
2023-12-12 05:42:05 +00:00
|
|
|
let mut tx = repo.start_transaction(&settings);
|
2022-05-21 17:23:46 +00:00
|
|
|
let mut_repo = tx.mut_repo();
|
2023-01-25 16:29:15 +00:00
|
|
|
let old_wc_commit = mut_repo
|
2022-12-25 16:36:13 +00:00
|
|
|
.new_commit(
|
|
|
|
&settings,
|
|
|
|
vec![repo.store().root_commit_id().clone()],
|
2023-08-30 14:57:13 +00:00
|
|
|
repo.store().empty_merged_tree_id(),
|
2022-12-25 16:36:13 +00:00
|
|
|
)
|
|
|
|
.set_description("not empty")
|
|
|
|
.write()
|
|
|
|
.unwrap();
|
2022-05-21 17:23:46 +00:00
|
|
|
let ws_id = WorkspaceId::default();
|
2023-01-25 16:29:15 +00:00
|
|
|
mut_repo.edit(ws_id.clone(), &old_wc_commit).unwrap();
|
2023-12-12 05:42:05 +00:00
|
|
|
let repo = tx.commit("test");
|
2022-05-21 17:23:46 +00:00
|
|
|
|
2023-12-12 05:42:05 +00:00
|
|
|
let mut tx = repo.start_transaction(&settings);
|
2022-05-21 17:23:46 +00:00
|
|
|
let mut_repo = tx.mut_repo();
|
2023-01-25 16:29:15 +00:00
|
|
|
let new_wc_commit = write_random_commit(mut_repo, &settings);
|
2023-07-12 05:35:40 +00:00
|
|
|
mut_repo.edit(ws_id, &new_wc_commit).unwrap();
|
|
|
|
mut_repo.rebase_descendants(&settings).unwrap();
|
|
|
|
assert!(mut_repo.view().heads().contains(old_wc_commit.id()));
|
|
|
|
}
|
|
|
|
|
2023-09-19 12:36:45 +00:00
|
|
|
#[test]
|
|
|
|
fn test_checkout_previous_empty_with_local_branch() {
|
2023-07-12 05:35:40 +00:00
|
|
|
// Test that MutableRepo::check_out() does not abandon the previous commit if it
|
|
|
|
// is pointed by local branch.
|
|
|
|
let settings = testutils::user_settings();
|
2023-09-19 12:36:45 +00:00
|
|
|
let test_repo = TestRepo::init();
|
2023-07-12 05:35:40 +00:00
|
|
|
let repo = &test_repo.repo;
|
|
|
|
|
2023-12-12 05:42:05 +00:00
|
|
|
let mut tx = repo.start_transaction(&settings);
|
2023-07-12 05:35:40 +00:00
|
|
|
let mut_repo = tx.mut_repo();
|
|
|
|
let old_wc_commit = mut_repo
|
|
|
|
.new_commit(
|
|
|
|
&settings,
|
|
|
|
vec![repo.store().root_commit_id().clone()],
|
2023-08-30 14:57:13 +00:00
|
|
|
repo.store().empty_merged_tree_id(),
|
2023-07-12 05:35:40 +00:00
|
|
|
)
|
|
|
|
.write()
|
|
|
|
.unwrap();
|
2023-07-11 13:14:59 +00:00
|
|
|
mut_repo.set_local_branch_target("b", RefTarget::normal(old_wc_commit.id().clone()));
|
2023-07-12 05:35:40 +00:00
|
|
|
let ws_id = WorkspaceId::default();
|
|
|
|
mut_repo.edit(ws_id.clone(), &old_wc_commit).unwrap();
|
2023-12-12 05:42:05 +00:00
|
|
|
let repo = tx.commit("test");
|
2023-07-12 05:35:40 +00:00
|
|
|
|
2023-12-12 05:42:05 +00:00
|
|
|
let mut tx = repo.start_transaction(&settings);
|
2023-07-12 05:35:40 +00:00
|
|
|
let mut_repo = tx.mut_repo();
|
|
|
|
let new_wc_commit = write_random_commit(mut_repo, &settings);
|
2023-01-25 16:29:15 +00:00
|
|
|
mut_repo.edit(ws_id, &new_wc_commit).unwrap();
|
2022-05-21 17:23:46 +00:00
|
|
|
mut_repo.rebase_descendants(&settings).unwrap();
|
2023-01-25 16:29:15 +00:00
|
|
|
assert!(mut_repo.view().heads().contains(old_wc_commit.id()));
|
2022-05-21 17:23:46 +00:00
|
|
|
}
|
|
|
|
|
2023-09-19 12:36:45 +00:00
|
|
|
#[test]
|
|
|
|
fn test_checkout_previous_empty_non_head() {
|
2022-03-26 21:53:51 +00:00
|
|
|
// Test that MutableRepo::check_out() does not abandon the previous commit if it
|
|
|
|
// was empty and is not a head
|
|
|
|
let settings = testutils::user_settings();
|
2023-09-19 12:36:45 +00:00
|
|
|
let test_repo = TestRepo::init();
|
2022-03-26 21:53:51 +00:00
|
|
|
let repo = &test_repo.repo;
|
|
|
|
|
2023-12-12 05:42:05 +00:00
|
|
|
let mut tx = repo.start_transaction(&settings);
|
2022-03-26 21:53:51 +00:00
|
|
|
let mut_repo = tx.mut_repo();
|
2023-01-25 16:29:15 +00:00
|
|
|
let old_wc_commit = mut_repo
|
2022-12-25 16:36:13 +00:00
|
|
|
.new_commit(
|
|
|
|
&settings,
|
|
|
|
vec![repo.store().root_commit_id().clone()],
|
2023-08-30 14:57:13 +00:00
|
|
|
repo.store().empty_merged_tree_id(),
|
2022-12-25 16:36:13 +00:00
|
|
|
)
|
|
|
|
.write()
|
|
|
|
.unwrap();
|
|
|
|
let old_child = mut_repo
|
|
|
|
.new_commit(
|
|
|
|
&settings,
|
2023-01-25 16:29:15 +00:00
|
|
|
vec![old_wc_commit.id().clone()],
|
2023-08-28 20:09:19 +00:00
|
|
|
old_wc_commit.tree_id().clone(),
|
2022-12-25 16:36:13 +00:00
|
|
|
)
|
|
|
|
.write()
|
|
|
|
.unwrap();
|
2022-03-26 21:53:51 +00:00
|
|
|
let ws_id = WorkspaceId::default();
|
2023-01-25 16:29:15 +00:00
|
|
|
mut_repo.edit(ws_id.clone(), &old_wc_commit).unwrap();
|
2023-12-12 05:42:05 +00:00
|
|
|
let repo = tx.commit("test");
|
2022-03-26 21:53:51 +00:00
|
|
|
|
2023-12-12 05:42:05 +00:00
|
|
|
let mut tx = repo.start_transaction(&settings);
|
2022-03-26 21:53:51 +00:00
|
|
|
let mut_repo = tx.mut_repo();
|
2023-01-25 16:29:15 +00:00
|
|
|
let new_wc_commit = write_random_commit(mut_repo, &settings);
|
|
|
|
mut_repo.edit(ws_id, &new_wc_commit).unwrap();
|
2022-04-28 05:30:03 +00:00
|
|
|
mut_repo.rebase_descendants(&settings).unwrap();
|
2022-03-26 21:53:51 +00:00
|
|
|
assert_eq!(
|
|
|
|
*mut_repo.view().heads(),
|
2023-01-25 16:29:15 +00:00
|
|
|
hashset! {old_child.id().clone(), new_wc_commit.id().clone()}
|
2022-03-26 21:53:51 +00:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2023-09-19 12:36:45 +00:00
|
|
|
#[test]
|
|
|
|
fn test_edit_initial() {
|
2023-01-25 16:29:15 +00:00
|
|
|
// Test that MutableRepo::edit() can be used on the initial working-copy commit
|
|
|
|
// in a workspace
|
2022-01-31 01:54:05 +00:00
|
|
|
let settings = testutils::user_settings();
|
2023-09-19 12:36:45 +00:00
|
|
|
let test_repo = TestRepo::init();
|
2022-02-05 23:13:07 +00:00
|
|
|
let repo = &test_repo.repo;
|
2022-01-31 01:54:05 +00:00
|
|
|
|
2023-12-12 05:42:05 +00:00
|
|
|
let mut tx = repo.start_transaction(&settings);
|
2023-01-25 16:29:15 +00:00
|
|
|
let wc_commit = write_random_commit(tx.mut_repo(), &settings);
|
2023-12-12 05:42:05 +00:00
|
|
|
let repo = tx.commit("test");
|
2022-01-31 01:54:05 +00:00
|
|
|
|
2023-12-12 05:42:05 +00:00
|
|
|
let mut tx = repo.start_transaction(&settings);
|
2022-01-31 01:54:05 +00:00
|
|
|
let workspace_id = WorkspaceId::new("new-workspace".to_string());
|
2023-01-25 16:29:15 +00:00
|
|
|
tx.mut_repo()
|
|
|
|
.edit(workspace_id.clone(), &wc_commit)
|
|
|
|
.unwrap();
|
2023-12-12 05:42:05 +00:00
|
|
|
let repo = tx.commit("test");
|
2022-09-18 21:46:12 +00:00
|
|
|
assert_eq!(
|
|
|
|
repo.view().get_wc_commit_id(&workspace_id),
|
2023-01-25 16:29:15 +00:00
|
|
|
Some(wc_commit.id())
|
2022-09-18 21:46:12 +00:00
|
|
|
);
|
2022-01-31 01:54:05 +00:00
|
|
|
}
|
|
|
|
|
2023-09-19 12:36:45 +00:00
|
|
|
#[test]
|
|
|
|
fn test_add_head_success() {
|
2021-03-16 22:57:47 +00:00
|
|
|
// Test that MutableRepo::add_head() adds the head, and that it's still there
|
2021-01-31 02:50:27 +00:00
|
|
|
// after commit. It should also be indexed.
|
2020-12-30 04:00:28 +00:00
|
|
|
let settings = testutils::user_settings();
|
2023-09-19 12:36:45 +00:00
|
|
|
let test_repo = TestRepo::init();
|
2022-02-05 23:13:07 +00:00
|
|
|
let repo = &test_repo.repo;
|
2020-12-30 04:00:28 +00:00
|
|
|
|
|
|
|
// Create a commit outside of the repo by using a temporary transaction. Then
|
|
|
|
// add that as a head.
|
2023-12-12 05:42:05 +00:00
|
|
|
let mut tx = repo.start_transaction(&settings);
|
2022-12-24 15:38:20 +00:00
|
|
|
let new_commit = write_random_commit(tx.mut_repo(), &settings);
|
2021-12-01 18:06:25 +00:00
|
|
|
drop(tx);
|
2020-12-30 04:00:28 +00:00
|
|
|
|
2023-12-12 05:42:05 +00:00
|
|
|
let mut tx = repo.start_transaction(&settings);
|
2021-03-16 23:21:31 +00:00
|
|
|
let mut_repo = tx.mut_repo();
|
|
|
|
assert!(!mut_repo.view().heads().contains(new_commit.id()));
|
|
|
|
assert!(!mut_repo.index().has_id(new_commit.id()));
|
2023-12-21 00:38:16 +00:00
|
|
|
mut_repo.add_head(&new_commit).unwrap();
|
2021-03-16 23:21:31 +00:00
|
|
|
assert!(mut_repo.view().heads().contains(new_commit.id()));
|
|
|
|
assert!(mut_repo.index().has_id(new_commit.id()));
|
2023-12-12 05:42:05 +00:00
|
|
|
let repo = tx.commit("test");
|
2021-01-16 20:15:06 +00:00
|
|
|
assert!(repo.view().heads().contains(new_commit.id()));
|
2021-01-31 02:50:27 +00:00
|
|
|
assert!(repo.index().has_id(new_commit.id()));
|
2020-12-30 04:00:28 +00:00
|
|
|
}
|
|
|
|
|
2023-09-19 12:36:45 +00:00
|
|
|
#[test]
|
|
|
|
fn test_add_head_ancestor() {
|
2021-03-16 22:57:47 +00:00
|
|
|
// Test that MutableRepo::add_head() does not add a head if it's an ancestor of
|
2020-12-30 04:00:28 +00:00
|
|
|
// an existing head.
|
|
|
|
let settings = testutils::user_settings();
|
2023-09-19 12:36:45 +00:00
|
|
|
let test_repo = TestRepo::init();
|
2022-02-05 23:13:07 +00:00
|
|
|
let repo = &test_repo.repo;
|
2020-12-30 04:00:28 +00:00
|
|
|
|
2023-12-12 05:42:05 +00:00
|
|
|
let mut tx = repo.start_transaction(&settings);
|
2021-05-01 04:41:27 +00:00
|
|
|
let mut graph_builder = CommitGraphBuilder::new(&settings, tx.mut_repo());
|
|
|
|
let commit1 = graph_builder.initial_commit();
|
|
|
|
let commit2 = graph_builder.commit_with_parents(&[&commit1]);
|
2023-03-12 17:16:41 +00:00
|
|
|
let commit3 = graph_builder.commit_with_parents(&[&commit2]);
|
2023-12-12 05:42:05 +00:00
|
|
|
let repo = tx.commit("test");
|
2020-12-30 04:00:28 +00:00
|
|
|
|
2023-03-12 17:16:41 +00:00
|
|
|
assert_eq!(repo.view().heads(), &hashset! {commit3.id().clone()});
|
2023-12-12 05:42:05 +00:00
|
|
|
let mut tx = repo.start_transaction(&settings);
|
2021-03-16 23:21:31 +00:00
|
|
|
let mut_repo = tx.mut_repo();
|
2023-12-21 00:38:16 +00:00
|
|
|
mut_repo.add_head(&commit1).unwrap();
|
2023-03-12 17:16:41 +00:00
|
|
|
assert_eq!(repo.view().heads(), &hashset! {commit3.id().clone()});
|
2020-12-30 04:00:28 +00:00
|
|
|
}
|
2021-01-15 07:54:11 +00:00
|
|
|
|
2023-09-19 12:36:45 +00:00
|
|
|
#[test]
|
|
|
|
fn test_add_head_not_immediate_child() {
|
2021-03-16 22:57:47 +00:00
|
|
|
// Test that MutableRepo::add_head() can be used for adding a head that is not
|
2021-02-14 08:11:04 +00:00
|
|
|
// an immediate child of a current head.
|
|
|
|
let settings = testutils::user_settings();
|
2023-09-19 12:36:45 +00:00
|
|
|
let test_repo = TestRepo::init();
|
2022-02-05 23:13:07 +00:00
|
|
|
let repo = &test_repo.repo;
|
2021-02-14 08:11:04 +00:00
|
|
|
|
2023-12-12 05:42:05 +00:00
|
|
|
let mut tx = repo.start_transaction(&settings);
|
2022-12-24 15:38:20 +00:00
|
|
|
let initial = write_random_commit(tx.mut_repo(), &settings);
|
2023-12-12 05:42:05 +00:00
|
|
|
let repo = tx.commit("test");
|
2021-02-14 08:11:04 +00:00
|
|
|
|
2023-03-12 17:16:41 +00:00
|
|
|
// Create some commits outside of the repo by using a temporary transaction.
|
|
|
|
// Then add one of them as a head.
|
2023-12-12 05:42:05 +00:00
|
|
|
let mut tx = repo.start_transaction(&settings);
|
2022-12-24 17:01:11 +00:00
|
|
|
let rewritten = create_random_commit(tx.mut_repo(), &settings)
|
2021-02-14 08:11:04 +00:00
|
|
|
.set_change_id(initial.change_id().clone())
|
|
|
|
.set_predecessors(vec![initial.id().clone()])
|
2022-12-24 05:09:19 +00:00
|
|
|
.write()
|
|
|
|
.unwrap();
|
2022-12-24 17:01:11 +00:00
|
|
|
let child = create_random_commit(tx.mut_repo(), &settings)
|
2021-02-14 08:11:04 +00:00
|
|
|
.set_parents(vec![rewritten.id().clone()])
|
2022-12-24 05:09:19 +00:00
|
|
|
.write()
|
|
|
|
.unwrap();
|
2021-12-01 18:06:25 +00:00
|
|
|
drop(tx);
|
2021-02-14 08:11:04 +00:00
|
|
|
|
2023-03-12 17:16:41 +00:00
|
|
|
assert_eq!(repo.view().heads(), &hashset! {initial.id().clone()});
|
2023-12-12 05:42:05 +00:00
|
|
|
let mut tx = repo.start_transaction(&settings);
|
2021-03-16 23:21:31 +00:00
|
|
|
let mut_repo = tx.mut_repo();
|
2023-12-21 00:38:16 +00:00
|
|
|
mut_repo.add_head(&child).unwrap();
|
2023-03-12 17:16:41 +00:00
|
|
|
assert_eq!(
|
|
|
|
mut_repo.view().heads(),
|
|
|
|
&hashset! {initial.id().clone(), child.id().clone()}
|
|
|
|
);
|
2021-03-16 23:21:31 +00:00
|
|
|
assert!(mut_repo.index().has_id(initial.id()));
|
|
|
|
assert!(mut_repo.index().has_id(rewritten.id()));
|
|
|
|
assert!(mut_repo.index().has_id(child.id()));
|
2021-02-14 08:11:04 +00:00
|
|
|
}
|
|
|
|
|
2023-09-19 12:36:45 +00:00
|
|
|
#[test]
|
|
|
|
fn test_remove_head() {
|
2021-03-16 22:57:47 +00:00
|
|
|
// Test that MutableRepo::remove_head() removes the head, and that it's still
|
2021-01-31 02:50:27 +00:00
|
|
|
// removed after commit. It should remain in the index, since we otherwise would
|
|
|
|
// have to reindex everything.
|
2021-01-15 07:54:11 +00:00
|
|
|
let settings = testutils::user_settings();
|
2023-09-19 12:36:45 +00:00
|
|
|
let test_repo = TestRepo::init();
|
2022-02-05 23:13:07 +00:00
|
|
|
let repo = &test_repo.repo;
|
2021-01-15 07:54:11 +00:00
|
|
|
|
2023-12-12 05:42:05 +00:00
|
|
|
let mut tx = repo.start_transaction(&settings);
|
2021-05-01 04:41:27 +00:00
|
|
|
let mut graph_builder = CommitGraphBuilder::new(&settings, tx.mut_repo());
|
|
|
|
let commit1 = graph_builder.initial_commit();
|
|
|
|
let commit2 = graph_builder.commit_with_parents(&[&commit1]);
|
|
|
|
let commit3 = graph_builder.commit_with_parents(&[&commit2]);
|
2023-12-12 05:42:05 +00:00
|
|
|
let repo = tx.commit("test");
|
2021-01-15 07:54:11 +00:00
|
|
|
|
2023-12-12 05:42:05 +00:00
|
|
|
let mut tx = repo.start_transaction(&settings);
|
2021-03-16 23:21:31 +00:00
|
|
|
let mut_repo = tx.mut_repo();
|
|
|
|
assert!(mut_repo.view().heads().contains(commit3.id()));
|
2021-09-25 16:25:42 +00:00
|
|
|
mut_repo.remove_head(commit3.id());
|
2021-03-16 23:21:31 +00:00
|
|
|
let heads = mut_repo.view().heads().clone();
|
2021-01-15 07:54:11 +00:00
|
|
|
assert!(!heads.contains(commit3.id()));
|
|
|
|
assert!(!heads.contains(commit2.id()));
|
|
|
|
assert!(!heads.contains(commit1.id()));
|
2021-03-16 23:21:31 +00:00
|
|
|
assert!(mut_repo.index().has_id(commit1.id()));
|
|
|
|
assert!(mut_repo.index().has_id(commit2.id()));
|
|
|
|
assert!(mut_repo.index().has_id(commit3.id()));
|
2023-12-12 05:42:05 +00:00
|
|
|
let repo = tx.commit("test");
|
2021-01-16 20:15:06 +00:00
|
|
|
let heads = repo.view().heads().clone();
|
2021-01-15 07:54:11 +00:00
|
|
|
assert!(!heads.contains(commit3.id()));
|
|
|
|
assert!(!heads.contains(commit2.id()));
|
|
|
|
assert!(!heads.contains(commit1.id()));
|
2021-01-31 02:50:27 +00:00
|
|
|
assert!(repo.index().has_id(commit1.id()));
|
|
|
|
assert!(repo.index().has_id(commit2.id()));
|
|
|
|
assert!(repo.index().has_id(commit3.id()));
|
2021-01-15 07:54:11 +00:00
|
|
|
}
|
2021-01-16 00:27:16 +00:00
|
|
|
|
2023-09-19 12:36:45 +00:00
|
|
|
#[test]
|
|
|
|
fn test_has_changed() {
|
2021-12-05 20:44:08 +00:00
|
|
|
// Test that MutableRepo::has_changed() reports changes iff the view has changed
|
|
|
|
// (e.g. not after setting a branch to point to where it was already
|
|
|
|
// pointing).
|
|
|
|
let settings = testutils::user_settings();
|
2023-09-19 12:36:45 +00:00
|
|
|
let test_repo = TestRepo::init();
|
2022-02-05 23:13:07 +00:00
|
|
|
let repo = &test_repo.repo;
|
2023-10-13 14:37:17 +00:00
|
|
|
let normal_remote_ref = |id: &CommitId| RemoteRef {
|
|
|
|
target: RefTarget::normal(id.clone()),
|
2023-10-11 17:15:17 +00:00
|
|
|
state: RemoteRefState::Tracking, // doesn't matter
|
2023-10-13 14:37:17 +00:00
|
|
|
};
|
2021-12-05 20:44:08 +00:00
|
|
|
|
2023-12-12 05:42:05 +00:00
|
|
|
let mut tx = repo.start_transaction(&settings);
|
2021-12-05 20:44:08 +00:00
|
|
|
let mut_repo = tx.mut_repo();
|
2022-12-24 15:38:20 +00:00
|
|
|
let commit1 = write_random_commit(mut_repo, &settings);
|
|
|
|
let commit2 = write_random_commit(mut_repo, &settings);
|
2021-12-05 20:44:08 +00:00
|
|
|
mut_repo.remove_head(commit2.id());
|
2022-02-05 23:13:07 +00:00
|
|
|
let ws_id = WorkspaceId::default();
|
2022-10-20 23:33:14 +00:00
|
|
|
mut_repo
|
|
|
|
.set_wc_commit(ws_id.clone(), commit1.id().clone())
|
|
|
|
.unwrap();
|
2023-07-11 13:14:59 +00:00
|
|
|
mut_repo.set_local_branch_target("main", RefTarget::normal(commit1.id().clone()));
|
2023-10-13 14:37:17 +00:00
|
|
|
mut_repo.set_remote_branch("main", "origin", normal_remote_ref(commit1.id()));
|
2023-12-12 05:42:05 +00:00
|
|
|
let repo = tx.commit("test");
|
2021-12-05 20:44:08 +00:00
|
|
|
// Test the setup
|
|
|
|
assert_eq!(repo.view().heads(), &hashset! {commit1.id().clone()});
|
|
|
|
|
2023-12-12 05:42:05 +00:00
|
|
|
let mut tx = repo.start_transaction(&settings);
|
2021-12-05 20:44:08 +00:00
|
|
|
let mut_repo = tx.mut_repo();
|
|
|
|
|
2023-12-21 00:38:16 +00:00
|
|
|
mut_repo.add_head(&commit1).unwrap();
|
2022-10-20 23:33:14 +00:00
|
|
|
mut_repo
|
|
|
|
.set_wc_commit(ws_id.clone(), commit1.id().clone())
|
|
|
|
.unwrap();
|
2023-07-11 13:14:59 +00:00
|
|
|
mut_repo.set_local_branch_target("main", RefTarget::normal(commit1.id().clone()));
|
2023-10-13 14:37:17 +00:00
|
|
|
mut_repo.set_remote_branch("main", "origin", normal_remote_ref(commit1.id()));
|
2021-12-05 20:44:08 +00:00
|
|
|
assert!(!mut_repo.has_changes());
|
|
|
|
|
|
|
|
mut_repo.remove_head(commit2.id());
|
2023-07-12 16:56:02 +00:00
|
|
|
mut_repo.set_local_branch_target("stable", RefTarget::absent());
|
2023-10-13 14:37:17 +00:00
|
|
|
mut_repo.set_remote_branch("stable", "origin", RemoteRef::absent());
|
2021-12-05 20:44:08 +00:00
|
|
|
assert!(!mut_repo.has_changes());
|
|
|
|
|
2023-12-21 00:38:16 +00:00
|
|
|
mut_repo.add_head(&commit2).unwrap();
|
2021-12-05 20:44:08 +00:00
|
|
|
assert!(mut_repo.has_changes());
|
|
|
|
mut_repo.remove_head(commit2.id());
|
|
|
|
assert!(!mut_repo.has_changes());
|
|
|
|
|
2022-10-20 23:33:14 +00:00
|
|
|
mut_repo
|
|
|
|
.set_wc_commit(ws_id.clone(), commit2.id().clone())
|
|
|
|
.unwrap();
|
2021-12-05 20:44:08 +00:00
|
|
|
assert!(mut_repo.has_changes());
|
2022-10-20 23:33:14 +00:00
|
|
|
mut_repo.set_wc_commit(ws_id, commit1.id().clone()).unwrap();
|
2021-12-05 20:44:08 +00:00
|
|
|
assert!(!mut_repo.has_changes());
|
|
|
|
|
2023-07-11 13:14:59 +00:00
|
|
|
mut_repo.set_local_branch_target("main", RefTarget::normal(commit2.id().clone()));
|
2021-12-05 20:44:08 +00:00
|
|
|
assert!(mut_repo.has_changes());
|
2023-07-11 13:14:59 +00:00
|
|
|
mut_repo.set_local_branch_target("main", RefTarget::normal(commit1.id().clone()));
|
2021-12-05 20:44:08 +00:00
|
|
|
assert!(!mut_repo.has_changes());
|
|
|
|
|
2023-10-13 14:37:17 +00:00
|
|
|
mut_repo.set_remote_branch("main", "origin", normal_remote_ref(commit2.id()));
|
2021-12-05 20:44:08 +00:00
|
|
|
assert!(mut_repo.has_changes());
|
2023-10-13 14:37:17 +00:00
|
|
|
mut_repo.set_remote_branch("main", "origin", normal_remote_ref(commit1.id()));
|
2021-12-05 20:44:08 +00:00
|
|
|
assert!(!mut_repo.has_changes());
|
|
|
|
}
|
|
|
|
|
2023-09-19 12:36:45 +00:00
|
|
|
#[test]
|
|
|
|
fn test_rebase_descendants_simple() {
|
2023-12-22 05:33:00 +00:00
|
|
|
// There are many additional tests of this functionality in `test_rewrite.rs`.
|
2021-09-29 15:44:00 +00:00
|
|
|
let settings = testutils::user_settings();
|
2023-09-19 12:36:45 +00:00
|
|
|
let test_repo = TestRepo::init();
|
2022-02-05 23:13:07 +00:00
|
|
|
let repo = &test_repo.repo;
|
2021-09-29 15:44:00 +00:00
|
|
|
|
2023-12-12 05:42:05 +00:00
|
|
|
let mut tx = repo.start_transaction(&settings);
|
2021-09-29 15:44:00 +00:00
|
|
|
let mut graph_builder = CommitGraphBuilder::new(&settings, tx.mut_repo());
|
|
|
|
let commit1 = graph_builder.initial_commit();
|
|
|
|
let commit2 = graph_builder.commit_with_parents(&[&commit1]);
|
|
|
|
let commit3 = graph_builder.commit_with_parents(&[&commit2]);
|
|
|
|
let commit4 = graph_builder.commit_with_parents(&[&commit1]);
|
|
|
|
let commit5 = graph_builder.commit_with_parents(&[&commit4]);
|
2023-12-12 05:42:05 +00:00
|
|
|
let repo = tx.commit("test");
|
2021-09-29 15:44:00 +00:00
|
|
|
|
2023-12-12 05:42:05 +00:00
|
|
|
let mut tx = repo.start_transaction(&settings);
|
2021-09-29 15:44:00 +00:00
|
|
|
let mut_repo = tx.mut_repo();
|
|
|
|
let mut graph_builder = CommitGraphBuilder::new(&settings, mut_repo);
|
|
|
|
let commit6 = graph_builder.commit_with_parents(&[&commit1]);
|
|
|
|
mut_repo.record_rewritten_commit(commit2.id().clone(), commit6.id().clone());
|
|
|
|
mut_repo.record_abandoned_commit(commit4.id().clone());
|
2023-12-22 05:33:00 +00:00
|
|
|
let rebase_map = tx
|
|
|
|
.mut_repo()
|
|
|
|
.rebase_descendants_return_map(&settings)
|
|
|
|
.unwrap();
|
2021-09-29 15:44:00 +00:00
|
|
|
// Commit 3 got rebased onto commit 2's replacement, i.e. commit 6
|
2023-12-22 05:33:00 +00:00
|
|
|
assert_rebased_onto(tx.mut_repo(), &rebase_map, &commit3, &[commit6.id()]);
|
2021-09-29 15:44:00 +00:00
|
|
|
// Commit 5 got rebased onto commit 4's parent, i.e. commit 1
|
2023-12-22 05:33:00 +00:00
|
|
|
assert_rebased_onto(tx.mut_repo(), &rebase_map, &commit5, &[commit1.id()]);
|
|
|
|
assert_eq!(rebase_map.len(), 2);
|
|
|
|
|
2021-09-29 15:44:00 +00:00
|
|
|
// No more descendants to rebase if we try again.
|
2023-12-22 05:33:00 +00:00
|
|
|
let rebase_map = tx
|
|
|
|
.mut_repo()
|
|
|
|
.rebase_descendants_return_map(&settings)
|
|
|
|
.unwrap();
|
|
|
|
assert_eq!(rebase_map.len(), 0);
|
2021-09-29 15:44:00 +00:00
|
|
|
}
|
|
|
|
|
2023-09-19 12:36:45 +00:00
|
|
|
#[test]
|
|
|
|
fn test_rebase_descendants_conflicting_rewrite() {
|
2023-12-22 05:33:00 +00:00
|
|
|
// Test rebasing descendants when one commit was rewritten to several other
|
|
|
|
// commits. There are many additional tests of this functionality in
|
|
|
|
// `test_rewrite.rs`.
|
2021-09-29 15:44:00 +00:00
|
|
|
let settings = testutils::user_settings();
|
2023-09-19 12:36:45 +00:00
|
|
|
let test_repo = TestRepo::init();
|
2022-02-05 23:13:07 +00:00
|
|
|
let repo = &test_repo.repo;
|
2021-09-29 15:44:00 +00:00
|
|
|
|
2023-12-12 05:42:05 +00:00
|
|
|
let mut tx = repo.start_transaction(&settings);
|
2021-09-29 15:44:00 +00:00
|
|
|
let mut graph_builder = CommitGraphBuilder::new(&settings, tx.mut_repo());
|
|
|
|
let commit1 = graph_builder.initial_commit();
|
|
|
|
let commit2 = graph_builder.commit_with_parents(&[&commit1]);
|
|
|
|
let _commit3 = graph_builder.commit_with_parents(&[&commit2]);
|
2023-12-12 05:42:05 +00:00
|
|
|
let repo = tx.commit("test");
|
2021-09-29 15:44:00 +00:00
|
|
|
|
2023-12-12 05:42:05 +00:00
|
|
|
let mut tx = repo.start_transaction(&settings);
|
2021-09-29 15:44:00 +00:00
|
|
|
let mut_repo = tx.mut_repo();
|
|
|
|
let mut graph_builder = CommitGraphBuilder::new(&settings, mut_repo);
|
|
|
|
let commit4 = graph_builder.commit_with_parents(&[&commit1]);
|
|
|
|
let commit5 = graph_builder.commit_with_parents(&[&commit1]);
|
|
|
|
mut_repo.record_rewritten_commit(commit2.id().clone(), commit4.id().clone());
|
|
|
|
mut_repo.record_rewritten_commit(commit2.id().clone(), commit5.id().clone());
|
|
|
|
// Commit 3 does *not* get rebased because it's unclear if it should go onto
|
|
|
|
// commit 4 or commit 5
|
2023-12-22 05:33:00 +00:00
|
|
|
let rebase_map = tx
|
|
|
|
.mut_repo()
|
|
|
|
.rebase_descendants_return_map(&settings)
|
|
|
|
.unwrap();
|
|
|
|
assert!(rebase_map.is_empty());
|
2021-09-29 15:44:00 +00:00
|
|
|
}
|
2022-10-19 18:44:39 +00:00
|
|
|
|
2023-09-19 12:36:45 +00:00
|
|
|
#[test]
|
|
|
|
fn test_rename_remote() {
|
2022-10-19 18:44:39 +00:00
|
|
|
let settings = testutils::user_settings();
|
2023-09-19 12:36:45 +00:00
|
|
|
let test_repo = TestRepo::init();
|
2022-10-19 18:44:39 +00:00
|
|
|
let repo = &test_repo.repo;
|
2023-12-12 05:42:05 +00:00
|
|
|
let mut tx = repo.start_transaction(&settings);
|
2022-10-19 18:44:39 +00:00
|
|
|
let mut_repo = tx.mut_repo();
|
2022-12-24 15:38:20 +00:00
|
|
|
let commit = write_random_commit(mut_repo, &settings);
|
2023-10-12 19:06:10 +00:00
|
|
|
let remote_ref = RemoteRef {
|
|
|
|
target: RefTarget::normal(commit.id().clone()),
|
2023-10-11 17:15:17 +00:00
|
|
|
state: RemoteRefState::Tracking, // doesn't matter
|
2023-10-12 19:06:10 +00:00
|
|
|
};
|
2023-10-13 14:37:17 +00:00
|
|
|
mut_repo.set_remote_branch("main", "origin", remote_ref.clone());
|
2022-10-19 18:44:39 +00:00
|
|
|
mut_repo.rename_remote("origin", "upstream");
|
2023-10-12 19:06:10 +00:00
|
|
|
assert_eq!(mut_repo.get_remote_branch("main", "upstream"), remote_ref);
|
2023-07-12 16:56:02 +00:00
|
|
|
assert_eq!(
|
|
|
|
mut_repo.get_remote_branch("main", "origin"),
|
2023-10-12 19:06:10 +00:00
|
|
|
RemoteRef::absent()
|
2023-07-12 16:56:02 +00:00
|
|
|
);
|
2022-10-19 18:44:39 +00:00
|
|
|
}
|