local_working_copy: when all sides of a conflict are executable, materialise the conflicted file as executable

Fixes #3579 and adds a testcase for an executable conflict treevalue.
This commit is contained in:
Thomas Castiglione 2024-04-26 12:42:20 +08:00 committed by gulbanana
parent 5041649cbb
commit 59d3a2c866
4 changed files with 68 additions and 7 deletions

View file

@ -40,6 +40,9 @@ to avoid letting the user edit the immutable one.
* `jj config list` now properly escapes TOML keys (#1322). * `jj config list` now properly escapes TOML keys (#1322).
* Files with conflicts are now checked out as executable if all sides of the
conflict are executable.
## [0.17.1] - 2024-05-07 ## [0.17.1] - 2024-05-07
### Fixed bugs ### Fixed bugs

View file

@ -1152,10 +1152,15 @@ impl TreeState {
.block_on()?; .block_on()?;
match new_file_ids.into_resolved() { match new_file_ids.into_resolved() {
Ok(file_id) => { Ok(file_id) => {
// On Windows, we preserve the executable bit from the merged trees.
#[cfg(windows)] #[cfg(windows)]
let executable = { let executable = {
let () = executable; // use the variable let () = executable; // use the variable
if let Some(merge) = current_tree_values.to_executable_merge() {
merge.resolve_trivial().copied().unwrap_or_default()
} else {
false false
}
}; };
Ok(Merge::normal(TreeValue::File { Ok(Merge::normal(TreeValue::File {
id: file_id.unwrap(), id: file_id.unwrap(),
@ -1224,6 +1229,7 @@ impl TreeState {
&self, &self,
disk_path: &Path, disk_path: &Path,
conflict_data: Vec<u8>, conflict_data: Vec<u8>,
executable: bool,
) -> Result<FileState, CheckoutError> { ) -> Result<FileState, CheckoutError> {
let mut file = OpenOptions::new() let mut file = OpenOptions::new()
.write(true) .write(true)
@ -1239,12 +1245,11 @@ impl TreeState {
err: err.into(), err: err.into(),
})?; })?;
let size = conflict_data.len() as u64; let size = conflict_data.len() as u64;
// TODO: Set the executable bit correctly (when possible) and preserve that on self.set_executable(disk_path, executable)?;
// Windows like we do with the executable bit for regular files.
let metadata = file let metadata = file
.metadata() .metadata()
.map_err(|err| checkout_error_for_stat_error(err, disk_path))?; .map_err(|err| checkout_error_for_stat_error(err, disk_path))?;
Ok(FileState::for_file(false, size, &metadata)) Ok(FileState::for_file(executable, size, &metadata))
} }
#[cfg_attr(windows, allow(unused_variables))] #[cfg_attr(windows, allow(unused_variables))]
@ -1393,8 +1398,13 @@ impl TreeState {
MaterializedTreeValue::Tree(_) => { MaterializedTreeValue::Tree(_) => {
panic!("unexpected tree entry in diff at {path:?}"); panic!("unexpected tree entry in diff at {path:?}");
} }
MaterializedTreeValue::Conflict { id: _, contents } => { MaterializedTreeValue::Conflict { id, contents } => {
self.write_conflict(&disk_path, contents)? let executable = if let Some(merge) = id.to_executable_merge() {
merge.resolve_trivial().copied().unwrap_or_default()
} else {
false
};
self.write_conflict(&disk_path, contents, executable)?
} }
}; };
changed_file_states.push((path, file_state)); changed_file_states.push((path, file_state));

View file

@ -512,6 +512,16 @@ impl MergedTreeValue {
}) })
} }
/// If this merge contains only files or absent entries, returns a merge of
/// the files' executable bits.
pub fn to_executable_merge(&self) -> Option<Merge<bool>> {
self.maybe_map(|term| match term {
None => Some(false),
Some(TreeValue::File { id: _, executable }) => Some(*executable),
_ => None,
})
}
/// Creates a new merge with the file ids from the given merge. In other /// Creates a new merge with the file ids from the given merge. In other
/// words, only the executable bits from `self` will be preserved. /// words, only the executable bits from `self` will be preserved.
pub fn with_new_file_ids(&self, file_ids: &Merge<Option<FileId>>) -> Self { pub fn with_new_file_ids(&self, file_ids: &Merge<Option<FileId>>) -> Self {

View file

@ -87,6 +87,8 @@ fn test_checkout_file_transitions(backend: TestRepoBackend) {
// Executable, but same content as Normal, to test transition where only the bit changed // Executable, but same content as Normal, to test transition where only the bit changed
ExecutableNormalContent, ExecutableNormalContent,
Conflict, Conflict,
// Same content as Executable, to test that transition preserves the executable bit
ConflictedExecutableContent,
Symlink, Symlink,
Tree, Tree,
GitSubmodule, GitSubmodule,
@ -110,7 +112,8 @@ fn test_checkout_file_transitions(backend: TestRepoBackend) {
}) })
} }
Kind::Executable => { Kind::Executable => {
let id = testutils::write_file(store, path, "executable file contents"); let id: jj_lib::backend::FileId =
testutils::write_file(store, path, "executable file contents");
Merge::normal(TreeValue::File { Merge::normal(TreeValue::File {
id, id,
executable: true, executable: true,
@ -144,6 +147,29 @@ fn test_checkout_file_transitions(backend: TestRepoBackend) {
], ],
) )
} }
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 => { Kind::Symlink => {
let id = store.write_symlink(path, "target").unwrap(); let id = store.write_symlink(path, "target").unwrap();
Merge::normal(TreeValue::Symlink(id)) Merge::normal(TreeValue::Symlink(id))
@ -174,6 +200,7 @@ fn test_checkout_file_transitions(backend: TestRepoBackend) {
Kind::Executable, Kind::Executable,
Kind::ExecutableNormalContent, Kind::ExecutableNormalContent,
Kind::Conflict, Kind::Conflict,
Kind::ConflictedExecutableContent,
Kind::Tree, Kind::Tree,
]; ];
kinds.push(Kind::Symlink); kinds.push(Kind::Symlink);
@ -246,6 +273,17 @@ fn test_checkout_file_transitions(backend: TestRepoBackend) {
"{path:?} should not be executable" "{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 => { Kind::Symlink => {
assert!(maybe_metadata.is_ok(), "{path:?} should exist"); assert!(maybe_metadata.is_ok(), "{path:?} should exist");
let metadata = maybe_metadata.unwrap(); let metadata = maybe_metadata.unwrap();