mirror of
https://github.com/Llewellynvdm/exa.git
synced 2025-01-13 17:02:58 +00:00
Re-prefix the paths found by following symlinks
Fixes #134, a bug that showed symlinks incorrectly as broken, but only when the file was listed directly on the command-line *and* the file was in a different directory to the one exa was being run in. I’m not sure why the old code used `String::new()`, but it doesn’t seem to affect anything.
This commit is contained in:
parent
d2b1499fb1
commit
108a402dbd
@ -165,44 +165,54 @@ impl<'dir> File<'dir> {
|
||||
self.name.starts_with('.')
|
||||
}
|
||||
|
||||
/// Assuming the current file is a symlink, follows the link and
|
||||
/// returns a File object from the path the link points to.
|
||||
/// Re-prefixes the path pointed to by this file, if it's a symlink, to
|
||||
/// make it an absolute path that can be accessed from whichever
|
||||
/// directory exa is being run from.
|
||||
fn reorient_target_path(&self, path: &Path) -> PathBuf {
|
||||
if path.is_absolute() {
|
||||
path.to_path_buf()
|
||||
}
|
||||
else if let Some(dir) = self.dir {
|
||||
dir.join(&*path)
|
||||
}
|
||||
else if let Some(parent) = self.path.parent() {
|
||||
parent.join(&*path)
|
||||
}
|
||||
else {
|
||||
self.path.join(&*path)
|
||||
}
|
||||
}
|
||||
|
||||
/// Again assuming this file is a symlink, follows that link and returns
|
||||
/// the result of following it.
|
||||
///
|
||||
/// If statting the file fails (usually because the file on the
|
||||
/// other end doesn't exist), returns the path to the file
|
||||
/// that should be there.
|
||||
/// For a working symlink that the user is allowed to follow,
|
||||
/// this will be the `File` object at the other end, which can then have
|
||||
/// its name, colour, and other details read.
|
||||
///
|
||||
/// For a broken symlink, returns where the file *would* be, if it
|
||||
/// existed. If this file cannot be read at all, returns the error that
|
||||
/// we got when we tried to read it.
|
||||
pub fn link_target(&self) -> FileTarget<'dir> {
|
||||
let path = match fs::read_link(&self.path) {
|
||||
|
||||
// We need to be careful to treat the path actually pointed to by
|
||||
// this file -- which could be absolute or relative -- to the path
|
||||
// we actually look up and turn into a `File` -- which needs to be
|
||||
// absolute to be accessible from any directory.
|
||||
let display_path = match fs::read_link(&self.path) {
|
||||
Ok(path) => path,
|
||||
Err(e) => return FileTarget::Err(e),
|
||||
};
|
||||
|
||||
let (metadata, ext) = {
|
||||
let target_path_ = match self.dir {
|
||||
Some(dir) if dir.path != Path::new(".") => Some(dir.join(&*path)),
|
||||
_ => None
|
||||
};
|
||||
let target_path = target_path_.as_ref().unwrap_or(&path);
|
||||
// Use plain `metadata` instead of `symlink_metadata` - we *want* to follow links.
|
||||
(fs::metadata(&target_path), ext(target_path))
|
||||
};
|
||||
let target_path = self.reorient_target_path(&*display_path);
|
||||
|
||||
let filename = match path.components().next_back() {
|
||||
Some(comp) => comp.as_os_str().to_string_lossy().to_string(),
|
||||
None => String::new(),
|
||||
};
|
||||
|
||||
if let Ok(metadata) = metadata {
|
||||
FileTarget::Ok(File {
|
||||
path: path,
|
||||
dir: self.dir,
|
||||
metadata: metadata,
|
||||
ext: ext,
|
||||
name: filename,
|
||||
})
|
||||
// Use plain `metadata` instead of `symlink_metadata` - we *want* to
|
||||
// follow links.
|
||||
if let Ok(metadata) = fs::metadata(&target_path) {
|
||||
FileTarget::Ok(File::with_metadata(metadata, &*display_path, None))
|
||||
}
|
||||
else {
|
||||
FileTarget::Broken(path)
|
||||
FileTarget::Broken(display_path)
|
||||
}
|
||||
}
|
||||
|
||||
@ -410,6 +420,10 @@ pub enum FileTarget<'dir> {
|
||||
/// file isn’t a link to begin with, but also if, say, we don’t have
|
||||
/// permission to follow it.
|
||||
Err(IOError),
|
||||
|
||||
// Err is its own variant, instead of having the whole thing be inside an
|
||||
// `IOResult`, because being unable to follow a symlink is not a serious
|
||||
// error -- we just display the error message and move on.
|
||||
}
|
||||
|
||||
impl<'dir> FileTarget<'dir> {
|
||||
|
10
xtests/links_1_files
Normal file
10
xtests/links_1_files
Normal file
@ -0,0 +1,10 @@
|
||||
[36m/testcases/links/broken[0m [31m->[0m [4;31mnowhere[0m
|
||||
[36m/testcases/links/current_dir[0m [38;5;244m->[0m [1;34m.[0m
|
||||
[36m/testcases/links/forbidden[0m [31m->[0m [4;31m/proc/1/root[0m
|
||||
[36m/testcases/links/itself[0m [31m->[0m [4;31mitself[0m
|
||||
[36m/testcases/links/parent_dir[0m [38;5;244m->[0m [1;34m..[0m
|
||||
[36m/testcases/links/root[0m [38;5;244m->[0m [1;34m/[0m
|
||||
[36m/testcases/links/[0msome_file
|
||||
[36m/testcases/links/some_file_absolute[0m [38;5;244m->[0m [36m/testcases/links/[0msome_file
|
||||
[36m/testcases/links/some_file_relative[0m [38;5;244m->[0m some_file
|
||||
[36m/testcases/links/usr[0m [38;5;244m->[0m [36m/[1;34musr[0m
|
@ -94,5 +94,9 @@ COLUMNS=80 $exa $testcases/links 2>&1 | diff -q - $results/links || ex
|
||||
$exa $testcases/links -T 2>&1 | diff -q - $results/links_T || exit 1
|
||||
$exa /proc/1/root -T 2>&1 | diff -q - $results/proc_1_root || exit 1
|
||||
|
||||
# There’ve been bugs where the target file wasn’t printed properly when the
|
||||
# symlink file was specified on the command-line directly.
|
||||
$exa $testcases/links/* -1 | diff -q - $results/links_1_files || exit 1
|
||||
|
||||
|
||||
echo "All the tests passed!"
|
||||
|
Loading…
Reference in New Issue
Block a user