1
0
Fork 0
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:
David Sherret 2023-12-14 16:09:05 +01:00 committed by GitHub
parent 19d52b9a55
commit ac04787c30
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
10 changed files with 205 additions and 124 deletions

View file

@ -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);

View file

@ -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

View file

@ -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,
}; };

View file

@ -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"
}
} }
} }

View file

@ -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;
} }

View file

@ -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 {

View file

@ -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)
} }

View file

@ -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()),
}
}

View file

@ -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] = &[

View file

@ -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: &regex::Captures| subpath) .replace(target, |_caps: &regex::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: &regex::Captures| subpath); .replace(&resolved_path_str, |_caps: &regex::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,