use std::path::Path;
use crate::{
AliasValue, PathUtil, Resolution, ResolveContext, ResolveError, ResolveOptions, Resolver,
};
#[allow(clippy::too_many_lines)]
#[test]
#[cfg(not(target_os = "windows"))] fn alias() {
use std::path::{Path, PathBuf};
use super::memory_fs::MemoryFS;
use crate::ResolverGeneric;
let f = Path::new("/");
let file_system = MemoryFS::new(&[
("/a/index.js", ""),
("/a/dir/index.js", ""),
("/recursive/index.js", ""),
("/recursive/dir/index.js", ""),
("/b/index.js", ""),
("/b/dir/index.js", ""),
("/c/index.js", ""),
("/c/dir/index.js", ""),
("/d/index.js.js", ""),
("/d/dir/.empty", ""),
("/e/index.js", ""),
("/e/anotherDir/index.js", ""),
("/e/dir/file", ""),
("/dashed-name", ""),
]);
let resolver = ResolverGeneric::new_with_file_system(
file_system,
ResolveOptions {
alias: vec![
("aliasA".into(), vec![AliasValue::from("a")]),
("b$".into(), vec![AliasValue::from("a/index.js")]),
("c$".into(), vec![AliasValue::from("/a/index.js")]),
(
"multiAlias".into(),
vec![
AliasValue::from("b"),
AliasValue::from("c"),
AliasValue::from("d"),
AliasValue::from("e"),
AliasValue::from("a"),
],
),
("recursive".into(), vec![AliasValue::from("recursive/dir")]),
("/d/dir".into(), vec![AliasValue::from("/c/dir")]),
("/d/index.js".into(), vec![AliasValue::from("/c/index.js")]),
("#".into(), vec![AliasValue::from("/c/dir")]),
("@".into(), vec![AliasValue::from("/c/dir")]),
("ignored".into(), vec![AliasValue::Ignore]),
("alias_query".into(), vec![AliasValue::from("a?query_after")]),
("alias_fragment".into(), vec![AliasValue::from("a#fragment_after")]),
("dash".into(), vec![AliasValue::Ignore]),
("@scope/package-name/file$".into(), vec![AliasValue::from("/c/dir")]),
("@adir/*".into(), vec![AliasValue::from("./a/")]), ("@*".into(), vec![AliasValue::from("/*")]),
("@e*".into(), vec![AliasValue::from("/e/*")]),
("@e*file".into(), vec![AliasValue::from("/e*file")]),
],
modules: vec!["/".into()],
..ResolveOptions::default()
},
);
#[rustfmt::skip]
let pass = [
("should resolve a not aliased module 1", "a", "/a/index.js"),
("should resolve a not aliased module 2", "a/index.js", "/a/index.js"),
("should resolve a not aliased module 3", "a/dir", "/a/dir/index.js"),
("should resolve a not aliased module 4", "a/dir/index.js", "/a/dir/index.js"),
("should resolve an aliased module 1", "aliasA", "/a/index.js"),
("should resolve an aliased module 2", "aliasA/index.js", "/a/index.js"),
("should resolve an aliased module 3", "aliasA/dir", "/a/dir/index.js"),
("should resolve an aliased module 4", "aliasA/dir/index.js", "/a/dir/index.js"),
("should resolve '#' alias 1", "#", "/c/dir/index.js"),
("should resolve '#' alias 2", "#/index.js", "/c/dir/index.js"),
("should resolve '@' alias 1", "@", "/c/dir/index.js"),
("should resolve '@' alias 2", "@/index.js", "/c/dir/index.js"),
("should resolve '@' alias 3", "@/", "/c/dir/index.js"),
("should resolve a recursive aliased module 1", "recursive", "/recursive/dir/index.js"),
("should resolve a recursive aliased module 2", "recursive/index.js", "/recursive/dir/index.js"),
("should resolve a recursive aliased module 3", "recursive/dir", "/recursive/dir/index.js"),
("should resolve a recursive aliased module 4", "recursive/dir/index.js", "/recursive/dir/index.js"),
("should resolve a file aliased module 1", "b", "/a/index.js"),
("should resolve a file aliased module 2", "c", "/a/index.js"),
("should resolve a file aliased module with a query 1", "b?query", "/a/index.js?query"),
("should resolve a file aliased module with a query 2", "c?query", "/a/index.js?query"),
("should resolve a path in a file aliased module 1", "b/index.js", "/b/index.js"),
("should resolve a path in a file aliased module 2", "b/dir", "/b/dir/index.js"),
("should resolve a path in a file aliased module 3", "b/dir/index.js", "/b/dir/index.js"),
("should resolve a path in a file aliased module 4", "c/index.js", "/c/index.js"),
("should resolve a path in a file aliased module 5", "c/dir", "/c/dir/index.js"),
("should resolve a path in a file aliased module 6", "c/dir/index.js", "/c/dir/index.js"),
("should resolve a file aliased file 1", "d", "/c/index.js"),
("should resolve a file aliased file 2", "d/dir/index.js", "/c/dir/index.js"),
("should resolve a file in multiple aliased dirs 1", "multiAlias/dir/file", "/e/dir/file"),
("should resolve a file in multiple aliased dirs 2", "multiAlias/anotherDir", "/e/anotherDir/index.js"),
("should resolve wildcard alias 1", "@a", "/a/index.js"),
("should resolve wildcard alias 2", "@a/dir", "/a/dir/index.js"),
("should resolve wildcard alias 3", "@e/dir/file", "/e/dir/file"),
("should resolve wildcard alias 4", "@e/anotherDir", "/e/anotherDir/index.js"),
("should resolve wildcard alias 5", "@e/dir/file", "/e/dir/file"),
("should resolve scoped package name with sub dir 1", "@adir/index.js", "/a/index.js"),
("should resolve scoped package name with sub dir 2", "@adir/dir", "/a/index.js"),
("should resolve query in alias value", "alias_query?query_before", "/a/index.js?query_after"),
("should resolve query in alias value", "alias_fragment#fragment_before", "/a/index.js#fragment_after"),
("should resolve dashed name", "dashed-name", "/dashed-name"),
("should resolve scoped package name with sub dir", "@scope/package-name/file", "/c/dir/index.js"),
];
for (comment, request, expected) in pass {
let resolved_path = resolver.resolve(f, request).map(|r| r.full_path());
assert_eq!(resolved_path, Ok(PathBuf::from(expected)), "{comment} {request}");
}
#[rustfmt::skip]
let ignore = [
("should resolve an ignore module", "ignored", ResolveError::Ignored(f.join("ignored")))
];
for (comment, request, expected) in ignore {
let resolution = resolver.resolve(f, request);
assert_eq!(resolution, Err(expected), "{comment} {request}");
}
}
#[test]
fn infinite_recursion() {
let f = super::fixture();
let resolver = Resolver::new(ResolveOptions {
alias: vec![
("./a".into(), vec![AliasValue::from("./b")]),
("./b".into(), vec![AliasValue::from("./a")]),
],
..ResolveOptions::default()
});
let resolution = resolver.resolve(f, "./a");
assert_eq!(resolution, Err(ResolveError::Recursion));
}
fn check_slash(path: &Path) {
let s = path.to_string_lossy().to_string();
#[cfg(target_os = "windows")]
{
assert!(!s.contains('/'), "{s}");
assert!(s.contains('\\'), "{s}");
}
#[cfg(not(target_os = "windows"))]
{
assert!(s.contains('/'), "{s}");
assert!(!s.contains('\\'), "{s}");
}
}
#[test]
fn absolute_path() {
let f = super::fixture();
let resolver = Resolver::new(ResolveOptions {
alias: vec![(f.join("foo").to_str().unwrap().to_string(), vec![AliasValue::Ignore])],
modules: vec![f.clone().to_str().unwrap().to_string()],
..ResolveOptions::default()
});
let resolution = resolver.resolve(&f, "foo/index");
assert_eq!(resolution, Err(ResolveError::Ignored(f.join("foo"))));
}
#[test]
fn system_path() {
let f = super::fixture();
let resolver = Resolver::new(ResolveOptions {
alias: vec![("@app".into(), vec![AliasValue::from(f.join("alias").to_string_lossy())])],
..ResolveOptions::default()
});
let specifiers = ["@app/files/a", "@app/files/a.js"];
for specifier in specifiers {
let path = resolver.resolve(&f, specifier).map(Resolution::into_path_buf).unwrap();
assert_eq!(path, f.join("alias/files/a.js"));
check_slash(&path);
}
}
#[test]
fn alias_is_full_path() {
let f = super::fixture();
let dir = f.join("foo");
let dir_str = dir.to_string_lossy().to_string();
let resolver = Resolver::new(ResolveOptions {
alias: vec![("@".into(), vec![AliasValue::Path(dir_str.clone())])],
..ResolveOptions::default()
});
let mut ctx = ResolveContext::default();
let specifiers = [
"@/index".to_string(),
"@/index.js".to_string(),
"@////index".to_string(),
dir_str,
];
for specifier in specifiers {
let resolution = resolver.resolve_with_context(&f, &specifier, None, &mut ctx);
assert_eq!(resolution.map(|r| r.full_path()), Ok(dir.join("index.js")));
}
for path in ctx.file_dependencies {
assert_eq!(path, path.normalize(), "{path:?}");
check_slash(&path);
}
for path in ctx.missing_dependencies {
assert_eq!(path, path.normalize(), "{path:?}");
check_slash(&path);
if let Some(path) = path.parent() {
assert!(!path.is_file(), "{path:?} must not be a file");
}
}
}
#[test]
fn all_alias_values_are_not_found() {
let f = super::fixture();
let resolver = Resolver::new(ResolveOptions {
alias: vec![(
"m1".to_string(),
vec![AliasValue::Path(f.join("node_modules").join("m2").to_string_lossy().to_string())],
)],
..ResolveOptions::default()
});
let resolution = resolver.resolve(&f, "m1/a.js");
assert_eq!(
resolution,
Err(ResolveError::MatchedAliasNotFound("m1/a.js".to_string(), "m1".to_string()))
);
}
#[test]
fn alias_fragment() {
let f = super::fixture();
let data = [
(
"handle fragment edge case (no fragment)",
"./no#fragment/#/#",
f.join("no#fragment/#/#.js"),
),
("handle fragment edge case (fragment)", "./no#fragment/#/", f.join("no.js#fragment/#/")),
(
"handle fragment escaping",
"./no\0#fragment/\0#/\0##fragment",
f.join("no#fragment/#/#.js#fragment"),
),
];
for (comment, request, expected) in data {
let resolver = Resolver::new(ResolveOptions {
alias: vec![("foo".to_string(), vec![AliasValue::Path(request.to_string())])],
..ResolveOptions::default()
});
let resolved_path = resolver.resolve(&f, "foo").map(|r| r.full_path());
assert_eq!(resolved_path, Ok(expected), "{comment} {request}");
}
}
#[test]
fn alias_try_fragment_as_path() {
let f = super::fixture();
let resolver = Resolver::new(ResolveOptions {
alias: vec![(
"#".to_string(),
vec![AliasValue::Path(f.join("#").to_string_lossy().to_string())],
)],
..ResolveOptions::default()
});
let resolution = resolver.resolve(&f, "#/a").map(|r| r.full_path());
assert_eq!(resolution, Ok(f.join("#").join("a.js")));
}
#[test]
fn alias_with_multiple_fallbacks() {
let f = super::fixture();
let resolver = Resolver::new(ResolveOptions {
alias: vec![(
"multi".to_string(),
vec![
AliasValue::Path(f.join("nonexistent").to_string_lossy().to_string()),
AliasValue::Path(f.join("foo").to_string_lossy().to_string()),
],
)],
..ResolveOptions::default()
});
let resolution = resolver.resolve(&f, "multi/index.js").map(|r| r.full_path());
assert_eq!(resolution, Ok(f.join("foo/index.js")));
}