mirror of
https://github.com/martinvonz/jj.git
synced 2025-01-15 16:53:25 +00:00
783e1f6512
I suspect that at least one reason that I didn't make `MutableRepo::base_repo` by an `Arc<ReadonlyRepo>` before was that I thought that that would mean that `start_transaction()` would need be moved off of `ReadonlyRepo` so it can be given an `&Arc<ReadonlyRepo>`, which would make it much less convenient to use. It turns out that a `self` argument can actually be of type `&Arc<ReadonlyRepo>`.
342 lines
12 KiB
Rust
342 lines
12 KiB
Rust
// Copyright 2020 Google LLC
|
|
//
|
|
// 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::OpenOptions;
|
|
use std::io::Write;
|
|
#[cfg(unix)]
|
|
use std::os::unix::fs::PermissionsExt;
|
|
use std::sync::Arc;
|
|
|
|
use jujube_lib::commit_builder::CommitBuilder;
|
|
use jujube_lib::repo::ReadonlyRepo;
|
|
use jujube_lib::repo_path::{FileRepoPath, RepoPath};
|
|
use jujube_lib::settings::UserSettings;
|
|
use jujube_lib::store::TreeValue;
|
|
use jujube_lib::testutils;
|
|
use jujube_lib::tree_builder::TreeBuilder;
|
|
use test_case::test_case;
|
|
|
|
#[test_case(false ; "local store")]
|
|
#[test_case(true ; "git store")]
|
|
fn test_root(use_git: bool) {
|
|
let settings = testutils::user_settings();
|
|
let (_temp_dir, repo) = testutils::init_repo(&settings, use_git);
|
|
|
|
let owned_wc = repo.working_copy().clone();
|
|
let wc = owned_wc.lock().unwrap();
|
|
assert_eq!(&wc.current_commit_id(), repo.view().checkout());
|
|
assert_ne!(&wc.current_commit_id(), repo.store().root_commit_id());
|
|
let (repo, wc_commit) = wc.commit(&settings, repo);
|
|
assert_eq!(wc_commit.id(), repo.view().checkout());
|
|
assert_eq!(wc_commit.tree().id(), repo.store().empty_tree_id());
|
|
assert_eq!(wc_commit.store_commit().parents, vec![]);
|
|
assert_eq!(wc_commit.predecessors(), vec![]);
|
|
assert_eq!(wc_commit.description(), "");
|
|
assert_eq!(wc_commit.is_open(), true);
|
|
assert_eq!(wc_commit.author().name, settings.user_name());
|
|
assert_eq!(wc_commit.author().email, settings.user_email());
|
|
assert_eq!(wc_commit.committer().name, settings.user_name());
|
|
assert_eq!(wc_commit.committer().email, settings.user_email());
|
|
}
|
|
|
|
#[cfg(unix)]
|
|
#[test_case(false ; "local store")]
|
|
#[test_case(true ; "git store")]
|
|
fn test_checkout_file_transitions(use_git: bool) {
|
|
// Tests switching between commits where a certain path is of one type in one
|
|
// commit and another type in the other. Includes a "missing" type, so we cover
|
|
// additions and removals as well.
|
|
|
|
let settings = testutils::user_settings();
|
|
let (_temp_dir, mut repo) = testutils::init_repo(&settings, use_git);
|
|
let store = repo.store().clone();
|
|
|
|
#[derive(Debug, Clone, Copy)]
|
|
enum Kind {
|
|
Missing,
|
|
Normal,
|
|
Executable,
|
|
Symlink,
|
|
Tree,
|
|
GitSubmodule,
|
|
}
|
|
|
|
fn write_path(
|
|
settings: &UserSettings,
|
|
repo: &Arc<ReadonlyRepo>,
|
|
tree_builder: &mut TreeBuilder,
|
|
kind: Kind,
|
|
path: &str,
|
|
) {
|
|
let store = repo.store();
|
|
let value = match kind {
|
|
Kind::Missing => {
|
|
return;
|
|
}
|
|
Kind::Normal => {
|
|
let id =
|
|
testutils::write_file(store, &FileRepoPath::from(path), "normal file contents");
|
|
TreeValue::Normal {
|
|
id,
|
|
executable: false,
|
|
}
|
|
}
|
|
Kind::Executable => {
|
|
let id = testutils::write_file(
|
|
store,
|
|
&FileRepoPath::from(path),
|
|
"executable file contents",
|
|
);
|
|
TreeValue::Normal {
|
|
id,
|
|
executable: true,
|
|
}
|
|
}
|
|
Kind::Symlink => {
|
|
let id = store
|
|
.write_symlink(&FileRepoPath::from(path), "target")
|
|
.unwrap();
|
|
TreeValue::Symlink(id)
|
|
}
|
|
Kind::Tree => {
|
|
let mut sub_tree_builder = store.tree_builder(store.empty_tree_id().clone());
|
|
let file_path = path.to_owned() + "/file";
|
|
write_path(
|
|
settings,
|
|
repo,
|
|
&mut sub_tree_builder,
|
|
Kind::Normal,
|
|
&file_path,
|
|
);
|
|
let id = sub_tree_builder.write_tree();
|
|
TreeValue::Tree(id)
|
|
}
|
|
Kind::GitSubmodule => {
|
|
let id = testutils::create_random_commit(&settings, &repo)
|
|
.write_to_new_transaction(&repo, "test")
|
|
.id()
|
|
.clone();
|
|
TreeValue::GitSubmodule(id)
|
|
}
|
|
};
|
|
tree_builder.set(RepoPath::from(path), value);
|
|
}
|
|
|
|
let mut kinds = vec![
|
|
Kind::Missing,
|
|
Kind::Normal,
|
|
Kind::Executable,
|
|
Kind::Symlink,
|
|
Kind::Tree,
|
|
];
|
|
if use_git {
|
|
kinds.push(Kind::GitSubmodule);
|
|
}
|
|
let mut left_tree_builder = store.tree_builder(store.empty_tree_id().clone());
|
|
let mut right_tree_builder = store.tree_builder(store.empty_tree_id().clone());
|
|
let mut files = vec![];
|
|
for left_kind in &kinds {
|
|
for right_kind in &kinds {
|
|
let path = format!("{:?}_{:?}", left_kind, right_kind);
|
|
write_path(&settings, &repo, &mut left_tree_builder, *left_kind, &path);
|
|
write_path(
|
|
&settings,
|
|
&repo,
|
|
&mut right_tree_builder,
|
|
*right_kind,
|
|
&path,
|
|
);
|
|
files.push((*left_kind, *right_kind, path));
|
|
}
|
|
}
|
|
let left_tree_id = left_tree_builder.write_tree();
|
|
let right_tree_id = right_tree_builder.write_tree();
|
|
|
|
let left_commit = CommitBuilder::for_new_commit(&settings, repo.store(), left_tree_id)
|
|
.set_parents(vec![store.root_commit_id().clone()])
|
|
.set_open(true)
|
|
.write_to_new_transaction(&repo, "test");
|
|
let right_commit = CommitBuilder::for_new_commit(&settings, repo.store(), right_tree_id)
|
|
.set_parents(vec![store.root_commit_id().clone()])
|
|
.set_open(true)
|
|
.write_to_new_transaction(&repo, "test");
|
|
|
|
let owned_wc = repo.working_copy().clone();
|
|
let wc = owned_wc.lock().unwrap();
|
|
wc.check_out(left_commit).unwrap();
|
|
repo = wc.commit(&settings, repo).0;
|
|
wc.check_out(right_commit.clone()).unwrap();
|
|
|
|
// Check that the working copy is clean.
|
|
let (reloaded_repo, after_commit) = wc.commit(&settings, repo);
|
|
repo = reloaded_repo;
|
|
let diff_summary = right_commit.tree().diff_summary(&after_commit.tree());
|
|
assert_eq!(diff_summary.modified, vec![]);
|
|
assert_eq!(diff_summary.added, vec![]);
|
|
assert_eq!(diff_summary.removed, vec![]);
|
|
|
|
for (_left_kind, right_kind, path) in &files {
|
|
let wc_path = repo.working_copy_path().join(path);
|
|
let maybe_metadata = wc_path.symlink_metadata();
|
|
match right_kind {
|
|
Kind::Missing => {
|
|
assert_eq!(maybe_metadata.is_ok(), false, "{:?} should not exist", path);
|
|
}
|
|
Kind::Normal => {
|
|
assert_eq!(maybe_metadata.is_ok(), true, "{:?} should exist", path);
|
|
let metadata = maybe_metadata.unwrap();
|
|
assert_eq!(metadata.is_file(), true, "{:?} should be a file", path);
|
|
#[cfg(unix)]
|
|
assert_eq!(
|
|
metadata.permissions().mode() & 0o111,
|
|
0,
|
|
"{:?} should not be executable",
|
|
path
|
|
);
|
|
}
|
|
Kind::Executable => {
|
|
assert_eq!(maybe_metadata.is_ok(), true, "{:?} should exist", path);
|
|
let metadata = maybe_metadata.unwrap();
|
|
assert_eq!(metadata.is_file(), true, "{:?} should be a file", path);
|
|
#[cfg(unix)]
|
|
assert_ne!(
|
|
metadata.permissions().mode() & 0o111,
|
|
0,
|
|
"{:?} should be executable",
|
|
path
|
|
);
|
|
}
|
|
Kind::Symlink => {
|
|
assert_eq!(maybe_metadata.is_ok(), true, "{:?} should exist", path);
|
|
let metadata = maybe_metadata.unwrap();
|
|
assert_eq!(
|
|
metadata.file_type().is_symlink(),
|
|
true,
|
|
"{:?} should be a symlink",
|
|
path
|
|
);
|
|
}
|
|
Kind::Tree => {
|
|
assert_eq!(maybe_metadata.is_ok(), true, "{:?} should exist", path);
|
|
let metadata = maybe_metadata.unwrap();
|
|
assert_eq!(metadata.is_dir(), true, "{:?} should be a directory", path);
|
|
}
|
|
Kind::GitSubmodule => {
|
|
// Not supported for now
|
|
assert_eq!(maybe_metadata.is_ok(), false, "{:?} should not exist", path);
|
|
}
|
|
};
|
|
}
|
|
}
|
|
|
|
#[test_case(false ; "local store")]
|
|
#[test_case(true ; "git store")]
|
|
fn test_commit_racy_timestamps(use_git: bool) {
|
|
// Tests that file modifications are detected even if they happen the same
|
|
// millisecond as the updated working copy state.
|
|
let _home_dir = testutils::new_user_home();
|
|
let settings = testutils::user_settings();
|
|
let (_temp_dir, mut repo) = testutils::init_repo(&settings, use_git);
|
|
|
|
let file_path = repo.working_copy_path().join("file");
|
|
let mut previous_tree_id = repo.store().empty_tree_id().clone();
|
|
let owned_wc = repo.working_copy().clone();
|
|
let wc = owned_wc.lock().unwrap();
|
|
for i in 0..100 {
|
|
{
|
|
let mut file = OpenOptions::new()
|
|
.create(true)
|
|
.write(true)
|
|
.open(&file_path)
|
|
.unwrap();
|
|
file.write_all(format!("contents {}", i).as_bytes())
|
|
.unwrap();
|
|
}
|
|
let (reloaded_repo, commit) = wc.commit(&settings, repo);
|
|
repo = reloaded_repo;
|
|
let new_tree_id = commit.tree().id().clone();
|
|
assert_ne!(new_tree_id, previous_tree_id);
|
|
previous_tree_id = new_tree_id;
|
|
}
|
|
}
|
|
|
|
#[test_case(false ; "local store")]
|
|
#[test_case(true ; "git store")]
|
|
fn test_gitignores(use_git: bool) {
|
|
// Tests that .gitignore files are respected.
|
|
|
|
let _home_dir = testutils::new_user_home();
|
|
let settings = testutils::user_settings();
|
|
let (_temp_dir, repo) = testutils::init_repo(&settings, use_git);
|
|
|
|
let gitignore_path = FileRepoPath::from(".gitignore");
|
|
let added_path = FileRepoPath::from("added");
|
|
let modified_path = FileRepoPath::from("modified");
|
|
let removed_path = FileRepoPath::from("removed");
|
|
let ignored_path = FileRepoPath::from("ignored");
|
|
let subdir_modified_path = FileRepoPath::from("dir/modified");
|
|
let subdir_ignored_path = FileRepoPath::from("dir/ignored");
|
|
|
|
testutils::write_working_copy_file(&repo, &gitignore_path, "ignored\n");
|
|
testutils::write_working_copy_file(&repo, &modified_path, "1");
|
|
testutils::write_working_copy_file(&repo, &removed_path, "1");
|
|
std::fs::create_dir(repo.working_copy_path().join("dir")).unwrap();
|
|
testutils::write_working_copy_file(&repo, &subdir_modified_path, "1");
|
|
|
|
let wc = repo.working_copy().clone();
|
|
let (repo, commit1) = wc.lock().unwrap().commit(&settings, repo);
|
|
let files1: Vec<_> = commit1
|
|
.tree()
|
|
.entries()
|
|
.map(|(name, _value)| name)
|
|
.collect();
|
|
assert_eq!(
|
|
files1,
|
|
vec![
|
|
gitignore_path.to_repo_path(),
|
|
subdir_modified_path.to_repo_path(),
|
|
modified_path.to_repo_path(),
|
|
removed_path.to_repo_path()
|
|
]
|
|
);
|
|
|
|
testutils::write_working_copy_file(&repo, &gitignore_path, "ignored\nmodified\nremoved\n");
|
|
testutils::write_working_copy_file(&repo, &added_path, "2");
|
|
testutils::write_working_copy_file(&repo, &modified_path, "2");
|
|
std::fs::remove_file(
|
|
repo.working_copy_path()
|
|
.join(removed_path.to_internal_string()),
|
|
)
|
|
.unwrap();
|
|
testutils::write_working_copy_file(&repo, &ignored_path, "2");
|
|
testutils::write_working_copy_file(&repo, &subdir_modified_path, "2");
|
|
testutils::write_working_copy_file(&repo, &subdir_ignored_path, "2");
|
|
|
|
let wc = repo.working_copy().clone();
|
|
let (_repo, commit2) = wc.lock().unwrap().commit(&settings, repo);
|
|
let files2: Vec<_> = commit2
|
|
.tree()
|
|
.entries()
|
|
.map(|(name, _value)| name)
|
|
.collect();
|
|
assert_eq!(
|
|
files2,
|
|
vec![
|
|
gitignore_path.to_repo_path(),
|
|
added_path.to_repo_path(),
|
|
subdir_modified_path.to_repo_path(),
|
|
modified_path.to_repo_path()
|
|
]
|
|
);
|
|
}
|