forked from mirrors/jj
working_copy: on checkout, record file state's type and size
This patch is essentially f6a516ff6d
taken further, to also apply to
when we write a symlink or a conflict. As with regular files, these
races seem very unlikely to happen, but I found these cases while
working on #258, so let's fix. Fixing it also means that we don't need
to handle these transition cases in the next patch (when
`file_states()` can indicate that the file is e.g. a socket).
This commit is contained in:
parent
242c506a9c
commit
a42b24c014
1 changed files with 39 additions and 17 deletions
|
@ -17,7 +17,7 @@ use std::collections::{BTreeMap, HashSet};
|
||||||
use std::ffi::OsString;
|
use std::ffi::OsString;
|
||||||
use std::fs;
|
use std::fs;
|
||||||
use std::fs::{DirEntry, File, Metadata, OpenOptions};
|
use std::fs::{DirEntry, File, Metadata, OpenOptions};
|
||||||
use std::io::Read;
|
use std::io::{Read, Write};
|
||||||
use std::ops::Bound;
|
use std::ops::Bound;
|
||||||
#[cfg(unix)]
|
#[cfg(unix)]
|
||||||
use std::os::unix::fs::symlink;
|
use std::os::unix::fs::symlink;
|
||||||
|
@ -62,6 +62,33 @@ pub struct FileState {
|
||||||
}
|
}
|
||||||
|
|
||||||
impl FileState {
|
impl FileState {
|
||||||
|
fn for_file(executable: bool, size: u64, metadata: &Metadata) -> Self {
|
||||||
|
FileState {
|
||||||
|
file_type: FileType::Normal { executable },
|
||||||
|
mtime: mtime_from_metadata(metadata),
|
||||||
|
size,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn for_symlink(metadata: &Metadata) -> Self {
|
||||||
|
// When using fscrypt, the reported size is not the content size. So if
|
||||||
|
// we were to record the content size here (like we do for regular files), we
|
||||||
|
// would end up thinking the file has changed everytime we snapshot.
|
||||||
|
FileState {
|
||||||
|
file_type: FileType::Symlink,
|
||||||
|
mtime: mtime_from_metadata(metadata),
|
||||||
|
size: metadata.len(),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn for_conflict(id: ConflictId, size: u64, metadata: &Metadata) -> Self {
|
||||||
|
FileState {
|
||||||
|
file_type: FileType::Conflict { id },
|
||||||
|
mtime: mtime_from_metadata(metadata),
|
||||||
|
size,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
#[cfg_attr(unix, allow(dead_code))]
|
#[cfg_attr(unix, allow(dead_code))]
|
||||||
fn is_executable(&self) -> bool {
|
fn is_executable(&self) -> bool {
|
||||||
if let FileType::Normal { executable } = &self.file_type {
|
if let FileType::Normal { executable } = &self.file_type {
|
||||||
|
@ -620,13 +647,7 @@ impl TreeState {
|
||||||
let metadata = file
|
let metadata = file
|
||||||
.metadata()
|
.metadata()
|
||||||
.map_err(|err| CheckoutError::for_stat_error(err, disk_path))?;
|
.map_err(|err| CheckoutError::for_stat_error(err, disk_path))?;
|
||||||
let mut file_state = file_state(&metadata);
|
Ok(FileState::for_file(executable, size, &metadata))
|
||||||
// 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.size = size;
|
|
||||||
Ok(file_state)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg_attr(windows, allow(unused_variables))]
|
#[cfg_attr(windows, allow(unused_variables))]
|
||||||
|
@ -637,13 +658,13 @@ impl TreeState {
|
||||||
id: &SymlinkId,
|
id: &SymlinkId,
|
||||||
) -> Result<FileState, CheckoutError> {
|
) -> Result<FileState, CheckoutError> {
|
||||||
create_parent_dirs(disk_path)?;
|
create_parent_dirs(disk_path)?;
|
||||||
|
let target = self.store.read_symlink(path, id)?;
|
||||||
#[cfg(windows)]
|
#[cfg(windows)]
|
||||||
{
|
{
|
||||||
println!("ignoring symlink at {:?}", path);
|
println!("ignoring symlink at {:?}", path);
|
||||||
}
|
}
|
||||||
#[cfg(unix)]
|
#[cfg(unix)]
|
||||||
{
|
{
|
||||||
let target = self.store.read_symlink(path, id)?;
|
|
||||||
let target = PathBuf::from(&target);
|
let target = PathBuf::from(&target);
|
||||||
symlink(&target, disk_path).map_err(|err| CheckoutError::IoError {
|
symlink(&target, disk_path).map_err(|err| CheckoutError::IoError {
|
||||||
message: format!(
|
message: format!(
|
||||||
|
@ -657,7 +678,7 @@ impl TreeState {
|
||||||
let metadata = disk_path
|
let metadata = disk_path
|
||||||
.symlink_metadata()
|
.symlink_metadata()
|
||||||
.map_err(|err| CheckoutError::for_stat_error(err, disk_path))?;
|
.map_err(|err| CheckoutError::for_stat_error(err, disk_path))?;
|
||||||
Ok(file_state(&metadata))
|
Ok(FileState::for_symlink(&metadata))
|
||||||
}
|
}
|
||||||
|
|
||||||
fn write_conflict(
|
fn write_conflict(
|
||||||
|
@ -679,20 +700,21 @@ impl TreeState {
|
||||||
message: format!("Failed to open file {} for writing", disk_path.display()),
|
message: format!("Failed to open file {} for writing", disk_path.display()),
|
||||||
err,
|
err,
|
||||||
})?;
|
})?;
|
||||||
materialize_conflict(self.store.as_ref(), path, &conflict, &mut file).map_err(|err| {
|
let mut conflict_data = vec![];
|
||||||
CheckoutError::IoError {
|
materialize_conflict(self.store.as_ref(), path, &conflict, &mut conflict_data)
|
||||||
|
.expect("Failed to materialize conflict to in-memory buffer");
|
||||||
|
file.write_all(&conflict_data)
|
||||||
|
.map_err(|err| CheckoutError::IoError {
|
||||||
message: format!("Failed to write conflict to file {}", disk_path.display()),
|
message: format!("Failed to write conflict to file {}", disk_path.display()),
|
||||||
err,
|
err,
|
||||||
}
|
})?;
|
||||||
})?;
|
let size = conflict_data.len() as u64;
|
||||||
// TODO: Set the executable bit correctly (when possible) and preserve that on
|
// TODO: Set the executable bit correctly (when possible) and preserve that on
|
||||||
// Windows like we do with the executable bit for regular files.
|
// Windows like we do with the executable bit for regular files.
|
||||||
let metadata = file
|
let metadata = file
|
||||||
.metadata()
|
.metadata()
|
||||||
.map_err(|err| CheckoutError::for_stat_error(err, disk_path))?;
|
.map_err(|err| CheckoutError::for_stat_error(err, disk_path))?;
|
||||||
let mut result = file_state(&metadata);
|
Ok(FileState::for_conflict(id.clone(), size, &metadata))
|
||||||
result.file_type = FileType::Conflict { id: id.clone() };
|
|
||||||
Ok(result)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg_attr(windows, allow(unused_variables))]
|
#[cfg_attr(windows, allow(unused_variables))]
|
||||||
|
|
Loading…
Reference in a new issue