use std::path::Component;
use std::path::Path;
use std::path::PathBuf;
pub(super) fn absolutize(path: &Path) -> std::io::Result<PathBuf> {
Ok(absolutize_from(path, &std::env::current_dir()?))
}
pub(super) fn absolutize_from(path: &Path, base_path: &Path) -> PathBuf {
normalize_path(&path_with_base(path, base_path))
}
fn normalize_path(path: &Path) -> PathBuf {
let mut normalized = PathBuf::new();
for component in path.components() {
match component {
Component::CurDir => {}
Component::ParentDir => {
normalized.pop();
}
Component::Prefix(_) | Component::RootDir | Component::Normal(_) => {
normalized.push(component.as_os_str());
}
}
}
if normalized.as_os_str().is_empty() {
PathBuf::from(".")
} else {
normalized
}
}
#[cfg(not(windows))]
fn path_with_base(path: &Path, base_path: &Path) -> PathBuf {
if path.is_absolute() {
path.to_path_buf()
} else {
base_path.join(path)
}
}
#[cfg(windows)]
fn path_with_base(path: &Path, base_path: &Path) -> PathBuf {
if path.is_absolute() || path.has_root() {
return base_path.join(path);
}
let mut components = path.components();
let Some(Component::Prefix(prefix)) = components.next() else {
return base_path.join(path);
};
let mut path = PathBuf::new();
path.push(prefix.as_os_str());
if components.clone().next().is_none() {
path.push(std::path::MAIN_SEPARATOR_STR);
return path;
}
let skip_base_prefix = matches!(base_path.components().next(), Some(Component::Prefix(_)));
for component in base_path
.components()
.skip(usize::from(skip_base_prefix))
.chain(components)
{
path.push(component.as_os_str());
}
path
}
#[cfg(test)]
mod tests {
use super::*;
use pretty_assertions::assert_eq;
#[cfg(unix)]
#[test]
fn absolute_path_without_dots_is_unchanged() {
assert_eq!(
absolutize_from(Path::new("/path/to/123/456"), Path::new("/base")),
PathBuf::from("/path/to/123/456")
);
}
#[cfg(unix)]
#[test]
fn absolute_path_dots_are_removed() {
assert_eq!(
absolutize_from(Path::new("/path/to/./123/../456"), Path::new("/base")),
PathBuf::from("/path/to/456")
);
}
#[cfg(unix)]
#[test]
fn relative_path_without_dot_uses_base() {
assert_eq!(
absolutize_from(Path::new("path/to/123/456"), Path::new("/base")),
PathBuf::from("/base/path/to/123/456")
);
}
#[cfg(unix)]
#[test]
fn relative_path_with_current_dir_uses_base() {
assert_eq!(
absolutize_from(Path::new("./path/to/123/456"), Path::new("/base")),
PathBuf::from("/base/path/to/123/456")
);
}
#[cfg(unix)]
#[test]
fn relative_path_with_parent_dir_uses_base_parent() {
assert_eq!(
absolutize_from(Path::new("../path/to/123/456"), Path::new("/base/cwd")),
PathBuf::from("/base/path/to/123/456")
);
}
#[cfg(unix)]
#[test]
fn parent_dir_above_root_stays_at_root() {
assert_eq!(
absolutize_from(Path::new("../../path/to/123/456"), Path::new("/")),
PathBuf::from("/path/to/123/456")
);
}
#[cfg(unix)]
#[test]
fn empty_path_uses_base() {
assert_eq!(
absolutize_from(Path::new(""), Path::new("/base/cwd")),
PathBuf::from("/base/cwd")
);
}
#[cfg(windows)]
#[test]
fn windows_root_relative_path_uses_base_prefix() {
assert_eq!(
absolutize_from(Path::new(r"\path\to\file"), Path::new(r"C:\base\cwd")),
PathBuf::from(r"C:\path\to\file")
);
}
#[cfg(windows)]
#[test]
fn windows_drive_relative_path_uses_path_prefix_and_base_tail() {
assert_eq!(
absolutize_from(Path::new(r"D:path\to\file"), Path::new(r"C:\base\cwd")),
PathBuf::from(r"D:\base\cwd\path\to\file")
);
}
}