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.
|
|
|
|
|
|
|
|
use std::fs;
|
|
|
|
use std::fs::OpenOptions;
|
2021-10-21 05:09:09 +00:00
|
|
|
use std::io::{Read, Write};
|
2022-03-30 05:14:39 +00:00
|
|
|
use std::path::{Path, PathBuf};
|
2022-11-01 01:29:45 +00:00
|
|
|
use std::sync::{Arc, Once};
|
2020-12-12 08:00:42 +00:00
|
|
|
|
2021-06-09 20:57:48 +00:00
|
|
|
use itertools::Itertools;
|
2022-11-08 12:35:16 +00:00
|
|
|
use jujutsu_lib::backend::{FileId, TreeId, TreeValue};
|
|
|
|
use jujutsu_lib::commit::Commit;
|
|
|
|
use jujutsu_lib::commit_builder::CommitBuilder;
|
|
|
|
use jujutsu_lib::git_backend::GitBackend;
|
|
|
|
use jujutsu_lib::local_backend::LocalBackend;
|
|
|
|
use jujutsu_lib::repo::{MutableRepo, ReadonlyRepo};
|
|
|
|
use jujutsu_lib::repo_path::RepoPath;
|
|
|
|
use jujutsu_lib::rewrite::RebasedDescendant;
|
|
|
|
use jujutsu_lib::settings::UserSettings;
|
|
|
|
use jujutsu_lib::store::Store;
|
|
|
|
use jujutsu_lib::tree::Tree;
|
|
|
|
use jujutsu_lib::tree_builder::TreeBuilder;
|
|
|
|
use jujutsu_lib::workspace::Workspace;
|
2020-12-12 08:00:42 +00:00
|
|
|
use tempfile::TempDir;
|
|
|
|
|
2022-11-01 01:29:45 +00:00
|
|
|
pub fn hermetic_libgit2() {
|
2022-10-23 17:16:14 +00:00
|
|
|
// libgit2 respects init.defaultBranch (and possibly other config
|
|
|
|
// variables) in the user's config files. Disable access to them to make
|
|
|
|
// our tests hermetic.
|
|
|
|
//
|
|
|
|
// set_search_path is unsafe because it cannot guarantee thread safety (as
|
|
|
|
// its documentation states). For the same reason, we wrap these invocations
|
2022-11-01 01:29:45 +00:00
|
|
|
// in `call_once`.
|
|
|
|
static CONFIGURE_GIT2: Once = Once::new();
|
|
|
|
CONFIGURE_GIT2.call_once(|| unsafe {
|
|
|
|
git2::opts::set_search_path(git2::ConfigLevel::System, "").unwrap();
|
|
|
|
git2::opts::set_search_path(git2::ConfigLevel::Global, "").unwrap();
|
|
|
|
git2::opts::set_search_path(git2::ConfigLevel::XDG, "").unwrap();
|
|
|
|
git2::opts::set_search_path(git2::ConfigLevel::ProgramData, "").unwrap();
|
|
|
|
});
|
2022-10-23 17:16:14 +00:00
|
|
|
}
|
|
|
|
|
2022-09-07 03:25:03 +00:00
|
|
|
pub fn new_temp_dir() -> TempDir {
|
2022-10-23 17:47:27 +00:00
|
|
|
hermetic_libgit2();
|
2022-09-07 03:25:03 +00:00
|
|
|
tempfile::Builder::new()
|
|
|
|
.prefix("jj-test-")
|
|
|
|
.tempdir()
|
|
|
|
.unwrap()
|
|
|
|
}
|
|
|
|
|
2020-12-12 08:00:42 +00:00
|
|
|
pub fn user_settings() -> UserSettings {
|
2022-03-10 02:04:19 +00:00
|
|
|
let config = config::Config::builder()
|
|
|
|
.set_override("user.name", "Test User")
|
|
|
|
.unwrap()
|
|
|
|
.set_override("user.email", "test.user@example.com")
|
|
|
|
.unwrap()
|
2022-11-14 07:41:11 +00:00
|
|
|
.set_override("operation.hostname", "host.example.com")
|
|
|
|
.unwrap()
|
|
|
|
.set_override("operation.username", "test-username")
|
|
|
|
.unwrap()
|
2022-03-10 02:04:19 +00:00
|
|
|
.build()
|
|
|
|
.unwrap();
|
2020-12-12 08:00:42 +00:00
|
|
|
UserSettings::from_config(config)
|
|
|
|
}
|
|
|
|
|
2022-02-05 21:04:30 +00:00
|
|
|
pub struct TestRepo {
|
2022-03-30 05:14:39 +00:00
|
|
|
_temp_dir: TempDir,
|
2022-02-05 21:04:30 +00:00
|
|
|
pub repo: Arc<ReadonlyRepo>,
|
|
|
|
}
|
|
|
|
|
2022-05-21 17:55:51 +00:00
|
|
|
impl TestRepo {
|
2022-05-21 18:20:51 +00:00
|
|
|
pub fn init(use_git: bool) -> Self {
|
|
|
|
let settings = user_settings();
|
2022-09-07 03:25:03 +00:00
|
|
|
let temp_dir = new_temp_dir();
|
2022-05-21 17:55:51 +00:00
|
|
|
|
|
|
|
let repo_dir = temp_dir.path().join("repo");
|
|
|
|
fs::create_dir(&repo_dir).unwrap();
|
|
|
|
|
|
|
|
let repo = if use_git {
|
|
|
|
let git_path = temp_dir.path().join("git-repo");
|
|
|
|
git2::Repository::init(&git_path).unwrap();
|
2022-12-14 18:22:12 +00:00
|
|
|
ReadonlyRepo::init(
|
|
|
|
&settings,
|
|
|
|
&repo_dir,
|
|
|
|
|store_path| Box::new(GitBackend::init_external(store_path, &git_path)),
|
|
|
|
ReadonlyRepo::default_op_store_factory(),
|
2022-12-15 23:47:31 +00:00
|
|
|
ReadonlyRepo::default_op_heads_store_factory(),
|
2022-12-14 18:22:12 +00:00
|
|
|
)
|
2022-10-28 04:27:53 +00:00
|
|
|
.unwrap()
|
2022-05-21 17:55:51 +00:00
|
|
|
} else {
|
2022-12-14 18:22:12 +00:00
|
|
|
ReadonlyRepo::init(
|
|
|
|
&settings,
|
|
|
|
&repo_dir,
|
|
|
|
|store_path| Box::new(LocalBackend::init(store_path)),
|
|
|
|
ReadonlyRepo::default_op_store_factory(),
|
2022-12-15 23:47:31 +00:00
|
|
|
ReadonlyRepo::default_op_heads_store_factory(),
|
2022-12-14 18:22:12 +00:00
|
|
|
)
|
2022-10-28 04:27:53 +00:00
|
|
|
.unwrap()
|
2022-05-21 17:55:51 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
Self {
|
|
|
|
_temp_dir: temp_dir,
|
|
|
|
repo,
|
|
|
|
}
|
2022-03-30 05:14:39 +00:00
|
|
|
}
|
2022-02-05 21:04:30 +00:00
|
|
|
}
|
|
|
|
|
2021-11-21 07:46:54 +00:00
|
|
|
pub struct TestWorkspace {
|
2022-03-30 05:14:39 +00:00
|
|
|
temp_dir: TempDir,
|
2021-11-21 07:46:54 +00:00
|
|
|
pub workspace: Workspace,
|
|
|
|
pub repo: Arc<ReadonlyRepo>,
|
|
|
|
}
|
|
|
|
|
2022-05-21 17:55:51 +00:00
|
|
|
impl TestWorkspace {
|
|
|
|
pub fn init(settings: &UserSettings, use_git: bool) -> Self {
|
2022-09-07 03:25:03 +00:00
|
|
|
let temp_dir = new_temp_dir();
|
2022-05-21 17:55:51 +00:00
|
|
|
|
|
|
|
let workspace_root = temp_dir.path().join("repo");
|
|
|
|
fs::create_dir(&workspace_root).unwrap();
|
|
|
|
|
|
|
|
let (workspace, repo) = if use_git {
|
|
|
|
let git_path = temp_dir.path().join("git-repo");
|
|
|
|
git2::Repository::init(&git_path).unwrap();
|
2022-09-23 21:03:43 +00:00
|
|
|
Workspace::init_external_git(settings, &workspace_root, &git_path).unwrap()
|
2022-05-21 17:55:51 +00:00
|
|
|
} else {
|
2022-09-23 21:03:43 +00:00
|
|
|
Workspace::init_local(settings, &workspace_root).unwrap()
|
2022-05-21 17:55:51 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
Self {
|
|
|
|
temp_dir,
|
|
|
|
workspace,
|
|
|
|
repo,
|
|
|
|
}
|
2021-11-21 07:46:54 +00:00
|
|
|
}
|
2020-12-12 08:00:42 +00:00
|
|
|
|
2022-03-30 05:14:39 +00:00
|
|
|
pub fn root_dir(&self) -> PathBuf {
|
|
|
|
self.temp_dir.path().join("repo").join("..")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-10-21 05:09:09 +00:00
|
|
|
pub fn read_file(store: &Store, path: &RepoPath, id: &FileId) -> Vec<u8> {
|
|
|
|
let mut reader = store.read_file(path, id).unwrap();
|
|
|
|
let mut content = vec![];
|
|
|
|
reader.read_to_end(&mut content).unwrap();
|
|
|
|
content
|
|
|
|
}
|
|
|
|
|
2021-09-12 06:52:38 +00:00
|
|
|
pub fn write_file(store: &Store, path: &RepoPath, contents: &str) -> FileId {
|
2020-12-12 08:00:42 +00:00
|
|
|
store.write_file(path, &mut contents.as_bytes()).unwrap()
|
|
|
|
}
|
|
|
|
|
2021-05-17 04:55:51 +00:00
|
|
|
pub fn write_normal_file(tree_builder: &mut TreeBuilder, path: &RepoPath, contents: &str) {
|
2021-10-30 05:13:35 +00:00
|
|
|
let id = write_file(tree_builder.store(), path, contents);
|
2020-12-12 08:00:42 +00:00
|
|
|
tree_builder.set(
|
2021-05-17 04:55:51 +00:00
|
|
|
path.clone(),
|
2022-11-14 21:27:18 +00:00
|
|
|
TreeValue::File {
|
2020-12-12 08:00:42 +00:00
|
|
|
id,
|
|
|
|
executable: false,
|
|
|
|
},
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2021-05-17 04:55:51 +00:00
|
|
|
pub fn write_executable_file(tree_builder: &mut TreeBuilder, path: &RepoPath, contents: &str) {
|
2021-10-30 05:13:35 +00:00
|
|
|
let id = write_file(tree_builder.store(), path, contents);
|
2020-12-12 08:00:42 +00:00
|
|
|
tree_builder.set(
|
2021-05-17 04:55:51 +00:00
|
|
|
path.clone(),
|
2022-11-14 21:27:18 +00:00
|
|
|
TreeValue::File {
|
2020-12-12 08:00:42 +00:00
|
|
|
id,
|
|
|
|
executable: true,
|
|
|
|
},
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2021-05-17 04:55:51 +00:00
|
|
|
pub fn write_symlink(tree_builder: &mut TreeBuilder, path: &RepoPath, target: &str) {
|
2021-10-30 05:13:35 +00:00
|
|
|
let id = tree_builder.store().write_symlink(path, target).unwrap();
|
2021-05-17 04:55:51 +00:00
|
|
|
tree_builder.set(path.clone(), TreeValue::Symlink(id));
|
2020-12-12 08:00:42 +00:00
|
|
|
}
|
|
|
|
|
2021-05-17 04:55:51 +00:00
|
|
|
pub fn create_tree(repo: &ReadonlyRepo, path_contents: &[(&RepoPath, &str)]) -> Tree {
|
2020-12-12 08:00:42 +00:00
|
|
|
let store = repo.store();
|
|
|
|
let mut tree_builder = store.tree_builder(store.empty_tree_id().clone());
|
|
|
|
for (path, contents) in path_contents {
|
|
|
|
write_normal_file(&mut tree_builder, path, contents);
|
|
|
|
}
|
|
|
|
let id = tree_builder.write_tree();
|
2021-05-19 16:41:25 +00:00
|
|
|
store.get_tree(&RepoPath::root(), &id).unwrap()
|
2020-12-12 08:00:42 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#[must_use]
|
|
|
|
pub fn create_random_tree(repo: &ReadonlyRepo) -> TreeId {
|
|
|
|
let mut tree_builder = repo
|
|
|
|
.store()
|
|
|
|
.tree_builder(repo.store().empty_tree_id().clone());
|
|
|
|
let number = rand::random::<u32>();
|
2022-12-15 02:30:06 +00:00
|
|
|
let path = RepoPath::from_internal_string(format!("file{number}").as_str());
|
2020-12-12 08:00:42 +00:00
|
|
|
write_normal_file(&mut tree_builder, &path, "contents");
|
|
|
|
tree_builder.write_tree()
|
|
|
|
}
|
|
|
|
|
|
|
|
#[must_use]
|
2022-12-24 17:01:11 +00:00
|
|
|
pub fn create_random_commit<'repo>(
|
|
|
|
mut_repo: &'repo mut MutableRepo,
|
|
|
|
settings: &UserSettings,
|
|
|
|
) -> CommitBuilder<'repo> {
|
|
|
|
let tree_id = create_random_tree(mut_repo.base_repo());
|
2020-12-12 08:00:42 +00:00
|
|
|
let number = rand::random::<u32>();
|
2022-09-19 04:59:49 +00:00
|
|
|
CommitBuilder::for_new_commit(
|
2022-12-24 17:01:11 +00:00
|
|
|
mut_repo,
|
2022-09-19 04:59:49 +00:00
|
|
|
settings,
|
2022-12-24 17:01:11 +00:00
|
|
|
vec![mut_repo.store().root_commit_id().clone()],
|
2022-09-19 04:59:49 +00:00
|
|
|
tree_id,
|
|
|
|
)
|
2022-12-15 02:30:06 +00:00
|
|
|
.set_description(format!("random commit {number}"))
|
2020-12-12 08:00:42 +00:00
|
|
|
}
|
|
|
|
|
2021-11-21 22:31:44 +00:00
|
|
|
pub fn write_working_copy_file(workspace_root: &Path, path: &RepoPath, contents: &str) {
|
2020-12-12 08:00:42 +00:00
|
|
|
let mut file = OpenOptions::new()
|
|
|
|
.write(true)
|
|
|
|
.create(true)
|
|
|
|
.truncate(true)
|
2021-11-21 22:31:44 +00:00
|
|
|
.open(path.to_fs_path(workspace_root))
|
2020-12-12 08:00:42 +00:00
|
|
|
.unwrap();
|
|
|
|
file.write_all(contents.as_bytes()).unwrap();
|
|
|
|
}
|
2021-05-01 04:41:27 +00:00
|
|
|
|
|
|
|
pub struct CommitGraphBuilder<'settings, 'repo> {
|
|
|
|
settings: &'settings UserSettings,
|
|
|
|
mut_repo: &'repo mut MutableRepo,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<'settings, 'repo> CommitGraphBuilder<'settings, 'repo> {
|
|
|
|
pub fn new(
|
|
|
|
settings: &'settings UserSettings,
|
|
|
|
mut_repo: &'repo mut MutableRepo,
|
|
|
|
) -> CommitGraphBuilder<'settings, 'repo> {
|
|
|
|
CommitGraphBuilder { settings, mut_repo }
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn initial_commit(&mut self) -> Commit {
|
2022-12-24 17:01:11 +00:00
|
|
|
create_random_commit(self.mut_repo, self.settings).write()
|
2021-05-01 04:41:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn commit_with_parents(&mut self, parents: &[&Commit]) -> Commit {
|
2021-06-09 20:57:48 +00:00
|
|
|
let parent_ids = parents
|
|
|
|
.iter()
|
|
|
|
.map(|commit| commit.id().clone())
|
|
|
|
.collect_vec();
|
2022-12-24 17:01:11 +00:00
|
|
|
create_random_commit(self.mut_repo, self.settings)
|
2021-05-01 04:41:27 +00:00
|
|
|
.set_parents(parent_ids)
|
2022-12-24 17:01:11 +00:00
|
|
|
.write()
|
2021-05-01 04:41:27 +00:00
|
|
|
}
|
|
|
|
}
|
2021-09-29 18:27:58 +00:00
|
|
|
|
|
|
|
pub fn assert_rebased(
|
|
|
|
rebased: Option<RebasedDescendant>,
|
|
|
|
expected_old_commit: &Commit,
|
|
|
|
expected_new_parents: &[&Commit],
|
|
|
|
) -> Commit {
|
|
|
|
if let Some(RebasedDescendant {
|
|
|
|
old_commit,
|
|
|
|
new_commit,
|
|
|
|
}) = rebased
|
|
|
|
{
|
|
|
|
assert_eq!(old_commit, *expected_old_commit);
|
|
|
|
assert_eq!(new_commit.change_id(), expected_old_commit.change_id());
|
|
|
|
assert_eq!(
|
|
|
|
new_commit.parent_ids(),
|
|
|
|
expected_new_parents
|
|
|
|
.iter()
|
|
|
|
.map(|commit| commit.id().clone())
|
|
|
|
.collect_vec()
|
|
|
|
);
|
|
|
|
new_commit
|
|
|
|
} else {
|
2022-12-15 02:30:06 +00:00
|
|
|
panic!("expected rebased commit: {rebased:?}");
|
2021-09-29 18:27:58 +00:00
|
|
|
}
|
|
|
|
}
|