ok/jj
1
0
Fork 0
forked from mirrors/jj

working_copy: preserve executable bit on Windows

Windows doesn't support recording the executable bit in the file
system. Before this commit, the code for reading and writing the
executable wouldn't even compile on Windows. This commit at least
makes it so we preserve whatever bit has been recorded in the repo.
At least I hope that's what it does -- I don't have access to a
Windows machine right now.
This commit is contained in:
Martin von Zweigbergk 2021-01-04 11:18:16 -08:00
parent d4aed83aa6
commit 302c66825f

View file

@ -69,6 +69,14 @@ impl FileState {
size: 0, size: 0,
} }
} }
fn mark_executable(&mut self, executable: bool) {
match (&self.file_type, executable) {
(FileType::Normal, true) => self.file_type = FileType::Executable,
(FileType::Executable, false) => self.file_type = FileType::Normal,
_ => {}
}
}
} }
pub struct TreeState { pub struct TreeState {
@ -306,26 +314,38 @@ impl TreeState {
let disk_file = disk_dir.join(file_name); let disk_file = disk_dir.join(file_name);
deleted_files.remove(&file); deleted_files.remove(&file);
let new_file_state = self.file_state(&entry.path()).unwrap(); let new_file_state = self.file_state(&entry.path()).unwrap();
let clean = match self.file_states.get(&file) { let clean;
let executable;
match self.file_states.get(&file) {
None => { None => {
// untracked // untracked
if git_repo.status_should_ignore(&disk_file).unwrap() { if git_repo.status_should_ignore(&disk_file).unwrap() {
continue; continue;
} }
false clean = false;
executable = new_file_state.file_type == FileType::Executable;
} }
Some(current_entry) => { Some(current_entry) => {
current_entry == &new_file_state && current_entry.mtime < self.read_time clean = current_entry == &new_file_state
&& current_entry.mtime < self.read_time;
#[cfg(windows)]
{
// On Windows, we preserve the state we had recorded
// when we wrote the file.
executable = current_entry.file_type == FileType::Executable
}
#[cfg(not(windows))]
{
executable = new_file_state.file_type == FileType::Executable
}
} }
}; };
if !clean { if !clean {
let file_value = match new_file_state.file_type { let file_value = match new_file_state.file_type {
FileType::Normal | FileType::Executable => { FileType::Normal | FileType::Executable => {
let id = self.write_file_to_store(&file, &disk_file); let id = self.write_file_to_store(&file, &disk_file);
TreeValue::Normal { TreeValue::Normal { id, executable }
id,
executable: new_file_state.file_type == FileType::Executable,
}
} }
FileType::Symlink => { FileType::Symlink => {
let id = self.write_symlink_to_store(&file, &disk_file); let id = self.write_symlink_to_store(&file, &disk_file);
@ -374,7 +394,12 @@ impl TreeState {
// the file exists, and the stat information is most likely accurate, // the file exists, and the stat information is most likely accurate,
// except for other processes modifying the file concurrently (The mtime is set // except for other processes modifying the file concurrently (The mtime is set
// at write time and won't change when we close the file.) // at write time and won't change when we close the file.)
self.file_state(&disk_path).unwrap() let mut file_state = self.file_state(&disk_path).unwrap();
// Make sure the state we record is what we tried to set above. This is mostly
// for Windows, since the executable bit is not reflected in the file system
// there.
file_state.mark_executable(executable);
file_state
} }
fn write_symlink(&self, disk_path: &PathBuf, path: &FileRepoPath, id: &SymlinkId) -> FileState { fn write_symlink(&self, disk_path: &PathBuf, path: &FileRepoPath, id: &SymlinkId) -> FileState {
@ -393,9 +418,16 @@ impl TreeState {
} }
fn set_executable(&self, disk_path: &PathBuf, executable: bool) { fn set_executable(&self, disk_path: &PathBuf, executable: bool) {
#[cfg(windows)]
{
return;
}
#[cfg(not(windows))]
{
let mode = if executable { 0o755 } else { 0o644 }; let mode = if executable { 0o755 } else { 0o644 };
fs::set_permissions(disk_path, fs::Permissions::from_mode(mode)).unwrap(); fs::set_permissions(disk_path, fs::Permissions::from_mode(mode)).unwrap();
} }
}
pub fn check_out(&mut self, tree_id: TreeId) -> Result<CheckoutStats, CheckoutError> { pub fn check_out(&mut self, tree_id: TreeId) -> Result<CheckoutStats, CheckoutError> {
let old_tree = self let old_tree = self
@ -475,11 +507,7 @@ impl TreeState {
assert_ne!(executable, old_executable); assert_ne!(executable, old_executable);
self.set_executable(&disk_path, *executable); self.set_executable(&disk_path, *executable);
let mut file_state = self.file_states.get(&path).unwrap().clone(); let mut file_state = self.file_states.get(&path).unwrap().clone();
file_state.file_type = if *executable { file_state.mark_executable(*executable);
FileType::Executable
} else {
FileType::Normal
};
file_state file_state
} }
(_, TreeValue::Normal { id, executable }) => { (_, TreeValue::Normal { id, executable }) => {