use std::cmp::max;
use std::thread;
use assert_matches::assert_matches;
use jj_lib::repo::Repo as _;
use jj_lib::working_copy::CheckoutError;
use jj_lib::workspace::Workspace;
use jj_lib::workspace::default_working_copy_factories;
use pollster::FutureExt as _;
use testutils::TestResult;
use testutils::TestWorkspace;
use testutils::assert_tree_eq;
use testutils::commit_with_tree;
use testutils::create_tree;
use testutils::empty_snapshot_options;
use testutils::repo_path;
use testutils::repo_path_buf;
use testutils::write_working_copy_file;
#[test]
fn test_concurrent_checkout() -> TestResult {
let settings = testutils::user_settings();
let mut test_workspace1 = TestWorkspace::init_with_settings(&settings);
let repo = test_workspace1.repo.clone();
let workspace1_root = test_workspace1.workspace.workspace_root().to_owned();
let tree1 = testutils::create_random_tree(&repo);
let tree2 = testutils::create_random_tree(&repo);
let tree3 = testutils::create_random_tree(&repo);
let commit1 = commit_with_tree(repo.store(), tree1.clone());
let commit2 = commit_with_tree(repo.store(), tree2.clone());
let commit3 = commit_with_tree(repo.store(), tree3);
let ws1 = &mut test_workspace1.workspace;
ws1.check_out(repo.op_id().clone(), None, &commit1)
.block_on()?;
{
let mut ws2 = Workspace::load(
&settings,
&workspace1_root,
&test_workspace1.env.default_store_factories(),
&default_working_copy_factories(),
)?;
let repo = ws2.repo_loader().load_at(repo.operation()).block_on()?;
let commit2 = repo.store().get_commit(commit2.id())?;
ws2.check_out(repo.op_id().clone(), Some(&tree1), &commit2)
.block_on()?;
}
assert_matches!(
ws1.check_out(repo.op_id().clone(), Some(&tree1), &commit3)
.block_on(),
Err(CheckoutError::ConcurrentCheckout)
);
let ws3 = Workspace::load(
&settings,
&workspace1_root,
&test_workspace1.env.default_store_factories(),
&default_working_copy_factories(),
)?;
assert_tree_eq!(*ws3.working_copy().tree()?, tree2);
Ok(())
}
#[test]
fn test_checkout_parallel() -> TestResult {
let settings = testutils::user_settings();
let mut test_workspace = TestWorkspace::init_with_settings(&settings);
let repo = &test_workspace.repo;
let workspace_root = test_workspace.workspace.workspace_root().to_owned();
let num_threads = max(num_cpus::get(), 4);
let mut trees = vec![];
for i in 0..num_threads {
let path = repo_path_buf(format!("file{i}"));
let tree = create_tree(repo, &[(&path, "contents")]);
trees.push(tree);
}
let tree = create_tree(repo, &[(repo_path("other file"), "contents")]);
let commit = commit_with_tree(repo.store(), tree);
test_workspace
.workspace
.check_out(repo.op_id().clone(), None, &commit)
.block_on()?;
thread::scope(|s| {
for tree in &trees {
let test_env = &test_workspace.env;
let op_id = repo.op_id().clone();
let trees = trees.clone();
let commit = commit_with_tree(repo.store(), tree.clone());
let settings = settings.clone();
let workspace_root = workspace_root.clone();
s.spawn(move || {
let mut workspace = Workspace::load(
&settings,
&workspace_root,
&test_env.default_store_factories(),
&default_working_copy_factories(),
)
.unwrap();
let repo = workspace
.repo_loader()
.load_at(repo.operation())
.block_on()
.unwrap();
let commit = repo.store().get_commit(commit.id()).unwrap();
let stats = workspace
.check_out(op_id, None, &commit)
.block_on()
.unwrap();
assert_eq!(stats.updated_files, 0);
assert_eq!(stats.added_files, 1);
assert_eq!(stats.removed_files, 1);
let mut locked_ws = workspace.start_working_copy_mutation().unwrap();
let (new_tree, _stats) = locked_ws
.locked_wc()
.snapshot(&empty_snapshot_options())
.block_on()
.unwrap();
assert!(
trees
.iter()
.any(|tree| tree.tree_ids() == new_tree.tree_ids())
);
});
}
});
Ok(())
}
#[test]
fn test_racy_checkout() -> TestResult {
let mut test_workspace = TestWorkspace::init();
let repo = &test_workspace.repo;
let op_id = repo.op_id().clone();
let workspace_root = test_workspace.workspace.workspace_root().to_owned();
let path = repo_path("file");
let tree = create_tree(repo, &[(path, "1")]);
let commit = commit_with_tree(repo.store(), tree.clone());
let mut num_matches = 0;
for _ in 0..100 {
let ws = &mut test_workspace.workspace;
ws.check_out(op_id.clone(), None, &commit).block_on()?;
assert_eq!(
std::fs::read(path.to_fs_path_unchecked(&workspace_root))?,
b"1".to_vec()
);
write_working_copy_file(&workspace_root, path, "x");
let modified_tree = test_workspace.snapshot()?;
if modified_tree.tree_ids() == tree.tree_ids() {
num_matches += 1;
}
write_working_copy_file(&workspace_root, path, "1");
}
assert_eq!(num_matches, 0);
Ok(())
}