2024-12-31 14:12:39 -05:00
// Copyright 2018-2025 the Deno authors. MIT license.
2024-09-30 09:33:32 -04:00
use std ::borrow ::Cow ;
use std ::path ::Path ;
use std ::path ::PathBuf ;
use deno_package_json ::PackageJson ;
use deno_package_json ::PackageJsonDepValue ;
2024-12-15 03:18:04 -05:00
use deno_package_json ::PackageJsonRc ;
2024-09-30 09:33:32 -04:00
use deno_path_util ::url_to_file_path ;
use deno_semver ::package ::PackageReq ;
2024-12-20 16:14:37 -05:00
use deno_semver ::StackString ;
2024-09-30 09:33:32 -04:00
use deno_semver ::Version ;
use node_resolver ::errors ::PackageFolderResolveError ;
use node_resolver ::errors ::PackageFolderResolveIoError ;
use node_resolver ::errors ::PackageJsonLoadError ;
use node_resolver ::errors ::PackageNotFoundError ;
2024-11-01 12:27:00 -04:00
use node_resolver ::InNpmPackageChecker ;
2024-11-14 15:24:25 -05:00
use node_resolver ::NpmPackageFolderResolver ;
2024-11-01 12:27:00 -04:00
use node_resolver ::PackageJsonResolverRc ;
2024-12-30 12:38:20 -05:00
use sys_traits ::FsCanonicalize ;
use sys_traits ::FsDirEntry ;
use sys_traits ::FsMetadata ;
use sys_traits ::FsRead ;
use sys_traits ::FsReadDir ;
2024-10-02 16:17:39 -04:00
use thiserror ::Error ;
2024-09-30 09:33:32 -04:00
use url ::Url ;
use super ::local ::normalize_pkg_name_for_node_modules_deno_folder ;
2024-11-14 15:24:25 -05:00
use super ::CliNpmReqResolver ;
use super ::ResolvePkgFolderFromDenoReqError ;
2024-09-30 09:33:32 -04:00
2024-10-02 16:17:39 -04:00
#[ derive(Debug, Error) ]
pub enum ByonmResolvePkgFolderFromDenoReqError {
#[ error( " Could not find \" {} \" in a node_modules folder. Deno expects the node_modules/ directory to be up to date. Did you forget to run `deno install`? " , .0) ]
2024-12-20 16:14:37 -05:00
MissingAlias ( StackString ) ,
2024-10-02 16:17:39 -04:00
#[ error(transparent) ]
PackageJson ( #[ from ] PackageJsonLoadError ) ,
#[ error( " Could not find a matching package for 'npm:{}' in the node_modules directory. Ensure you have all your JSR and npm dependencies listed in your deno.json or package.json, then run `deno install`. Alternatively, turn on auto-install by specifying ` \" nodeModulesDir \" : \" auto \" ` in your deno.json file. " , .0) ]
UnmatchedReq ( PackageReq ) ,
#[ error(transparent) ]
Io ( #[ from ] std ::io ::Error ) ,
}
2024-12-30 12:38:20 -05:00
pub struct ByonmNpmResolverCreateOptions < TSys : FsRead > {
2024-09-30 09:33:32 -04:00
// todo(dsherret): investigate removing this
pub root_node_modules_dir : Option < PathBuf > ,
2024-12-30 12:38:20 -05:00
pub sys : TSys ,
pub pkg_json_resolver : PackageJsonResolverRc < TSys > ,
2024-09-30 09:33:32 -04:00
}
2024-12-15 03:18:04 -05:00
#[ allow(clippy::disallowed_types) ]
2024-12-30 12:38:20 -05:00
pub type ByonmNpmResolverRc < TSys > =
crate ::sync ::MaybeArc < ByonmNpmResolver < TSys > > ;
2024-12-15 03:18:04 -05:00
2024-09-30 09:33:32 -04:00
#[ derive(Debug) ]
2024-12-30 12:38:20 -05:00
pub struct ByonmNpmResolver <
TSys : FsCanonicalize + FsRead + FsMetadata + FsReadDir ,
> {
sys : TSys ,
pkg_json_resolver : PackageJsonResolverRc < TSys > ,
2024-09-30 09:33:32 -04:00
root_node_modules_dir : Option < PathBuf > ,
}
2024-12-30 12:38:20 -05:00
impl < TSys : Clone + FsCanonicalize + FsRead + FsMetadata + FsReadDir > Clone
for ByonmNpmResolver < TSys >
2024-11-01 12:27:00 -04:00
{
2024-09-30 09:33:32 -04:00
fn clone ( & self ) -> Self {
Self {
2024-12-30 12:38:20 -05:00
sys : self . sys . clone ( ) ,
2024-11-01 12:27:00 -04:00
pkg_json_resolver : self . pkg_json_resolver . clone ( ) ,
2024-09-30 09:33:32 -04:00
root_node_modules_dir : self . root_node_modules_dir . clone ( ) ,
}
}
}
2024-12-30 12:38:20 -05:00
impl < TSys : FsCanonicalize + FsRead + FsMetadata + FsReadDir >
ByonmNpmResolver < TSys >
{
pub fn new ( options : ByonmNpmResolverCreateOptions < TSys > ) -> Self {
2024-09-30 09:33:32 -04:00
Self {
root_node_modules_dir : options . root_node_modules_dir ,
2024-12-30 12:38:20 -05:00
sys : options . sys ,
2024-11-01 12:27:00 -04:00
pkg_json_resolver : options . pkg_json_resolver ,
2024-09-30 09:33:32 -04:00
}
}
pub fn root_node_modules_dir ( & self ) -> Option < & Path > {
self . root_node_modules_dir . as_deref ( )
}
fn load_pkg_json (
& self ,
path : & Path ,
2024-12-15 03:18:04 -05:00
) -> Result < Option < PackageJsonRc > , PackageJsonLoadError > {
2024-11-01 12:27:00 -04:00
self . pkg_json_resolver . load_package_json ( path )
2024-09-30 09:33:32 -04:00
}
/// Finds the ancestor package.json that contains the specified dependency.
pub fn find_ancestor_package_json_with_dep (
& self ,
dep_name : & str ,
referrer : & Url ,
2024-12-15 03:18:04 -05:00
) -> Option < PackageJsonRc > {
2024-09-30 09:33:32 -04:00
let referrer_path = url_to_file_path ( referrer ) . ok ( ) ? ;
let mut current_folder = referrer_path . parent ( ) ? ;
loop {
let pkg_json_path = current_folder . join ( " package.json " ) ;
if let Ok ( Some ( pkg_json ) ) = self . load_pkg_json ( & pkg_json_path ) {
if let Some ( deps ) = & pkg_json . dependencies {
if deps . contains_key ( dep_name ) {
return Some ( pkg_json ) ;
}
}
if let Some ( deps ) = & pkg_json . dev_dependencies {
if deps . contains_key ( dep_name ) {
return Some ( pkg_json ) ;
}
}
}
if let Some ( parent ) = current_folder . parent ( ) {
current_folder = parent ;
} else {
return None ;
}
}
}
pub fn resolve_pkg_folder_from_deno_module_req (
& self ,
req : & PackageReq ,
referrer : & Url ,
2024-10-02 16:17:39 -04:00
) -> Result < PathBuf , ByonmResolvePkgFolderFromDenoReqError > {
2024-12-30 12:38:20 -05:00
fn node_resolve_dir < TSys : FsCanonicalize + FsMetadata > (
sys : & TSys ,
2024-09-30 09:33:32 -04:00
alias : & str ,
start_dir : & Path ,
2024-10-02 16:17:39 -04:00
) -> std ::io ::Result < Option < PathBuf > > {
2024-09-30 09:33:32 -04:00
for ancestor in start_dir . ancestors ( ) {
let node_modules_folder = ancestor . join ( " node_modules " ) ;
let sub_dir = join_package_name ( & node_modules_folder , alias ) ;
2024-12-30 12:38:20 -05:00
if sys . fs_is_dir_no_err ( & sub_dir ) {
return Ok ( Some (
deno_path_util ::fs ::canonicalize_path_maybe_not_exists (
sys , & sub_dir ,
) ? ,
) ) ;
2024-09-30 09:33:32 -04:00
}
}
Ok ( None )
}
// now attempt to resolve if it's found in any package.json
let maybe_pkg_json_and_alias =
self . resolve_pkg_json_and_alias_for_req ( req , referrer ) ? ;
match maybe_pkg_json_and_alias {
Some ( ( pkg_json , alias ) ) = > {
// now try node resolution
if let Some ( resolved ) =
2024-12-30 12:38:20 -05:00
node_resolve_dir ( & self . sys , & alias , pkg_json . dir_path ( ) ) ?
2024-09-30 09:33:32 -04:00
{
return Ok ( resolved ) ;
}
2024-10-02 16:17:39 -04:00
Err ( ByonmResolvePkgFolderFromDenoReqError ::MissingAlias ( alias ) )
2024-09-30 09:33:32 -04:00
}
None = > {
// now check if node_modules/.deno/ matches this constraint
if let Some ( folder ) = self . resolve_folder_in_root_node_modules ( req ) {
return Ok ( folder ) ;
}
2024-10-02 16:17:39 -04:00
Err ( ByonmResolvePkgFolderFromDenoReqError ::UnmatchedReq (
req . clone ( ) ,
) )
2024-09-30 09:33:32 -04:00
}
}
}
fn resolve_pkg_json_and_alias_for_req (
& self ,
req : & PackageReq ,
referrer : & Url ,
2024-12-20 16:14:37 -05:00
) -> Result < Option < ( PackageJsonRc , StackString ) > , PackageJsonLoadError > {
2024-09-30 09:33:32 -04:00
fn resolve_alias_from_pkg_json (
req : & PackageReq ,
pkg_json : & PackageJson ,
2024-12-20 16:14:37 -05:00
) -> Option < StackString > {
2024-09-30 09:33:32 -04:00
let deps = pkg_json . resolve_local_package_json_deps ( ) ;
2024-12-20 16:14:37 -05:00
for ( key , value ) in
deps . dependencies . iter ( ) . chain ( deps . dev_dependencies . iter ( ) )
2024-11-29 18:54:26 -05:00
{
2024-09-30 09:33:32 -04:00
if let Ok ( value ) = value {
match value {
PackageJsonDepValue ::Req ( dep_req ) = > {
if dep_req . name = = req . name
& & dep_req . version_req . intersects ( & req . version_req )
{
2024-12-20 16:14:37 -05:00
return Some ( key . clone ( ) ) ;
2024-09-30 09:33:32 -04:00
}
}
PackageJsonDepValue ::Workspace ( _workspace ) = > {
2024-12-20 16:14:37 -05:00
if key . as_str ( ) = = req . name
& & req . version_req . tag ( ) = = Some ( " workspace " )
{
return Some ( key . clone ( ) ) ;
2024-09-30 09:33:32 -04:00
}
}
}
}
}
None
}
// attempt to resolve the npm specifier from the referrer's package.json,
2024-12-12 18:58:14 -05:00
let maybe_referrer_path = url_to_file_path ( referrer ) . ok ( ) ;
if let Some ( file_path ) = maybe_referrer_path {
for dir_path in file_path . as_path ( ) . ancestors ( ) . skip ( 1 ) {
2024-09-30 09:33:32 -04:00
let package_json_path = dir_path . join ( " package.json " ) ;
if let Some ( pkg_json ) = self . load_pkg_json ( & package_json_path ) ? {
if let Some ( alias ) =
resolve_alias_from_pkg_json ( req , pkg_json . as_ref ( ) )
{
return Ok ( Some ( ( pkg_json , alias ) ) ) ;
}
}
}
}
2024-12-12 18:58:14 -05:00
// fall fallback to the project's package.json
2024-09-30 09:33:32 -04:00
if let Some ( root_node_modules_dir ) = & self . root_node_modules_dir {
let root_pkg_json_path =
root_node_modules_dir . parent ( ) . unwrap ( ) . join ( " package.json " ) ;
if let Some ( pkg_json ) = self . load_pkg_json ( & root_pkg_json_path ) ? {
if let Some ( alias ) = resolve_alias_from_pkg_json ( req , pkg_json . as_ref ( ) )
{
return Ok ( Some ( ( pkg_json , alias ) ) ) ;
}
}
}
2024-12-12 18:58:14 -05:00
// now try to resolve based on the closest node_modules directory
let maybe_referrer_path = url_to_file_path ( referrer ) . ok ( ) ;
let search_node_modules = | node_modules : & Path | {
if req . version_req . tag ( ) . is_some ( ) {
return None ;
}
let pkg_folder = node_modules . join ( & req . name ) ;
if let Ok ( Some ( dep_pkg_json ) ) =
self . load_pkg_json ( & pkg_folder . join ( " package.json " ) )
{
2024-12-20 16:14:37 -05:00
if dep_pkg_json . name . as_deref ( ) = = Some ( req . name . as_str ( ) ) {
2024-12-12 18:58:14 -05:00
let matches_req = dep_pkg_json
. version
. as_ref ( )
. and_then ( | v | Version ::parse_from_npm ( v ) . ok ( ) )
. map ( | version | req . version_req . matches ( & version ) )
. unwrap_or ( true ) ;
if matches_req {
return Some ( ( dep_pkg_json , req . name . clone ( ) ) ) ;
}
}
}
None
} ;
if let Some ( file_path ) = & maybe_referrer_path {
for dir_path in file_path . as_path ( ) . ancestors ( ) . skip ( 1 ) {
if let Some ( result ) =
search_node_modules ( & dir_path . join ( " node_modules " ) )
{
return Ok ( Some ( result ) ) ;
}
}
}
// and finally check the root node_modules directory
if let Some ( root_node_modules_dir ) = & self . root_node_modules_dir {
let already_searched = maybe_referrer_path
. as_ref ( )
. and_then ( | referrer_path | {
root_node_modules_dir
. parent ( )
. map ( | root_dir | referrer_path . starts_with ( root_dir ) )
} )
. unwrap_or ( false ) ;
if ! already_searched {
if let Some ( result ) = search_node_modules ( root_node_modules_dir ) {
return Ok ( Some ( result ) ) ;
}
}
}
2024-09-30 09:33:32 -04:00
Ok ( None )
}
fn resolve_folder_in_root_node_modules (
& self ,
req : & PackageReq ,
) -> Option < PathBuf > {
// now check if node_modules/.deno/ matches this constraint
let root_node_modules_dir = self . root_node_modules_dir . as_ref ( ) ? ;
let node_modules_deno_dir = root_node_modules_dir . join ( " .deno " ) ;
2024-12-30 12:38:20 -05:00
let Ok ( entries ) = self . sys . fs_read_dir ( & node_modules_deno_dir ) else {
2024-09-30 09:33:32 -04:00
return None ;
} ;
let search_prefix = format! (
" {}@ " ,
normalize_pkg_name_for_node_modules_deno_folder ( & req . name )
) ;
let mut best_version = None ;
// example entries:
// - @denotest+add@1.0.0
// - @denotest+add@1.0.0_1
for entry in entries {
2024-12-30 12:38:20 -05:00
let Ok ( entry ) = entry else {
continue ;
} ;
let Ok ( file_type ) = entry . file_type ( ) else {
continue ;
} ;
if ! file_type . is_dir ( ) {
2024-09-30 09:33:32 -04:00
continue ;
}
2024-12-30 12:38:20 -05:00
let entry_name = entry . file_name ( ) . to_string_lossy ( ) . into_owned ( ) ;
let Some ( version_and_copy_idx ) = entry_name . strip_prefix ( & search_prefix )
2024-09-30 09:33:32 -04:00
else {
continue ;
} ;
let version = version_and_copy_idx
. rsplit_once ( '_' )
. map ( | ( v , _ ) | v )
. unwrap_or ( version_and_copy_idx ) ;
let Ok ( version ) = Version ::parse_from_npm ( version ) else {
continue ;
} ;
fix(install): store tags associated with package in node_modules dir (#26000)
Fixes #25998. Fixes https://github.com/denoland/deno/issues/25928.
Originally I was just going to make this an error message instead of a
panic, but once I got to a minimal repro I felt that this really should
work.
The panic occurs when you have `nodeModulesDir: manual` (or a
package.json present), and you have an npm package with a tag in your
deno.json (see the spec test that illustrates this).
This code path only actually executes when trying to choose an
appropriate package version from `node_modules/.deno`, so we should be
able to fix it by storing some extra data at install time.
The fix proposed here is to repurpose the `.initialized` file that we
store in `node_modules` to store the tags associated with a package.
Basically, if you have a version requirement with a tag (e.g.
`npm:chalk@latest`), when we set up the node_modules folder for that
package, we store the tag (`latest`) in `.initialized`. Then, when doing
BYONM resolution, if we have a version requirement with a tag, we read
that file and check if the tag is present.
The downside is that we do more work when setting up `node_modules`. We
_could_ do this only when BYONM is enabled, but that would have the
downside of needing to re-run `deno install` when you switch from auto
-> manual, though maybe that's not a big deal.
2024-10-02 20:16:46 -04:00
if let Some ( tag ) = req . version_req . tag ( ) {
let initialized_file =
2024-12-30 12:38:20 -05:00
node_modules_deno_dir . join ( & entry_name ) . join ( " .initialized " ) ;
let Ok ( contents ) = self . sys . fs_read_to_string_lossy ( & initialized_file )
fix(install): store tags associated with package in node_modules dir (#26000)
Fixes #25998. Fixes https://github.com/denoland/deno/issues/25928.
Originally I was just going to make this an error message instead of a
panic, but once I got to a minimal repro I felt that this really should
work.
The panic occurs when you have `nodeModulesDir: manual` (or a
package.json present), and you have an npm package with a tag in your
deno.json (see the spec test that illustrates this).
This code path only actually executes when trying to choose an
appropriate package version from `node_modules/.deno`, so we should be
able to fix it by storing some extra data at install time.
The fix proposed here is to repurpose the `.initialized` file that we
store in `node_modules` to store the tags associated with a package.
Basically, if you have a version requirement with a tag (e.g.
`npm:chalk@latest`), when we set up the node_modules folder for that
package, we store the tag (`latest`) in `.initialized`. Then, when doing
BYONM resolution, if we have a version requirement with a tag, we read
that file and check if the tag is present.
The downside is that we do more work when setting up `node_modules`. We
_could_ do this only when BYONM is enabled, but that would have the
downside of needing to re-run `deno install` when you switch from auto
-> manual, though maybe that's not a big deal.
2024-10-02 20:16:46 -04:00
else {
continue ;
} ;
let mut tags = contents . split ( ',' ) . map ( str ::trim ) ;
if tags . any ( | t | t = = tag ) {
if let Some ( ( best_version_version , _ ) ) = & best_version {
if version > * best_version_version {
2024-12-30 12:38:20 -05:00
best_version = Some ( ( version , entry_name ) ) ;
fix(install): store tags associated with package in node_modules dir (#26000)
Fixes #25998. Fixes https://github.com/denoland/deno/issues/25928.
Originally I was just going to make this an error message instead of a
panic, but once I got to a minimal repro I felt that this really should
work.
The panic occurs when you have `nodeModulesDir: manual` (or a
package.json present), and you have an npm package with a tag in your
deno.json (see the spec test that illustrates this).
This code path only actually executes when trying to choose an
appropriate package version from `node_modules/.deno`, so we should be
able to fix it by storing some extra data at install time.
The fix proposed here is to repurpose the `.initialized` file that we
store in `node_modules` to store the tags associated with a package.
Basically, if you have a version requirement with a tag (e.g.
`npm:chalk@latest`), when we set up the node_modules folder for that
package, we store the tag (`latest`) in `.initialized`. Then, when doing
BYONM resolution, if we have a version requirement with a tag, we read
that file and check if the tag is present.
The downside is that we do more work when setting up `node_modules`. We
_could_ do this only when BYONM is enabled, but that would have the
downside of needing to re-run `deno install` when you switch from auto
-> manual, though maybe that's not a big deal.
2024-10-02 20:16:46 -04:00
}
} else {
2024-12-30 12:38:20 -05:00
best_version = Some ( ( version , entry_name ) ) ;
fix(install): store tags associated with package in node_modules dir (#26000)
Fixes #25998. Fixes https://github.com/denoland/deno/issues/25928.
Originally I was just going to make this an error message instead of a
panic, but once I got to a minimal repro I felt that this really should
work.
The panic occurs when you have `nodeModulesDir: manual` (or a
package.json present), and you have an npm package with a tag in your
deno.json (see the spec test that illustrates this).
This code path only actually executes when trying to choose an
appropriate package version from `node_modules/.deno`, so we should be
able to fix it by storing some extra data at install time.
The fix proposed here is to repurpose the `.initialized` file that we
store in `node_modules` to store the tags associated with a package.
Basically, if you have a version requirement with a tag (e.g.
`npm:chalk@latest`), when we set up the node_modules folder for that
package, we store the tag (`latest`) in `.initialized`. Then, when doing
BYONM resolution, if we have a version requirement with a tag, we read
that file and check if the tag is present.
The downside is that we do more work when setting up `node_modules`. We
_could_ do this only when BYONM is enabled, but that would have the
downside of needing to re-run `deno install` when you switch from auto
-> manual, though maybe that's not a big deal.
2024-10-02 20:16:46 -04:00
}
}
} else if req . version_req . matches ( & version ) {
2024-09-30 09:33:32 -04:00
if let Some ( ( best_version_version , _ ) ) = & best_version {
if version > * best_version_version {
2024-12-30 12:38:20 -05:00
best_version = Some ( ( version , entry_name ) ) ;
2024-09-30 09:33:32 -04:00
}
} else {
2024-12-30 12:38:20 -05:00
best_version = Some ( ( version , entry_name ) ) ;
2024-09-30 09:33:32 -04:00
}
}
}
best_version . map ( | ( _version , entry_name ) | {
join_package_name (
& node_modules_deno_dir . join ( entry_name ) . join ( " node_modules " ) ,
& req . name ,
)
} )
}
}
2024-11-01 12:27:00 -04:00
impl <
2024-12-30 12:38:20 -05:00
Sys : FsCanonicalize
+ FsMetadata
+ FsRead
+ FsReadDir
+ Send
+ Sync
+ std ::fmt ::Debug ,
> CliNpmReqResolver for ByonmNpmResolver < Sys >
2024-11-14 15:24:25 -05:00
{
fn resolve_pkg_folder_from_deno_module_req (
& self ,
req : & PackageReq ,
referrer : & Url ,
) -> Result < PathBuf , ResolvePkgFolderFromDenoReqError > {
ByonmNpmResolver ::resolve_pkg_folder_from_deno_module_req (
self , req , referrer ,
)
. map_err ( ResolvePkgFolderFromDenoReqError ::Byonm )
}
}
impl <
2024-12-30 12:38:20 -05:00
Sys : FsCanonicalize
+ FsMetadata
+ FsRead
+ FsReadDir
+ Send
+ Sync
+ std ::fmt ::Debug ,
> NpmPackageFolderResolver for ByonmNpmResolver < Sys >
2024-09-30 09:33:32 -04:00
{
fn resolve_package_folder_from_package (
& self ,
name : & str ,
referrer : & Url ,
) -> Result < PathBuf , PackageFolderResolveError > {
2024-12-30 12:38:20 -05:00
fn inner < TSys : FsMetadata > (
sys : & TSys ,
2024-09-30 09:33:32 -04:00
name : & str ,
referrer : & Url ,
) -> Result < PathBuf , PackageFolderResolveError > {
let maybe_referrer_file = url_to_file_path ( referrer ) . ok ( ) ;
let maybe_start_folder =
maybe_referrer_file . as_ref ( ) . and_then ( | f | f . parent ( ) ) ;
if let Some ( start_folder ) = maybe_start_folder {
for current_folder in start_folder . ancestors ( ) {
let node_modules_folder = if current_folder . ends_with ( " node_modules " )
{
Cow ::Borrowed ( current_folder )
} else {
Cow ::Owned ( current_folder . join ( " node_modules " ) )
} ;
let sub_dir = join_package_name ( & node_modules_folder , name ) ;
2024-12-30 12:38:20 -05:00
if sys . fs_is_dir_no_err ( & sub_dir ) {
2024-09-30 09:33:32 -04:00
return Ok ( sub_dir ) ;
}
}
}
Err (
PackageNotFoundError {
package_name : name . to_string ( ) ,
referrer : referrer . clone ( ) ,
referrer_extra : None ,
}
. into ( ) ,
)
}
2024-12-30 12:38:20 -05:00
let path = inner ( & self . sys , name , referrer ) ? ;
self . sys . fs_canonicalize ( & path ) . map_err ( | err | {
2024-09-30 09:33:32 -04:00
PackageFolderResolveIoError {
package_name : name . to_string ( ) ,
referrer : referrer . clone ( ) ,
source : err ,
}
. into ( )
} )
}
2024-11-01 12:27:00 -04:00
}
#[ derive(Debug) ]
pub struct ByonmInNpmPackageChecker ;
2024-09-30 09:33:32 -04:00
2024-11-01 12:27:00 -04:00
impl InNpmPackageChecker for ByonmInNpmPackageChecker {
2024-09-30 09:33:32 -04:00
fn in_npm_package ( & self , specifier : & Url ) -> bool {
specifier . scheme ( ) = = " file "
& & specifier
. path ( )
. to_ascii_lowercase ( )
. contains ( " /node_modules/ " )
}
}
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
}