2025-01-01 04:12:39 +09:00
|
|
|
// Copyright 2018-2025 the Deno authors. MIT license.
|
2022-09-22 11:17:02 -04:00
|
|
|
|
|
|
|
//! Code for local node_modules resolution.
|
|
|
|
|
2022-11-16 13:44:31 -05:00
|
|
|
use std::borrow::Cow;
|
2022-09-22 11:17:02 -04:00
|
|
|
use std::path::Path;
|
|
|
|
use std::path::PathBuf;
|
2023-04-14 16:22:33 -04:00
|
|
|
use std::sync::Arc;
|
2022-09-22 11:17:02 -04:00
|
|
|
|
2023-02-22 14:15:25 -05:00
|
|
|
use async_trait::async_trait;
|
2022-09-22 11:17:02 -04:00
|
|
|
use deno_ast::ModuleSpecifier;
|
2024-09-28 08:50:16 -04:00
|
|
|
use deno_cache_dir::npm::mixed_case_package_name_decode;
|
2022-09-22 11:17:02 -04:00
|
|
|
use deno_core::url::Url;
|
2023-04-06 18:46:44 -04:00
|
|
|
use deno_npm::NpmPackageCacheFolderId;
|
|
|
|
use deno_npm::NpmPackageId;
|
2024-12-30 12:38:20 -05:00
|
|
|
use deno_path_util::fs::canonicalize_path_maybe_not_exists;
|
2024-09-30 09:33:32 -04:00
|
|
|
use deno_resolver::npm::normalize_pkg_name_for_node_modules_deno_folder;
|
2023-08-21 11:53:52 +02:00
|
|
|
use deno_semver::package::PackageNv;
|
2024-12-20 16:14:37 -05:00
|
|
|
use deno_semver::StackString;
|
2024-07-25 19:08:14 -04:00
|
|
|
use node_resolver::errors::PackageFolderResolveError;
|
|
|
|
use node_resolver::errors::PackageFolderResolveIoError;
|
|
|
|
use node_resolver::errors::PackageNotFoundError;
|
|
|
|
use node_resolver::errors::ReferrerNotFoundError;
|
2024-12-30 12:38:20 -05:00
|
|
|
use sys_traits::FsMetadata;
|
2022-09-22 11:17:02 -04:00
|
|
|
|
2024-12-31 12:13:39 -05:00
|
|
|
use super::super::resolution::NpmResolution;
|
|
|
|
use super::common::NpmPackageFsResolver;
|
2024-12-31 11:29:07 -05:00
|
|
|
use crate::sys::CliSys;
|
2022-09-22 11:17:02 -04:00
|
|
|
|
|
|
|
/// Resolver that creates a local node_modules directory
|
|
|
|
/// and resolves packages from it.
|
2023-04-14 16:22:33 -04:00
|
|
|
#[derive(Debug)]
|
2022-09-22 11:17:02 -04:00
|
|
|
pub struct LocalNpmPackageResolver {
|
2023-04-14 16:22:33 -04:00
|
|
|
resolution: Arc<NpmResolution>,
|
2024-12-31 11:29:07 -05:00
|
|
|
sys: CliSys,
|
2022-09-22 11:17:02 -04:00
|
|
|
root_node_modules_path: PathBuf,
|
2023-02-23 10:58:10 -05:00
|
|
|
root_node_modules_url: Url,
|
2022-09-22 11:17:02 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
impl LocalNpmPackageResolver {
|
2024-07-03 20:54:33 -04:00
|
|
|
#[allow(clippy::too_many_arguments)]
|
2022-09-22 11:17:02 -04:00
|
|
|
pub fn new(
|
2023-04-14 16:22:33 -04:00
|
|
|
resolution: Arc<NpmResolution>,
|
2024-12-31 11:29:07 -05:00
|
|
|
sys: CliSys,
|
2024-06-02 21:39:13 -04:00
|
|
|
node_modules_folder: PathBuf,
|
2022-09-22 11:17:02 -04:00
|
|
|
) -> Self {
|
|
|
|
Self {
|
|
|
|
resolution,
|
2024-12-30 12:38:20 -05:00
|
|
|
sys,
|
2024-06-02 21:39:13 -04:00
|
|
|
root_node_modules_url: Url::from_directory_path(&node_modules_folder)
|
|
|
|
.unwrap(),
|
|
|
|
root_node_modules_path: node_modules_folder,
|
2022-09-22 11:17:02 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn resolve_package_root(&self, path: &Path) -> PathBuf {
|
|
|
|
let mut last_found = path;
|
|
|
|
loop {
|
|
|
|
let parent = last_found.parent().unwrap();
|
|
|
|
if parent.file_name().unwrap() == "node_modules" {
|
|
|
|
return last_found.to_path_buf();
|
|
|
|
} else {
|
|
|
|
last_found = parent;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn resolve_folder_for_specifier(
|
|
|
|
&self,
|
|
|
|
specifier: &ModuleSpecifier,
|
2024-07-09 12:15:03 -04:00
|
|
|
) -> Result<Option<PathBuf>, std::io::Error> {
|
2023-08-27 12:04:12 +08:00
|
|
|
let Some(relative_url) =
|
|
|
|
self.root_node_modules_url.make_relative(specifier)
|
|
|
|
else {
|
2023-07-17 14:00:44 -04:00
|
|
|
return Ok(None);
|
|
|
|
};
|
2022-09-22 11:17:02 -04:00
|
|
|
if relative_url.starts_with("../") {
|
2023-07-17 14:00:44 -04:00
|
|
|
return Ok(None);
|
2022-09-22 11:17:02 -04:00
|
|
|
}
|
|
|
|
// it's within the directory, so use it
|
2023-07-17 14:00:44 -04:00
|
|
|
let Some(path) = specifier.to_file_path().ok() else {
|
|
|
|
return Ok(None);
|
|
|
|
};
|
|
|
|
// Canonicalize the path so it's not pointing to the symlinked directory
|
|
|
|
// in `node_modules` directory of the referrer.
|
2024-12-30 12:38:20 -05:00
|
|
|
canonicalize_path_maybe_not_exists(&self.sys, &path).map(Some)
|
2022-09-22 11:17:02 -04:00
|
|
|
}
|
2024-05-23 22:26:23 +01:00
|
|
|
|
|
|
|
fn resolve_package_folder_from_specifier(
|
|
|
|
&self,
|
|
|
|
specifier: &ModuleSpecifier,
|
2025-01-08 14:52:32 -08:00
|
|
|
) -> Result<Option<PathBuf>, std::io::Error> {
|
2024-05-23 22:26:23 +01:00
|
|
|
let Some(local_path) = self.resolve_folder_for_specifier(specifier)? else {
|
|
|
|
return Ok(None);
|
|
|
|
};
|
|
|
|
let package_root_path = self.resolve_package_root(&local_path);
|
|
|
|
Ok(Some(package_root_path))
|
|
|
|
}
|
2023-04-14 16:22:33 -04:00
|
|
|
}
|
2022-11-08 14:17:24 -05:00
|
|
|
|
2024-06-06 18:37:41 -04:00
|
|
|
#[async_trait(?Send)]
|
2023-04-14 16:22:33 -04:00
|
|
|
impl NpmPackageFsResolver for LocalNpmPackageResolver {
|
2024-09-30 09:33:32 -04:00
|
|
|
fn node_modules_path(&self) -> Option<&Path> {
|
|
|
|
Some(self.root_node_modules_path.as_ref())
|
2023-04-14 16:22:33 -04:00
|
|
|
}
|
|
|
|
|
2024-07-09 12:15:03 -04:00
|
|
|
fn maybe_package_folder(&self, id: &NpmPackageId) -> Option<PathBuf> {
|
|
|
|
let cache_folder_id = self
|
|
|
|
.resolution
|
|
|
|
.resolve_pkg_cache_folder_id_from_pkg_id(id)?;
|
|
|
|
// package is stored at:
|
|
|
|
// node_modules/.deno/<package_cache_folder_id_folder_name>/node_modules/<package_name>
|
|
|
|
Some(
|
|
|
|
self
|
|
|
|
.root_node_modules_path
|
|
|
|
.join(".deno")
|
|
|
|
.join(get_package_folder_id_folder_name(&cache_folder_id))
|
|
|
|
.join("node_modules")
|
|
|
|
.join(&cache_folder_id.nv.name),
|
|
|
|
)
|
2022-11-08 14:17:24 -05:00
|
|
|
}
|
2022-09-22 11:17:02 -04:00
|
|
|
|
|
|
|
fn resolve_package_folder_from_package(
|
|
|
|
&self,
|
|
|
|
name: &str,
|
|
|
|
referrer: &ModuleSpecifier,
|
2024-07-09 12:15:03 -04:00
|
|
|
) -> Result<PathBuf, PackageFolderResolveError> {
|
|
|
|
let maybe_local_path = self
|
|
|
|
.resolve_folder_for_specifier(referrer)
|
2024-07-23 20:22:24 -04:00
|
|
|
.map_err(|err| PackageFolderResolveIoError {
|
2024-07-09 12:15:03 -04:00
|
|
|
package_name: name.to_string(),
|
|
|
|
referrer: referrer.clone(),
|
|
|
|
source: err,
|
|
|
|
})?;
|
|
|
|
let Some(local_path) = maybe_local_path else {
|
|
|
|
return Err(
|
2024-07-23 20:22:24 -04:00
|
|
|
ReferrerNotFoundError {
|
2024-07-09 12:15:03 -04:00
|
|
|
referrer: referrer.clone(),
|
|
|
|
referrer_extra: None,
|
|
|
|
}
|
|
|
|
.into(),
|
|
|
|
);
|
2023-07-17 14:00:44 -04:00
|
|
|
};
|
2022-09-22 11:17:02 -04:00
|
|
|
let package_root_path = self.resolve_package_root(&local_path);
|
|
|
|
let mut current_folder = package_root_path.as_path();
|
2024-04-25 10:13:55 -04:00
|
|
|
while let Some(parent_folder) = current_folder.parent() {
|
|
|
|
current_folder = parent_folder;
|
2023-05-24 15:04:21 -04:00
|
|
|
let node_modules_folder = if current_folder.ends_with("node_modules") {
|
|
|
|
Cow::Borrowed(current_folder)
|
|
|
|
} else {
|
|
|
|
Cow::Owned(current_folder.join("node_modules"))
|
|
|
|
};
|
2022-10-21 11:20:18 -04:00
|
|
|
|
2023-10-25 14:39:00 -04:00
|
|
|
let sub_dir = join_package_name(&node_modules_folder, name);
|
2024-12-30 12:38:20 -05:00
|
|
|
if self.sys.fs_is_dir_no_err(&sub_dir) {
|
2023-10-25 14:39:00 -04:00
|
|
|
return Ok(sub_dir);
|
|
|
|
}
|
|
|
|
|
2022-09-22 11:17:02 -04:00
|
|
|
if current_folder == self.root_node_modules_path {
|
2024-04-25 10:13:55 -04:00
|
|
|
break;
|
2022-09-22 11:17:02 -04:00
|
|
|
}
|
|
|
|
}
|
2024-04-25 10:13:55 -04:00
|
|
|
|
2024-07-09 12:15:03 -04:00
|
|
|
Err(
|
2024-07-23 20:22:24 -04:00
|
|
|
PackageNotFoundError {
|
2024-07-09 12:15:03 -04:00
|
|
|
package_name: name.to_string(),
|
|
|
|
referrer: referrer.clone(),
|
|
|
|
referrer_extra: None,
|
|
|
|
}
|
|
|
|
.into(),
|
|
|
|
)
|
2022-09-22 11:17:02 -04:00
|
|
|
}
|
|
|
|
|
2023-07-01 21:07:57 -04:00
|
|
|
fn resolve_package_cache_folder_id_from_specifier(
|
|
|
|
&self,
|
|
|
|
specifier: &ModuleSpecifier,
|
2025-01-08 14:52:32 -08:00
|
|
|
) -> Result<Option<NpmPackageCacheFolderId>, std::io::Error> {
|
2023-08-27 12:04:12 +08:00
|
|
|
let Some(folder_path) =
|
|
|
|
self.resolve_package_folder_from_specifier(specifier)?
|
|
|
|
else {
|
2023-07-17 14:00:44 -04:00
|
|
|
return Ok(None);
|
|
|
|
};
|
2024-12-04 12:05:34 -05:00
|
|
|
// ex. project/node_modules/.deno/preact@10.24.3/node_modules/preact/
|
|
|
|
let Some(node_modules_ancestor) = folder_path
|
|
|
|
.ancestors()
|
|
|
|
.find(|ancestor| ancestor.ends_with("node_modules"))
|
|
|
|
else {
|
|
|
|
return Ok(None);
|
|
|
|
};
|
|
|
|
let Some(folder_name) =
|
|
|
|
node_modules_ancestor.parent().and_then(|p| p.file_name())
|
|
|
|
else {
|
|
|
|
return Ok(None);
|
|
|
|
};
|
|
|
|
Ok(get_package_folder_id_from_folder_name(
|
|
|
|
&folder_name.to_string_lossy(),
|
|
|
|
))
|
2023-07-01 21:07:57 -04:00
|
|
|
}
|
2022-09-22 11:17:02 -04:00
|
|
|
}
|
|
|
|
|
2025-01-08 18:46:37 -05:00
|
|
|
pub fn get_package_folder_id_folder_name(
|
2023-02-21 12:03:48 -05:00
|
|
|
folder_id: &NpmPackageCacheFolderId,
|
|
|
|
) -> String {
|
|
|
|
let copy_str = if folder_id.copy_index == 0 {
|
2024-09-04 16:00:44 +02:00
|
|
|
Cow::Borrowed("")
|
2022-11-08 14:17:24 -05:00
|
|
|
} else {
|
2024-09-04 16:00:44 +02:00
|
|
|
Cow::Owned(format!("_{}", folder_id.copy_index))
|
2022-11-08 14:17:24 -05:00
|
|
|
};
|
2023-02-21 12:03:48 -05:00
|
|
|
let nv = &folder_id.nv;
|
2024-09-04 16:00:44 +02:00
|
|
|
let name = normalize_pkg_name_for_node_modules_deno_folder(&nv.name);
|
|
|
|
format!("{}@{}{}", name, nv.version, copy_str)
|
2022-11-08 14:17:24 -05:00
|
|
|
}
|
|
|
|
|
2023-07-01 21:07:57 -04:00
|
|
|
fn get_package_folder_id_from_folder_name(
|
|
|
|
folder_name: &str,
|
|
|
|
) -> Option<NpmPackageCacheFolderId> {
|
|
|
|
let folder_name = folder_name.replace('+', "/");
|
|
|
|
let (name, ending) = folder_name.rsplit_once('@')?;
|
2024-12-20 16:14:37 -05:00
|
|
|
let name: StackString = if let Some(encoded_name) = name.strip_prefix('_') {
|
|
|
|
StackString::from_string(mixed_case_package_name_decode(encoded_name)?)
|
2023-07-01 21:07:57 -04:00
|
|
|
} else {
|
2024-12-20 16:14:37 -05:00
|
|
|
name.into()
|
2023-07-01 21:07:57 -04:00
|
|
|
};
|
|
|
|
let (raw_version, copy_index) = match ending.split_once('_') {
|
|
|
|
Some((raw_version, copy_index)) => {
|
|
|
|
let copy_index = copy_index.parse::<u8>().ok()?;
|
|
|
|
(raw_version, copy_index)
|
|
|
|
}
|
|
|
|
None => (ending, 0),
|
|
|
|
};
|
|
|
|
let version = deno_semver::Version::parse_from_npm(raw_version).ok()?;
|
|
|
|
Some(NpmPackageCacheFolderId {
|
2023-08-21 11:53:52 +02:00
|
|
|
nv: PackageNv { name, version },
|
2023-07-01 21:07:57 -04:00
|
|
|
copy_index,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2022-09-22 11:17:02 -04:00
|
|
|
fn join_package_name(path: &Path, package_name: &str) -> PathBuf {
|
|
|
|
let mut path = path.to_path_buf();
|
|
|
|
// ensure backslashes are used on windows
|
|
|
|
for part in package_name.split('/') {
|
|
|
|
path = path.join(part);
|
|
|
|
}
|
|
|
|
path
|
|
|
|
}
|
2023-07-01 21:07:57 -04:00
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
mod test {
|
|
|
|
use deno_npm::NpmPackageCacheFolderId;
|
2023-08-21 11:53:52 +02:00
|
|
|
use deno_semver::package::PackageNv;
|
2023-07-01 21:07:57 -04:00
|
|
|
|
|
|
|
use super::*;
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_get_package_folder_id_folder_name() {
|
|
|
|
let cases = vec![
|
|
|
|
(
|
|
|
|
NpmPackageCacheFolderId {
|
2023-08-21 11:53:52 +02:00
|
|
|
nv: PackageNv::from_str("@types/foo@1.2.3").unwrap(),
|
2023-07-01 21:07:57 -04:00
|
|
|
copy_index: 1,
|
|
|
|
},
|
|
|
|
"@types+foo@1.2.3_1".to_string(),
|
|
|
|
),
|
|
|
|
(
|
|
|
|
NpmPackageCacheFolderId {
|
2023-08-21 11:53:52 +02:00
|
|
|
nv: PackageNv::from_str("JSON@3.2.1").unwrap(),
|
2023-07-01 21:07:57 -04:00
|
|
|
copy_index: 0,
|
|
|
|
},
|
|
|
|
"_jjju6tq@3.2.1".to_string(),
|
|
|
|
),
|
|
|
|
];
|
|
|
|
for (input, output) in cases {
|
|
|
|
assert_eq!(get_package_folder_id_folder_name(&input), output);
|
|
|
|
let folder_id = get_package_folder_id_from_folder_name(&output).unwrap();
|
|
|
|
assert_eq!(folder_id, input);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|