// Copyright 2020 The Jujutsu Authors // // 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. #[cfg(unix)] use std::os::unix::fs::PermissionsExt; #[cfg(unix)] use std::os::unix::net::UnixListener; use std::path::Path; use std::path::PathBuf; use std::sync::Arc; use assert_matches::assert_matches; use indoc::indoc; use itertools::Itertools; use jj_lib::backend::MergedTreeId; use jj_lib::backend::TreeId; use jj_lib::backend::TreeValue; use jj_lib::file_util::check_symlink_support; use jj_lib::file_util::try_symlink; use jj_lib::fsmonitor::FsmonitorSettings; use jj_lib::local_working_copy::LocalWorkingCopy; use jj_lib::merge::Merge; use jj_lib::merge::MergedTreeValue; use jj_lib::merged_tree::MergedTree; use jj_lib::merged_tree::MergedTreeBuilder; use jj_lib::op_store::OperationId; use jj_lib::op_store::WorkspaceId; use jj_lib::repo::ReadonlyRepo; use jj_lib::repo::Repo; use jj_lib::repo_path::RepoPath; use jj_lib::repo_path::RepoPathBuf; use jj_lib::repo_path::RepoPathComponent; use jj_lib::secret_backend::SecretBackend; use jj_lib::settings::UserSettings; use jj_lib::working_copy::CheckoutError; use jj_lib::working_copy::CheckoutStats; use jj_lib::working_copy::SnapshotError; use jj_lib::working_copy::SnapshotOptions; use jj_lib::workspace::default_working_copy_factories; use jj_lib::workspace::LockedWorkspace; use jj_lib::workspace::Workspace; use pollster::FutureExt; use test_case::test_case; use testutils::commit_with_tree; use testutils::create_tree; use testutils::write_random_commit; use testutils::TestRepoBackend; use testutils::TestWorkspace; fn check_icase_fs(dir: &Path) -> bool { let test_file = tempfile::Builder::new() .prefix("icase-") .tempfile_in(dir) .unwrap(); let orig_name = test_file.path().file_name().unwrap().to_str().unwrap(); let upper_name = orig_name.to_ascii_uppercase(); assert_ne!(orig_name, upper_name); dir.join(upper_name).try_exists().unwrap() } /// Returns true if the directory appears to ignore some unicode zero-width /// characters, as in HFS+. fn check_hfs_plus(dir: &Path) -> bool { let test_file = tempfile::Builder::new() .prefix("hfs-plus-\u{200c}-") .tempfile_in(dir) .unwrap(); let orig_name = test_file.path().file_name().unwrap().to_str().unwrap(); let stripped_name = orig_name.replace('\u{200c}', ""); assert_ne!(orig_name, stripped_name); dir.join(stripped_name).try_exists().unwrap() } /// Returns true if the directory appears to support Windows short file names. fn check_vfat(dir: &Path) -> bool { let _test_file = tempfile::Builder::new() .prefix("vfattest-") .tempfile_in(dir) .unwrap(); let short_name = "VFATTE~1"; dir.join(short_name).try_exists().unwrap() } fn to_owned_path_vec(paths: &[&RepoPath]) -> Vec { paths.iter().map(|&path| path.to_owned()).collect() } fn tree_entries(tree: &MergedTree) -> Vec<(RepoPathBuf, Option)> { tree.entries() .map(|(path, result)| (path, result.ok())) .collect_vec() } #[test] fn test_root() { // Test that the working copy is clean and empty after init. let settings = testutils::user_settings(); let mut test_workspace = TestWorkspace::init(&settings); let wc = test_workspace.workspace.working_copy(); assert_eq!(wc.sparse_patterns().unwrap(), vec![RepoPathBuf::root()]); let new_tree = test_workspace.snapshot().unwrap(); let repo = &test_workspace.repo; let wc_commit_id = repo .view() .get_wc_commit_id(&WorkspaceId::default()) .unwrap(); let wc_commit = repo.store().get_commit(wc_commit_id).unwrap(); assert_eq!(new_tree.id(), *wc_commit.tree_id()); assert_eq!(new_tree.id(), repo.store().empty_merged_tree_id()); } #[test_case(TestRepoBackend::Local ; "local backend")] #[test_case(TestRepoBackend::Git ; "git backend")] fn test_checkout_file_transitions(backend: TestRepoBackend) { // 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 mut test_workspace = TestWorkspace::init_with_backend(&settings, backend); let repo = &test_workspace.repo; let store = repo.store().clone(); let workspace_root = test_workspace.workspace.workspace_root().to_owned(); #[derive(Debug, PartialEq, Eq, Clone, Copy)] enum Kind { Missing, Normal, Executable, // Executable, but same content as Normal, to test transition where only the bit changed ExecutableNormalContent, Conflict, // Same content as Executable, to test that transition preserves the executable bit ConflictedExecutableContent, Symlink, Tree, GitSubmodule, } fn write_path( settings: &UserSettings, repo: &Arc, tree_builder: &mut MergedTreeBuilder, kind: Kind, path: &RepoPath, ) { let store = repo.store(); let value = match kind { Kind::Missing => Merge::absent(), Kind::Normal => { let id = testutils::write_file(store, path, "normal file contents"); Merge::normal(TreeValue::File { id, executable: false, }) } Kind::Executable => { let id: jj_lib::backend::FileId = testutils::write_file(store, path, "executable file contents"); Merge::normal(TreeValue::File { id, executable: true, }) } Kind::ExecutableNormalContent => { let id = testutils::write_file(store, path, "normal file contents"); Merge::normal(TreeValue::File { id, executable: true, }) } Kind::Conflict => { let base_file_id = testutils::write_file(store, path, "base file contents"); let left_file_id = testutils::write_file(store, path, "left file contents"); let right_file_id = testutils::write_file(store, path, "right file contents"); Merge::from_removes_adds( vec![Some(TreeValue::File { id: base_file_id, executable: false, })], vec![ Some(TreeValue::File { id: left_file_id, executable: false, }), Some(TreeValue::File { id: right_file_id, executable: false, }), ], ) } Kind::ConflictedExecutableContent => { let base_file_id = testutils::write_file(store, path, "executable file contents"); let left_file_id = testutils::write_file(store, path, "left executable file contents"); let right_file_id = testutils::write_file(store, path, "right executable file contents"); Merge::from_removes_adds( vec![Some(TreeValue::File { id: base_file_id, executable: true, })], vec![ Some(TreeValue::File { id: left_file_id, executable: true, }), Some(TreeValue::File { id: right_file_id, executable: true, }), ], ) } Kind::Symlink => { let id = store.write_symlink(path, "target").block_on().unwrap(); Merge::normal(TreeValue::Symlink(id)) } Kind::Tree => { let file_path = path.join(RepoPathComponent::new("file")); let id = testutils::write_file(store, &file_path, "normal file contents"); let value = TreeValue::File { id, executable: false, }; tree_builder.set_or_remove(file_path, Merge::normal(value)); return; } Kind::GitSubmodule => { let mut tx = repo.start_transaction(settings); let id = write_random_commit(tx.repo_mut(), settings).id().clone(); tx.commit("test"); Merge::normal(TreeValue::GitSubmodule(id)) } }; tree_builder.set_or_remove(path.to_owned(), value); } let mut kinds = vec![ Kind::Missing, Kind::Normal, Kind::Executable, Kind::ExecutableNormalContent, Kind::Conflict, Kind::ConflictedExecutableContent, Kind::Tree, ]; kinds.push(Kind::Symlink); if backend == TestRepoBackend::Git { kinds.push(Kind::GitSubmodule); } let mut left_tree_builder = MergedTreeBuilder::new(store.empty_merged_tree_id()); let mut right_tree_builder = MergedTreeBuilder::new(store.empty_merged_tree_id()); let mut files = vec![]; for left_kind in &kinds { for right_kind in &kinds { let path = RepoPathBuf::from_internal_string(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.clone())); } } let left_tree_id = left_tree_builder.write_tree(&store).unwrap(); let right_tree_id = right_tree_builder.write_tree(&store).unwrap(); let left_commit = commit_with_tree(&store, left_tree_id); let right_commit = commit_with_tree(&store, right_tree_id.clone()); let ws = &mut test_workspace.workspace; ws.check_out(repo.op_id().clone(), None, &left_commit) .unwrap(); ws.check_out(repo.op_id().clone(), None, &right_commit) .unwrap(); // Check that the working copy is clean. let new_tree = test_workspace.snapshot().unwrap(); assert_eq!(new_tree.id(), right_tree_id); for (_left_kind, right_kind, path) in &files { let wc_path = workspace_root.join(path.as_internal_file_string()); let maybe_metadata = wc_path.symlink_metadata(); match right_kind { Kind::Missing => { assert!(maybe_metadata.is_err(), "{path:?} should not exist"); } Kind::Normal => { assert!(maybe_metadata.is_ok(), "{path:?} should exist"); let metadata = maybe_metadata.unwrap(); assert!(metadata.is_file(), "{path:?} should be a file"); #[cfg(unix)] assert_eq!( metadata.permissions().mode() & 0o111, 0, "{path:?} should not be executable" ); } Kind::Executable | Kind::ExecutableNormalContent => { assert!(maybe_metadata.is_ok(), "{path:?} should exist"); let metadata = maybe_metadata.unwrap(); assert!(metadata.is_file(), "{path:?} should be a file"); #[cfg(unix)] assert_ne!( metadata.permissions().mode() & 0o111, 0, "{path:?} should be executable" ); } Kind::Conflict => { assert!(maybe_metadata.is_ok(), "{path:?} should exist"); let metadata = maybe_metadata.unwrap(); assert!(metadata.is_file(), "{path:?} should be a file"); #[cfg(unix)] assert_eq!( metadata.permissions().mode() & 0o111, 0, "{path:?} should not be executable" ); } Kind::ConflictedExecutableContent => { assert!(maybe_metadata.is_ok(), "{path:?} should exist"); let metadata = maybe_metadata.unwrap(); assert!(metadata.is_file(), "{path:?} should be a file"); #[cfg(unix)] assert_ne!( metadata.permissions().mode() & 0o111, 0, "{path:?} should be executable" ); } Kind::Symlink => { assert!(maybe_metadata.is_ok(), "{path:?} should exist"); let metadata = maybe_metadata.unwrap(); if check_symlink_support().unwrap_or(false) { assert!( metadata.file_type().is_symlink(), "{path:?} should be a symlink" ); } } Kind::Tree => { assert!(maybe_metadata.is_ok(), "{path:?} should exist"); let metadata = maybe_metadata.unwrap(); assert!(metadata.is_dir(), "{path:?} should be a directory"); } Kind::GitSubmodule => { // Not supported for now assert!(maybe_metadata.is_err(), "{path:?} should not exist"); } }; } } // Test case for issue #2165 #[test] fn test_conflict_subdirectory() { let settings = testutils::user_settings(); let mut test_workspace = TestWorkspace::init(&settings); let repo = &test_workspace.repo; let path = RepoPath::from_internal_string("sub/file"); let empty_tree = create_tree(repo, &[]); let tree1 = create_tree(repo, &[(path, "0")]); let tree2 = create_tree(repo, &[(path, "1")]); let merged_tree = tree1.merge(&empty_tree, &tree2).unwrap(); let commit1 = commit_with_tree(repo.store(), tree1.id()); let merged_commit = commit_with_tree(repo.store(), merged_tree.id()); let repo = &test_workspace.repo; let ws = &mut test_workspace.workspace; ws.check_out(repo.op_id().clone(), None, &commit1).unwrap(); ws.check_out(repo.op_id().clone(), None, &merged_commit) .unwrap(); } #[test] fn test_acl() { let settings = testutils::user_settings(); let test_workspace = TestWorkspace::init_with_backend(&settings, TestRepoBackend::Git); let repo = &test_workspace.repo; let workspace_root = test_workspace.workspace.workspace_root().to_owned(); let secret_modified_path = RepoPath::from_internal_string("secret/modified"); let secret_added_path = RepoPath::from_internal_string("secret/added"); let secret_deleted_path = RepoPath::from_internal_string("secret/deleted"); let became_secret_path = RepoPath::from_internal_string("file1"); let became_public_path = RepoPath::from_internal_string("file2"); let tree1 = create_tree( repo, &[ (secret_modified_path, "0"), (secret_deleted_path, "0"), (became_secret_path, "public"), (became_public_path, "secret"), ], ); let tree2 = create_tree( repo, &[ (secret_modified_path, "1"), (secret_added_path, "1"), (became_secret_path, "secret"), (became_public_path, "public"), ], ); let commit1 = commit_with_tree(repo.store(), tree1.id()); let commit2 = commit_with_tree(repo.store(), tree2.id()); SecretBackend::adopt_git_repo(&workspace_root); let mut ws = Workspace::load( &settings, &workspace_root, &test_workspace.env.default_store_factories(), &default_working_copy_factories(), ) .unwrap(); // Reload commits from the store associated with the workspace let repo = ws.repo_loader().load_at(repo.operation()).unwrap(); let commit1 = repo.store().get_commit(commit1.id()).unwrap(); let commit2 = repo.store().get_commit(commit2.id()).unwrap(); ws.check_out(repo.op_id().clone(), None, &commit1).unwrap(); assert!(!secret_modified_path .to_fs_path_unchecked(&workspace_root) .is_file()); assert!(!secret_added_path .to_fs_path_unchecked(&workspace_root) .is_file()); assert!(!secret_deleted_path .to_fs_path_unchecked(&workspace_root) .is_file()); assert!(became_secret_path .to_fs_path_unchecked(&workspace_root) .is_file()); assert!(!became_public_path .to_fs_path_unchecked(&workspace_root) .is_file()); ws.check_out(repo.op_id().clone(), None, &commit2).unwrap(); assert!(!secret_modified_path .to_fs_path_unchecked(&workspace_root) .is_file()); assert!(!secret_added_path .to_fs_path_unchecked(&workspace_root) .is_file()); assert!(!secret_deleted_path .to_fs_path_unchecked(&workspace_root) .is_file()); assert!(!became_secret_path .to_fs_path_unchecked(&workspace_root) .is_file()); assert!(became_public_path .to_fs_path_unchecked(&workspace_root) .is_file()); } #[test] fn test_tree_builder_file_directory_transition() { let settings = testutils::user_settings(); let test_workspace = TestWorkspace::init(&settings); let repo = &test_workspace.repo; let store = repo.store(); let mut ws = test_workspace.workspace; let workspace_root = ws.workspace_root().to_owned(); let mut check_out_tree = |tree_id: &TreeId| { let tree = repo.store().get_tree(RepoPath::root(), tree_id).unwrap(); let commit = commit_with_tree(repo.store(), MergedTreeId::Legacy(tree.id().clone())); ws.check_out(repo.op_id().clone(), None, &commit).unwrap(); }; let parent_path = RepoPath::from_internal_string("foo/bar"); let child_path = RepoPath::from_internal_string("foo/bar/baz"); // Add file at parent_path let mut tree_builder = store.tree_builder(store.empty_tree_id().clone()); testutils::write_normal_file(&mut tree_builder, parent_path, ""); let tree_id = tree_builder.write_tree().unwrap(); check_out_tree(&tree_id); assert!(parent_path.to_fs_path_unchecked(&workspace_root).is_file()); assert!(!child_path.to_fs_path_unchecked(&workspace_root).exists()); // Turn parent_path into directory, add file at child_path let mut tree_builder = store.tree_builder(tree_id); tree_builder.remove(parent_path.to_owned()); testutils::write_normal_file(&mut tree_builder, child_path, ""); let tree_id = tree_builder.write_tree().unwrap(); check_out_tree(&tree_id); assert!(parent_path.to_fs_path_unchecked(&workspace_root).is_dir()); assert!(child_path.to_fs_path_unchecked(&workspace_root).is_file()); // Turn parent_path back to file let mut tree_builder = store.tree_builder(tree_id); tree_builder.remove(child_path.to_owned()); testutils::write_normal_file(&mut tree_builder, parent_path, ""); let tree_id = tree_builder.write_tree().unwrap(); check_out_tree(&tree_id); assert!(parent_path.to_fs_path_unchecked(&workspace_root).is_file()); assert!(!child_path.to_fs_path_unchecked(&workspace_root).exists()); } #[test] fn test_conflicting_changes_on_disk() { let settings = testutils::user_settings(); let test_workspace = TestWorkspace::init(&settings); let repo = &test_workspace.repo; let mut ws = test_workspace.workspace; let workspace_root = ws.workspace_root().to_owned(); // file on disk conflicts with file in target commit let file_file_path = RepoPath::from_internal_string("file-file"); // file on disk conflicts with directory in target commit let file_dir_path = RepoPath::from_internal_string("file-dir"); // directory on disk conflicts with file in target commit let dir_file_path = RepoPath::from_internal_string("dir-file"); let tree = create_tree( repo, &[ (file_file_path, "committed contents"), ( &file_dir_path.join(RepoPathComponent::new("file")), "committed contents", ), (dir_file_path, "committed contents"), ], ); let commit = commit_with_tree(repo.store(), tree.id()); std::fs::write( file_file_path.to_fs_path_unchecked(&workspace_root), "contents on disk", ) .unwrap(); std::fs::write( file_dir_path.to_fs_path_unchecked(&workspace_root), "contents on disk", ) .unwrap(); std::fs::create_dir(dir_file_path.to_fs_path_unchecked(&workspace_root)).unwrap(); std::fs::write( dir_file_path .to_fs_path_unchecked(&workspace_root) .join("file"), "contents on disk", ) .unwrap(); let stats = ws.check_out(repo.op_id().clone(), None, &commit).unwrap(); assert_eq!( stats, CheckoutStats { updated_files: 0, added_files: 3, removed_files: 0, skipped_files: 3 } ); assert_eq!( std::fs::read_to_string(file_file_path.to_fs_path_unchecked(&workspace_root)).ok(), Some("contents on disk".to_string()) ); assert_eq!( std::fs::read_to_string(file_dir_path.to_fs_path_unchecked(&workspace_root)).ok(), Some("contents on disk".to_string()) ); assert_eq!( std::fs::read_to_string( dir_file_path .to_fs_path_unchecked(&workspace_root) .join("file") ) .ok(), Some("contents on disk".to_string()) ); } #[test] fn test_reset() { let settings = testutils::user_settings(); let mut test_workspace = TestWorkspace::init(&settings); let repo = &test_workspace.repo; let op_id = repo.op_id().clone(); let workspace_root = test_workspace.workspace.workspace_root().to_owned(); let ignored_path = RepoPath::from_internal_string("ignored"); let gitignore_path = RepoPath::from_internal_string(".gitignore"); let tree_without_file = create_tree(repo, &[(gitignore_path, "ignored\n")]); let commit_without_file = commit_with_tree(repo.store(), tree_without_file.id().clone()); let tree_with_file = create_tree( repo, &[(gitignore_path, "ignored\n"), (ignored_path, "code")], ); let commit_with_file = commit_with_tree(repo.store(), tree_with_file.id().clone()); let ws = &mut test_workspace.workspace; let commit = commit_with_tree(repo.store(), tree_with_file.id()); ws.check_out(repo.op_id().clone(), None, &commit).unwrap(); // Test the setup: the file should exist on disk and in the tree state. assert!(ignored_path.to_fs_path_unchecked(&workspace_root).is_file()); let wc: &LocalWorkingCopy = ws.working_copy().as_any().downcast_ref().unwrap(); assert!(wc.file_states().unwrap().contains_path(ignored_path)); // After we reset to the commit without the file, it should still exist on disk, // but it should not be in the tree state, and it should not get added when we // commit the working copy (because it's ignored). let mut locked_ws = ws.start_working_copy_mutation().unwrap(); locked_ws.locked_wc().reset(&commit_without_file).unwrap(); locked_ws.finish(op_id.clone()).unwrap(); assert!(ignored_path.to_fs_path_unchecked(&workspace_root).is_file()); let wc: &LocalWorkingCopy = ws.working_copy().as_any().downcast_ref().unwrap(); assert!(!wc.file_states().unwrap().contains_path(ignored_path)); let new_tree = test_workspace.snapshot().unwrap(); assert_eq!(new_tree.id(), tree_without_file.id()); // Now test the opposite direction: resetting to a commit where the file is // tracked. The file should become tracked (even though it's ignored). let ws = &mut test_workspace.workspace; let mut locked_ws = ws.start_working_copy_mutation().unwrap(); locked_ws.locked_wc().reset(&commit_with_file).unwrap(); locked_ws.finish(op_id.clone()).unwrap(); assert!(ignored_path.to_fs_path_unchecked(&workspace_root).is_file()); let wc: &LocalWorkingCopy = ws.working_copy().as_any().downcast_ref().unwrap(); assert!(wc.file_states().unwrap().contains_path(ignored_path)); let new_tree = test_workspace.snapshot().unwrap(); assert_eq!(new_tree.id(), tree_with_file.id()); } #[test] fn test_checkout_discard() { // Start a mutation, do a checkout, and then discard the mutation. The working // copy files should remain changed, but the state files should not be // written. let settings = testutils::user_settings(); let mut test_workspace = TestWorkspace::init(&settings); let repo = test_workspace.repo.clone(); let workspace_root = test_workspace.workspace.workspace_root().to_owned(); let file1_path = RepoPath::from_internal_string("file1"); let file2_path = RepoPath::from_internal_string("file2"); let store = repo.store(); let tree1 = create_tree(&repo, &[(file1_path, "contents")]); let tree2 = create_tree(&repo, &[(file2_path, "contents")]); let commit1 = commit_with_tree(repo.store(), tree1.id()); let commit2 = commit_with_tree(repo.store(), tree2.id()); let ws = &mut test_workspace.workspace; ws.check_out(repo.op_id().clone(), None, &commit1).unwrap(); let wc: &LocalWorkingCopy = ws.working_copy().as_any().downcast_ref().unwrap(); let state_path = wc.state_path().to_path_buf(); // Test the setup: the file should exist on disk and in the tree state. assert!(file1_path.to_fs_path_unchecked(&workspace_root).is_file()); let wc: &LocalWorkingCopy = ws.working_copy().as_any().downcast_ref().unwrap(); assert!(wc.file_states().unwrap().contains_path(file1_path)); // Start a checkout let mut locked_ws = ws.start_working_copy_mutation().unwrap(); locked_ws.locked_wc().check_out(&commit2).unwrap(); // The change should be reflected in the working copy but not saved assert!(!file1_path.to_fs_path_unchecked(&workspace_root).is_file()); assert!(file2_path.to_fs_path_unchecked(&workspace_root).is_file()); let reloaded_wc = LocalWorkingCopy::load(store.clone(), workspace_root.clone(), state_path.clone()); assert!(reloaded_wc.file_states().unwrap().contains_path(file1_path)); assert!(!reloaded_wc.file_states().unwrap().contains_path(file2_path)); drop(locked_ws); // The change should remain in the working copy, but not in memory and not saved let wc: &LocalWorkingCopy = ws.working_copy().as_any().downcast_ref().unwrap(); assert!(wc.file_states().unwrap().contains_path(file1_path)); assert!(!wc.file_states().unwrap().contains_path(file2_path)); assert!(!file1_path.to_fs_path_unchecked(&workspace_root).is_file()); assert!(file2_path.to_fs_path_unchecked(&workspace_root).is_file()); let reloaded_wc = LocalWorkingCopy::load(store.clone(), workspace_root, state_path); assert!(reloaded_wc.file_states().unwrap().contains_path(file1_path)); assert!(!reloaded_wc.file_states().unwrap().contains_path(file2_path)); } #[test] fn test_materialize_snapshot_conflicted_files() { let settings = testutils::user_settings(); let mut test_workspace = TestWorkspace::init(&settings); let repo = &test_workspace.repo.clone(); let ws = &mut test_workspace.workspace; let workspace_root = ws.workspace_root().to_owned(); // Create tree with 3-sided conflict, with file1 and file2 having different // conflicts: // file1: A - A + A - B + C // file2: A - B + C - D + D let file1_path = RepoPath::from_internal_string("file1"); let file2_path = RepoPath::from_internal_string("file2"); let side1_tree = create_tree(repo, &[(file1_path, "a\n"), (file2_path, "1\n")]); let base1_tree = create_tree(repo, &[(file1_path, "a\n"), (file2_path, "2\n")]); let side2_tree = create_tree(repo, &[(file1_path, "a\n"), (file2_path, "4\n")]); let base2_tree = create_tree(repo, &[(file1_path, "b\n"), (file2_path, "3\n")]); let side3_tree = create_tree(repo, &[(file1_path, "c\n"), (file2_path, "3\n")]); let merged_tree = side1_tree .merge(&base1_tree, &side2_tree) .unwrap() .merge(&base2_tree, &side3_tree) .unwrap(); let commit = commit_with_tree(repo.store(), merged_tree.id()); let stats = ws.check_out(repo.op_id().clone(), None, &commit).unwrap(); assert_eq!( stats, CheckoutStats { updated_files: 0, added_files: 2, removed_files: 0, skipped_files: 0 } ); // Even though the tree-level conflict is a 3-sided conflict, each file is // materialized as a 2-sided conflict. let file1_value = merged_tree.path_value(file1_path).unwrap(); let file2_value = merged_tree.path_value(file2_path).unwrap(); assert_eq!(file1_value.num_sides(), 3); assert_eq!(file2_value.num_sides(), 3); insta::assert_snapshot!( std::fs::read_to_string(file1_path.to_fs_path_unchecked(&workspace_root)).ok().unwrap(), @r###" <<<<<<< Conflict 1 of 1 %%%%%%% Changes from base to side #1 -b +a +++++++ Contents of side #2 c >>>>>>> Conflict 1 of 1 ends "###); insta::assert_snapshot!( std::fs::read_to_string(file2_path.to_fs_path_unchecked(&workspace_root)).ok().unwrap(), @r###" <<<<<<< Conflict 1 of 1 %%%%%%% Changes from base to side #1 -2 +1 +++++++ Contents of side #2 4 >>>>>>> Conflict 1 of 1 ends "###); // Editing a conflicted file should correctly propagate updates to each of // the conflicting trees. testutils::write_working_copy_file( &workspace_root, file1_path, indoc! {" <<<<<<< Conflict 1 of 1 %%%%%%% Changes from base to side #1 -b_edited +a_edited +++++++ Contents of side #2 c_edited >>>>>>> Conflict 1 of 1 ends "}, ); let edited_tree = test_workspace.snapshot().unwrap(); let edited_file_value = edited_tree.path_value(file1_path).unwrap(); let edited_file_values = edited_file_value.iter().flatten().collect_vec(); assert_eq!(edited_file_values.len(), 5); let get_file_id = |value: &TreeValue| match value { TreeValue::File { id, .. } => id.clone(), _ => panic!("unexpected value: {value:#?}"), }; // The file IDs with indices 0 and 1 are the original unedited file values // which were simplified. let edited_file_file_id_0 = get_file_id(edited_file_values[0]); assert_eq!( testutils::read_file(repo.store(), file1_path, &edited_file_file_id_0), b"a\n" ); assert_eq!(edited_file_values[0], edited_file_values[1]); let edited_file_file_id_2 = get_file_id(edited_file_values[2]); assert_eq!( testutils::read_file(repo.store(), file1_path, &edited_file_file_id_2), b"a_edited\n" ); let edited_file_file_id_3 = get_file_id(edited_file_values[3]); assert_eq!( testutils::read_file(repo.store(), file1_path, &edited_file_file_id_3), b"b_edited\n" ); let edited_file_file_id_4 = get_file_id(edited_file_values[4]); assert_eq!( testutils::read_file(repo.store(), file1_path, &edited_file_file_id_4), b"c_edited\n" ); } #[test] fn test_snapshot_racy_timestamps() { // Tests that file modifications are detected even if they happen the same // millisecond as the updated working copy state. let settings = testutils::user_settings(); let mut test_workspace = TestWorkspace::init(&settings); let repo = &test_workspace.repo; let workspace_root = test_workspace.workspace.workspace_root().to_owned(); let file_path = workspace_root.join("file"); let mut previous_tree_id = repo.store().empty_merged_tree_id(); for i in 0..100 { std::fs::write(&file_path, format!("contents {i}").as_bytes()).unwrap(); let mut locked_ws = test_workspace .workspace .start_working_copy_mutation() .unwrap(); let new_tree_id = locked_ws .locked_wc() .snapshot(&SnapshotOptions::empty_for_test()) .unwrap(); assert_ne!(new_tree_id, previous_tree_id); previous_tree_id = new_tree_id; } } #[cfg(unix)] #[test] fn test_snapshot_special_file() { // Tests that we ignore when special files (such as sockets and pipes) exist on // disk. let settings = testutils::user_settings(); let mut test_workspace = TestWorkspace::init(&settings); let workspace_root = test_workspace.workspace.workspace_root().to_owned(); let store = test_workspace.repo.store(); let ws = &mut test_workspace.workspace; let file1_path = RepoPath::from_internal_string("file1"); let file1_disk_path = file1_path.to_fs_path_unchecked(&workspace_root); std::fs::write(&file1_disk_path, "contents".as_bytes()).unwrap(); let file2_path = RepoPath::from_internal_string("file2"); let file2_disk_path = file2_path.to_fs_path_unchecked(&workspace_root); std::fs::write(file2_disk_path, "contents".as_bytes()).unwrap(); let socket_disk_path = workspace_root.join("socket"); UnixListener::bind(&socket_disk_path).unwrap(); // Test the setup assert!(socket_disk_path.exists()); assert!(!socket_disk_path.is_file()); // Snapshot the working copy with the socket file let mut locked_ws = ws.start_working_copy_mutation().unwrap(); let tree_id = locked_ws .locked_wc() .snapshot(&SnapshotOptions::empty_for_test()) .unwrap(); locked_ws.finish(OperationId::from_hex("abc123")).unwrap(); let tree = store.get_root_tree(&tree_id).unwrap(); // Only the regular files should be in the tree assert_eq!( tree.entries().map(|(path, _value)| path).collect_vec(), to_owned_path_vec(&[file1_path, file2_path]) ); let wc: &LocalWorkingCopy = ws.working_copy().as_any().downcast_ref().unwrap(); assert_eq!( wc.file_states().unwrap().paths().collect_vec(), vec![file1_path, file2_path] ); // Replace a regular file by a socket and snapshot the working copy again std::fs::remove_file(&file1_disk_path).unwrap(); UnixListener::bind(&file1_disk_path).unwrap(); let tree = test_workspace.snapshot().unwrap(); // Only the regular file should be in the tree assert_eq!( tree.entries().map(|(path, _value)| path).collect_vec(), to_owned_path_vec(&[file2_path]) ); let ws = &mut test_workspace.workspace; let wc: &LocalWorkingCopy = ws.working_copy().as_any().downcast_ref().unwrap(); assert_eq!( wc.file_states().unwrap().paths().collect_vec(), vec![file2_path] ); } #[test] fn test_gitignores() { // Tests that .gitignore files are respected. let settings = testutils::user_settings(); let mut test_workspace = TestWorkspace::init(&settings); let workspace_root = test_workspace.workspace.workspace_root().to_owned(); let gitignore_path = RepoPath::from_internal_string(".gitignore"); let added_path = RepoPath::from_internal_string("added"); let modified_path = RepoPath::from_internal_string("modified"); let removed_path = RepoPath::from_internal_string("removed"); let ignored_path = RepoPath::from_internal_string("ignored"); let subdir_modified_path = RepoPath::from_internal_string("dir/modified"); let subdir_ignored_path = RepoPath::from_internal_string("dir/ignored"); testutils::write_working_copy_file(&workspace_root, gitignore_path, "ignored\n"); testutils::write_working_copy_file(&workspace_root, modified_path, "1"); testutils::write_working_copy_file(&workspace_root, removed_path, "1"); std::fs::create_dir(workspace_root.join("dir")).unwrap(); testutils::write_working_copy_file(&workspace_root, subdir_modified_path, "1"); let tree1 = test_workspace.snapshot().unwrap(); let files1 = tree1.entries().map(|(name, _value)| name).collect_vec(); assert_eq!( files1, to_owned_path_vec(&[ gitignore_path, subdir_modified_path, modified_path, removed_path, ]) ); testutils::write_working_copy_file( &workspace_root, gitignore_path, "ignored\nmodified\nremoved\n", ); testutils::write_working_copy_file(&workspace_root, added_path, "2"); testutils::write_working_copy_file(&workspace_root, modified_path, "2"); std::fs::remove_file(removed_path.to_fs_path_unchecked(&workspace_root)).unwrap(); testutils::write_working_copy_file(&workspace_root, ignored_path, "2"); testutils::write_working_copy_file(&workspace_root, subdir_modified_path, "2"); testutils::write_working_copy_file(&workspace_root, subdir_ignored_path, "2"); let tree2 = test_workspace.snapshot().unwrap(); let files2 = tree2.entries().map(|(name, _value)| name).collect_vec(); assert_eq!( files2, to_owned_path_vec(&[ gitignore_path, added_path, subdir_modified_path, modified_path, ]) ); } #[test] fn test_gitignores_in_ignored_dir() { // Tests that .gitignore files in an ignored directory are ignored, i.e. that // they cannot override the ignores from the parent let settings = testutils::user_settings(); let mut test_workspace = TestWorkspace::init(&settings); let op_id = test_workspace.repo.op_id().clone(); let workspace_root = test_workspace.workspace.workspace_root().to_owned(); let gitignore_path = RepoPath::from_internal_string(".gitignore"); let nested_gitignore_path = RepoPath::from_internal_string("ignored/.gitignore"); let ignored_path = RepoPath::from_internal_string("ignored/file"); let tree1 = create_tree(&test_workspace.repo, &[(gitignore_path, "ignored\n")]); let commit1 = commit_with_tree(test_workspace.repo.store(), tree1.id()); let ws = &mut test_workspace.workspace; ws.check_out(op_id.clone(), None, &commit1).unwrap(); testutils::write_working_copy_file(&workspace_root, nested_gitignore_path, "!file\n"); testutils::write_working_copy_file(&workspace_root, ignored_path, "contents"); let new_tree = test_workspace.snapshot().unwrap(); assert_eq!(tree_entries(&new_tree), tree_entries(&tree1)); // The nested .gitignore is ignored even if it's tracked let tree2 = create_tree( &test_workspace.repo, &[ (gitignore_path, "ignored\n"), (nested_gitignore_path, "!file\n"), ], ); let commit2 = commit_with_tree(test_workspace.repo.store(), tree2.id().clone()); let mut locked_ws = test_workspace .workspace .start_working_copy_mutation() .unwrap(); locked_ws.locked_wc().reset(&commit2).unwrap(); locked_ws.finish(OperationId::from_hex("abc123")).unwrap(); let new_tree = test_workspace.snapshot().unwrap(); assert_eq!(tree_entries(&new_tree), tree_entries(&tree2)); } #[test] fn test_gitignores_checkout_never_overwrites_ignored() { // Tests that a .gitignore'd file doesn't get overwritten if check out a commit // where the file is tracked. let settings = testutils::user_settings(); let mut test_workspace = TestWorkspace::init(&settings); let repo = &test_workspace.repo; let workspace_root = test_workspace.workspace.workspace_root().to_owned(); // Write an ignored file called "modified" to disk let gitignore_path = RepoPath::from_internal_string(".gitignore"); testutils::write_working_copy_file(&workspace_root, gitignore_path, "modified\n"); let modified_path = RepoPath::from_internal_string("modified"); testutils::write_working_copy_file(&workspace_root, modified_path, "garbage"); // Create a tree that adds the same file but with different contents let tree = create_tree(repo, &[(modified_path, "contents")]); let commit = commit_with_tree(repo.store(), tree.id()); // Now check out the tree that adds the file "modified" with contents // "contents". The exiting contents ("garbage") shouldn't be replaced in the // working copy. let ws = &mut test_workspace.workspace; assert!(ws.check_out(repo.op_id().clone(), None, &commit).is_ok()); // Check that the old contents are in the working copy let path = workspace_root.join("modified"); assert!(path.is_file()); assert_eq!(std::fs::read(&path).unwrap(), b"garbage"); } #[test] fn test_gitignores_ignored_directory_already_tracked() { // Tests that a .gitignore'd directory that already has a tracked file in it // does not get removed when snapshotting the working directory. let settings = testutils::user_settings(); let mut test_workspace = TestWorkspace::init(&settings); let workspace_root = test_workspace.workspace.workspace_root().to_owned(); let repo = test_workspace.repo.clone(); #[derive(Clone, Copy, Debug, Eq, PartialEq)] enum Kind { Normal, Executable, Symlink, } let create_tree_with_kind = |path_contents: &[(&RepoPath, Kind, &str)]| { let store = repo.store(); let mut tree_builder = store.tree_builder(store.empty_tree_id().clone()); for (path, kind, contents) in path_contents { match kind { Kind::Normal => { testutils::write_normal_file(&mut tree_builder, path, contents); } Kind::Executable => { testutils::write_executable_file(&mut tree_builder, path, contents); } Kind::Symlink => { testutils::write_symlink(&mut tree_builder, path, contents); } } } let id = tree_builder.write_tree().unwrap(); MergedTree::resolved(store.get_tree(RepoPath::root(), &id).unwrap()) }; let gitignore_path = RepoPath::from_internal_string(".gitignore"); let unchanged_normal_path = RepoPath::from_internal_string("ignored/unchanged_normal"); let modified_normal_path = RepoPath::from_internal_string("ignored/modified_normal"); let deleted_normal_path = RepoPath::from_internal_string("ignored/deleted_normal"); let unchanged_executable_path = RepoPath::from_internal_string("ignored/unchanged_executable"); let modified_executable_path = RepoPath::from_internal_string("ignored/modified_executable"); let deleted_executable_path = RepoPath::from_internal_string("ignored/deleted_executable"); let unchanged_symlink_path = RepoPath::from_internal_string("ignored/unchanged_symlink"); let modified_symlink_path = RepoPath::from_internal_string("ignored/modified_symlink"); let deleted_symlink_path = RepoPath::from_internal_string("ignored/deleted_symlink"); let tree = create_tree_with_kind(&[ (gitignore_path, Kind::Normal, "/ignored/\n"), (unchanged_normal_path, Kind::Normal, "contents"), (modified_normal_path, Kind::Normal, "contents"), (deleted_normal_path, Kind::Normal, "contents"), (unchanged_executable_path, Kind::Executable, "contents"), (modified_executable_path, Kind::Executable, "contents"), (deleted_executable_path, Kind::Executable, "contents"), (unchanged_symlink_path, Kind::Symlink, "contents"), (modified_symlink_path, Kind::Symlink, "contents"), (deleted_symlink_path, Kind::Symlink, "contents"), ]); let commit = commit_with_tree(repo.store(), tree.id()); // Check out the tree with the files in `ignored/` let ws = &mut test_workspace.workspace; ws.check_out(repo.op_id().clone(), None, &commit).unwrap(); // Make some changes inside the ignored directory and check that they are // detected when we snapshot. The files that are still there should not be // deleted from the resulting tree. std::fs::write( modified_normal_path.to_fs_path_unchecked(&workspace_root), "modified", ) .unwrap(); std::fs::remove_file(deleted_normal_path.to_fs_path_unchecked(&workspace_root)).unwrap(); std::fs::write( modified_executable_path.to_fs_path_unchecked(&workspace_root), "modified", ) .unwrap(); std::fs::remove_file(deleted_executable_path.to_fs_path_unchecked(&workspace_root)).unwrap(); let fs_path = modified_symlink_path.to_fs_path_unchecked(&workspace_root); std::fs::remove_file(&fs_path).unwrap(); if check_symlink_support().unwrap_or(false) { try_symlink("modified", &fs_path).unwrap(); } else { std::fs::write(fs_path, "modified").unwrap(); } std::fs::remove_file(deleted_symlink_path.to_fs_path_unchecked(&workspace_root)).unwrap(); let new_tree = test_workspace.snapshot().unwrap(); let expected_tree = create_tree_with_kind(&[ (gitignore_path, Kind::Normal, "/ignored/\n"), (unchanged_normal_path, Kind::Normal, "contents"), (modified_normal_path, Kind::Normal, "modified"), (unchanged_executable_path, Kind::Executable, "contents"), (modified_executable_path, Kind::Executable, "modified"), (unchanged_symlink_path, Kind::Symlink, "contents"), (modified_symlink_path, Kind::Symlink, "modified"), ]); assert_eq!(tree_entries(&new_tree), tree_entries(&expected_tree)); } #[test] fn test_dotgit_ignored() { // Tests that .git directories and files are always ignored (we could accept // them if the backend is not git). let settings = testutils::user_settings(); let mut test_workspace = TestWorkspace::init(&settings); let store = test_workspace.repo.store().clone(); let workspace_root = test_workspace.workspace.workspace_root().to_owned(); // Test with a .git/ directory (with a file in, since we don't write empty // trees) let dotgit_path = workspace_root.join(".git"); std::fs::create_dir(&dotgit_path).unwrap(); testutils::write_working_copy_file( &workspace_root, RepoPath::from_internal_string(".git/file"), "contents", ); let new_tree = test_workspace.snapshot().unwrap(); let empty_tree_id = store.empty_merged_tree_id(); assert_eq!(new_tree.id(), empty_tree_id); std::fs::remove_dir_all(&dotgit_path).unwrap(); // Test with a .git file testutils::write_working_copy_file( &workspace_root, RepoPath::from_internal_string(".git"), "contents", ); let new_tree = test_workspace.snapshot().unwrap(); assert_eq!(new_tree.id(), empty_tree_id); } #[test] fn test_gitsubmodule() { // Tests that git submodules are ignored. let settings = testutils::user_settings(); let mut test_workspace = TestWorkspace::init_with_backend(&settings, TestRepoBackend::Git); let repo = &test_workspace.repo; let store = repo.store().clone(); let workspace_root = test_workspace.workspace.workspace_root().to_owned(); let mut tree_builder = store.tree_builder(store.empty_tree_id().clone()); let added_path = RepoPath::from_internal_string("added"); let submodule_path = RepoPath::from_internal_string("submodule"); let added_submodule_path = RepoPath::from_internal_string("submodule/added"); tree_builder.set( added_path.to_owned(), TreeValue::File { id: testutils::write_file(repo.store(), added_path, "added\n"), executable: false, }, ); let mut tx = repo.start_transaction(&settings); let submodule_id = write_random_commit(tx.repo_mut(), &settings).id().clone(); tx.commit("create submodule commit"); tree_builder.set( submodule_path.to_owned(), TreeValue::GitSubmodule(submodule_id), ); let tree_id = MergedTreeId::Legacy(tree_builder.write_tree().unwrap()); let tree = store.get_root_tree(&tree_id).unwrap(); let commit = commit_with_tree(repo.store(), tree.id()); let ws = &mut test_workspace.workspace; ws.check_out(repo.op_id().clone(), None, &commit).unwrap(); std::fs::create_dir(submodule_path.to_fs_path_unchecked(&workspace_root)).unwrap(); testutils::write_working_copy_file( &workspace_root, added_submodule_path, "i am a file in a submodule\n", ); // Check that the files present in the submodule are not tracked // when we snapshot let new_tree = test_workspace.snapshot().unwrap(); assert_eq!(new_tree.id(), tree_id); // Check that the files in the submodule are not deleted let file_in_submodule_path = added_submodule_path.to_fs_path_unchecked(&workspace_root); assert!( file_in_submodule_path.metadata().is_ok(), "{file_in_submodule_path:?} should exist" ); } #[test] fn test_check_out_existing_file_cannot_be_removed() { let settings = testutils::user_settings(); let mut test_workspace = TestWorkspace::init(&settings); let repo = &test_workspace.repo; let workspace_root = test_workspace.workspace.workspace_root().to_owned(); let file_path = RepoPath::from_internal_string("file"); let tree1 = create_tree(repo, &[(file_path, "0")]); let tree2 = create_tree(repo, &[(file_path, "1")]); let commit1 = commit_with_tree(repo.store(), tree1.id()); let commit2 = commit_with_tree(repo.store(), tree2.id()); let ws = &mut test_workspace.workspace; ws.check_out(repo.op_id().clone(), None, &commit1).unwrap(); // Trigger IO error by replacing file with directory. std::fs::remove_file(file_path.to_fs_path_unchecked(&workspace_root)).unwrap(); std::fs::create_dir(file_path.to_fs_path_unchecked(&workspace_root)).unwrap(); let result = ws.check_out(repo.op_id().clone(), None, &commit2); assert_matches!( result, Err(CheckoutError::Other { message, .. }) if message.contains("Failed to remove") ); } #[test] fn test_check_out_existing_directory_symlink() { if !check_symlink_support().unwrap() { eprintln!("Skipping test because symlink isn't supported"); return; } let settings = testutils::user_settings(); let mut test_workspace = TestWorkspace::init(&settings); let repo = &test_workspace.repo; let workspace_root = test_workspace.workspace.workspace_root().to_owned(); // Creates a symlink in working directory, and a tree that will add a file // under the symlinked directory. try_symlink("..", workspace_root.join("parent")).unwrap(); let file_path = RepoPath::from_internal_string("parent/escaped"); let tree = create_tree(repo, &[(file_path, "contents")]); let commit = commit_with_tree(repo.store(), tree.id()); // Checkout doesn't fail, but the file should be skipped. let ws = &mut test_workspace.workspace; let stats = ws.check_out(repo.op_id().clone(), None, &commit).unwrap(); assert_eq!(stats.skipped_files, 1); // Therefore, "../escaped" shouldn't be created. assert!(!workspace_root.parent().unwrap().join("escaped").exists()); } #[test] fn test_check_out_existing_directory_symlink_icase_fs() { if !check_symlink_support().unwrap() { eprintln!("Skipping test because symlink isn't supported"); return; } let settings = testutils::user_settings(); let mut test_workspace = TestWorkspace::init(&settings); let repo = &test_workspace.repo; let workspace_root = test_workspace.workspace.workspace_root().to_owned(); let is_icase_fs = check_icase_fs(&workspace_root); // Creates a symlink in working directory, and a tree that will add a file // under the symlinked directory. try_symlink("..", workspace_root.join("parent")).unwrap(); let file_path = RepoPath::from_internal_string("PARENT/escaped"); let tree = create_tree(repo, &[(file_path, "contents")]); let commit = commit_with_tree(repo.store(), tree.id()); // Checkout doesn't fail, but the file should be skipped on icase fs. let ws = &mut test_workspace.workspace; let stats = ws.check_out(repo.op_id().clone(), None, &commit).unwrap(); if is_icase_fs { assert_eq!(stats.skipped_files, 1); } else { assert_eq!(stats.skipped_files, 0); } // Therefore, "../escaped" shouldn't be created. assert!(!workspace_root.parent().unwrap().join("escaped").exists()); } #[test_case(false; "symlink target does not exist")] #[test_case(true; "symlink target exists")] fn test_check_out_existing_file_symlink_icase_fs(victim_exists: bool) { if !check_symlink_support().unwrap() { eprintln!("Skipping test because symlink isn't supported"); return; } let settings = testutils::user_settings(); let mut test_workspace = TestWorkspace::init(&settings); let repo = &test_workspace.repo; let workspace_root = test_workspace.workspace.workspace_root().to_owned(); let is_icase_fs = check_icase_fs(&workspace_root); // Creates a symlink in working directory, and a tree that will overwrite // the symlink content. try_symlink( PathBuf::from_iter(["..", "pwned"]), workspace_root.join("parent"), ) .unwrap(); let victim_file_path = workspace_root.parent().unwrap().join("pwned"); if victim_exists { std::fs::write(&victim_file_path, "old").unwrap(); } assert_eq!(workspace_root.join("parent").exists(), victim_exists); let file_path = RepoPath::from_internal_string("PARENT"); let tree = create_tree(repo, &[(file_path, "bad")]); let commit = commit_with_tree(repo.store(), tree.id()); // Checkout doesn't fail, but the file should be skipped on icase fs. let ws = &mut test_workspace.workspace; let stats = ws.check_out(repo.op_id().clone(), None, &commit).unwrap(); if is_icase_fs { assert_eq!(stats.skipped_files, 1); } else { assert_eq!(stats.skipped_files, 0); } // Therefore, "../pwned" shouldn't be updated. if victim_exists { assert_eq!(std::fs::read(&victim_file_path).unwrap(), b"old"); } else { assert!(!victim_file_path.exists()); } } #[test] fn test_check_out_file_removal_over_existing_directory_symlink() { if !check_symlink_support().unwrap() { eprintln!("Skipping test because symlink isn't supported"); return; } let settings = testutils::user_settings(); let mut test_workspace = TestWorkspace::init(&settings); let repo = &test_workspace.repo; let workspace_root = test_workspace.workspace.workspace_root().to_owned(); let file_path = RepoPath::from_internal_string("parent/escaped"); let tree1 = create_tree(repo, &[(file_path, "contents")]); let tree2 = create_tree(repo, &[]); let commit1 = commit_with_tree(repo.store(), tree1.id()); let commit2 = commit_with_tree(repo.store(), tree2.id()); // Check out "parent/escaped". let ws = &mut test_workspace.workspace; ws.check_out(repo.op_id().clone(), None, &commit1).unwrap(); // Pretend that "parent" was a symlink, which might be created by // e.g. checking out "PARENT" on case-insensitive fs. The file // "parent/escaped" would be skipped in that case. std::fs::remove_file(file_path.to_fs_path_unchecked(&workspace_root)).unwrap(); std::fs::remove_dir(workspace_root.join("parent")).unwrap(); try_symlink("..", workspace_root.join("parent")).unwrap(); let victim_file_path = workspace_root.parent().unwrap().join("escaped"); std::fs::write(&victim_file_path, "").unwrap(); assert!(file_path.to_fs_path_unchecked(&workspace_root).exists()); // Check out empty tree, which tries to remove "parent/escaped". let stats = ws.check_out(repo.op_id().clone(), None, &commit2).unwrap(); assert_eq!(stats.skipped_files, 1); // "../escaped" shouldn't be removed. assert!(victim_file_path.exists()); } #[test_case("../pwned"; "escape from root")] #[test_case("sub/../../pwned"; "escape from sub dir")] fn test_check_out_malformed_file_path(file_path_str: &str) { let settings = testutils::user_settings(); let mut test_workspace = TestWorkspace::init(&settings); let repo = &test_workspace.repo; let workspace_root = test_workspace.workspace.workspace_root().to_owned(); let file_path = RepoPath::from_internal_string(file_path_str); let tree = create_tree(repo, &[(file_path, "contents")]); let commit = commit_with_tree(repo.store(), tree.id()); // Checkout should fail let ws = &mut test_workspace.workspace; let result = ws.check_out(repo.op_id().clone(), None, &commit); assert_matches!(result, Err(CheckoutError::InvalidRepoPath(_))); // Therefore, "pwned" file shouldn't be created. assert!(!workspace_root.join(file_path_str).exists()); assert!(!workspace_root.parent().unwrap().join("pwned").exists()); } #[test_case(r"sub\..\../pwned"; "path separator")] #[test_case("d:/pwned"; "drive letter")] fn test_check_out_malformed_file_path_windows(file_path_str: &str) { let settings = testutils::user_settings(); let mut test_workspace = TestWorkspace::init(&settings); let repo = &test_workspace.repo; let workspace_root = test_workspace.workspace.workspace_root().to_owned(); let file_path = RepoPath::from_internal_string(file_path_str); let tree = create_tree(repo, &[(file_path, "contents")]); let commit = commit_with_tree(repo.store(), tree.id()); // Checkout should fail on Windows let ws = &mut test_workspace.workspace; let result = ws.check_out(repo.op_id().clone(), None, &commit); if cfg!(windows) { assert_matches!(result, Err(CheckoutError::InvalidRepoPath(_))); } else { assert_matches!(result, Ok(_)); } // Therefore, "pwned" file shouldn't be created. if cfg!(windows) { assert!(!workspace_root.join(file_path_str).exists()); } assert!(!workspace_root.parent().unwrap().join("pwned").exists()); } #[test_case(".git"; "root .git file")] #[test_case(".jj"; "root .jj file")] #[test_case(".git/pwned"; "root .git dir")] #[test_case(".jj/pwned"; "root .jj dir")] #[test_case("sub/.git"; "sub .git file")] #[test_case("sub/.jj"; "sub .jj file")] #[test_case("sub/.git/pwned"; "sub .git dir")] #[test_case("sub/.jj/pwned"; "sub .jj dir")] fn test_check_out_reserved_file_path(file_path_str: &str) { let settings = testutils::user_settings(); let mut test_workspace = TestWorkspace::init(&settings); let repo = &test_workspace.repo; let workspace_root = test_workspace.workspace.workspace_root().to_owned(); std::fs::create_dir(workspace_root.join(".git")).unwrap(); let file_path = RepoPath::from_internal_string(file_path_str); let disk_path = file_path.to_fs_path_unchecked(&workspace_root); let tree1 = create_tree(repo, &[(file_path, "contents")]); let tree2 = create_tree(repo, &[]); let commit1 = commit_with_tree(repo.store(), tree1.id()); let commit2 = commit_with_tree(repo.store(), tree2.id()); // Checkout should fail. let ws = &mut test_workspace.workspace; let result = ws.check_out(repo.op_id().clone(), None, &commit1); assert_matches!(result, Err(CheckoutError::ReservedPathComponent { .. })); // Therefore, "pwned" file shouldn't be created. if ![".git", ".jj"].contains(&file_path_str) { assert!(!disk_path.exists()); } assert!(!workspace_root.join(".git").join("pwned").exists()); assert!(!workspace_root.join(".jj").join("pwned").exists()); assert!(!workspace_root.join("sub").join(".git").exists()); assert!(!workspace_root.join("sub").join(".jj").exists()); // Pretend that the checkout somehow succeeded. let mut locked_ws = ws.start_working_copy_mutation().unwrap(); locked_ws.locked_wc().reset(&commit1).unwrap(); locked_ws.finish(repo.op_id().clone()).unwrap(); if ![".git", ".jj"].contains(&file_path_str) { std::fs::create_dir_all(disk_path.parent().unwrap()).unwrap(); std::fs::write(&disk_path, "").unwrap(); } // Check out empty tree, which tries to remove the file. let result = ws.check_out(repo.op_id().clone(), None, &commit2); assert_matches!(result, Err(CheckoutError::ReservedPathComponent { .. })); // The existing file shouldn't be removed. assert!(disk_path.exists()); } #[test_case(".Git/pwned"; "root .git dir")] #[test_case(".jJ/pwned"; "root .jj dir")] #[test_case("sub/.GIt"; "sub .git file")] #[test_case("sub/.JJ"; "sub .jj file")] #[test_case("sub/.gIT/pwned"; "sub .git dir")] #[test_case("sub/.Jj/pwned"; "sub .jj dir")] fn test_check_out_reserved_file_path_icase_fs(file_path_str: &str) { let settings = testutils::user_settings(); let mut test_workspace = TestWorkspace::init(&settings); let repo = &test_workspace.repo; let workspace_root = test_workspace.workspace.workspace_root().to_owned(); std::fs::create_dir(workspace_root.join(".git")).unwrap(); let is_icase_fs = check_icase_fs(&workspace_root); let file_path = RepoPath::from_internal_string(file_path_str); let disk_path = file_path.to_fs_path_unchecked(&workspace_root); let tree1 = create_tree(repo, &[(file_path, "contents")]); let tree2 = create_tree(repo, &[]); let commit1 = commit_with_tree(repo.store(), tree1.id()); let commit2 = commit_with_tree(repo.store(), tree2.id()); // Checkout should fail on icase fs. let ws = &mut test_workspace.workspace; let result = ws.check_out(repo.op_id().clone(), None, &commit1); if is_icase_fs { assert_matches!(result, Err(CheckoutError::ReservedPathComponent { .. })); } else { assert_matches!(result, Ok(_)); } // Therefore, "pwned" file shouldn't be created. if is_icase_fs { assert!(!disk_path.exists()); } assert!(!workspace_root.join(".git").join("pwned").exists()); assert!(!workspace_root.join(".jj").join("pwned").exists()); assert!(!workspace_root.join("sub").join(".git").exists()); assert!(!workspace_root.join("sub").join(".jj").exists()); // Pretend that the checkout somehow succeeded. let mut locked_ws = ws.start_working_copy_mutation().unwrap(); locked_ws.locked_wc().reset(&commit1).unwrap(); locked_ws.finish(repo.op_id().clone()).unwrap(); std::fs::create_dir_all(disk_path.parent().unwrap()).unwrap(); std::fs::write(&disk_path, "").unwrap(); // Check out empty tree, which tries to remove the file. let result = ws.check_out(repo.op_id().clone(), None, &commit2); if is_icase_fs { assert_matches!(result, Err(CheckoutError::ReservedPathComponent { .. })); } else { assert_matches!(result, Ok(_)); } // The existing file shouldn't be removed on icase fs. if is_icase_fs { assert!(disk_path.exists()); } } // Here we don't test ignored characters exhaustively because our implementation // isn't using deny list. #[test_case("\u{200c}.git/pwned"; "root .git dir")] #[test_case(".\u{200d}jj/pwned"; "root .jj dir")] #[test_case("sub/.g\u{200c}it"; "sub .git file")] #[test_case("sub/.jj\u{200d}"; "sub .jj file")] #[test_case("sub/.gi\u{200e}t/pwned"; "sub .git dir")] #[test_case("sub/.jj\u{200f}/pwned"; "sub .jj dir")] fn test_check_out_reserved_file_path_hfs_plus(file_path_str: &str) { let settings = testutils::user_settings(); let mut test_workspace = TestWorkspace::init(&settings); let repo = &test_workspace.repo; let workspace_root = test_workspace.workspace.workspace_root().to_owned(); std::fs::create_dir(workspace_root.join(".git")).unwrap(); let is_hfs_plus = check_hfs_plus(&workspace_root); let file_path = RepoPath::from_internal_string(file_path_str); let disk_path = file_path.to_fs_path_unchecked(&workspace_root); let tree1 = create_tree(repo, &[(file_path, "contents")]); let tree2 = create_tree(repo, &[]); let commit1 = commit_with_tree(repo.store(), tree1.id()); let commit2 = commit_with_tree(repo.store(), tree2.id()); // Checkout should fail on HFS+-like fs. let ws = &mut test_workspace.workspace; let result = ws.check_out(repo.op_id().clone(), None, &commit1); if is_hfs_plus { assert_matches!(result, Err(CheckoutError::ReservedPathComponent { .. })); } else { assert_matches!(result, Ok(_)); } // Therefore, "pwned" file shouldn't be created. if is_hfs_plus { assert!(!disk_path.exists()); } assert!(!workspace_root.join(".git").join("pwned").exists()); assert!(!workspace_root.join(".jj").join("pwned").exists()); assert!(!workspace_root.join("sub").join(".git").exists()); assert!(!workspace_root.join("sub").join(".jj").exists()); // Pretend that the checkout somehow succeeded. let mut locked_ws = ws.start_working_copy_mutation().unwrap(); locked_ws.locked_wc().reset(&commit1).unwrap(); locked_ws.finish(repo.op_id().clone()).unwrap(); std::fs::create_dir_all(disk_path.parent().unwrap()).unwrap(); std::fs::write(&disk_path, "").unwrap(); // Check out empty tree, which tries to remove the file. let result = ws.check_out(repo.op_id().clone(), None, &commit2); if is_hfs_plus { assert_matches!(result, Err(CheckoutError::ReservedPathComponent { .. })); } else { assert_matches!(result, Ok(_)); } // The existing file shouldn't be removed on HFS+-like fs. if is_hfs_plus { assert!(disk_path.exists()); } } #[test_case(".git/pwned", &["GIT~1/pwned", "GI2837~1/pwned"]; "root .git dir short name")] #[test_case(".jj/pwned", &["JJ~1/pwned", "JJ2E09~1/pwned"]; "root .jj dir short name")] #[test_case(".git/pwned", &[".GIT./pwned"]; "root .git dir trailing dots")] #[test_case(".jj/pwned", &[".JJ../pwned"]; "root .jj dir trailing dots")] #[test_case("sub/.git", &["sub/.GIT.."]; "sub .git file trailing dots")] #[test_case("sub/.jj", &["sub/.JJ."]; "sub .jj file trailing dots")] // TODO: Add more weird patterns? // - https://en.wikipedia.org/wiki/8.3_filename // - See is_ntfs_dotgit() of Git and pathauditor of Mercurial fn test_check_out_reserved_file_path_vfat(vfat_path_str: &str, file_path_strs: &[&str]) { let settings = testutils::user_settings(); let mut test_workspace = TestWorkspace::init(&settings); let repo = &test_workspace.repo; let workspace_root = test_workspace.workspace.workspace_root().to_owned(); std::fs::create_dir(workspace_root.join(".git")).unwrap(); let is_vfat = check_vfat(&workspace_root); let vfat_disk_path = workspace_root.join(vfat_path_str); let file_paths = file_path_strs .iter() .map(|&s| RepoPath::from_internal_string(s)) .collect_vec(); let tree1 = create_tree( repo, &file_paths .iter() .map(|&path| (path, "contents")) .collect_vec(), ); let tree2 = create_tree(repo, &[]); let commit1 = commit_with_tree(repo.store(), tree1.id()); let commit2 = commit_with_tree(repo.store(), tree2.id()); // Checkout should fail on VFAT-like fs. let ws = &mut test_workspace.workspace; let result = ws.check_out(repo.op_id().clone(), None, &commit1); if is_vfat { assert_matches!(result, Err(CheckoutError::ReservedPathComponent { .. })); } else { assert_matches!(result, Ok(_)); } // Therefore, "pwned" file shouldn't be created. if is_vfat { assert!(!vfat_disk_path.exists()); } assert!(!workspace_root.join(".git").join("pwned").exists()); assert!(!workspace_root.join(".jj").join("pwned").exists()); assert!(!workspace_root.join("sub").join(".git").exists()); assert!(!workspace_root.join("sub").join(".jj").exists()); // Pretend that the checkout somehow succeeded. let mut locked_ws = ws.start_working_copy_mutation().unwrap(); locked_ws.locked_wc().reset(&commit1).unwrap(); locked_ws.finish(repo.op_id().clone()).unwrap(); if is_vfat { std::fs::create_dir_all(vfat_disk_path.parent().unwrap()).unwrap(); std::fs::write(&vfat_disk_path, "").unwrap(); } // Check out empty tree, which tries to remove the file. let result = ws.check_out(repo.op_id().clone(), None, &commit2); if is_vfat { assert_matches!(result, Err(CheckoutError::ReservedPathComponent { .. })); } else { assert_matches!(result, Ok(_)); } // The existing file shouldn't be removed on VFAT-like fs. if is_vfat { assert!(vfat_disk_path.exists()); } } #[test] fn test_fsmonitor() { let settings = testutils::user_settings(); let mut test_workspace = TestWorkspace::init(&settings); let repo = &test_workspace.repo; let workspace_root = test_workspace.workspace.workspace_root().to_owned(); let ws = &mut test_workspace.workspace; assert_eq!( ws.working_copy().sparse_patterns().unwrap(), vec![RepoPathBuf::root()] ); let foo_path = RepoPath::from_internal_string("foo"); let bar_path = RepoPath::from_internal_string("bar"); let nested_path = RepoPath::from_internal_string("path/to/nested"); testutils::write_working_copy_file(&workspace_root, foo_path, "foo\n"); testutils::write_working_copy_file(&workspace_root, bar_path, "bar\n"); testutils::write_working_copy_file(&workspace_root, nested_path, "nested\n"); let ignored_path = RepoPath::from_internal_string("path/to/ignored"); let gitignore_path = RepoPath::from_internal_string("path/.gitignore"); testutils::write_working_copy_file(&workspace_root, ignored_path, "ignored\n"); testutils::write_working_copy_file(&workspace_root, gitignore_path, "to/ignored\n"); let snapshot = |locked_ws: &mut LockedWorkspace, paths: &[&RepoPath]| { let fs_paths = paths .iter() .map(|p| p.to_fs_path_unchecked(Path::new(""))) .collect(); locked_ws .locked_wc() .snapshot(&SnapshotOptions { fsmonitor_settings: FsmonitorSettings::Test { changed_files: fs_paths, }, ..SnapshotOptions::empty_for_test() }) .unwrap() }; { let mut locked_ws = ws.start_working_copy_mutation().unwrap(); let tree_id = snapshot(&mut locked_ws, &[]); assert_eq!(tree_id, repo.store().empty_merged_tree_id()); } { let mut locked_ws = ws.start_working_copy_mutation().unwrap(); let tree_id = snapshot(&mut locked_ws, &[foo_path]); insta::assert_snapshot!(testutils::dump_tree(repo.store(), &tree_id), @r###" tree d5e38c0a1b0ee5de47c5 file "foo" (e99c2057c15160add351): "foo\n" "###); } { let mut locked_ws = ws.start_working_copy_mutation().unwrap(); let tree_id = snapshot( &mut locked_ws, &[foo_path, bar_path, nested_path, ignored_path], ); insta::assert_snapshot!(testutils::dump_tree(repo.store(), &tree_id), @r###" tree f408c8d080414f8e90e1 file "bar" (94cc973e7e1aefb7eff6): "bar\n" file "foo" (e99c2057c15160add351): "foo\n" file "path/to/nested" (6209060941cd770c8d46): "nested\n" "###); locked_ws.finish(repo.op_id().clone()).unwrap(); } { testutils::write_working_copy_file(&workspace_root, foo_path, "updated foo\n"); testutils::write_working_copy_file(&workspace_root, bar_path, "updated bar\n"); let mut locked_ws = ws.start_working_copy_mutation().unwrap(); let tree_id = snapshot(&mut locked_ws, &[foo_path]); insta::assert_snapshot!(testutils::dump_tree(repo.store(), &tree_id), @r###" tree e994a93c46f41dc91704 file "bar" (94cc973e7e1aefb7eff6): "bar\n" file "foo" (e0fbd106147cc04ccd05): "updated foo\n" file "path/to/nested" (6209060941cd770c8d46): "nested\n" "###); } { std::fs::remove_file(foo_path.to_fs_path_unchecked(&workspace_root)).unwrap(); let mut locked_ws = ws.start_working_copy_mutation().unwrap(); let tree_id = snapshot(&mut locked_ws, &[foo_path]); insta::assert_snapshot!(testutils::dump_tree(repo.store(), &tree_id), @r###" tree 1df764981d4d74a4ecfa file "bar" (94cc973e7e1aefb7eff6): "bar\n" file "path/to/nested" (6209060941cd770c8d46): "nested\n" "###); locked_ws.finish(repo.op_id().clone()).unwrap(); } } #[test] fn test_snapshot_max_new_file_size() { let settings = testutils::user_settings(); let mut test_workspace = TestWorkspace::init(&settings); let workspace_root = test_workspace.workspace.workspace_root().to_owned(); let small_path = RepoPath::from_internal_string("small"); let large_path = RepoPath::from_internal_string("large"); let limit: usize = 1024; std::fs::write( small_path.to_fs_path_unchecked(&workspace_root), vec![0; limit], ) .unwrap(); let options = SnapshotOptions { max_new_file_size: limit as u64, ..SnapshotOptions::empty_for_test() }; test_workspace .snapshot_with_options(&options) .expect("files exactly matching the size limit should succeed"); std::fs::write( small_path.to_fs_path_unchecked(&workspace_root), vec![0; limit + 1], ) .unwrap(); test_workspace .snapshot_with_options(&options) .expect("existing files may grow beyond the size limit"); // A new file of 1KiB + 1 bytes should fail std::fs::write( large_path.to_fs_path_unchecked(&workspace_root), vec![0; limit + 1], ) .unwrap(); let err = test_workspace .snapshot_with_options(&options) .expect_err("new files beyond the size limit should fail"); assert!( matches!(err, SnapshotError::NewFileTooLarge { .. }), "the failure should be attributed to new file size" ); }