2020-04-11 19:05:29 +00:00
|
|
|
use std::io;
|
|
|
|
use std::path::{Component, Path, PathBuf};
|
|
|
|
|
2020-04-15 23:29:22 +00:00
|
|
|
pub fn absolutize<P, Q>(relative_to: P, path: Q) -> PathBuf
|
2020-04-11 19:05:29 +00:00
|
|
|
where
|
|
|
|
P: AsRef<Path>,
|
|
|
|
Q: AsRef<Path>,
|
|
|
|
{
|
2020-04-15 23:29:22 +00:00
|
|
|
let path = relative_to.as_ref().join(path);
|
|
|
|
|
|
|
|
let (relative_to, path) = {
|
2020-04-11 19:05:29 +00:00
|
|
|
let components: Vec<_> = path.components().collect();
|
|
|
|
let separator = components
|
|
|
|
.iter()
|
|
|
|
.enumerate()
|
|
|
|
.find(|(_, c)| c == &&Component::CurDir || c == &&Component::ParentDir);
|
|
|
|
|
|
|
|
if let Some((index, _)) = separator {
|
|
|
|
let (absolute, relative) = components.split_at(index);
|
|
|
|
let absolute: PathBuf = absolute.iter().collect();
|
|
|
|
let relative: PathBuf = relative.iter().collect();
|
|
|
|
|
|
|
|
(absolute, relative)
|
|
|
|
} else {
|
|
|
|
(relative_to.as_ref().to_path_buf(), path)
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2020-04-15 23:29:22 +00:00
|
|
|
let path = if path.is_relative() {
|
2020-04-11 19:05:29 +00:00
|
|
|
let mut result = relative_to;
|
|
|
|
path.components().for_each(|component| match component {
|
|
|
|
Component::ParentDir => {
|
|
|
|
result.pop();
|
|
|
|
}
|
|
|
|
Component::Normal(normal) => result.push(normal),
|
|
|
|
_ => {}
|
|
|
|
});
|
|
|
|
|
|
|
|
result
|
|
|
|
} else {
|
|
|
|
path
|
2020-04-15 23:29:22 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
dunce::simplified(&path).to_path_buf()
|
2020-04-12 08:33:38 +00:00
|
|
|
}
|
2020-04-11 19:05:29 +00:00
|
|
|
|
2020-04-15 23:29:22 +00:00
|
|
|
pub fn canonicalize<P, Q>(relative_to: P, path: Q) -> io::Result<PathBuf>
|
2020-04-12 08:33:38 +00:00
|
|
|
where
|
|
|
|
P: AsRef<Path>,
|
|
|
|
Q: AsRef<Path>,
|
|
|
|
{
|
2020-04-15 23:29:22 +00:00
|
|
|
let canonicalized = absolutize(relative_to, path);
|
2020-04-12 08:33:38 +00:00
|
|
|
let path = match std::fs::read_link(&canonicalized) {
|
2020-04-11 19:05:29 +00:00
|
|
|
Ok(resolved) => resolved,
|
|
|
|
Err(e) => {
|
2020-04-12 08:33:38 +00:00
|
|
|
if canonicalized.exists() {
|
|
|
|
canonicalized
|
2020-04-11 19:05:29 +00:00
|
|
|
} else {
|
|
|
|
return Err(e);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
Ok(dunce::simplified(&path).to_path_buf())
|
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
mod tests {
|
|
|
|
use super::*;
|
|
|
|
use std::io;
|
|
|
|
|
|
|
|
#[test]
|
2020-04-15 23:29:22 +00:00
|
|
|
fn absolutize_two_dots() {
|
2020-04-12 08:33:38 +00:00
|
|
|
let relative_to = Path::new("/foo/bar");
|
2020-04-11 19:05:29 +00:00
|
|
|
let path = Path::new("..");
|
|
|
|
|
|
|
|
assert_eq!(
|
2020-04-12 08:33:38 +00:00
|
|
|
PathBuf::from("/foo"), // missing path
|
2020-04-15 23:29:22 +00:00
|
|
|
absolutize(relative_to, path)
|
2020-04-11 19:05:29 +00:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
2020-04-15 23:29:22 +00:00
|
|
|
fn canonicalize_should_succeed() -> io::Result<()> {
|
|
|
|
let relative_to = Path::new("/foo/bar");
|
|
|
|
let path = Path::new("../..");
|
2020-04-11 19:05:29 +00:00
|
|
|
|
|
|
|
assert_eq!(
|
2020-04-12 08:33:38 +00:00
|
|
|
PathBuf::from("/"), // existing path
|
2020-04-15 23:29:22 +00:00
|
|
|
canonicalize(relative_to, path)?,
|
2020-04-11 19:05:29 +00:00
|
|
|
);
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
2020-04-15 23:29:22 +00:00
|
|
|
fn canonicalize_should_fail() {
|
2020-04-12 08:33:38 +00:00
|
|
|
let relative_to = Path::new("/foo/bar/baz"); // '/foo' is missing
|
2020-04-15 23:29:22 +00:00
|
|
|
let path = Path::new("../..");
|
2020-04-11 19:05:29 +00:00
|
|
|
|
2020-04-15 23:29:22 +00:00
|
|
|
assert!(canonicalize(relative_to, path).is_err());
|
2020-04-11 19:05:29 +00:00
|
|
|
}
|
|
|
|
}
|