mirror of
https://github.com/Llewellynvdm/exa.git
synced 2025-01-27 23:58:25 +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('.')
|
self.name.starts_with('.')
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Assuming the current file is a symlink, follows the link and
|
/// Re-prefixes the path pointed to by this file, if it's a symlink, to
|
||||||
/// returns a File object from the path the link points 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
|
/// For a working symlink that the user is allowed to follow,
|
||||||
/// other end doesn't exist), returns the path to the file
|
/// this will be the `File` object at the other end, which can then have
|
||||||
/// that should be there.
|
/// 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> {
|
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,
|
Ok(path) => path,
|
||||||
Err(e) => return FileTarget::Err(e),
|
Err(e) => return FileTarget::Err(e),
|
||||||
};
|
};
|
||||||
|
|
||||||
let (metadata, ext) = {
|
let target_path = self.reorient_target_path(&*display_path);
|
||||||
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 filename = match path.components().next_back() {
|
// Use plain `metadata` instead of `symlink_metadata` - we *want* to
|
||||||
Some(comp) => comp.as_os_str().to_string_lossy().to_string(),
|
// follow links.
|
||||||
None => String::new(),
|
if let Ok(metadata) = fs::metadata(&target_path) {
|
||||||
};
|
FileTarget::Ok(File::with_metadata(metadata, &*display_path, None))
|
||||||
|
|
||||||
if let Ok(metadata) = metadata {
|
|
||||||
FileTarget::Ok(File {
|
|
||||||
path: path,
|
|
||||||
dir: self.dir,
|
|
||||||
metadata: metadata,
|
|
||||||
ext: ext,
|
|
||||||
name: filename,
|
|
||||||
})
|
|
||||||
}
|
}
|
||||||
else {
|
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
|
/// file isn’t a link to begin with, but also if, say, we don’t have
|
||||||
/// permission to follow it.
|
/// permission to follow it.
|
||||||
Err(IOError),
|
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> {
|
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 $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
|
$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!"
|
echo "All the tests passed!"
|
||||||
|
Loading…
x
Reference in New Issue
Block a user