mirror of
https://github.com/denoland/deno.git
synced 2025-01-11 16:42:21 -05:00
fix(node): support resolving a package.json import to a builtin node module (#21576)
Closes https://github.com/denoland/deno/issues/21501
This commit is contained in:
parent
19d52b9a55
commit
ac04787c30
10 changed files with 205 additions and 124 deletions
|
@ -2,3 +2,6 @@ import data from "@denotest/imports-package-json";
|
||||||
|
|
||||||
console.log(data.hi);
|
console.log(data.hi);
|
||||||
console.log(data.bye);
|
console.log(data.bye);
|
||||||
|
console.log(typeof data.fs.readFile);
|
||||||
|
console.log(typeof data.path.join);
|
||||||
|
console.log(typeof data.fs2.writeFile);
|
||||||
|
|
|
@ -2,3 +2,6 @@ Download http://localhost:4545/npm/registry/@denotest/imports-package-json
|
||||||
Download http://localhost:4545/npm/registry/@denotest/imports-package-json/1.0.0.tgz
|
Download http://localhost:4545/npm/registry/@denotest/imports-package-json/1.0.0.tgz
|
||||||
hi
|
hi
|
||||||
bye
|
bye
|
||||||
|
function
|
||||||
|
function
|
||||||
|
function
|
||||||
|
|
|
@ -1,7 +1,13 @@
|
||||||
import hi from "#hi";
|
import hi from "#hi";
|
||||||
import bye from "./sub_path/main.js";
|
import bye from "./sub_path/main.js";
|
||||||
|
import fs from "#fs";
|
||||||
|
import path from "#path";
|
||||||
|
import fs2 from "#fs2";
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
hi,
|
hi,
|
||||||
bye,
|
bye,
|
||||||
|
fs,
|
||||||
|
path,
|
||||||
|
fs2,
|
||||||
};
|
};
|
||||||
|
|
|
@ -11,6 +11,11 @@
|
||||||
"./sub-path-import-not-defined": "./sub_path/import_not_defined.js"
|
"./sub-path-import-not-defined": "./sub_path/import_not_defined.js"
|
||||||
},
|
},
|
||||||
"imports": {
|
"imports": {
|
||||||
"#hi": "./hi.js"
|
"#hi": "./hi.js",
|
||||||
|
"#fs": "fs",
|
||||||
|
"#path": "node:path",
|
||||||
|
"#fs2": {
|
||||||
|
"node": "fs"
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -11,6 +11,7 @@ use once_cell::sync::Lazy;
|
||||||
|
|
||||||
use deno_core::error::AnyError;
|
use deno_core::error::AnyError;
|
||||||
|
|
||||||
|
use crate::path::to_file_specifier;
|
||||||
use crate::resolution::NodeResolverRc;
|
use crate::resolution::NodeResolverRc;
|
||||||
use crate::NodeModuleKind;
|
use crate::NodeModuleKind;
|
||||||
use crate::NodePermissions;
|
use crate::NodePermissions;
|
||||||
|
@ -106,7 +107,7 @@ impl<TCjsCodeAnalyzer: CjsCodeAnalyzer> NodeCodeTranslator<TCjsCodeAnalyzer> {
|
||||||
handled_reexports.insert(reexport.to_string());
|
handled_reexports.insert(reexport.to_string());
|
||||||
|
|
||||||
// First, resolve the reexport specifier
|
// First, resolve the reexport specifier
|
||||||
let resolved_reexport = self.resolve(
|
let reexport_specifier = self.resolve(
|
||||||
&reexport,
|
&reexport,
|
||||||
&referrer,
|
&referrer,
|
||||||
// FIXME(bartlomieju): check if these conditions are okay, probably
|
// FIXME(bartlomieju): check if these conditions are okay, probably
|
||||||
|
@ -117,8 +118,6 @@ impl<TCjsCodeAnalyzer: CjsCodeAnalyzer> NodeCodeTranslator<TCjsCodeAnalyzer> {
|
||||||
)?;
|
)?;
|
||||||
|
|
||||||
// Second, resolve its exports and re-exports
|
// Second, resolve its exports and re-exports
|
||||||
let reexport_specifier =
|
|
||||||
ModuleSpecifier::from_file_path(&resolved_reexport).unwrap();
|
|
||||||
let analysis = self
|
let analysis = self
|
||||||
.cjs_code_analyzer
|
.cjs_code_analyzer
|
||||||
.analyze_cjs(&reexport_specifier, None)
|
.analyze_cjs(&reexport_specifier, None)
|
||||||
|
@ -177,7 +176,7 @@ impl<TCjsCodeAnalyzer: CjsCodeAnalyzer> NodeCodeTranslator<TCjsCodeAnalyzer> {
|
||||||
conditions: &[&str],
|
conditions: &[&str],
|
||||||
mode: NodeResolutionMode,
|
mode: NodeResolutionMode,
|
||||||
permissions: &dyn NodePermissions,
|
permissions: &dyn NodePermissions,
|
||||||
) -> Result<PathBuf, AnyError> {
|
) -> Result<ModuleSpecifier, AnyError> {
|
||||||
if specifier.starts_with('/') {
|
if specifier.starts_with('/') {
|
||||||
todo!();
|
todo!();
|
||||||
}
|
}
|
||||||
|
@ -186,7 +185,8 @@ impl<TCjsCodeAnalyzer: CjsCodeAnalyzer> NodeCodeTranslator<TCjsCodeAnalyzer> {
|
||||||
if specifier.starts_with("./") || specifier.starts_with("../") {
|
if specifier.starts_with("./") || specifier.starts_with("../") {
|
||||||
if let Some(parent) = referrer_path.parent() {
|
if let Some(parent) = referrer_path.parent() {
|
||||||
return self
|
return self
|
||||||
.file_extension_probe(parent.join(specifier), &referrer_path);
|
.file_extension_probe(parent.join(specifier), &referrer_path)
|
||||||
|
.map(|p| to_file_specifier(&p));
|
||||||
} else {
|
} else {
|
||||||
todo!();
|
todo!();
|
||||||
}
|
}
|
||||||
|
@ -238,17 +238,19 @@ impl<TCjsCodeAnalyzer: CjsCodeAnalyzer> NodeCodeTranslator<TCjsCodeAnalyzer> {
|
||||||
)?;
|
)?;
|
||||||
if package_json.exists {
|
if package_json.exists {
|
||||||
if let Some(main) = package_json.main(NodeModuleKind::Cjs) {
|
if let Some(main) = package_json.main(NodeModuleKind::Cjs) {
|
||||||
return Ok(d.join(main).clean());
|
return Ok(to_file_specifier(&d.join(main).clean()));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return Ok(d.join("index.js").clean());
|
return Ok(to_file_specifier(&d.join("index.js").clean()));
|
||||||
}
|
}
|
||||||
return self.file_extension_probe(d, &referrer_path);
|
return self
|
||||||
|
.file_extension_probe(d, &referrer_path)
|
||||||
|
.map(|p| to_file_specifier(&p));
|
||||||
} else if let Some(main) = package_json.main(NodeModuleKind::Cjs) {
|
} else if let Some(main) = package_json.main(NodeModuleKind::Cjs) {
|
||||||
return Ok(module_dir.join(main).clean());
|
return Ok(to_file_specifier(&module_dir.join(main).clean()));
|
||||||
} else {
|
} else {
|
||||||
return Ok(module_dir.join("index.js").clean());
|
return Ok(to_file_specifier(&module_dir.join("index.js").clean()));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -264,7 +266,7 @@ impl<TCjsCodeAnalyzer: CjsCodeAnalyzer> NodeCodeTranslator<TCjsCodeAnalyzer> {
|
||||||
parent.join("node_modules").join(specifier)
|
parent.join("node_modules").join(specifier)
|
||||||
};
|
};
|
||||||
if let Ok(path) = self.file_extension_probe(path, &referrer_path) {
|
if let Ok(path) = self.file_extension_probe(path, &referrer_path) {
|
||||||
return Ok(path);
|
return Ok(to_file_specifier(&path));
|
||||||
}
|
}
|
||||||
last = parent;
|
last = parent;
|
||||||
}
|
}
|
||||||
|
|
|
@ -54,7 +54,7 @@ pub fn err_module_not_found(path: &str, base: &str, typ: &str) -> AnyError {
|
||||||
pub fn err_invalid_package_target(
|
pub fn err_invalid_package_target(
|
||||||
pkg_path: &str,
|
pkg_path: &str,
|
||||||
key: &str,
|
key: &str,
|
||||||
target: String,
|
target: &str,
|
||||||
is_import: bool,
|
is_import: bool,
|
||||||
maybe_referrer: Option<String>,
|
maybe_referrer: Option<String>,
|
||||||
) -> AnyError {
|
) -> AnyError {
|
||||||
|
|
|
@ -431,7 +431,13 @@ where
|
||||||
NodeResolutionMode::Execution,
|
NodeResolutionMode::Execution,
|
||||||
permissions,
|
permissions,
|
||||||
)
|
)
|
||||||
.map(|r| Some(r.to_string_lossy().to_string()))
|
.map(|r| {
|
||||||
|
Some(if r.scheme() == "file" {
|
||||||
|
r.to_file_path().unwrap().to_string_lossy().to_string()
|
||||||
|
} else {
|
||||||
|
r.to_string()
|
||||||
|
})
|
||||||
|
})
|
||||||
} else {
|
} else {
|
||||||
Ok(None)
|
Ok(None)
|
||||||
}
|
}
|
||||||
|
@ -515,7 +521,13 @@ where
|
||||||
NodeResolutionMode::Execution,
|
NodeResolutionMode::Execution,
|
||||||
permissions,
|
permissions,
|
||||||
)
|
)
|
||||||
.map(|r| Some(r.to_string_lossy().to_string()))
|
.map(|r| {
|
||||||
|
Some(if r.scheme() == "file" {
|
||||||
|
r.to_file_path().unwrap().to_string_lossy().to_string()
|
||||||
|
} else {
|
||||||
|
r.to_string()
|
||||||
|
})
|
||||||
|
})
|
||||||
} else {
|
} else {
|
||||||
Ok(None)
|
Ok(None)
|
||||||
}
|
}
|
||||||
|
@ -588,7 +600,7 @@ where
|
||||||
NodeResolutionMode::Execution,
|
NodeResolutionMode::Execution,
|
||||||
permissions,
|
permissions,
|
||||||
)
|
)
|
||||||
.map(|r| Some(Url::from_file_path(r).unwrap().to_string()))
|
.map(|r| Some(r.to_string()))
|
||||||
} else {
|
} else {
|
||||||
Ok(None)
|
Ok(None)
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,8 +1,11 @@
|
||||||
// Copyright 2018-2023 the Deno authors. All rights reserved. MIT license.
|
// Copyright 2018-2023 the Deno authors. All rights reserved. MIT license.
|
||||||
|
|
||||||
use std::path::Component;
|
use std::path::Component;
|
||||||
|
use std::path::Path;
|
||||||
use std::path::PathBuf;
|
use std::path::PathBuf;
|
||||||
|
|
||||||
|
use deno_core::ModuleSpecifier;
|
||||||
|
|
||||||
/// Extension to path_clean::PathClean
|
/// Extension to path_clean::PathClean
|
||||||
pub trait PathClean<T> {
|
pub trait PathClean<T> {
|
||||||
fn clean(&self) -> T;
|
fn clean(&self) -> T;
|
||||||
|
@ -38,3 +41,10 @@ impl PathClean<PathBuf> for PathBuf {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub(crate) fn to_file_specifier(path: &Path) -> ModuleSpecifier {
|
||||||
|
match ModuleSpecifier::from_file_path(path) {
|
||||||
|
Ok(url) => url,
|
||||||
|
Err(_) => panic!("Invalid path: {}", path.display()),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
|
@ -1,5 +1,7 @@
|
||||||
// Copyright 2018-2023 the Deno authors. All rights reserved. MIT license.
|
// Copyright 2018-2023 the Deno authors. All rights reserved. MIT license.
|
||||||
|
|
||||||
|
use deno_core::ModuleSpecifier;
|
||||||
|
|
||||||
/// e.g. `is_builtin_node_module("assert")`
|
/// e.g. `is_builtin_node_module("assert")`
|
||||||
pub fn is_builtin_node_module(module_name: &str) -> bool {
|
pub fn is_builtin_node_module(module_name: &str) -> bool {
|
||||||
SUPPORTED_BUILTIN_NODE_MODULES
|
SUPPORTED_BUILTIN_NODE_MODULES
|
||||||
|
@ -7,6 +9,18 @@ pub fn is_builtin_node_module(module_name: &str) -> bool {
|
||||||
.any(|m| *m == module_name)
|
.any(|m| *m == module_name)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Ex. returns `fs` for `node:fs`
|
||||||
|
pub fn get_module_name_from_builtin_node_module_specifier(
|
||||||
|
specifier: &ModuleSpecifier,
|
||||||
|
) -> Option<&str> {
|
||||||
|
if specifier.scheme() != "node" {
|
||||||
|
return None;
|
||||||
|
}
|
||||||
|
|
||||||
|
let (_, specifier) = specifier.as_str().split_once(':')?;
|
||||||
|
Some(specifier)
|
||||||
|
}
|
||||||
|
|
||||||
macro_rules! generate_builtin_node_module_lists {
|
macro_rules! generate_builtin_node_module_lists {
|
||||||
($( $module_name:literal ,)+) => {
|
($( $module_name:literal ,)+) => {
|
||||||
pub static SUPPORTED_BUILTIN_NODE_MODULES: &[&str] = &[
|
pub static SUPPORTED_BUILTIN_NODE_MODULES: &[&str] = &[
|
||||||
|
|
|
@ -17,6 +17,9 @@ use deno_fs::FileSystemRc;
|
||||||
use deno_media_type::MediaType;
|
use deno_media_type::MediaType;
|
||||||
|
|
||||||
use crate::errors;
|
use crate::errors;
|
||||||
|
use crate::is_builtin_node_module;
|
||||||
|
use crate::path::to_file_specifier;
|
||||||
|
use crate::polyfill::get_module_name_from_builtin_node_module_specifier;
|
||||||
use crate::AllowAllNodePermissions;
|
use crate::AllowAllNodePermissions;
|
||||||
use crate::NodePermissions;
|
use crate::NodePermissions;
|
||||||
use crate::NpmResolverRc;
|
use crate::NpmResolverRc;
|
||||||
|
@ -164,17 +167,14 @@ impl NodeResolver {
|
||||||
return Ok(Some(NodeResolution::Esm(url)));
|
return Ok(Some(NodeResolution::Esm(url)));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if let Some(module_name) =
|
||||||
|
get_module_name_from_builtin_node_module_specifier(&url)
|
||||||
|
{
|
||||||
|
return Ok(Some(NodeResolution::BuiltIn(module_name.to_string())));
|
||||||
|
}
|
||||||
|
|
||||||
let protocol = url.scheme();
|
let protocol = url.scheme();
|
||||||
|
|
||||||
if protocol == "node" {
|
|
||||||
let split_specifier = url.as_str().split(':');
|
|
||||||
let specifier = split_specifier.skip(1).collect::<String>();
|
|
||||||
|
|
||||||
if crate::is_builtin_node_module(&specifier) {
|
|
||||||
return Ok(Some(NodeResolution::BuiltIn(specifier)));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if protocol != "file" && protocol != "data" {
|
if protocol != "file" && protocol != "data" {
|
||||||
return Err(errors::err_unsupported_esm_url_scheme(&url));
|
return Err(errors::err_unsupported_esm_url_scheme(&url));
|
||||||
}
|
}
|
||||||
|
@ -203,12 +203,10 @@ impl NodeResolver {
|
||||||
let path = url.to_file_path().unwrap();
|
let path = url.to_file_path().unwrap();
|
||||||
// todo(16370): the module kind is not correct here. I think we need
|
// todo(16370): the module kind is not correct here. I think we need
|
||||||
// typescript to tell us if the referrer is esm or cjs
|
// typescript to tell us if the referrer is esm or cjs
|
||||||
let path =
|
|
||||||
match self.path_to_declaration_path(path, NodeModuleKind::Esm) {
|
match self.path_to_declaration_path(path, NodeModuleKind::Esm) {
|
||||||
Some(path) => path,
|
Some(path) => to_file_specifier(&path),
|
||||||
None => return Ok(None),
|
None => return Ok(None),
|
||||||
};
|
}
|
||||||
ModuleSpecifier::from_file_path(path).unwrap()
|
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -235,30 +233,24 @@ impl NodeResolver {
|
||||||
// should use the value provided by typescript instead
|
// should use the value provided by typescript instead
|
||||||
let declaration_path =
|
let declaration_path =
|
||||||
self.path_to_declaration_path(file_path, NodeModuleKind::Esm);
|
self.path_to_declaration_path(file_path, NodeModuleKind::Esm);
|
||||||
declaration_path.map(|declaration_path| {
|
declaration_path
|
||||||
ModuleSpecifier::from_file_path(declaration_path).unwrap()
|
.map(|declaration_path| to_file_specifier(&declaration_path))
|
||||||
})
|
|
||||||
} else {
|
} else {
|
||||||
Some(resolved_specifier)
|
Some(resolved_specifier)
|
||||||
}
|
}
|
||||||
} else if specifier.starts_with('#') {
|
} else if specifier.starts_with('#') {
|
||||||
Some(
|
Some(self.package_imports_resolve(
|
||||||
self
|
|
||||||
.package_imports_resolve(
|
|
||||||
specifier,
|
specifier,
|
||||||
referrer,
|
referrer,
|
||||||
NodeModuleKind::Esm,
|
NodeModuleKind::Esm,
|
||||||
conditions,
|
conditions,
|
||||||
mode,
|
mode,
|
||||||
permissions,
|
permissions,
|
||||||
)
|
)?)
|
||||||
.map(|p| ModuleSpecifier::from_file_path(p).unwrap())?,
|
|
||||||
)
|
|
||||||
} else if let Ok(resolved) = Url::parse(specifier) {
|
} else if let Ok(resolved) = Url::parse(specifier) {
|
||||||
Some(resolved)
|
Some(resolved)
|
||||||
} else {
|
} else {
|
||||||
self
|
self.package_resolve(
|
||||||
.package_resolve(
|
|
||||||
specifier,
|
specifier,
|
||||||
referrer,
|
referrer,
|
||||||
NodeModuleKind::Esm,
|
NodeModuleKind::Esm,
|
||||||
|
@ -266,7 +258,6 @@ impl NodeResolver {
|
||||||
mode,
|
mode,
|
||||||
permissions,
|
permissions,
|
||||||
)?
|
)?
|
||||||
.map(|p| ModuleSpecifier::from_file_path(p).unwrap())
|
|
||||||
};
|
};
|
||||||
Ok(match url {
|
Ok(match url {
|
||||||
Some(url) => Some(self.finalize_resolution(url, referrer)?),
|
Some(url) => Some(self.finalize_resolution(url, referrer)?),
|
||||||
|
@ -289,6 +280,10 @@ impl NodeResolver {
|
||||||
));
|
));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if resolved.scheme() == "node" {
|
||||||
|
return Ok(resolved);
|
||||||
|
}
|
||||||
|
|
||||||
let path = to_file_path(&resolved);
|
let path = to_file_path(&resolved);
|
||||||
|
|
||||||
// TODO(bartlomieju): currently not supported
|
// TODO(bartlomieju): currently not supported
|
||||||
|
@ -340,7 +335,7 @@ impl NodeResolver {
|
||||||
let package_subpath = package_subpath
|
let package_subpath = package_subpath
|
||||||
.map(|s| format!("./{s}"))
|
.map(|s| format!("./{s}"))
|
||||||
.unwrap_or_else(|| ".".to_string());
|
.unwrap_or_else(|| ".".to_string());
|
||||||
let maybe_resolved_path = self
|
let maybe_resolved_url = self
|
||||||
.resolve_package_subpath(
|
.resolve_package_subpath(
|
||||||
&package_json,
|
&package_json,
|
||||||
&package_subpath,
|
&package_subpath,
|
||||||
|
@ -357,21 +352,25 @@ impl NodeResolver {
|
||||||
package_json.path.display()
|
package_json.path.display()
|
||||||
)
|
)
|
||||||
})?;
|
})?;
|
||||||
let resolved_path = match maybe_resolved_path {
|
let resolved_url = match maybe_resolved_url {
|
||||||
Some(resolved_path) => resolved_path,
|
Some(resolved_path) => resolved_path,
|
||||||
None => return Ok(None),
|
None => return Ok(None),
|
||||||
};
|
};
|
||||||
let resolved_path = match mode {
|
let resolved_url = match mode {
|
||||||
NodeResolutionMode::Execution => resolved_path,
|
NodeResolutionMode::Execution => resolved_url,
|
||||||
NodeResolutionMode::Types => {
|
NodeResolutionMode::Types => {
|
||||||
match self.path_to_declaration_path(resolved_path, node_module_kind) {
|
if resolved_url.scheme() == "file" {
|
||||||
Some(path) => path,
|
let path = resolved_url.to_file_path().unwrap();
|
||||||
|
match self.path_to_declaration_path(path, node_module_kind) {
|
||||||
|
Some(path) => to_file_specifier(&path),
|
||||||
None => return Ok(None),
|
None => return Ok(None),
|
||||||
}
|
}
|
||||||
|
} else {
|
||||||
|
resolved_url
|
||||||
|
}
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
let url = ModuleSpecifier::from_file_path(resolved_path).unwrap();
|
let resolve_response = self.url_to_node_resolution(resolved_url)?;
|
||||||
let resolve_response = self.url_to_node_resolution(url)?;
|
|
||||||
// TODO(bartlomieju): skipped checking errors for commonJS resolution and
|
// TODO(bartlomieju): skipped checking errors for commonJS resolution and
|
||||||
// "preserveSymlinksMain"/"preserveSymlinks" options.
|
// "preserveSymlinksMain"/"preserveSymlinks" options.
|
||||||
Ok(Some(resolve_response))
|
Ok(Some(resolve_response))
|
||||||
|
@ -408,8 +407,7 @@ impl NodeResolver {
|
||||||
let package_json = self
|
let package_json = self
|
||||||
.load_package_json(&AllowAllNodePermissions, package_json_path.clone())?;
|
.load_package_json(&AllowAllNodePermissions, package_json_path.clone())?;
|
||||||
let bin_entry = resolve_bin_entry_value(&package_json, sub_path)?;
|
let bin_entry = resolve_bin_entry_value(&package_json, sub_path)?;
|
||||||
let url =
|
let url = to_file_specifier(&package_folder.join(bin_entry));
|
||||||
ModuleSpecifier::from_file_path(package_folder.join(bin_entry)).unwrap();
|
|
||||||
|
|
||||||
let resolve_response = self.url_to_node_resolution(url)?;
|
let resolve_response = self.url_to_node_resolution(url)?;
|
||||||
// TODO(bartlomieju): skipped checking errors for commonJS resolution and
|
// TODO(bartlomieju): skipped checking errors for commonJS resolution and
|
||||||
|
@ -531,7 +529,7 @@ impl NodeResolver {
|
||||||
conditions: &[&str],
|
conditions: &[&str],
|
||||||
mode: NodeResolutionMode,
|
mode: NodeResolutionMode,
|
||||||
permissions: &dyn NodePermissions,
|
permissions: &dyn NodePermissions,
|
||||||
) -> Result<PathBuf, AnyError> {
|
) -> Result<ModuleSpecifier, AnyError> {
|
||||||
if name == "#" || name.starts_with("#/") || name.ends_with('/') {
|
if name == "#" || name.starts_with("#/") || name.ends_with('/') {
|
||||||
let reason = "is not a valid internal imports specifier name";
|
let reason = "is not a valid internal imports specifier name";
|
||||||
return Err(errors::err_invalid_module_specifier(
|
return Err(errors::err_invalid_module_specifier(
|
||||||
|
@ -549,9 +547,10 @@ impl NodeResolver {
|
||||||
package_json_path = Some(package_config.path.clone());
|
package_json_path = Some(package_config.path.clone());
|
||||||
if let Some(imports) = &package_config.imports {
|
if let Some(imports) = &package_config.imports {
|
||||||
if imports.contains_key(name) && !name.contains('*') {
|
if imports.contains_key(name) && !name.contains('*') {
|
||||||
|
let target = imports.get(name).unwrap();
|
||||||
let maybe_resolved = self.resolve_package_target(
|
let maybe_resolved = self.resolve_package_target(
|
||||||
package_json_path.as_ref().unwrap(),
|
package_json_path.as_ref().unwrap(),
|
||||||
imports.get(name).unwrap().to_owned(),
|
target,
|
||||||
"",
|
"",
|
||||||
name,
|
name,
|
||||||
referrer,
|
referrer,
|
||||||
|
@ -591,7 +590,7 @@ impl NodeResolver {
|
||||||
}
|
}
|
||||||
|
|
||||||
if !best_match.is_empty() {
|
if !best_match.is_empty() {
|
||||||
let target = imports.get(best_match).unwrap().to_owned();
|
let target = imports.get(best_match).unwrap();
|
||||||
let maybe_resolved = self.resolve_package_target(
|
let maybe_resolved = self.resolve_package_target(
|
||||||
package_json_path.as_ref().unwrap(),
|
package_json_path.as_ref().unwrap(),
|
||||||
target,
|
target,
|
||||||
|
@ -624,7 +623,7 @@ impl NodeResolver {
|
||||||
#[allow(clippy::too_many_arguments)]
|
#[allow(clippy::too_many_arguments)]
|
||||||
fn resolve_package_target_string(
|
fn resolve_package_target_string(
|
||||||
&self,
|
&self,
|
||||||
target: String,
|
target: &str,
|
||||||
subpath: &str,
|
subpath: &str,
|
||||||
match_: &str,
|
match_: &str,
|
||||||
package_json_path: &Path,
|
package_json_path: &Path,
|
||||||
|
@ -635,7 +634,7 @@ impl NodeResolver {
|
||||||
conditions: &[&str],
|
conditions: &[&str],
|
||||||
mode: NodeResolutionMode,
|
mode: NodeResolutionMode,
|
||||||
permissions: &dyn NodePermissions,
|
permissions: &dyn NodePermissions,
|
||||||
) -> Result<PathBuf, AnyError> {
|
) -> Result<ModuleSpecifier, AnyError> {
|
||||||
if !subpath.is_empty() && !pattern && !target.ends_with('/') {
|
if !subpath.is_empty() && !pattern && !target.ends_with('/') {
|
||||||
return Err(throw_invalid_package_target(
|
return Err(throw_invalid_package_target(
|
||||||
match_,
|
match_,
|
||||||
|
@ -650,18 +649,25 @@ impl NodeResolver {
|
||||||
let pattern_re = lazy_regex::regex!(r"\*");
|
let pattern_re = lazy_regex::regex!(r"\*");
|
||||||
if !target.starts_with("./") {
|
if !target.starts_with("./") {
|
||||||
if internal && !target.starts_with("../") && !target.starts_with('/') {
|
if internal && !target.starts_with("../") && !target.starts_with('/') {
|
||||||
let is_url = Url::parse(&target).is_ok();
|
let target_url = Url::parse(target);
|
||||||
if !is_url {
|
match target_url {
|
||||||
|
Ok(url) => {
|
||||||
|
if get_module_name_from_builtin_node_module_specifier(&url)
|
||||||
|
.is_some()
|
||||||
|
{
|
||||||
|
return Ok(url);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Err(_) => {
|
||||||
let export_target = if pattern {
|
let export_target = if pattern {
|
||||||
pattern_re
|
pattern_re
|
||||||
.replace(&target, |_caps: ®ex::Captures| subpath)
|
.replace(target, |_caps: ®ex::Captures| subpath)
|
||||||
.to_string()
|
.to_string()
|
||||||
} else {
|
} else {
|
||||||
format!("{target}{subpath}")
|
format!("{target}{subpath}")
|
||||||
};
|
};
|
||||||
let package_json_url =
|
let package_json_url = to_file_specifier(package_json_path);
|
||||||
ModuleSpecifier::from_file_path(package_json_path).unwrap();
|
let result = match self.package_resolve(
|
||||||
return match self.package_resolve(
|
|
||||||
&export_target,
|
&export_target,
|
||||||
&package_json_url,
|
&package_json_url,
|
||||||
referrer_kind,
|
referrer_kind,
|
||||||
|
@ -669,10 +675,25 @@ impl NodeResolver {
|
||||||
mode,
|
mode,
|
||||||
permissions,
|
permissions,
|
||||||
) {
|
) {
|
||||||
Ok(Some(path)) => Ok(path),
|
Ok(Some(url)) => Ok(url),
|
||||||
Ok(None) => Err(generic_error("not found")),
|
Ok(None) => Err(generic_error("not found")),
|
||||||
Err(err) => Err(err),
|
Err(err) => Err(err),
|
||||||
};
|
};
|
||||||
|
|
||||||
|
return match result {
|
||||||
|
Ok(url) => Ok(url),
|
||||||
|
Err(err) => {
|
||||||
|
if is_builtin_node_module(target) {
|
||||||
|
Ok(
|
||||||
|
ModuleSpecifier::parse(&format!("node:{}", target))
|
||||||
|
.unwrap(),
|
||||||
|
)
|
||||||
|
} else {
|
||||||
|
Err(err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return Err(throw_invalid_package_target(
|
return Err(throw_invalid_package_target(
|
||||||
|
@ -693,7 +714,7 @@ impl NodeResolver {
|
||||||
));
|
));
|
||||||
}
|
}
|
||||||
let package_path = package_json_path.parent().unwrap();
|
let package_path = package_json_path.parent().unwrap();
|
||||||
let resolved_path = package_path.join(&target).clean();
|
let resolved_path = package_path.join(target).clean();
|
||||||
if !resolved_path.starts_with(package_path) {
|
if !resolved_path.starts_with(package_path) {
|
||||||
return Err(throw_invalid_package_target(
|
return Err(throw_invalid_package_target(
|
||||||
match_,
|
match_,
|
||||||
|
@ -704,7 +725,7 @@ impl NodeResolver {
|
||||||
));
|
));
|
||||||
}
|
}
|
||||||
if subpath.is_empty() {
|
if subpath.is_empty() {
|
||||||
return Ok(resolved_path);
|
return Ok(to_file_specifier(&resolved_path));
|
||||||
}
|
}
|
||||||
if invalid_segment_re.is_match(subpath) {
|
if invalid_segment_re.is_match(subpath) {
|
||||||
let request = if pattern {
|
let request = if pattern {
|
||||||
|
@ -723,16 +744,16 @@ impl NodeResolver {
|
||||||
let resolved_path_str = resolved_path.to_string_lossy();
|
let resolved_path_str = resolved_path.to_string_lossy();
|
||||||
let replaced = pattern_re
|
let replaced = pattern_re
|
||||||
.replace(&resolved_path_str, |_caps: ®ex::Captures| subpath);
|
.replace(&resolved_path_str, |_caps: ®ex::Captures| subpath);
|
||||||
return Ok(PathBuf::from(replaced.to_string()));
|
return Ok(to_file_specifier(&PathBuf::from(replaced.to_string())));
|
||||||
}
|
}
|
||||||
Ok(resolved_path.join(subpath).clean())
|
Ok(to_file_specifier(&resolved_path.join(subpath).clean()))
|
||||||
}
|
}
|
||||||
|
|
||||||
#[allow(clippy::too_many_arguments)]
|
#[allow(clippy::too_many_arguments)]
|
||||||
fn resolve_package_target(
|
fn resolve_package_target(
|
||||||
&self,
|
&self,
|
||||||
package_json_path: &Path,
|
package_json_path: &Path,
|
||||||
target: Value,
|
target: &Value,
|
||||||
subpath: &str,
|
subpath: &str,
|
||||||
package_subpath: &str,
|
package_subpath: &str,
|
||||||
referrer: &ModuleSpecifier,
|
referrer: &ModuleSpecifier,
|
||||||
|
@ -742,11 +763,11 @@ impl NodeResolver {
|
||||||
conditions: &[&str],
|
conditions: &[&str],
|
||||||
mode: NodeResolutionMode,
|
mode: NodeResolutionMode,
|
||||||
permissions: &dyn NodePermissions,
|
permissions: &dyn NodePermissions,
|
||||||
) -> Result<Option<PathBuf>, AnyError> {
|
) -> Result<Option<ModuleSpecifier>, AnyError> {
|
||||||
if let Some(target) = target.as_str() {
|
if let Some(target) = target.as_str() {
|
||||||
return self
|
return self
|
||||||
.resolve_package_target_string(
|
.resolve_package_target_string(
|
||||||
target.to_string(),
|
target,
|
||||||
subpath,
|
subpath,
|
||||||
package_subpath,
|
package_subpath,
|
||||||
package_json_path,
|
package_json_path,
|
||||||
|
@ -758,11 +779,14 @@ impl NodeResolver {
|
||||||
mode,
|
mode,
|
||||||
permissions,
|
permissions,
|
||||||
)
|
)
|
||||||
.map(|path| {
|
.map(|url| {
|
||||||
if mode.is_types() {
|
if mode.is_types() && url.scheme() == "file" {
|
||||||
self.path_to_declaration_path(path, referrer_kind)
|
let path = url.to_file_path().unwrap();
|
||||||
|
self
|
||||||
|
.path_to_declaration_path(path, referrer_kind)
|
||||||
|
.map(|path| to_file_specifier(&path))
|
||||||
} else {
|
} else {
|
||||||
Some(path)
|
Some(url)
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
} else if let Some(target_arr) = target.as_array() {
|
} else if let Some(target_arr) = target.as_array() {
|
||||||
|
@ -774,7 +798,7 @@ impl NodeResolver {
|
||||||
for target_item in target_arr {
|
for target_item in target_arr {
|
||||||
let resolved_result = self.resolve_package_target(
|
let resolved_result = self.resolve_package_target(
|
||||||
package_json_path,
|
package_json_path,
|
||||||
target_item.to_owned(),
|
target_item,
|
||||||
subpath,
|
subpath,
|
||||||
package_subpath,
|
package_subpath,
|
||||||
referrer,
|
referrer,
|
||||||
|
@ -819,7 +843,7 @@ impl NodeResolver {
|
||||||
|| conditions.contains(&key.as_str())
|
|| conditions.contains(&key.as_str())
|
||||||
|| mode.is_types() && key.as_str() == "types"
|
|| mode.is_types() && key.as_str() == "types"
|
||||||
{
|
{
|
||||||
let condition_target = target_obj.get(key).unwrap().to_owned();
|
let condition_target = target_obj.get(key).unwrap();
|
||||||
|
|
||||||
let resolved = self.resolve_package_target(
|
let resolved = self.resolve_package_target(
|
||||||
package_json_path,
|
package_json_path,
|
||||||
|
@ -848,7 +872,7 @@ impl NodeResolver {
|
||||||
|
|
||||||
Err(throw_invalid_package_target(
|
Err(throw_invalid_package_target(
|
||||||
package_subpath,
|
package_subpath,
|
||||||
target.to_string(),
|
&target.to_string(),
|
||||||
package_json_path,
|
package_json_path,
|
||||||
internal,
|
internal,
|
||||||
referrer,
|
referrer,
|
||||||
|
@ -866,12 +890,12 @@ impl NodeResolver {
|
||||||
conditions: &[&str],
|
conditions: &[&str],
|
||||||
mode: NodeResolutionMode,
|
mode: NodeResolutionMode,
|
||||||
permissions: &dyn NodePermissions,
|
permissions: &dyn NodePermissions,
|
||||||
) -> Result<PathBuf, AnyError> {
|
) -> Result<ModuleSpecifier, AnyError> {
|
||||||
if package_exports.contains_key(package_subpath)
|
if package_exports.contains_key(package_subpath)
|
||||||
&& package_subpath.find('*').is_none()
|
&& package_subpath.find('*').is_none()
|
||||||
&& !package_subpath.ends_with('/')
|
&& !package_subpath.ends_with('/')
|
||||||
{
|
{
|
||||||
let target = package_exports.get(package_subpath).unwrap().to_owned();
|
let target = package_exports.get(package_subpath).unwrap();
|
||||||
let resolved = self.resolve_package_target(
|
let resolved = self.resolve_package_target(
|
||||||
package_json_path,
|
package_json_path,
|
||||||
target,
|
target,
|
||||||
|
@ -885,15 +909,15 @@ impl NodeResolver {
|
||||||
mode,
|
mode,
|
||||||
permissions,
|
permissions,
|
||||||
)?;
|
)?;
|
||||||
if resolved.is_none() {
|
return match resolved {
|
||||||
return Err(throw_exports_not_found(
|
Some(resolved) => Ok(resolved),
|
||||||
|
None => Err(throw_exports_not_found(
|
||||||
package_subpath,
|
package_subpath,
|
||||||
package_json_path,
|
package_json_path,
|
||||||
referrer,
|
referrer,
|
||||||
mode,
|
mode,
|
||||||
));
|
)),
|
||||||
}
|
};
|
||||||
return Ok(resolved.unwrap());
|
|
||||||
}
|
}
|
||||||
|
|
||||||
let mut best_match = "";
|
let mut best_match = "";
|
||||||
|
@ -931,7 +955,7 @@ impl NodeResolver {
|
||||||
}
|
}
|
||||||
|
|
||||||
if !best_match.is_empty() {
|
if !best_match.is_empty() {
|
||||||
let target = package_exports.get(best_match).unwrap().to_owned();
|
let target = package_exports.get(best_match).unwrap();
|
||||||
let maybe_resolved = self.resolve_package_target(
|
let maybe_resolved = self.resolve_package_target(
|
||||||
package_json_path,
|
package_json_path,
|
||||||
target,
|
target,
|
||||||
|
@ -973,7 +997,7 @@ impl NodeResolver {
|
||||||
conditions: &[&str],
|
conditions: &[&str],
|
||||||
mode: NodeResolutionMode,
|
mode: NodeResolutionMode,
|
||||||
permissions: &dyn NodePermissions,
|
permissions: &dyn NodePermissions,
|
||||||
) -> Result<Option<PathBuf>, AnyError> {
|
) -> Result<Option<ModuleSpecifier>, AnyError> {
|
||||||
let (package_name, package_subpath, _is_scoped) =
|
let (package_name, package_subpath, _is_scoped) =
|
||||||
parse_npm_pkg_name(specifier, referrer)?;
|
parse_npm_pkg_name(specifier, referrer)?;
|
||||||
|
|
||||||
|
@ -1044,7 +1068,7 @@ impl NodeResolver {
|
||||||
conditions: &[&str],
|
conditions: &[&str],
|
||||||
mode: NodeResolutionMode,
|
mode: NodeResolutionMode,
|
||||||
permissions: &dyn NodePermissions,
|
permissions: &dyn NodePermissions,
|
||||||
) -> Result<Option<PathBuf>, AnyError> {
|
) -> Result<Option<ModuleSpecifier>, AnyError> {
|
||||||
if let Some(exports) = &package_json.exports {
|
if let Some(exports) = &package_json.exports {
|
||||||
let result = self.package_exports_resolve(
|
let result = self.package_exports_resolve(
|
||||||
&package_json.path,
|
&package_json.path,
|
||||||
|
@ -1063,7 +1087,7 @@ impl NodeResolver {
|
||||||
if let Ok(Some(path)) =
|
if let Ok(Some(path)) =
|
||||||
self.legacy_main_resolve(package_json, referrer_kind, mode)
|
self.legacy_main_resolve(package_json, referrer_kind, mode)
|
||||||
{
|
{
|
||||||
return Ok(Some(path));
|
return Ok(Some(to_file_specifier(&path)));
|
||||||
} else {
|
} else {
|
||||||
return Ok(None);
|
return Ok(None);
|
||||||
}
|
}
|
||||||
|
@ -1073,16 +1097,18 @@ impl NodeResolver {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if package_subpath == "." {
|
if package_subpath == "." {
|
||||||
return self.legacy_main_resolve(package_json, referrer_kind, mode);
|
return self
|
||||||
|
.legacy_main_resolve(package_json, referrer_kind, mode)
|
||||||
|
.map(|maybe_resolved| maybe_resolved.map(|p| to_file_specifier(&p)));
|
||||||
}
|
}
|
||||||
|
|
||||||
let file_path = package_json.path.parent().unwrap().join(package_subpath);
|
let file_path = package_json.path.parent().unwrap().join(package_subpath);
|
||||||
if mode.is_types() {
|
if mode.is_types() {
|
||||||
let maybe_declaration_path =
|
let maybe_declaration_path =
|
||||||
self.path_to_declaration_path(file_path, referrer_kind);
|
self.path_to_declaration_path(file_path, referrer_kind);
|
||||||
Ok(maybe_declaration_path)
|
Ok(maybe_declaration_path.map(|p| to_file_specifier(&p)))
|
||||||
} else {
|
} else {
|
||||||
Ok(Some(file_path))
|
Ok(Some(to_file_specifier(&file_path)))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1416,7 +1442,7 @@ fn throw_import_not_defined(
|
||||||
|
|
||||||
fn throw_invalid_package_target(
|
fn throw_invalid_package_target(
|
||||||
subpath: &str,
|
subpath: &str,
|
||||||
target: String,
|
target: &str,
|
||||||
package_json_path: &Path,
|
package_json_path: &Path,
|
||||||
internal: bool,
|
internal: bool,
|
||||||
referrer: &ModuleSpecifier,
|
referrer: &ModuleSpecifier,
|
||||||
|
|
Loading…
Reference in a new issue