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.
|
|
|
|
|
2021-03-14 17:37:28 +00:00
|
|
|
use std::path::Path;
|
|
|
|
|
2021-09-12 06:52:38 +00:00
|
|
|
use jujutsu_lib::backend::CommitId;
|
2021-05-15 16:16:31 +00:00
|
|
|
use jujutsu_lib::repo::RepoRef;
|
2020-12-12 08:00:42 +00:00
|
|
|
use test_case::test_case;
|
2022-12-24 15:38:20 +00:00
|
|
|
use testutils::{create_random_commit, write_random_commit, TestRepo};
|
2020-12-12 08:00:42 +00:00
|
|
|
|
|
|
|
fn list_dir(dir: &Path) -> Vec<String> {
|
|
|
|
std::fs::read_dir(dir)
|
|
|
|
.unwrap()
|
|
|
|
.map(|entry| entry.unwrap().file_name().to_str().unwrap().to_owned())
|
|
|
|
.collect()
|
|
|
|
}
|
|
|
|
|
2021-09-12 06:52:38 +00:00
|
|
|
#[test_case(false ; "local backend")]
|
|
|
|
#[test_case(true ; "git backend")]
|
2021-03-12 23:46:06 +00:00
|
|
|
fn test_unpublished_operation(use_git: bool) {
|
|
|
|
// Test that the operation doesn't get published until that's requested.
|
|
|
|
let settings = testutils::user_settings();
|
2022-05-21 18:20:51 +00:00
|
|
|
let test_repo = TestRepo::init(use_git);
|
2022-02-05 23:31:20 +00:00
|
|
|
let repo = &test_repo.repo;
|
2021-03-12 23:46:06 +00:00
|
|
|
|
2023-01-25 04:20:46 +00:00
|
|
|
let op_heads_dir = repo.repo_path().join("op_heads").join("heads");
|
2021-03-14 05:38:37 +00:00
|
|
|
let op_id0 = repo.op_id().clone();
|
|
|
|
assert_eq!(list_dir(&op_heads_dir), vec![repo.op_id().hex()]);
|
2021-03-12 23:46:06 +00:00
|
|
|
|
2022-11-13 06:29:06 +00:00
|
|
|
let mut tx1 = repo.start_transaction(&settings, "transaction 1");
|
2022-12-24 15:38:20 +00:00
|
|
|
write_random_commit(tx1.mut_repo(), &settings);
|
2021-03-12 23:46:06 +00:00
|
|
|
let unpublished_op = tx1.write();
|
|
|
|
let op_id1 = unpublished_op.operation().id().clone();
|
|
|
|
assert_ne!(op_id1, op_id0);
|
|
|
|
assert_eq!(list_dir(&op_heads_dir), vec![op_id0.hex()]);
|
|
|
|
unpublished_op.publish();
|
|
|
|
assert_eq!(list_dir(&op_heads_dir), vec![op_id1.hex()]);
|
|
|
|
}
|
|
|
|
|
2021-09-12 06:52:38 +00:00
|
|
|
#[test_case(false ; "local backend")]
|
|
|
|
#[test_case(true ; "git backend")]
|
2020-12-12 08:00:42 +00:00
|
|
|
fn test_consecutive_operations(use_git: bool) {
|
|
|
|
// Test that consecutive operations result in a single op-head on disk after
|
|
|
|
// each operation
|
|
|
|
let settings = testutils::user_settings();
|
2022-05-21 18:20:51 +00:00
|
|
|
let test_repo = TestRepo::init(use_git);
|
2022-02-05 23:31:20 +00:00
|
|
|
let repo = &test_repo.repo;
|
2020-12-12 08:00:42 +00:00
|
|
|
|
2023-01-25 04:20:46 +00:00
|
|
|
let op_heads_dir = repo.repo_path().join("op_heads").join("heads");
|
2021-03-14 05:38:37 +00:00
|
|
|
let op_id0 = repo.op_id().clone();
|
|
|
|
assert_eq!(list_dir(&op_heads_dir), vec![repo.op_id().hex()]);
|
2020-12-12 08:00:42 +00:00
|
|
|
|
2022-11-13 06:29:06 +00:00
|
|
|
let mut tx1 = repo.start_transaction(&settings, "transaction 1");
|
2022-12-24 15:38:20 +00:00
|
|
|
write_random_commit(tx1.mut_repo(), &settings);
|
2021-05-07 06:29:53 +00:00
|
|
|
let op_id1 = tx1.commit().operation().id().clone();
|
2021-03-10 23:22:04 +00:00
|
|
|
assert_ne!(op_id1, op_id0);
|
|
|
|
assert_eq!(list_dir(&op_heads_dir), vec![op_id1.hex()]);
|
2020-12-12 08:00:42 +00:00
|
|
|
|
2022-04-28 05:30:03 +00:00
|
|
|
let repo = repo.reload_at_head(&settings).unwrap();
|
2022-11-13 06:29:06 +00:00
|
|
|
let mut tx2 = repo.start_transaction(&settings, "transaction 2");
|
2022-12-24 15:38:20 +00:00
|
|
|
write_random_commit(tx2.mut_repo(), &settings);
|
2021-05-07 06:29:53 +00:00
|
|
|
let op_id2 = tx2.commit().operation().id().clone();
|
2021-03-10 23:22:04 +00:00
|
|
|
assert_ne!(op_id2, op_id0);
|
|
|
|
assert_ne!(op_id2, op_id1);
|
|
|
|
assert_eq!(list_dir(&op_heads_dir), vec![op_id2.hex()]);
|
2020-12-12 08:00:42 +00:00
|
|
|
|
|
|
|
// Reloading the repo makes no difference (there are no conflicting operations
|
|
|
|
// to resolve).
|
2022-04-28 05:30:03 +00:00
|
|
|
let _repo = repo.reload_at_head(&settings).unwrap();
|
2021-03-10 23:22:04 +00:00
|
|
|
assert_eq!(list_dir(&op_heads_dir), vec![op_id2.hex()]);
|
2020-12-12 08:00:42 +00:00
|
|
|
}
|
|
|
|
|
2021-09-12 06:52:38 +00:00
|
|
|
#[test_case(false ; "local backend")]
|
|
|
|
#[test_case(true ; "git backend")]
|
2020-12-12 08:00:42 +00:00
|
|
|
fn test_concurrent_operations(use_git: bool) {
|
|
|
|
// Test that consecutive operations result in multiple op-heads on disk until
|
|
|
|
// the repo has been reloaded (which currently happens right away).
|
|
|
|
let settings = testutils::user_settings();
|
2022-05-21 18:20:51 +00:00
|
|
|
let test_repo = TestRepo::init(use_git);
|
2022-02-05 23:31:20 +00:00
|
|
|
let repo = &test_repo.repo;
|
2020-12-12 08:00:42 +00:00
|
|
|
|
2023-01-25 04:20:46 +00:00
|
|
|
let op_heads_dir = repo.repo_path().join("op_heads").join("heads");
|
2021-03-14 05:38:37 +00:00
|
|
|
let op_id0 = repo.op_id().clone();
|
|
|
|
assert_eq!(list_dir(&op_heads_dir), vec![repo.op_id().hex()]);
|
2020-12-12 08:00:42 +00:00
|
|
|
|
2022-11-13 06:29:06 +00:00
|
|
|
let mut tx1 = repo.start_transaction(&settings, "transaction 1");
|
2022-12-24 15:38:20 +00:00
|
|
|
write_random_commit(tx1.mut_repo(), &settings);
|
2021-05-07 06:29:53 +00:00
|
|
|
let op_id1 = tx1.commit().operation().id().clone();
|
2021-03-10 23:22:04 +00:00
|
|
|
assert_ne!(op_id1, op_id0);
|
|
|
|
assert_eq!(list_dir(&op_heads_dir), vec![op_id1.hex()]);
|
2020-12-12 08:00:42 +00:00
|
|
|
|
|
|
|
// After both transactions have committed, we should have two op-heads on disk,
|
|
|
|
// since they were run in parallel.
|
2022-11-13 06:29:06 +00:00
|
|
|
let mut tx2 = repo.start_transaction(&settings, "transaction 2");
|
2022-12-24 15:38:20 +00:00
|
|
|
write_random_commit(tx2.mut_repo(), &settings);
|
2021-05-07 06:29:53 +00:00
|
|
|
let op_id2 = tx2.commit().operation().id().clone();
|
2021-03-10 23:22:04 +00:00
|
|
|
assert_ne!(op_id2, op_id0);
|
|
|
|
assert_ne!(op_id2, op_id1);
|
2020-12-12 08:00:42 +00:00
|
|
|
let mut actual_heads_on_disk = list_dir(&op_heads_dir);
|
|
|
|
actual_heads_on_disk.sort();
|
2021-03-10 23:22:04 +00:00
|
|
|
let mut expected_heads_on_disk = vec![op_id1.hex(), op_id2.hex()];
|
2020-12-12 08:00:42 +00:00
|
|
|
expected_heads_on_disk.sort();
|
|
|
|
assert_eq!(actual_heads_on_disk, expected_heads_on_disk);
|
|
|
|
|
|
|
|
// Reloading the repo causes the operations to be merged
|
2022-04-28 05:30:03 +00:00
|
|
|
let repo = repo.reload_at_head(&settings).unwrap();
|
2021-03-14 05:38:37 +00:00
|
|
|
let merged_op_id = repo.op_id().clone();
|
2021-03-10 23:22:04 +00:00
|
|
|
assert_ne!(merged_op_id, op_id0);
|
|
|
|
assert_ne!(merged_op_id, op_id1);
|
|
|
|
assert_ne!(merged_op_id, op_id2);
|
|
|
|
assert_eq!(list_dir(&op_heads_dir), vec![merged_op_id.hex()]);
|
2020-12-12 08:00:42 +00:00
|
|
|
}
|
|
|
|
|
2021-01-31 07:44:31 +00:00
|
|
|
fn assert_heads(repo: RepoRef, expected: Vec<&CommitId>) {
|
2020-12-12 08:00:42 +00:00
|
|
|
let expected = expected.iter().cloned().cloned().collect();
|
2021-01-16 20:15:06 +00:00
|
|
|
assert_eq!(*repo.view().heads(), expected);
|
2020-12-12 08:00:42 +00:00
|
|
|
}
|
|
|
|
|
2021-09-12 06:52:38 +00:00
|
|
|
#[test_case(false ; "local backend")]
|
|
|
|
#[test_case(true ; "git backend")]
|
2020-12-12 08:00:42 +00:00
|
|
|
fn test_isolation(use_git: bool) {
|
|
|
|
// Test that two concurrent transactions don't see each other's changes.
|
|
|
|
let settings = testutils::user_settings();
|
2022-05-21 18:20:51 +00:00
|
|
|
let test_repo = TestRepo::init(use_git);
|
2022-02-05 23:31:20 +00:00
|
|
|
let repo = &test_repo.repo;
|
2020-12-12 08:00:42 +00:00
|
|
|
|
2022-11-13 06:29:06 +00:00
|
|
|
let mut tx = repo.start_transaction(&settings, "test");
|
2022-12-24 17:01:11 +00:00
|
|
|
let initial = create_random_commit(tx.mut_repo(), &settings)
|
2020-12-12 08:00:42 +00:00
|
|
|
.set_parents(vec![repo.store().root_commit_id().clone()])
|
2022-12-24 05:09:19 +00:00
|
|
|
.write()
|
|
|
|
.unwrap();
|
2021-09-11 16:45:06 +00:00
|
|
|
let repo = tx.commit();
|
2020-12-12 08:00:42 +00:00
|
|
|
|
2022-11-13 06:29:06 +00:00
|
|
|
let mut tx1 = repo.start_transaction(&settings, "transaction 1");
|
2021-03-16 23:32:51 +00:00
|
|
|
let mut_repo1 = tx1.mut_repo();
|
2022-11-13 06:29:06 +00:00
|
|
|
let mut tx2 = repo.start_transaction(&settings, "transaction 2");
|
2021-03-16 23:32:51 +00:00
|
|
|
let mut_repo2 = tx2.mut_repo();
|
2020-12-12 08:00:42 +00:00
|
|
|
|
2022-02-05 23:31:20 +00:00
|
|
|
assert_heads(repo.as_repo_ref(), vec![initial.id()]);
|
|
|
|
assert_heads(mut_repo1.as_repo_ref(), vec![initial.id()]);
|
|
|
|
assert_heads(mut_repo2.as_repo_ref(), vec![initial.id()]);
|
2020-12-12 08:00:42 +00:00
|
|
|
|
2022-12-25 20:58:08 +00:00
|
|
|
let rewrite1 = mut_repo1
|
|
|
|
.rewrite_commit(&settings, &initial)
|
2022-12-21 09:13:56 +00:00
|
|
|
.set_description("rewrite1")
|
2022-12-24 05:09:19 +00:00
|
|
|
.write()
|
|
|
|
.unwrap();
|
2022-04-28 05:30:03 +00:00
|
|
|
mut_repo1.rebase_descendants(&settings).unwrap();
|
2022-12-25 20:58:08 +00:00
|
|
|
let rewrite2 = mut_repo2
|
|
|
|
.rewrite_commit(&settings, &initial)
|
2022-12-21 09:13:56 +00:00
|
|
|
.set_description("rewrite2")
|
2022-12-24 05:09:19 +00:00
|
|
|
.write()
|
|
|
|
.unwrap();
|
2022-04-28 05:30:03 +00:00
|
|
|
mut_repo2.rebase_descendants(&settings).unwrap();
|
2020-12-12 08:00:42 +00:00
|
|
|
|
|
|
|
// Neither transaction has committed yet, so each transaction sees its own
|
|
|
|
// commit.
|
2022-02-05 23:31:20 +00:00
|
|
|
assert_heads(repo.as_repo_ref(), vec![initial.id()]);
|
2022-03-21 05:16:53 +00:00
|
|
|
assert_heads(mut_repo1.as_repo_ref(), vec![rewrite1.id()]);
|
|
|
|
assert_heads(mut_repo2.as_repo_ref(), vec![rewrite2.id()]);
|
2020-12-12 08:00:42 +00:00
|
|
|
|
|
|
|
// The base repo and tx2 don't see the commits from tx1.
|
|
|
|
tx1.commit();
|
2022-02-05 23:31:20 +00:00
|
|
|
assert_heads(repo.as_repo_ref(), vec![initial.id()]);
|
2022-03-21 05:16:53 +00:00
|
|
|
assert_heads(mut_repo2.as_repo_ref(), vec![rewrite2.id()]);
|
2020-12-12 08:00:42 +00:00
|
|
|
|
|
|
|
// The base repo still doesn't see the commits after both transactions commit.
|
|
|
|
tx2.commit();
|
2022-02-05 23:31:20 +00:00
|
|
|
assert_heads(repo.as_repo_ref(), vec![initial.id()]);
|
2020-12-12 08:00:42 +00:00
|
|
|
// After reload, the base repo sees both rewrites.
|
2022-04-28 05:30:03 +00:00
|
|
|
let repo = repo.reload_at_head(&settings).unwrap();
|
2022-03-21 05:16:53 +00:00
|
|
|
assert_heads(repo.as_repo_ref(), vec![rewrite1.id(), rewrite2.id()]);
|
2020-12-12 08:00:42 +00:00
|
|
|
}
|