use std::borrow::Cow;
use std::io::ErrorKind;
use std::path::Path;
use std::path::PathBuf;
use deno_path_util::normalize_path;
use deno_path_util::url_from_directory_path;
use sys_traits::FsCanonicalize;
use sys_traits::FsCreateDirAll;
use url::Url;
pub struct NpmCacheFolderId {
pub name: String,
pub version: String,
pub copy_index: u8,
}
#[derive(Clone, Debug)]
pub struct NpmCacheDir {
root_dir: PathBuf,
root_dir_url: Url,
known_registries_dirnames: Vec<String>,
}
impl NpmCacheDir {
pub fn new<Sys: FsCanonicalize + FsCreateDirAll>(
sys: &Sys,
root_dir: PathBuf,
known_registries_urls: Vec<Url>,
) -> Self {
fn try_get_canonicalized_root_dir<Sys: FsCanonicalize + FsCreateDirAll>(
sys: &Sys,
root_dir: &Path,
) -> Result<PathBuf, std::io::Error> {
match sys.fs_canonicalize(root_dir) {
Ok(path) => Ok(path),
Err(err) if err.kind() == ErrorKind::NotFound => {
sys.fs_create_dir_all(root_dir)?;
sys.fs_canonicalize(root_dir)
}
Err(err) => Err(err),
}
}
let root_dir = normalize_path(Cow::Owned(root_dir));
let root_dir = try_get_canonicalized_root_dir(sys, &root_dir)
.map(Cow::Owned)
.unwrap_or(root_dir);
let root_dir_url = url_from_directory_path(&root_dir).unwrap();
let known_registries_dirnames: Vec<_> = known_registries_urls
.into_iter()
.map(|url| {
root_url_to_safe_local_dirname(&url)
.to_string_lossy()
.replace('\\', "/")
})
.collect();
Self {
root_dir: root_dir.into_owned(),
root_dir_url,
known_registries_dirnames,
}
}
pub fn root_dir(&self) -> &Path {
&self.root_dir
}
pub fn root_dir_url(&self) -> &Url {
&self.root_dir_url
}
pub fn package_folder_for_id(
&self,
package_name: &str,
package_version: &str,
package_copy_index: u8,
registry_url: &Url,
) -> PathBuf {
if package_copy_index == 0 {
self
.package_name_folder(package_name, registry_url)
.join(package_version)
} else {
self
.package_name_folder(package_name, registry_url)
.join(format!("{}_{}", package_version, package_copy_index))
}
}
pub fn package_name_folder(&self, name: &str, registry_url: &Url) -> PathBuf {
let mut dir = self.registry_folder(registry_url);
if name.to_lowercase() != name {
let encoded_name = mixed_case_package_name_encode(name);
dir.join(format!("_{encoded_name}"))
} else {
for part in name.split('/') {
dir = dir.join(part);
}
dir
}
}
fn registry_folder(&self, registry_url: &Url) -> PathBuf {
self
.root_dir
.join(root_url_to_safe_local_dirname(registry_url))
}
pub fn resolve_package_folder_id_from_specifier(
&self,
specifier: &Url,
) -> Option<NpmCacheFolderId> {
let mut maybe_relative_url = None;
for registry_dirname in &self.known_registries_dirnames {
let registry_root_dir = self
.root_dir_url
.join(&format!("{}/", registry_dirname))
.unwrap();
let Some(relative_url) = registry_root_dir.make_relative(specifier)
else {
continue;
};
if relative_url.starts_with("../") {
continue;
}
maybe_relative_url = Some(relative_url);
break;
}
let mut relative_url = maybe_relative_url?;
if let Some(end_url) = relative_url.strip_prefix('_') {
let mut parts = end_url
.split('/')
.map(ToOwned::to_owned)
.collect::<Vec<_>>();
match mixed_case_package_name_decode(&parts[0]) {
Some(part) => {
parts[0] = part;
}
None => return None,
}
relative_url = parts.join("/");
}
let is_scoped_package = relative_url.starts_with('@');
let mut parts = relative_url
.split('/')
.enumerate()
.take(if is_scoped_package { 3 } else { 2 })
.map(|(_, part)| part)
.collect::<Vec<_>>();
if parts.len() < 2 {
return None;
}
let version_part = parts.pop().unwrap();
let name = parts.join("/");
let (version, copy_index) =
if let Some((version, copy_count)) = version_part.split_once('_') {
(version, copy_count.parse::<u8>().ok()?)
} else {
(version_part, 0)
};
Some(NpmCacheFolderId {
name,
version: version.to_string(),
copy_index,
})
}
}
pub fn mixed_case_package_name_encode(name: &str) -> String {
base32::encode(
base32::Alphabet::Rfc4648Lower { padding: false },
name.as_bytes(),
)
.to_lowercase()
}
pub fn mixed_case_package_name_decode(name: &str) -> Option<String> {
base32::decode(base32::Alphabet::Rfc4648Lower { padding: false }, name)
.and_then(|b| String::from_utf8(b).ok())
}
fn root_url_to_safe_local_dirname(root: &Url) -> PathBuf {
fn sanitize_segment(text: &str) -> String {
text
.chars()
.map(|c| if is_banned_segment_char(c) { '_' } else { c })
.collect()
}
fn is_banned_segment_char(c: char) -> bool {
matches!(c, '/' | '\\') || is_banned_path_char(c)
}
let mut result = String::new();
if let Some(domain) = root.domain() {
result.push_str(&sanitize_segment(domain));
}
if let Some(port) = root.port() {
if !result.is_empty() {
result.push('_');
}
result.push_str(&port.to_string());
}
let mut result = PathBuf::from(result);
if let Some(segments) = root.path_segments() {
for segment in segments.filter(|s| !s.is_empty()) {
result = result.join(sanitize_segment(segment));
}
}
result
}
fn is_banned_path_char(c: char) -> bool {
matches!(c, '<' | '>' | ':' | '"' | '|' | '?' | '*')
}
#[cfg(test)]
mod test {
use std::path::PathBuf;
use sys_traits::FsCreateDirAll;
use url::Url;
use super::NpmCacheDir;
#[test]
fn should_get_package_folder() {
let sys = sys_traits::impls::InMemorySys::default();
let root_dir = if cfg!(windows) {
PathBuf::from("C:\\cache")
} else {
PathBuf::from("/cache")
};
sys.fs_create_dir_all(&root_dir).unwrap();
let registry_url = Url::parse("https://registry.npmjs.org/").unwrap();
let cache =
NpmCacheDir::new(&sys, root_dir.clone(), vec![registry_url.clone()]);
assert_eq!(
cache.package_folder_for_id("json", "1.2.5", 0, ®istry_url,),
root_dir
.join("registry.npmjs.org")
.join("json")
.join("1.2.5"),
);
assert_eq!(
cache.package_folder_for_id("json", "1.2.5", 1, ®istry_url,),
root_dir
.join("registry.npmjs.org")
.join("json")
.join("1.2.5_1"),
);
assert_eq!(
cache.package_folder_for_id("JSON", "2.1.5", 0, ®istry_url,),
root_dir
.join("registry.npmjs.org")
.join("_jjju6tq")
.join("2.1.5"),
);
assert_eq!(
cache.package_folder_for_id("@types/JSON", "2.1.5", 0, ®istry_url,),
root_dir
.join("registry.npmjs.org")
.join("_ib2hs4dfomxuuu2pjy")
.join("2.1.5"),
);
}
}