From c11079d5570ca805d48f4aae8269af8fe704dbdc Mon Sep 17 00:00:00 2001 From: David Sherret Date: Mon, 21 Oct 2024 20:10:31 -0400 Subject: [PATCH 01/42] refactor: prefer parsing as `Program` over `Module` --- Cargo.lock | 4 ---- Cargo.toml | 4 ++++ cli/args/mod.rs | 2 ++ cli/emit.rs | 2 +- cli/lsp/analysis.rs | 13 +++++-------- cli/node.rs | 9 +-------- cli/tools/repl/session.rs | 1 + cli/util/extract.rs | 8 ++++---- 8 files changed, 18 insertions(+), 25 deletions(-) diff --git a/Cargo.lock b/Cargo.lock index dd77a331334cb0..2a86623a846bb9 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -1284,8 +1284,6 @@ dependencies = [ [[package]] name = "deno_ast" version = "0.42.2" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "b2b9d03b1bbeeecdac54367f075d572131736d06c5be3bc49037855bc5ab1bbb" dependencies = [ "base64 0.21.7", "deno_media_type", @@ -1610,8 +1608,6 @@ dependencies = [ [[package]] name = "deno_graph" version = "0.83.3" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "77163c46755676d8f793fc19e365537ba660a8db173cd1e02d21eb010c0b3cef" dependencies = [ "anyhow", "async-trait", diff --git a/Cargo.toml b/Cargo.toml index 7d5c507cae9685..669e0a181e8583 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -326,3 +326,7 @@ opt-level = 3 opt-level = 3 [profile.release.package.zstd-sys] opt-level = 3 + +[patch.crates-io] +deno_graph = { path = "../deno_graph" } +deno_ast = { path = "../deno_ast" } diff --git a/cli/args/mod.rs b/cli/args/mod.rs index f905e186ba8969..8468e54d52c869 100644 --- a/cli/args/mod.rs +++ b/cli/args/mod.rs @@ -200,6 +200,8 @@ pub fn ts_config_to_transpile_and_emit_options( precompile_jsx_dynamic_props: None, transform_jsx, var_decl_imports: false, + // todo(dsherret): support verbatim_module_syntax here properly + verbatim_module_syntax: false, }, deno_ast::EmitOptions { inline_sources: options.inline_sources, diff --git a/cli/emit.rs b/cli/emit.rs index b3f4a4477aed94..f9129d85621294 100644 --- a/cli/emit.rs +++ b/cli/emit.rs @@ -317,7 +317,7 @@ fn ensure_no_import_assertion( deno_core::anyhow::anyhow!("{}", msg) } - let Some(module) = parsed_source.program_ref().as_module() else { + let deno_ast::ProgramRef::Module(module) = parsed_source.program_ref() else { return Ok(()); }; diff --git a/cli/lsp/analysis.rs b/cli/lsp/analysis.rs index 8c2e8bb1dccbd9..0ec3d340b6c4d9 100644 --- a/cli/lsp/analysis.rs +++ b/cli/lsp/analysis.rs @@ -1206,14 +1206,11 @@ impl CodeActionCollection { }), ); - match parsed_source.program_ref() { - deno_ast::swc::ast::Program::Module(module) => module - .body - .iter() - .find(|i| i.range().contains(&specifier_range)) - .map(|i| text_info.line_and_column_index(i.range().start)), - deno_ast::swc::ast::Program::Script(_) => None, - } + parsed_source + .program_ref() + .body() + .find(|i| i.range().contains(&specifier_range)) + .map(|i| text_info.line_and_column_index(i.range().start)) } async fn deno_types_for_npm_action( diff --git a/cli/node.rs b/cli/node.rs index 733d5f8717f555..a9e9191f1d4c23 100644 --- a/cli/node.rs +++ b/cli/node.rs @@ -132,19 +132,12 @@ impl CliCjsCodeAnalyzer { maybe_syntax: None, }) })?; - if parsed_source.is_script() { + if parsed_source.is_script() || media_type == MediaType::Cjs { let analysis = parsed_source.analyze_cjs(); Ok(CliCjsAnalysis::Cjs { exports: analysis.exports, reexports: analysis.reexports, }) - } else if media_type == MediaType::Cjs { - // FIXME: `deno_ast` should internally handle MediaType::Cjs implying that - // the result must never be Esm - Ok(CliCjsAnalysis::Cjs { - exports: vec![], - reexports: vec![], - }) } else { Ok(CliCjsAnalysis::Esm) } diff --git a/cli/tools/repl/session.rs b/cli/tools/repl/session.rs index 484664dae493a1..f306f0197478fc 100644 --- a/cli/tools/repl/session.rs +++ b/cli/tools/repl/session.rs @@ -641,6 +641,7 @@ impl ReplSession { jsx_fragment_factory: self.jsx.frag_factory.clone(), jsx_import_source: self.jsx.import_source.clone(), var_decl_imports: true, + verbatim_module_syntax: false, }, &deno_ast::EmitOptions { source_map: deno_ast::SourceMapOption::None, diff --git a/cli/util/extract.rs b/cli/util/extract.rs index 873b7e7f2d4061..f577cbefec33ce 100644 --- a/cli/util/extract.rs +++ b/cli/util/extract.rs @@ -64,7 +64,7 @@ fn extract_inner( }) { Ok(parsed) => { let mut c = ExportCollector::default(); - c.visit_program(parsed.program_ref()); + c.visit_program(parsed.program().as_ref()); c } Err(_) => ExportCollector::default(), @@ -570,14 +570,14 @@ fn generate_pseudo_file( })?; let top_level_atoms = swc_utils::collect_decls_with_ctxt::( - parsed.program_ref(), + &parsed.program_ref(), parsed.top_level_context(), ); let transformed = parsed .program_ref() - .clone() + .to_owned() .fold_with(&mut as_folder(Transform { specifier: &file.specifier, base_file_specifier, @@ -1416,7 +1416,7 @@ console.log(Foo); }) .unwrap(); - collector.visit_program(parsed.program_ref()); + parsed.program_ref().visit_with(&mut collector); collector } From 904c3168a9334fa4413817f36daa2f63edaffbd5 Mon Sep 17 00:00:00 2001 From: David Sherret Date: Tue, 22 Oct 2024 11:57:21 -0400 Subject: [PATCH 02/42] more updates --- Cargo.lock | 2 -- Cargo.toml | 1 + cli/lsp/code_lens.rs | 4 ++-- cli/lsp/documents.rs | 2 +- cli/lsp/testing/collectors.rs | 2 +- 5 files changed, 5 insertions(+), 6 deletions(-) diff --git a/Cargo.lock b/Cargo.lock index 2a86623a846bb9..c245b8a9804687 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -1509,8 +1509,6 @@ dependencies = [ [[package]] name = "deno_doc" version = "0.154.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "17e204e45b0d79750880114e37b34abe19ad0710d8435a8da8f23a528fe98de4" dependencies = [ "anyhow", "cfg-if", diff --git a/Cargo.toml b/Cargo.toml index 669e0a181e8583..b528d59cd08114 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -330,3 +330,4 @@ opt-level = 3 [patch.crates-io] deno_graph = { path = "../deno_graph" } deno_ast = { path = "../deno_ast" } +deno_doc = { path = "../deno_doc" } diff --git a/cli/lsp/code_lens.rs b/cli/lsp/code_lens.rs index e117888fba66f4..a57ca3ac9fe141 100644 --- a/cli/lsp/code_lens.rs +++ b/cli/lsp/code_lens.rs @@ -421,7 +421,7 @@ pub fn collect_test( ) -> Result, AnyError> { let mut collector = DenoTestCollector::new(specifier.clone(), parsed_source.clone()); - parsed_source.module().visit_with(&mut collector); + parsed_source.program().visit_with(&mut collector); Ok(collector.take()) } @@ -581,7 +581,7 @@ mod tests { .unwrap(); let mut collector = DenoTestCollector::new(specifier, parsed_module.clone()); - parsed_module.module().visit_with(&mut collector); + parsed_module.program().visit_with(&mut collector); assert_eq!( collector.take(), vec![ diff --git a/cli/lsp/documents.rs b/cli/lsp/documents.rs index 7d1ca6810df79f..b7578e9614d428 100644 --- a/cli/lsp/documents.rs +++ b/cli/lsp/documents.rs @@ -272,7 +272,7 @@ fn get_maybe_test_module_fut( parsed_source.specifier().clone(), parsed_source.text_info_lazy().clone(), ); - parsed_source.module().visit_with(&mut collector); + parsed_source.program().visit_with(&mut collector); Arc::new(collector.take()) }) .map(Result::ok) diff --git a/cli/lsp/testing/collectors.rs b/cli/lsp/testing/collectors.rs index 2f2ddb8773ba77..2dd7ec0d96d342 100644 --- a/cli/lsp/testing/collectors.rs +++ b/cli/lsp/testing/collectors.rs @@ -650,7 +650,7 @@ pub mod tests { .unwrap(); let text_info = parsed_module.text_info_lazy().clone(); let mut collector = TestCollector::new(specifier, text_info); - parsed_module.module().visit_with(&mut collector); + parsed_module.program().visit_with(&mut collector); collector.take() } From 4335aa09c377955234be8358e9581d6b6f82f24a Mon Sep 17 00:00:00 2001 From: David Sherret Date: Tue, 22 Oct 2024 14:18:14 -0400 Subject: [PATCH 03/42] Starting on this, but going to revert. --- Cargo.lock | 2 - Cargo.toml | 1 + cli/factory.rs | 6 +- cli/module_loader.rs | 43 +++++++----- cli/npm/byonm.rs | 38 +++++----- cli/npm/managed/mod.rs | 23 +++---- cli/npm/mod.rs | 41 ++++++++++- cli/standalone/mod.rs | 80 ++++++++++++++++++---- cli/worker.rs | 41 +++++++---- ext/node/lib.rs | 12 ++-- ext/node/ops/require.rs | 11 +-- ext/node/ops/worker_threads.rs | 6 +- tests/specs/run/remote_cjs_main/output.out | 2 +- 13 files changed, 205 insertions(+), 101 deletions(-) diff --git a/Cargo.lock b/Cargo.lock index c245b8a9804687..86b9233996c31f 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -2887,8 +2887,6 @@ checksum = "31ae425815400e5ed474178a7a22e275a9687086a12ca63ec793ff292d8fdae8" [[package]] name = "eszip" version = "0.79.1" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "8eb55c89bdde75a3826a79d49c9d847623ae7fbdb2695b542982982da990d33e" dependencies = [ "anyhow", "async-trait", diff --git a/Cargo.toml b/Cargo.toml index b528d59cd08114..4c5c1faa5d2929 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -331,3 +331,4 @@ opt-level = 3 deno_graph = { path = "../deno_graph" } deno_ast = { path = "../deno_ast" } deno_doc = { path = "../deno_doc" } +eszip = { path = "../eszip" } diff --git a/cli/factory.rs b/cli/factory.rs index 25f3551102253f..867191713591a5 100644 --- a/cli/factory.rs +++ b/cli/factory.rs @@ -34,6 +34,7 @@ use crate::node::CliCjsCodeAnalyzer; use crate::node::CliNodeCodeTranslator; use crate::npm::create_cli_npm_resolver; use crate::npm::CliByonmNpmResolverCreateOptions; +use crate::npm::CliNodeRequireLoader; use crate::npm::CliNpmResolver; use crate::npm::CliNpmResolverCreateOptions; use crate::npm::CliNpmResolverManagedCreateOptions; @@ -790,9 +791,9 @@ impl CliFactory { &self, ) -> Result { let cli_options = self.cli_options()?; + let fs = self.fs(); let node_resolver = self.node_resolver().await?; let npm_resolver = self.npm_resolver().await?; - let fs = self.fs(); let cli_node_resolver = self.cli_node_resolver().await?; let cli_npm_resolver = self.npm_resolver().await?.clone(); let maybe_file_watcher_communicator = if cli_options.has_hmr() { @@ -810,7 +811,7 @@ impl CliFactory { None }, self.feature_checker()?.clone(), - self.fs().clone(), + fs.clone(), maybe_file_watcher_communicator, self.maybe_inspector_server()?.clone(), cli_options.maybe_lockfile().cloned(), @@ -822,6 +823,7 @@ impl CliFactory { None }, self.emitter()?.clone(), + fs.clone(), self.main_module_graph_container().await?.clone(), self.module_load_preparer().await?.clone(), cli_node_resolver.clone(), diff --git a/cli/module_loader.rs b/cli/module_loader.rs index 37d42f78e5e93b..9f2ed09dc3361c 100644 --- a/cli/module_loader.rs +++ b/cli/module_loader.rs @@ -23,6 +23,7 @@ use crate::graph_container::ModuleGraphUpdatePermit; use crate::graph_util::CreateGraphOptions; use crate::graph_util::ModuleGraphBuilder; use crate::node; +use crate::npm::CliNodeRequireLoader; use crate::npm::CliNpmResolver; use crate::resolver::CliGraphResolver; use crate::resolver::CliNodeResolver; @@ -33,7 +34,7 @@ use crate::tools::check::TypeChecker; use crate::util::progress_bar::ProgressBar; use crate::util::text_encoding::code_without_source_map; use crate::util::text_encoding::source_map_from_code; -use crate::worker::ModuleLoaderAndSourceMapGetter; +use crate::worker::CreateModuleLoaderResult; use crate::worker::ModuleLoaderFactory; use deno_ast::MediaType; use deno_core::anyhow::anyhow; @@ -63,6 +64,7 @@ use deno_graph::Module; use deno_graph::ModuleGraph; use deno_graph::Resolution; use deno_runtime::code_cache; +use deno_runtime::deno_fs::FileSystem; use deno_runtime::deno_node::create_host_defined_options; use deno_runtime::deno_permissions::PermissionsContainer; use deno_semver::npm::NpmPackageReqReference; @@ -201,6 +203,7 @@ struct SharedCliModuleLoaderState { is_repl: bool, code_cache: Option>, emitter: Arc, + fs: Arc, main_module_graph_container: Arc, module_load_preparer: Arc, node_resolver: Arc, @@ -220,6 +223,7 @@ impl CliModuleLoaderFactory { options: &CliOptions, code_cache: Option>, emitter: Arc, + fs: Arc, main_module_graph_container: Arc, module_load_preparer: Arc, node_resolver: Arc, @@ -241,6 +245,7 @@ impl CliModuleLoaderFactory { ), code_cache, emitter, + fs, main_module_graph_container, module_load_preparer, node_resolver, @@ -259,19 +264,25 @@ impl CliModuleLoaderFactory { is_worker: bool, parent_permissions: PermissionsContainer, permissions: PermissionsContainer, - ) -> ModuleLoaderAndSourceMapGetter { - let loader = Rc::new(CliModuleLoader(Rc::new(CliModuleLoaderInner { - lib, - is_worker, - parent_permissions, - permissions, - graph_container, - emitter: self.shared.emitter.clone(), - parsed_source_cache: self.shared.parsed_source_cache.clone(), - shared: self.shared.clone(), - }))); - ModuleLoaderAndSourceMapGetter { - module_loader: loader, + ) -> CreateModuleLoaderResult { + let module_loader = + Rc::new(CliModuleLoader(Rc::new(CliModuleLoaderInner { + lib, + is_worker, + parent_permissions, + permissions, + graph_container, + emitter: self.shared.emitter.clone(), + parsed_source_cache: self.shared.parsed_source_cache.clone(), + shared: self.shared.clone(), + }))); + let node_require_loader = Rc::new(CliNodeRequireLoader::new( + self.shared.fs.clone(), + self.shared.npm_resolver.clone(), + )); + CreateModuleLoaderResult { + module_loader, + node_require_loader, } } } @@ -280,7 +291,7 @@ impl ModuleLoaderFactory for CliModuleLoaderFactory { fn create_for_main( &self, root_permissions: PermissionsContainer, - ) -> ModuleLoaderAndSourceMapGetter { + ) -> CreateModuleLoaderResult { self.create_with_lib( (*self.shared.main_module_graph_container).clone(), self.shared.lib_window, @@ -294,7 +305,7 @@ impl ModuleLoaderFactory for CliModuleLoaderFactory { &self, parent_permissions: PermissionsContainer, permissions: PermissionsContainer, - ) -> ModuleLoaderAndSourceMapGetter { + ) -> CreateModuleLoaderResult { self.create_with_lib( // create a fresh module graph for the worker WorkerModuleGraphContainer::new(Arc::new(ModuleGraph::new( diff --git a/cli/npm/byonm.rs b/cli/npm/byonm.rs index fc095ab16f20d1..0fde6f9c5dc917 100644 --- a/cli/npm/byonm.rs +++ b/cli/npm/byonm.rs @@ -11,7 +11,7 @@ use deno_core::url::Url; use deno_resolver::npm::ByonmNpmResolver; use deno_resolver::npm::ByonmNpmResolverCreateOptions; use deno_runtime::deno_node::NodePermissions; -use deno_runtime::deno_node::NodeRequireResolver; +use deno_runtime::deno_node::NodeRequireLoader; use deno_runtime::ops::process::NpmProcessStateProvider; use deno_semver::package::PackageReq; use node_resolver::NpmResolver; @@ -32,23 +32,6 @@ pub type CliByonmNpmResolver = ByonmNpmResolver; #[derive(Debug)] struct CliByonmWrapper(Arc); -impl NodeRequireResolver for CliByonmWrapper { - fn ensure_read_permission<'a>( - &self, - permissions: &mut dyn NodePermissions, - path: &'a Path, - ) -> Result, AnyError> { - if !path - .components() - .any(|c| c.as_os_str().to_ascii_lowercase() == "node_modules") - { - permissions.check_read_path(path) - } else { - Ok(Cow::Borrowed(path)) - } - } -} - impl NpmProcessStateProvider for CliByonmWrapper { fn get_npm_process_state(&self) -> String { serde_json::to_string(&NpmProcessState { @@ -67,10 +50,6 @@ impl CliNpmResolver for CliByonmNpmResolver { self } - fn into_require_resolver(self: Arc) -> Arc { - Arc::new(CliByonmWrapper(self)) - } - fn into_process_state_provider( self: Arc, ) -> Arc { @@ -100,6 +79,21 @@ impl CliNpmResolver for CliByonmNpmResolver { .map_err(ResolvePkgFolderFromDenoReqError::Byonm) } + fn ensure_read_permission<'a>( + &self, + permissions: &mut dyn NodePermissions, + path: &'a Path, + ) -> Result, AnyError> { + if !path + .components() + .any(|c| c.as_os_str().to_ascii_lowercase() == "node_modules") + { + permissions.check_read_path(path) + } else { + Ok(Cow::Borrowed(path)) + } + } + fn check_state_hash(&self) -> Option { // it is very difficult to determine the check state hash for byonm // so we just return None to signify check caching is not supported diff --git a/cli/npm/managed/mod.rs b/cli/npm/managed/mod.rs index ec50a9c65a14d3..2bd4b34aa5aa4c 100644 --- a/cli/npm/managed/mod.rs +++ b/cli/npm/managed/mod.rs @@ -24,7 +24,6 @@ use deno_npm::NpmSystemInfo; use deno_runtime::colors; use deno_runtime::deno_fs::FileSystem; use deno_runtime::deno_node::NodePermissions; -use deno_runtime::deno_node::NodeRequireResolver; use deno_runtime::ops::process::NpmProcessStateProvider; use deno_semver::package::PackageNv; use deno_semver::package::PackageReq; @@ -593,16 +592,6 @@ impl NpmResolver for ManagedCliNpmResolver { } } -impl NodeRequireResolver for ManagedCliNpmResolver { - fn ensure_read_permission<'a>( - &self, - permissions: &mut dyn NodePermissions, - path: &'a Path, - ) -> Result, AnyError> { - self.fs_resolver.ensure_read_permission(permissions, path) - } -} - impl NpmProcessStateProvider for ManagedCliNpmResolver { fn get_npm_process_state(&self) -> String { npm_process_state( @@ -617,10 +606,6 @@ impl CliNpmResolver for ManagedCliNpmResolver { self } - fn into_require_resolver(self: Arc) -> Arc { - self - } - fn into_process_state_provider( self: Arc, ) -> Arc { @@ -681,6 +666,14 @@ impl CliNpmResolver for ManagedCliNpmResolver { .map_err(ResolvePkgFolderFromDenoReqError::Managed) } + fn ensure_read_permission<'a>( + &self, + permissions: &mut dyn NodePermissions, + path: &'a Path, + ) -> Result, AnyError> { + self.fs_resolver.ensure_read_permission(permissions, path) + } + fn check_state_hash(&self) -> Option { // We could go further and check all the individual // npm packages, but that's probably overkill. diff --git a/cli/npm/mod.rs b/cli/npm/mod.rs index 53baaf77b4a29a..7dc18e0947883d 100644 --- a/cli/npm/mod.rs +++ b/cli/npm/mod.rs @@ -4,6 +4,7 @@ mod byonm; mod common; mod managed; +use std::borrow::Cow; use std::path::Path; use std::path::PathBuf; use std::sync::Arc; @@ -15,7 +16,9 @@ use deno_core::serde_json; use deno_npm::registry::NpmPackageInfo; use deno_resolver::npm::ByonmNpmResolver; use deno_resolver::npm::ByonmResolvePkgFolderFromDenoReqError; -use deno_runtime::deno_node::NodeRequireResolver; +use deno_runtime::deno_fs::FileSystem; +use deno_runtime::deno_node::NodePermissions; +use deno_runtime::deno_node::NodeRequireLoader; use deno_runtime::ops::process::NpmProcessStateProvider; use deno_semver::package::PackageNv; use deno_semver::package::PackageReq; @@ -74,7 +77,6 @@ pub enum InnerCliNpmResolverRef<'a> { pub trait CliNpmResolver: NpmResolver { fn into_npm_resolver(self: Arc) -> Arc; - fn into_require_resolver(self: Arc) -> Arc; fn into_process_state_provider( self: Arc, ) -> Arc; @@ -105,6 +107,12 @@ pub trait CliNpmResolver: NpmResolver { referrer: &ModuleSpecifier, ) -> Result; + fn ensure_read_permission<'a>( + &self, + permissions: &mut dyn NodePermissions, + path: &'a Path, + ) -> Result, AnyError>; + /// Returns a hash returning the state of the npm resolver /// or `None` if the state currently can't be determined. fn check_state_hash(&self) -> Option; @@ -172,3 +180,32 @@ impl NpmFetchResolver { info } } + +#[derive(Debug)] +pub struct CliNodeRequireLoader { + fs: Arc, + npm_resolver: Arc, +} + +impl CliNodeRequireLoader { + pub fn new( + fs: Arc, + npm_resolver: Arc, + ) -> Self { + Self { fs, npm_resolver } + } +} + +impl NodeRequireLoader for CliNodeRequireLoader { + fn ensure_read_permission<'a>( + &self, + permissions: &mut dyn deno_runtime::deno_node::NodePermissions, + path: &'a Path, + ) -> Result, AnyError> { + self.npm_resolver.ensure_read_permission(permissions, path) + } + + fn load_text_file_lossy(&self, path: &Path) -> Result { + Ok(self.fs.read_text_file_lossy_sync(path, None)?) + } +} diff --git a/cli/standalone/mod.rs b/cli/standalone/mod.rs index 60018228b7cb72..f299d4dcb8db5f 100644 --- a/cli/standalone/mod.rs +++ b/cli/standalone/mod.rs @@ -28,6 +28,7 @@ use deno_npm::npm_rc::ResolvedNpmRc; use deno_package_json::PackageJsonDepValue; use deno_runtime::deno_fs; use deno_runtime::deno_node::create_host_defined_options; +use deno_runtime::deno_node::NodeRequireLoader; use deno_runtime::deno_node::NodeResolver; use deno_runtime::deno_permissions::Permissions; use deno_runtime::deno_permissions::PermissionsContainer; @@ -61,6 +62,8 @@ use crate::http_util::HttpClientProvider; use crate::node::CliCjsCodeAnalyzer; use crate::npm::create_cli_npm_resolver; use crate::npm::CliByonmNpmResolverCreateOptions; +use crate::npm::CliNodeRequireLoader; +use crate::npm::CliNpmResolver; use crate::npm::CliNpmResolverCreateOptions; use crate::npm::CliNpmResolverManagedCreateOptions; use crate::npm::CliNpmResolverManagedSnapshotOption; @@ -73,7 +76,7 @@ use crate::util::progress_bar::ProgressBarStyle; use crate::util::v8::construct_v8_flags; use crate::worker::CliMainWorkerFactory; use crate::worker::CliMainWorkerOptions; -use crate::worker::ModuleLoaderAndSourceMapGetter; +use crate::worker::CreateModuleLoaderResult; use crate::worker::ModuleLoaderFactory; pub mod binary; @@ -123,10 +126,12 @@ impl WorkspaceEszip { } struct SharedModuleLoaderState { + fs: Arc, eszip: WorkspaceEszip, - workspace_resolver: WorkspaceResolver, node_resolver: Arc, npm_module_loader: Arc, + npm_resolver: Arc, + workspace_resolver: WorkspaceResolver, } #[derive(Clone)] @@ -134,6 +139,12 @@ struct EmbeddedModuleLoader { shared: Arc, } +impl std::fmt::Debug for EmbeddedModuleLoader { + fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { + f.debug_struct("EmbeddedModuleLoader").finish() + } +} + pub const MODULE_NOT_FOUND: &str = "Module not found"; pub const UNSUPPORTED_SCHEME: &str = "Unsupported scheme"; @@ -382,6 +393,39 @@ impl ModuleLoader for EmbeddedModuleLoader { } } +impl NodeRequireLoader for EmbeddedModuleLoader { + fn ensure_read_permission<'a>( + &self, + permissions: &mut dyn deno_runtime::deno_node::NodePermissions, + path: &'a std::path::Path, + ) -> Result, AnyError> { + self + .shared + .npm_resolver + .ensure_read_permission(permissions, path) + } + + fn load_text_file_lossy( + &self, + path: &std::path::Path, + ) -> Result { + eprintln!("Reading: {}", path.display()); + if let Ok(url) = deno_path_util::url_from_file_path(&path) { + eprintln!("Looking at: {}", url); + if let Some(module) = self.shared.eszip.get_module(&url) { + eprintln!("FOUND"); + let source = module.inner.get_source_if_ready_sync()?; + eprintln!("YES"); + // todo(THIS PR): do not clone if not necessary + return Ok(String::from_utf8_lossy(&source).into_owned()); + } + eprintln!("NOPE 2"); + } + eprintln!("NOPE"); + Ok(self.shared.fs.read_text_file_lossy_sync(path, None)?) + } +} + fn arc_u8_to_arc_str( arc_u8: Arc<[u8]>, ) -> Result, std::str::Utf8Error> { @@ -399,28 +443,32 @@ struct StandaloneModuleLoaderFactory { shared: Arc, } +impl StandaloneModuleLoaderFactory { + pub fn create_result(&self) -> CreateModuleLoaderResult { + let loader = Rc::new(EmbeddedModuleLoader { + shared: self.shared.clone(), + }); + CreateModuleLoaderResult { + module_loader: loader.clone(), + node_require_loader: loader, + } + } +} + impl ModuleLoaderFactory for StandaloneModuleLoaderFactory { fn create_for_main( &self, _root_permissions: PermissionsContainer, - ) -> ModuleLoaderAndSourceMapGetter { - ModuleLoaderAndSourceMapGetter { - module_loader: Rc::new(EmbeddedModuleLoader { - shared: self.shared.clone(), - }), - } + ) -> CreateModuleLoaderResult { + self.create_result() } fn create_for_worker( &self, _parent_permissions: PermissionsContainer, _permissions: PermissionsContainer, - ) -> ModuleLoaderAndSourceMapGetter { - ModuleLoaderAndSourceMapGetter { - module_loader: Rc::new(EmbeddedModuleLoader { - shared: self.shared.clone(), - }), - } + ) -> CreateModuleLoaderResult { + self.create_result() } } @@ -645,11 +693,11 @@ pub async fn run( }; let module_loader_factory = StandaloneModuleLoaderFactory { shared: Arc::new(SharedModuleLoaderState { + fs: fs.clone(), eszip: WorkspaceEszip { eszip, root_dir_url, }, - workspace_resolver, node_resolver: cli_node_resolver.clone(), npm_module_loader: Arc::new(NpmModuleLoader::new( cjs_resolutions.clone(), @@ -657,6 +705,8 @@ pub async fn run( fs.clone(), cli_node_resolver, )), + npm_resolver: npm_resolver.clone(), + workspace_resolver, }), }; diff --git a/cli/worker.rs b/cli/worker.rs index e230197d2b56b5..bbfed670a282c0 100644 --- a/cli/worker.rs +++ b/cli/worker.rs @@ -23,6 +23,8 @@ use deno_runtime::deno_broadcast_channel::InMemoryBroadcastChannel; use deno_runtime::deno_fs; use deno_runtime::deno_node; use deno_runtime::deno_node::NodeExtInitServices; +use deno_runtime::deno_node::NodeRequireLoader; +use deno_runtime::deno_node::NodeRequireLoaderRc; use deno_runtime::deno_node::NodeResolver; use deno_runtime::deno_permissions::PermissionsContainer; use deno_runtime::deno_tls::RootCertStoreProvider; @@ -58,21 +60,22 @@ use crate::util::file_watcher::WatcherRestartMode; use crate::util::path::specifier_has_extension; use crate::version; -pub struct ModuleLoaderAndSourceMapGetter { +pub struct CreateModuleLoaderResult { pub module_loader: Rc, + pub node_require_loader: Rc, } pub trait ModuleLoaderFactory: Send + Sync { fn create_for_main( &self, root_permissions: PermissionsContainer, - ) -> ModuleLoaderAndSourceMapGetter; + ) -> CreateModuleLoaderResult; fn create_for_worker( &self, parent_permissions: PermissionsContainer, permissions: PermissionsContainer, - ) -> ModuleLoaderAndSourceMapGetter; + ) -> CreateModuleLoaderResult; } #[async_trait::async_trait(?Send)] @@ -148,9 +151,12 @@ struct SharedWorkerState { } impl SharedWorkerState { - pub fn create_node_init_services(&self) -> NodeExtInitServices { + pub fn create_node_init_services( + &self, + node_require_loader: NodeRequireLoaderRc, + ) -> NodeExtInitServices { NodeExtInitServices { - node_require_resolver: self.npm_resolver.clone().into_require_resolver(), + node_require_loader: node_require_loader, node_resolver: self.node_resolver.clone(), npm_resolver: self.npm_resolver.clone().into_npm_resolver(), } @@ -492,7 +498,10 @@ impl CliMainWorkerFactory { stdio: deno_runtime::deno_io::Stdio, ) -> Result { let shared = &self.shared; - let ModuleLoaderAndSourceMapGetter { module_loader } = shared + let CreateModuleLoaderResult { + module_loader, + node_require_loader, + } = shared .module_loader_factory .create_for_main(permissions.clone()); let (main_module, is_main_cjs) = if let Ok(package_ref) = @@ -597,7 +606,9 @@ impl CliMainWorkerFactory { root_cert_store_provider: Some(shared.root_cert_store_provider.clone()), module_loader, fs: shared.fs.clone(), - node_services: Some(shared.create_node_init_services()), + node_services: Some( + shared.create_node_init_services(node_require_loader), + ), npm_process_state_provider: Some(shared.npm_process_state_provider()), blob_store: shared.blob_store.clone(), broadcast_channel: shared.broadcast_channel.clone(), @@ -761,11 +772,13 @@ fn create_web_worker_callback( Arc::new(move |args| { let maybe_inspector_server = shared.maybe_inspector_server.clone(); - let ModuleLoaderAndSourceMapGetter { module_loader } = - shared.module_loader_factory.create_for_worker( - args.parent_permissions.clone(), - args.permissions.clone(), - ); + let CreateModuleLoaderResult { + module_loader, + node_require_loader, + } = shared.module_loader_factory.create_for_worker( + args.parent_permissions.clone(), + args.permissions.clone(), + ); let create_web_worker_cb = create_web_worker_callback(shared.clone(), stdio.clone()); @@ -795,7 +808,9 @@ fn create_web_worker_callback( root_cert_store_provider: Some(shared.root_cert_store_provider.clone()), module_loader, fs: shared.fs.clone(), - node_services: Some(shared.create_node_init_services()), + node_services: Some( + shared.create_node_init_services(node_require_loader), + ), blob_store: shared.blob_store.clone(), broadcast_channel: shared.broadcast_channel.clone(), shared_array_buffer_store: Some(shared.shared_array_buffer_store.clone()), diff --git a/ext/node/lib.rs b/ext/node/lib.rs index 9b22add4538aff..c6c91825500240 100644 --- a/ext/node/lib.rs +++ b/ext/node/lib.rs @@ -125,16 +125,17 @@ impl NodePermissions for deno_permissions::PermissionsContainer { } #[allow(clippy::disallowed_types)] -pub type NodeRequireResolverRc = - deno_fs::sync::MaybeArc; +pub type NodeRequireLoaderRc = std::rc::Rc; -pub trait NodeRequireResolver: std::fmt::Debug + MaybeSend + MaybeSync { +pub trait NodeRequireLoader: std::fmt::Debug + MaybeSend + MaybeSync { #[must_use = "the resolved return value to mitigate time-of-check to time-of-use issues"] fn ensure_read_permission<'a>( &self, permissions: &mut dyn NodePermissions, path: &'a Path, ) -> Result, AnyError>; + + fn load_text_file_lossy(&self, path: &Path) -> Result; } pub static NODE_ENV_VAR_ALLOWLIST: Lazy> = Lazy::new(|| { @@ -152,8 +153,9 @@ fn op_node_build_os() -> String { env!("TARGET").split('-').nth(2).unwrap().to_string() } +#[derive(Debug, Clone)] pub struct NodeExtInitServices { - pub node_require_resolver: NodeRequireResolverRc, + pub node_require_loader: NodeRequireLoaderRc, pub node_resolver: NodeResolverRc, pub npm_resolver: NpmResolverRc, } @@ -639,7 +641,7 @@ deno_core::extension!(deno_node, state.put(options.fs.clone()); if let Some(init) = &options.maybe_init { - state.put(init.node_require_resolver.clone()); + state.put(init.node_require_loader.clone()); state.put(init.node_resolver.clone()); state.put(init.npm_resolver.clone()); } diff --git a/ext/node/ops/require.rs b/ext/node/ops/require.rs index 7d85ee85328df2..0f8da52a056450 100644 --- a/ext/node/ops/require.rs +++ b/ext/node/ops/require.rs @@ -22,7 +22,7 @@ use std::path::PathBuf; use std::rc::Rc; use crate::NodePermissions; -use crate::NodeRequireResolverRc; +use crate::NodeRequireLoaderRc; use crate::NodeResolverRc; use crate::NpmResolverRc; @@ -34,9 +34,9 @@ fn ensure_read_permission<'a, P>( where P: NodePermissions + 'static, { - let resolver = state.borrow::().clone(); + let loader = state.borrow::().clone(); let permissions = state.borrow_mut::

(); - resolver.ensure_read_permission(permissions, file_path) + loader.ensure_read_permission(permissions, file_path) } #[op2] @@ -445,9 +445,10 @@ where P: NodePermissions + 'static, { let file_path = PathBuf::from(file_path); + // todo(dsherret): there's multiple borrows to NodeRequireLoaderRc here let file_path = ensure_read_permission::

(state, &file_path)?; - let fs = state.borrow::(); - Ok(fs.read_text_file_lossy_sync(&file_path, None)?) + let loader = state.borrow::(); + Ok(loader.load_text_file_lossy(&file_path)?) } #[op2] diff --git a/ext/node/ops/worker_threads.rs b/ext/node/ops/worker_threads.rs index 5f139c5dc8eda4..c6aba33cc63147 100644 --- a/ext/node/ops/worker_threads.rs +++ b/ext/node/ops/worker_threads.rs @@ -12,7 +12,7 @@ use std::path::Path; use std::path::PathBuf; use crate::NodePermissions; -use crate::NodeRequireResolverRc; +use crate::NodeRequireLoaderRc; use crate::NodeResolverRc; #[must_use = "the resolved return value to mitigate time-of-check to time-of-use issues"] @@ -23,9 +23,9 @@ fn ensure_read_permission<'a, P>( where P: NodePermissions + 'static, { - let resolver = state.borrow::().clone(); + let loader = state.borrow::().clone(); let permissions = state.borrow_mut::

(); - resolver.ensure_read_permission(permissions, file_path) + loader.ensure_read_permission(permissions, file_path) } #[op2] diff --git a/tests/specs/run/remote_cjs_main/output.out b/tests/specs/run/remote_cjs_main/output.out index f75c33907acea5..360934acb7faa0 100644 --- a/tests/specs/run/remote_cjs_main/output.out +++ b/tests/specs/run/remote_cjs_main/output.out @@ -1,3 +1,3 @@ Download http://localhost:4545/run/add.cjs -error: Expected a JavaScript or TypeScript module, but identified a Cjs module. Importing these types of modules is currently not supported. +error: Remote CJS modules are not supported. Specifier: http://localhost:4545/run/add.cjs From 7410544a69749d053496fc3e29545c80e816961a Mon Sep 17 00:00:00 2001 From: David Sherret Date: Fri, 25 Oct 2024 10:52:06 -0400 Subject: [PATCH 04/42] compiling --- Cargo.lock | 6 ------ Cargo.toml | 1 - cli/standalone/mod.rs | 15 ++------------- ext/node/ops/require.rs | 18 ++++++++---------- runtime/errors.rs | 11 ++++++----- 5 files changed, 16 insertions(+), 35 deletions(-) diff --git a/Cargo.lock b/Cargo.lock index 6e76f27c2f6696..bcee863cb44238 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -1280,8 +1280,6 @@ dependencies = [ [[package]] name = "deno_ast" version = "0.42.2" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "b2b9d03b1bbeeecdac54367f075d572131736d06c5be3bc49037855bc5ab1bbb" dependencies = [ "base64 0.21.7", "deno_media_type", @@ -1507,8 +1505,6 @@ dependencies = [ [[package]] name = "deno_doc" version = "0.154.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "17e204e45b0d79750880114e37b34abe19ad0710d8435a8da8f23a528fe98de4" dependencies = [ "anyhow", "cfg-if", @@ -1607,8 +1603,6 @@ dependencies = [ [[package]] name = "deno_graph" version = "0.83.3" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "77163c46755676d8f793fc19e365537ba660a8db173cd1e02d21eb010c0b3cef" dependencies = [ "anyhow", "async-trait", diff --git a/Cargo.toml b/Cargo.toml index 8f7cd820249429..3b830af0b13967 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -331,4 +331,3 @@ opt-level = 3 deno_graph = { path = "../deno_graph" } deno_ast = { path = "../deno_ast" } deno_doc = { path = "../deno_doc" } -eszip = { path = "../eszip" } diff --git a/cli/standalone/mod.rs b/cli/standalone/mod.rs index 6914a4e87fa095..8ab173b3e80b2c 100644 --- a/cli/standalone/mod.rs +++ b/cli/standalone/mod.rs @@ -94,6 +94,7 @@ use self::binary::Metadata; use self::file_system::DenoCompileFileSystem; struct SharedModuleLoaderState { + fs: Arc, modules: StandaloneModules, node_resolver: Arc, npm_module_loader: Arc, @@ -365,19 +366,6 @@ impl NodeRequireLoader for EmbeddedModuleLoader { &self, path: &std::path::Path, ) -> Result { - eprintln!("Reading: {}", path.display()); - if let Ok(url) = deno_path_util::url_from_file_path(&path) { - eprintln!("Looking at: {}", url); - if let Some(module) = self.shared.eszip.get_module(&url) { - eprintln!("FOUND"); - let source = module.inner.get_source_if_ready_sync()?; - eprintln!("YES"); - // todo(THIS PR): do not clone if not necessary - return Ok(String::from_utf8_lossy(&source).into_owned()); - } - eprintln!("NOPE 2"); - } - eprintln!("NOPE"); Ok(self.shared.fs.read_text_file_lossy_sync(path, None)?) } } @@ -609,6 +597,7 @@ pub async fn run(data: StandaloneData) -> Result { }; let module_loader_factory = StandaloneModuleLoaderFactory { shared: Arc::new(SharedModuleLoaderState { + fs: fs.clone(), modules, node_resolver: cli_node_resolver.clone(), npm_module_loader: Arc::new(NpmModuleLoader::new( diff --git a/ext/node/ops/require.rs b/ext/node/ops/require.rs index 659c88744319a7..0338723a4f4de4 100644 --- a/ext/node/ops/require.rs +++ b/ext/node/ops/require.rs @@ -9,6 +9,7 @@ use deno_core::OpState; use deno_fs::FileSystemRc; use deno_package_json::PackageJsonRc; use deno_path_util::normalize_path; +use deno_path_util::url_to_file_path; use node_resolver::NodeModuleKind; use node_resolver::NodeResolutionMode; use node_resolver::REQUIRE_CONDITIONS; @@ -54,10 +55,12 @@ pub enum RequireError { PackageImportsResolve( #[from] node_resolver::errors::PackageImportsResolveError, ), - #[error("failed to convert '{0}' to file path")] - FilePathConversion(Url), + #[error(transparent)] + FilePathConversion(#[from] deno_path_util::UrlToFilePathError), #[error(transparent)] Fs(#[from] deno_io::fs::FsError), + #[error(transparent)] + ReadModule(deno_core::error::AnyError), #[error("Unable to get CWD: {0}")] UnableToGetCwd(deno_io::fs::FsError), } @@ -477,7 +480,9 @@ where let file_path = ensure_read_permission::

(state, &file_path) .map_err(RequireError::Permission)?; let loader = state.borrow::(); - Ok(loader.load_text_file_lossy(&file_path)?) + loader + .load_text_file_lossy(&file_path) + .map_err(RequireError::ReadModule) } #[op2] @@ -637,13 +642,6 @@ fn url_to_file_path_string(url: &Url) -> Result { Ok(file_path.to_string_lossy().into_owned()) } -fn url_to_file_path(url: &Url) -> Result { - match url.to_file_path() { - Ok(file_path) => Ok(file_path), - Err(()) => Err(RequireError::FilePathConversion(url.clone())), - } -} - #[op2(fast)] pub fn op_require_can_parse_as_esm( scope: &mut v8::HandleScope, diff --git a/runtime/errors.rs b/runtime/errors.rs index 07bf694dc17ebf..325990c7831065 100644 --- a/runtime/errors.rs +++ b/runtime/errors.rs @@ -1058,11 +1058,12 @@ mod node { match error { RequireError::UrlParse(e) => get_url_parse_error_class(e), RequireError::Permission(e) => get_error_class_name(e).unwrap_or("Error"), - RequireError::PackageExportsResolve(_) => "Error", - RequireError::PackageJsonLoad(_) => "Error", - RequireError::ClosestPkgJson(_) => "Error", - RequireError::FilePathConversion(_) => "Error", - RequireError::PackageImportsResolve(_) => "Error", + RequireError::PackageExportsResolve(_) + | RequireError::PackageJsonLoad(_) + | RequireError::ClosestPkgJson(_) + | RequireError::FilePathConversion(_) + | RequireError::ReadModule(_) + | RequireError::PackageImportsResolve(_) => "Error", RequireError::Fs(e) | RequireError::UnableToGetCwd(e) => { super::get_fs_error(e) } From 34b1a883271342c2b32d29fb6a0789b7b3ff42e1 Mon Sep 17 00:00:00 2001 From: David Sherret Date: Fri, 25 Oct 2024 11:03:08 -0400 Subject: [PATCH 05/42] working --- cli/cache/mod.rs | 9 --------- 1 file changed, 9 deletions(-) diff --git a/cli/cache/mod.rs b/cli/cache/mod.rs index bf8f1b1f0ba273..e150bbef543b07 100644 --- a/cli/cache/mod.rs +++ b/cli/cache/mod.rs @@ -287,15 +287,6 @@ impl Loader for FetchCacher { } } - // make local CJS modules external to the graph - if specifier_has_extension(specifier, "cjs") { - return Box::pin(futures::future::ready(Ok(Some( - LoadResponse::External { - specifier: specifier.clone(), - }, - )))); - } - if self.unstable_detect_cjs && specifier_has_extension(specifier, "js") { if let Ok(Some(pkg_json)) = self.node_resolver.get_closest_package_json(specifier) From 873da6b03a82122e9a2580ea8143c63cc38f941e Mon Sep 17 00:00:00 2001 From: David Sherret Date: Fri, 25 Oct 2024 16:41:02 -0400 Subject: [PATCH 06/42] Start on translating code --- cli/factory.rs | 4 +++- cli/module_loader.rs | 6 ++++++ tests/specs/compile/cjs/__test__.jsonc | 24 ++++++++++++++++++++++++ tests/specs/compile/cjs/add.cjs | 1 + tests/specs/compile/cjs/file.cjs | 1 + tests/specs/compile/cjs/main.js | 3 +++ tests/specs/compile/cjs/output.out | 1 + 7 files changed, 39 insertions(+), 1 deletion(-) create mode 100644 tests/specs/compile/cjs/__test__.jsonc create mode 100644 tests/specs/compile/cjs/add.cjs create mode 100644 tests/specs/compile/cjs/file.cjs create mode 100644 tests/specs/compile/cjs/main.js create mode 100644 tests/specs/compile/cjs/output.out diff --git a/cli/factory.rs b/cli/factory.rs index e2080366b78f75..659c297fa3f67d 100644 --- a/cli/factory.rs +++ b/cli/factory.rs @@ -802,6 +802,7 @@ impl CliFactory { } else { None }; + let node_code_translator = self.node_code_translator().await?; Ok(CliMainWorkerFactory::new( self.blob_store().clone(), @@ -827,11 +828,12 @@ impl CliFactory { fs.clone(), self.main_module_graph_container().await?.clone(), self.module_load_preparer().await?.clone(), + node_code_translator.clone(), cli_node_resolver.clone(), cli_npm_resolver.clone(), NpmModuleLoader::new( self.cjs_resolutions().clone(), - self.node_code_translator().await?.clone(), + node_code_translator.clone(), fs.clone(), cli_node_resolver.clone(), ), diff --git a/cli/module_loader.rs b/cli/module_loader.rs index 9f2ed09dc3361c..4493b79d587652 100644 --- a/cli/module_loader.rs +++ b/cli/module_loader.rs @@ -23,6 +23,7 @@ use crate::graph_container::ModuleGraphUpdatePermit; use crate::graph_util::CreateGraphOptions; use crate::graph_util::ModuleGraphBuilder; use crate::node; +use crate::node::CliNodeCodeTranslator; use crate::npm::CliNodeRequireLoader; use crate::npm::CliNpmResolver; use crate::resolver::CliGraphResolver; @@ -206,6 +207,7 @@ struct SharedCliModuleLoaderState { fs: Arc, main_module_graph_container: Arc, module_load_preparer: Arc, + node_code_translator: Arc, node_resolver: Arc, npm_resolver: Arc, npm_module_loader: NpmModuleLoader, @@ -226,6 +228,7 @@ impl CliModuleLoaderFactory { fs: Arc, main_module_graph_container: Arc, module_load_preparer: Arc, + node_code_translator: Arc, node_resolver: Arc, npm_resolver: Arc, npm_module_loader: NpmModuleLoader, @@ -248,6 +251,7 @@ impl CliModuleLoaderFactory { fs, main_module_graph_container, module_load_preparer, + node_code_translator, node_resolver, npm_resolver, npm_module_loader, @@ -272,6 +276,7 @@ impl CliModuleLoaderFactory { parent_permissions, permissions, graph_container, + node_code_translator: self.shared.node_code_translator.clone(), emitter: self.shared.emitter.clone(), parsed_source_cache: self.shared.parsed_source_cache.clone(), shared: self.shared.clone(), @@ -329,6 +334,7 @@ struct CliModuleLoaderInner { permissions: PermissionsContainer, shared: Arc, emitter: Arc, + node_code_translator: Arc, parsed_source_cache: Arc, graph_container: TGraphContainer, } diff --git a/tests/specs/compile/cjs/__test__.jsonc b/tests/specs/compile/cjs/__test__.jsonc new file mode 100644 index 00000000000000..eda3a23d4fb1aa --- /dev/null +++ b/tests/specs/compile/cjs/__test__.jsonc @@ -0,0 +1,24 @@ +{ + "tempDir": true, + "steps": [{ + "if": "unix", + "args": "compile -A --output main main.js", + "output": "[WILDCARD]" + }, { + "if": "unix", + "commandName": "./main", + "args": [], + "output": "output.out", + "exitCode": 1 + }, { + "if": "windows", + "args": "compile -A --output main.exe main.js", + "output": "[WILDCARD]" + }, { + "if": "windows", + "commandName": "./main.exe", + "args": [], + "output": "output.out", + "exitCode": 1 + }] +} diff --git a/tests/specs/compile/cjs/add.cjs b/tests/specs/compile/cjs/add.cjs new file mode 100644 index 00000000000000..bf906012673a39 --- /dev/null +++ b/tests/specs/compile/cjs/add.cjs @@ -0,0 +1 @@ +module.exports = (a, b) => a + b; diff --git a/tests/specs/compile/cjs/file.cjs b/tests/specs/compile/cjs/file.cjs new file mode 100644 index 00000000000000..af7cecfdf47876 --- /dev/null +++ b/tests/specs/compile/cjs/file.cjs @@ -0,0 +1 @@ +module.exports.add = require("./add.cjs"); diff --git a/tests/specs/compile/cjs/main.js b/tests/specs/compile/cjs/main.js new file mode 100644 index 00000000000000..ac78584fed0efa --- /dev/null +++ b/tests/specs/compile/cjs/main.js @@ -0,0 +1,3 @@ +import { add } from "./file.cjs"; + +console.log(add(1, 2)); diff --git a/tests/specs/compile/cjs/output.out b/tests/specs/compile/cjs/output.out new file mode 100644 index 00000000000000..00750edc07d641 --- /dev/null +++ b/tests/specs/compile/cjs/output.out @@ -0,0 +1 @@ +3 From 342c81cd8a58c468c13839ec42b625319008b8b5 Mon Sep 17 00:00:00 2001 From: David Sherret Date: Fri, 25 Oct 2024 17:27:44 -0400 Subject: [PATCH 07/42] perf: pass transpiled modules to deno_core as a string --- cli/cache/emit.rs | 25 +++++++------------------ cli/emit.rs | 17 +++++++++-------- cli/module_loader.rs | 6 ++++-- cli/standalone/binary.rs | 2 +- cli/tools/coverage/mod.rs | 5 ++--- 5 files changed, 23 insertions(+), 32 deletions(-) diff --git a/cli/cache/emit.rs b/cli/cache/emit.rs index 6807f06c1063e4..74e1c1101bb6fd 100644 --- a/cli/cache/emit.rs +++ b/cli/cache/emit.rs @@ -39,7 +39,7 @@ impl EmitCache { &self, specifier: &ModuleSpecifier, expected_source_hash: u64, - ) -> Option> { + ) -> Option { let emit_filename = self.get_emit_filename(specifier)?; let bytes = self.disk_cache.get(&emit_filename).ok()?; self @@ -100,7 +100,7 @@ impl EmitFileSerializer { &self, mut bytes: Vec, expected_source_hash: u64, - ) -> Option> { + ) -> Option { let last_newline_index = bytes.iter().rposition(|&b| b == b'\n')?; let (content, last_line) = bytes.split_at(last_newline_index); let hashes = last_line.strip_prefix(LAST_LINE_PREFIX.as_bytes())?; @@ -120,7 +120,7 @@ impl EmitFileSerializer { // everything looks good, truncate and return it bytes.truncate(content.len()); - Some(bytes) + String::from_utf8(bytes).ok() } pub fn serialize(&self, code: &[u8], source_hash: u64) -> Vec { @@ -170,8 +170,6 @@ mod test { }, emit_failed_flag: Default::default(), }; - let to_string = - |bytes: Vec| -> String { String::from_utf8(bytes).unwrap() }; let specifier1 = ModuleSpecifier::from_file_path(temp_dir.path().join("file1.ts")) @@ -188,13 +186,10 @@ mod test { assert_eq!(cache.get_emit_code(&specifier1, 5), None); // providing the correct source hash assert_eq!( - cache.get_emit_code(&specifier1, 10).map(to_string), + cache.get_emit_code(&specifier1, 10), Some(emit_code1.clone()), ); - assert_eq!( - cache.get_emit_code(&specifier2, 2).map(to_string), - Some(emit_code2) - ); + assert_eq!(cache.get_emit_code(&specifier2, 2), Some(emit_code2)); // try changing the cli version (should not load previous ones) let cache = EmitCache { @@ -215,18 +210,12 @@ mod test { }, emit_failed_flag: Default::default(), }; - assert_eq!( - cache.get_emit_code(&specifier1, 5).map(to_string), - Some(emit_code1) - ); + assert_eq!(cache.get_emit_code(&specifier1, 5), Some(emit_code1)); // adding when already exists should not cause issue let emit_code3 = "asdf".to_string(); cache.set_emit_code(&specifier1, 20, emit_code3.as_bytes()); assert_eq!(cache.get_emit_code(&specifier1, 5), None); - assert_eq!( - cache.get_emit_code(&specifier1, 20).map(to_string), - Some(emit_code3) - ); + assert_eq!(cache.get_emit_code(&specifier1, 20), Some(emit_code3)); } } diff --git a/cli/emit.rs b/cli/emit.rs index ad200af0504222..9cb407f9e9c01f 100644 --- a/cli/emit.rs +++ b/cli/emit.rs @@ -13,7 +13,6 @@ use deno_core::error::AnyError; use deno_core::futures::stream::FuturesUnordered; use deno_core::futures::FutureExt; use deno_core::futures::StreamExt; -use deno_core::ModuleCodeBytes; use deno_core::ModuleSpecifier; use deno_graph::MediaType; use deno_graph::Module; @@ -94,7 +93,7 @@ impl Emitter { &self, specifier: &ModuleSpecifier, source: &str, - ) -> Option> { + ) -> Option { let source_hash = self.get_source_hash(source); self.emit_cache.get_emit_code(specifier, source_hash) } @@ -104,7 +103,7 @@ impl Emitter { specifier: &ModuleSpecifier, media_type: MediaType, source: &Arc, - ) -> Result { + ) -> Result { // Note: keep this in sync with the sync version below let helper = EmitParsedSourceHelper(self); match helper.pre_emit_parsed_source(specifier, source) { @@ -143,7 +142,7 @@ impl Emitter { specifier: &ModuleSpecifier, media_type: MediaType, source: &Arc, - ) -> Result { + ) -> Result { // Note: keep this in sync with the async version above let helper = EmitParsedSourceHelper(self); match helper.pre_emit_parsed_source(specifier, source) { @@ -227,7 +226,7 @@ impl Emitter { } enum PreEmitResult { - Cached(ModuleCodeBytes), + Cached(String), NotCached { source_hash: u64 }, } @@ -245,7 +244,7 @@ impl<'a> EmitParsedSourceHelper<'a> { if let Some(emit_code) = self.0.emit_cache.get_emit_code(specifier, source_hash) { - PreEmitResult::Cached(emit_code.into_boxed_slice().into()) + PreEmitResult::Cached(emit_code) } else { PreEmitResult::NotCached { source_hash } } @@ -272,7 +271,7 @@ impl<'a> EmitParsedSourceHelper<'a> { specifier: &ModuleSpecifier, transpile_result: TranspileResult, source_hash: u64, - ) -> ModuleCodeBytes { + ) -> String { let transpiled_source = match transpile_result { TranspileResult::Owned(source) => source, TranspileResult::Cloned(source) => { @@ -286,7 +285,9 @@ impl<'a> EmitParsedSourceHelper<'a> { source_hash, &transpiled_source.source, ); - transpiled_source.source.into_boxed_slice().into() + // todo(https://github.com/denoland/deno_ast/issues/282): move to deno_ast + // SAFETY: This is fine because swc is working off of Strings + unsafe { String::from_utf8_unchecked(transpiled_source.source) } } } diff --git a/cli/module_loader.rs b/cli/module_loader.rs index 37d42f78e5e93b..4a020516e70a59 100644 --- a/cli/module_loader.rs +++ b/cli/module_loader.rs @@ -541,7 +541,8 @@ impl self.parsed_source_cache.free(specifier); Ok(Some(ModuleCodeStringSource { - code: ModuleSourceCode::Bytes(transpile_result), + // note: it's faster to provide a string if we know it's a string + code: ModuleSourceCode::String(transpile_result.into()), found_url: specifier.clone(), media_type, })) @@ -571,7 +572,8 @@ impl self.parsed_source_cache.free(specifier); Ok(Some(ModuleCodeStringSource { - code: ModuleSourceCode::Bytes(transpile_result), + // note: it's faster to provide a string if we know it's a string + code: ModuleSourceCode::String(transpile_result.into()), found_url: specifier.clone(), media_type, })) diff --git a/cli/standalone/binary.rs b/cli/standalone/binary.rs index 0f8b0b49d4a408..f41f3003ff1b3e 100644 --- a/cli/standalone/binary.rs +++ b/cli/standalone/binary.rs @@ -613,7 +613,7 @@ impl<'a> DenoCompileBinaryWriter<'a> { .emitter .emit_parsed_source(&m.specifier, m.media_type, &m.source) .await?; - source.to_vec() + source.into_bytes() } else { m.source.as_bytes().to_vec() }; diff --git a/cli/tools/coverage/mod.rs b/cli/tools/coverage/mod.rs index 260c0c842477a7..3b08f2c77adc90 100644 --- a/cli/tools/coverage/mod.rs +++ b/cli/tools/coverage/mod.rs @@ -571,7 +571,7 @@ pub async fn cover_files( | MediaType::Cjs | MediaType::Mjs | MediaType::Json => None, - MediaType::Dts | MediaType::Dmts | MediaType::Dcts => Some(Vec::new()), + MediaType::Dts | MediaType::Dmts | MediaType::Dcts => Some(String::new()), MediaType::TypeScript | MediaType::Jsx | MediaType::Mts @@ -593,8 +593,7 @@ pub async fn cover_files( } }; let runtime_code: String = match transpiled_code { - Some(code) => String::from_utf8(code) - .with_context(|| format!("Failed decoding {}", file.specifier))?, + Some(code) => code, None => original_source.to_string(), }; From 3cd7c94b722d88ac83289f7c88cec4c0ab90ffda Mon Sep 17 00:00:00 2001 From: David Sherret Date: Fri, 25 Oct 2024 17:34:01 -0400 Subject: [PATCH 08/42] update --- cli/emit.rs | 46 ++++++++++++++++++++++++---------------------- 1 file changed, 24 insertions(+), 22 deletions(-) diff --git a/cli/emit.rs b/cli/emit.rs index 9cb407f9e9c01f..8e93092e67b1cd 100644 --- a/cli/emit.rs +++ b/cli/emit.rs @@ -112,7 +112,7 @@ impl Emitter { let parsed_source_cache = self.parsed_source_cache.clone(); let transpile_and_emit_options = self.transpile_and_emit_options.clone(); - let transpile_result = deno_core::unsync::spawn_blocking({ + let transpiled_source = deno_core::unsync::spawn_blocking({ let specifier = specifier.clone(); let source = source.clone(); move || -> Result<_, AnyError> { @@ -128,11 +128,12 @@ impl Emitter { }) .await .unwrap()?; - Ok(helper.post_emit_parsed_source( + helper.post_emit_parsed_source( specifier, - transpile_result, + &transpiled_source, source_hash, - )) + ); + Ok(transpiled_source) } } } @@ -148,7 +149,7 @@ impl Emitter { match helper.pre_emit_parsed_source(specifier, source) { PreEmitResult::Cached(emitted_text) => Ok(emitted_text), PreEmitResult::NotCached { source_hash } => { - let transpile_result = EmitParsedSourceHelper::transpile( + let transpiled_source = EmitParsedSourceHelper::transpile( &self.parsed_source_cache, specifier, source.clone(), @@ -156,11 +157,12 @@ impl Emitter { &self.transpile_and_emit_options.0, &self.transpile_and_emit_options.1, )?; - Ok(helper.post_emit_parsed_source( + helper.post_emit_parsed_source( specifier, - transpile_result, + &transpiled_source, source_hash, - )) + ); + Ok(transpiled_source) } } } @@ -257,21 +259,14 @@ impl<'a> EmitParsedSourceHelper<'a> { media_type: MediaType, transpile_options: &deno_ast::TranspileOptions, emit_options: &deno_ast::EmitOptions, - ) -> Result { + ) -> Result { // nothing else needs the parsed source at this point, so remove from // the cache in order to not transpile owned let parsed_source = parsed_source_cache .remove_or_parse_module(specifier, source, media_type)?; ensure_no_import_assertion(&parsed_source)?; - Ok(parsed_source.transpile(transpile_options, emit_options)?) - } - - pub fn post_emit_parsed_source( - &self, - specifier: &ModuleSpecifier, - transpile_result: TranspileResult, - source_hash: u64, - ) -> String { + let transpile_result = + parsed_source.transpile(transpile_options, emit_options)?; let transpiled_source = match transpile_result { TranspileResult::Owned(source) => source, TranspileResult::Cloned(source) => { @@ -280,14 +275,21 @@ impl<'a> EmitParsedSourceHelper<'a> { } }; debug_assert!(transpiled_source.source_map.is_none()); + let text = String::from_utf8(transpiled_source.source)?; + Ok(text) + } + + pub fn post_emit_parsed_source( + &self, + specifier: &ModuleSpecifier, + transpiled_source: &str, + source_hash: u64, + ) { self.0.emit_cache.set_emit_code( specifier, source_hash, - &transpiled_source.source, + transpiled_source.as_bytes(), ); - // todo(https://github.com/denoland/deno_ast/issues/282): move to deno_ast - // SAFETY: This is fine because swc is working off of Strings - unsafe { String::from_utf8_unchecked(transpiled_source.source) } } } From 85403444397b29f69c75d2b0287777f272fa3005 Mon Sep 17 00:00:00 2001 From: David Sherret Date: Fri, 25 Oct 2024 18:20:29 -0400 Subject: [PATCH 09/42] update --- Cargo.lock | 8 ++++---- cli/module_loader.rs | 41 +++++++++++++++++++++++++++++++++++++---- 2 files changed, 41 insertions(+), 8 deletions(-) diff --git a/Cargo.lock b/Cargo.lock index 623d028ce65fc9..fe890341d7e919 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -1280,8 +1280,6 @@ dependencies = [ [[package]] name = "deno_ast" version = "0.42.2" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "b2b9d03b1bbeeecdac54367f075d572131736d06c5be3bc49037855bc5ab1bbb" dependencies = [ "base64 0.21.7", "deno_media_type", @@ -1507,8 +1505,6 @@ dependencies = [ [[package]] name = "deno_doc" version = "0.154.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "17e204e45b0d79750880114e37b34abe19ad0710d8435a8da8f23a528fe98de4" dependencies = [ "anyhow", "cfg-if", @@ -8616,3 +8612,7 @@ dependencies = [ "cc", "pkg-config", ] + +[[patch.unused]] +name = "deno_graph" +version = "0.83.3" diff --git a/cli/module_loader.rs b/cli/module_loader.rs index 4493b79d587652..7dc7a71b0e12c6 100644 --- a/cli/module_loader.rs +++ b/cli/module_loader.rs @@ -563,11 +563,21 @@ impl media_type, })) } + Some(CodeOrDeferredEmit::Cjs { + specifier, + is_cts, + source, + }) => { + // let transpile_result = match is_cts { + // true => String::from_utf8(self.emitter.emit_parsed_source(specifier, MediaType::Cts, source).await?.into_bytes())?, + // false => + // } + } None => Ok(None), } } - fn load_prepared_module_sync( + fn load_prepared_module_for_source_map_sync( &self, specifier: &ModuleSpecifier, ) -> Result, AnyError> { @@ -593,6 +603,16 @@ impl media_type, })) } + Some(CodeOrDeferredEmit::Cjs { + specifier, + is_cts, + source, + }) => { + // todo(dsherret): to make this work, we should probably just + // rely on the CJS export cache. At the moment this is hard because + // cjs export analysis is only async + Ok(None) + } None => Ok(None), } } @@ -627,15 +647,20 @@ impl let code: ModuleCodeString = match media_type { MediaType::JavaScript | MediaType::Unknown - | MediaType::Cjs | MediaType::Mjs | MediaType::Json => source.clone().into(), MediaType::Dts | MediaType::Dcts | MediaType::Dmts => { Default::default() } + MediaType::Cjs | MediaType::Cts => { + return Ok(Some(CodeOrDeferredEmit::Cjs { + specifier, + is_cts: media_type == MediaType::Cts, + source, + })); + } MediaType::TypeScript | MediaType::Mts - | MediaType::Cts | MediaType::Jsx | MediaType::Tsx => { return Ok(Some(CodeOrDeferredEmit::DeferredEmit { @@ -675,6 +700,11 @@ enum CodeOrDeferredEmit<'a> { media_type: MediaType, source: &'a Arc, }, + Cjs { + specifier: &'a ModuleSpecifier, + is_cts: bool, + source: &'a Arc, + }, } // todo(dsherret): this double Rc boxing is not ideal @@ -836,7 +866,10 @@ impl ModuleLoader "wasm" | "file" | "http" | "https" | "data" | "blob" => (), _ => return None, } - let source = self.0.load_prepared_module_sync(&specifier).ok()??; + let source = self + .0 + .load_prepared_module_for_source_map_sync(&specifier) + .ok()??; source_map_from_code(source.code.as_bytes()) } From af06589cd847519017086710e8bd2e745a32e5f6 Mon Sep 17 00:00:00 2001 From: David Sherret Date: Fri, 25 Oct 2024 19:26:57 -0400 Subject: [PATCH 10/42] better cts support --- Cargo.lock | 6 ------ cli/cache/emit.rs | 2 ++ cli/cache/parsed_source.rs | 2 +- cli/emit.rs | 1 + cli/module_loader.rs | 41 ++++++++++++++++++++++++++----------- cli/npm/mod.rs | 22 ++++++++++++++++++-- cli/resolver.rs | 5 +++-- cli/tsc/99_main_compiler.js | 5 ----- cli/tsc/mod.rs | 3 +++ cli/util/path.rs | 13 +++++++----- cli/worker.rs | 3 ++- 11 files changed, 69 insertions(+), 34 deletions(-) diff --git a/Cargo.lock b/Cargo.lock index fe890341d7e919..1058a366cdf83b 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -1603,8 +1603,6 @@ dependencies = [ [[package]] name = "deno_graph" version = "0.83.4" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "5bd20bc0780071989c622cbfd5d4fb2e4fd05a247ccd7f791f13c8d2c3792228" dependencies = [ "anyhow", "async-trait", @@ -8612,7 +8610,3 @@ dependencies = [ "cc", "pkg-config", ] - -[[patch.unused]] -name = "deno_graph" -version = "0.83.3" diff --git a/cli/cache/emit.rs b/cli/cache/emit.rs index 74e1c1101bb6fd..3c9eecfcbdb6b2 100644 --- a/cli/cache/emit.rs +++ b/cli/cache/emit.rs @@ -10,6 +10,7 @@ use deno_core::unsync::sync::AtomicFlag; use super::DiskCache; /// The cache that stores previously emitted files. +#[derive(Debug)] pub struct EmitCache { disk_cache: DiskCache, emit_failed_flag: AtomicFlag, @@ -91,6 +92,7 @@ impl EmitCache { const LAST_LINE_PREFIX: &str = "\n// denoCacheMetadata="; +#[derive(Debug)] struct EmitFileSerializer { cli_version: &'static str, } diff --git a/cli/cache/parsed_source.rs b/cli/cache/parsed_source.rs index df6e45c35e565b..3104a0df3e4863 100644 --- a/cli/cache/parsed_source.rs +++ b/cli/cache/parsed_source.rs @@ -47,7 +47,7 @@ impl<'a> LazyGraphSourceParser<'a> { } } -#[derive(Default)] +#[derive(Debug, Default)] pub struct ParsedSourceCache { sources: Mutex>, } diff --git a/cli/emit.rs b/cli/emit.rs index 4c4dc83ca43d5d..653d5afd08d3d0 100644 --- a/cli/emit.rs +++ b/cli/emit.rs @@ -19,6 +19,7 @@ use deno_graph::Module; use deno_graph::ModuleGraph; use std::sync::Arc; +#[derive(Debug)] pub struct Emitter { emit_cache: Arc, parsed_source_cache: Arc, diff --git a/cli/module_loader.rs b/cli/module_loader.rs index 32cb8848504758..adc261c221f59f 100644 --- a/cli/module_loader.rs +++ b/cli/module_loader.rs @@ -282,6 +282,7 @@ impl CliModuleLoaderFactory { shared: self.shared.clone(), }))); let node_require_loader = Rc::new(CliNodeRequireLoader::new( + self.shared.emitter.clone(), self.shared.fs.clone(), self.shared.npm_resolver.clone(), )); @@ -566,13 +567,31 @@ impl } Some(CodeOrDeferredEmit::Cjs { specifier, - is_cts, + media_type, source, }) => { - // let transpile_result = match is_cts { - // true => String::from_utf8(self.emitter.emit_parsed_source(specifier, MediaType::Cts, source).await?.into_bytes())?, - // false => - // } + let js_source = match media_type { + MediaType::Cts => { + self + .emitter + .emit_parsed_source(specifier, MediaType::Cts, source) + .await? + } + MediaType::Cjs => source.to_string(), + _ => unreachable!(), + }; + let text = self + .node_code_translator + .translate_cjs_to_esm(specifier, Some(js_source)) + .await?; + // at this point, we no longer need the parsed source in memory, so free it + self.parsed_source_cache.free(specifier); + + Ok(Some(ModuleCodeStringSource { + code: ModuleSourceCode::String(text.into()), + found_url: specifier.clone(), + media_type, + })) } None => Ok(None), } @@ -605,11 +624,9 @@ impl media_type, })) } - Some(CodeOrDeferredEmit::Cjs { - specifier, - is_cts, - source, - }) => { + Some(CodeOrDeferredEmit::Cjs { .. }) => { + self.parsed_source_cache.free(specifier); + // todo(dsherret): to make this work, we should probably just // rely on the CJS export cache. At the moment this is hard because // cjs export analysis is only async @@ -657,7 +674,7 @@ impl MediaType::Cjs | MediaType::Cts => { return Ok(Some(CodeOrDeferredEmit::Cjs { specifier, - is_cts: media_type == MediaType::Cts, + media_type: *media_type, source, })); } @@ -704,7 +721,7 @@ enum CodeOrDeferredEmit<'a> { }, Cjs { specifier: &'a ModuleSpecifier, - is_cts: bool, + media_type: MediaType, source: &'a Arc, }, } diff --git a/cli/npm/mod.rs b/cli/npm/mod.rs index 11785cbf642fa0..08362698b6d2ba 100644 --- a/cli/npm/mod.rs +++ b/cli/npm/mod.rs @@ -11,6 +11,7 @@ use std::sync::Arc; use common::maybe_auth_header_for_npm_registry; use dashmap::DashMap; +use deno_ast::MediaType; use deno_ast::ModuleSpecifier; use deno_core::error::AnyError; use deno_core::serde_json; @@ -28,6 +29,7 @@ use managed::cache::registry_info::get_package_url; use node_resolver::NpmResolver; use thiserror::Error; +use crate::emit::Emitter; use crate::file_fetcher::FileFetcher; pub use self::byonm::CliByonmNpmResolver; @@ -200,16 +202,22 @@ impl NpmFetchResolver { #[derive(Debug)] pub struct CliNodeRequireLoader { + emitter: Arc, fs: Arc, npm_resolver: Arc, } impl CliNodeRequireLoader { pub fn new( + emitter: Arc, fs: Arc, npm_resolver: Arc, ) -> Self { - Self { fs, npm_resolver } + Self { + emitter, + fs, + npm_resolver, + } } } @@ -223,6 +231,16 @@ impl NodeRequireLoader for CliNodeRequireLoader { } fn load_text_file_lossy(&self, path: &Path) -> Result { - Ok(self.fs.read_text_file_lossy_sync(path, None)?) + // todo(dsherret): use the preloaded module from the graph if available + let specifier = deno_path_util::url_from_file_path(&path)?; + let media_type = MediaType::from_specifier(&specifier); + let text = self.fs.read_text_file_lossy_sync(path, None)?; + if media_type == MediaType::Cts { + self + .emitter + .emit_parsed_source_sync(&specifier, media_type, &text.into()) + } else { + Ok(text) + } } } diff --git a/cli/resolver.rs b/cli/resolver.rs index 84c671268a2896..606c2c14669ef7 100644 --- a/cli/resolver.rs +++ b/cli/resolver.rs @@ -368,7 +368,6 @@ impl NpmModuleLoader { pub fn if_in_npm_package(&self, specifier: &ModuleSpecifier) -> bool { self.node_resolver.in_npm_package(specifier) - || self.cjs_resolutions.is_known_cjs(specifier) } pub async fn load( @@ -445,7 +444,9 @@ impl CjsResolutionStore { return false; } - specifier_has_extension(specifier, "cjs") || self.0.contains(specifier) + specifier_has_extension(specifier, "cjs") + || specifier_has_extension(specifier, "cts") + || self.0.contains(specifier) } pub fn insert(&self, specifier: ModuleSpecifier) { diff --git a/cli/tsc/99_main_compiler.js b/cli/tsc/99_main_compiler.js index e3a0bee597105c..a6de96763bf9ae 100644 --- a/cli/tsc/99_main_compiler.js +++ b/cli/tsc/99_main_compiler.js @@ -1342,13 +1342,8 @@ delete Object.prototype.__proto__; "Global", "ImportMeta", "localStorage", - "module", - "NodeModule", - "NodeRequire", - "process", "queueMicrotask", "RequestInit", - "require", "ResponseInit", "sessionStorage", "setImmediate", diff --git a/cli/tsc/mod.rs b/cli/tsc/mod.rs index de91889b623b20..a55594460c34f0 100644 --- a/cli/tsc/mod.rs +++ b/cli/tsc/mod.rs @@ -560,6 +560,9 @@ fn op_load_inner( match module { Module::Js(module) => { media_type = module.media_type; + if matches!(media_type, MediaType::Cjs | MediaType::Cts) { + is_cjs = true; + } let source = module .fast_check_module() .map(|m| &*m.source) diff --git a/cli/util/path.rs b/cli/util/path.rs index e4ae6e7cb14d7d..b68e1361b7a0e6 100644 --- a/cli/util/path.rs +++ b/cli/util/path.rs @@ -42,19 +42,22 @@ pub fn get_extension(file_path: &Path) -> Option { .map(|e| e.to_lowercase()); } +// todo(dsherret): replace with https://github.com/denoland/deno_path_util/pull/4 pub fn specifier_has_extension( specifier: &ModuleSpecifier, searching_ext: &str, ) -> bool { - let Some((_, ext)) = specifier.path().rsplit_once('.') else { - return false; - }; let searching_ext = searching_ext.strip_prefix('.').unwrap_or(searching_ext); debug_assert!(!searching_ext.contains('.')); // exts like .d.ts are not implemented here - if ext.len() != searching_ext.len() { + let path = specifier.path(); + if path.len() < searching_ext.len() { return false; } - ext.eq_ignore_ascii_case(searching_ext) + let ext_pos = path.len() - searching_ext.len(); + let (start_path, end_path) = path.split_at(ext_pos); + end_path.eq_ignore_ascii_case(searching_ext) + && start_path.ends_with('.') + && !start_path.ends_with("/.") } pub fn get_atomic_dir_path(file_path: &Path) -> PathBuf { diff --git a/cli/worker.rs b/cli/worker.rs index bbfed670a282c0..fa2dbde94a2e7c 100644 --- a/cli/worker.rs +++ b/cli/worker.rs @@ -562,7 +562,8 @@ impl CliMainWorkerFactory { self.shared.cjs_resolution_store.is_known_cjs(&main_module) } else { main_module.scheme() == "file" - && specifier_has_extension(&main_module, "cjs") + && (specifier_has_extension(&main_module, "cjs") + || specifier_has_extension(&main_module, "cts")) }; (main_module, is_cjs) }; From eb9f9ff312bf21bf0ab0853d4b2807e16a1f4701 Mon Sep 17 00:00:00 2001 From: David Sherret Date: Mon, 28 Oct 2024 10:33:00 -0400 Subject: [PATCH 11/42] fix(check): expose more globals from @types/node --- cli/tsc/99_main_compiler.js | 6 ------ tests/specs/npm/compare_globals/__test__.jsonc | 4 ++-- .../compare_globals/{compare_globals => }/main.out | 3 ++- .../npm/compare_globals/{compare_globals => }/main.ts | 11 +++++++++-- 4 files changed, 13 insertions(+), 11 deletions(-) rename tests/specs/npm/compare_globals/{compare_globals => }/main.out (90%) rename tests/specs/npm/compare_globals/{compare_globals => }/main.ts (84%) diff --git a/cli/tsc/99_main_compiler.js b/cli/tsc/99_main_compiler.js index e3a0bee597105c..6011dece76b6c6 100644 --- a/cli/tsc/99_main_compiler.js +++ b/cli/tsc/99_main_compiler.js @@ -1337,18 +1337,12 @@ delete Object.prototype.__proto__; "console", "Console", "ErrorConstructor", - "exports", "gc", "Global", "ImportMeta", "localStorage", - "module", - "NodeModule", - "NodeRequire", - "process", "queueMicrotask", "RequestInit", - "require", "ResponseInit", "sessionStorage", "setImmediate", diff --git a/tests/specs/npm/compare_globals/__test__.jsonc b/tests/specs/npm/compare_globals/__test__.jsonc index e096ddeb221a5a..96458306a79143 100644 --- a/tests/specs/npm/compare_globals/__test__.jsonc +++ b/tests/specs/npm/compare_globals/__test__.jsonc @@ -1,4 +1,4 @@ { - "args": "run --allow-read --check=all compare_globals/main.ts", - "output": "compare_globals/main.out" + "args": "run --allow-read --check=all main.ts", + "output": "main.out" } diff --git a/tests/specs/npm/compare_globals/compare_globals/main.out b/tests/specs/npm/compare_globals/main.out similarity index 90% rename from tests/specs/npm/compare_globals/compare_globals/main.out rename to tests/specs/npm/compare_globals/main.out index 5be6125b08468b..290e9c3b23d18f 100644 --- a/tests/specs/npm/compare_globals/compare_globals/main.out +++ b/tests/specs/npm/compare_globals/main.out @@ -8,10 +8,11 @@ Download http://localhost:4260/@denotest/globals/1.0.0.tgz Download http://localhost:4260/@types/node/node-22.5.4.tgz Download http://localhost:4260/undici-types/undici-types-6.19.8.tgz [UNORDERED_END] -Check file:///[WILDCARD]/compare_globals/main.ts +Check file:///[WILDCARD]/main.ts true true [] +process equals process true setTimeout 1 false setTimeout 2 function setTimeout 3 function diff --git a/tests/specs/npm/compare_globals/compare_globals/main.ts b/tests/specs/npm/compare_globals/main.ts similarity index 84% rename from tests/specs/npm/compare_globals/compare_globals/main.ts rename to tests/specs/npm/compare_globals/main.ts index 9482798d8c9a24..308ce9b23f1672 100644 --- a/tests/specs/npm/compare_globals/compare_globals/main.ts +++ b/tests/specs/npm/compare_globals/main.ts @@ -5,10 +5,17 @@ console.log(globals.global === globals.globalThis); // @ts-expect-error even though these are the same object, they have different types console.log(globals.globalThis === globalThis); console.log(globals.process.execArgv); +console.log("process equals process", process === globals.process); type AssertTrue = never; -type _TestNoProcessGlobal = AssertTrue< - typeof globalThis extends { process: any } ? false : true +type _TestHasProcessGlobal = AssertTrue< + typeof globalThis extends { process: any } ? true : false +>; +type _TestProcessGlobalVersion = AssertTrue< + typeof process.versions.node extends string ? true : false +>; +type _TestNoBufferGlogal = AssertTrue< + typeof globalThis extends { Buffer: any } ? false : true >; type _TestHasNodeJsGlobal = NodeJS.Architecture; From 9e7ad6f13b944fac24a61c143ca4c6e4a48e33de Mon Sep 17 00:00:00 2001 From: David Sherret Date: Mon, 28 Oct 2024 11:37:48 -0400 Subject: [PATCH 12/42] support in deno compile --- cli/module_loader.rs | 10 ++--- cli/node.rs | 9 +++-- cli/resolver.rs | 4 +- cli/standalone/mod.rs | 69 ++++++++++++++++++++++++++++----- cli/standalone/serialization.rs | 45 ++++++++++++++------- cli/util/logger.rs | 2 + resolvers/node/analyze.rs | 20 +++++----- 7 files changed, 115 insertions(+), 44 deletions(-) diff --git a/cli/module_loader.rs b/cli/module_loader.rs index adc261c221f59f..aa433467213931 100644 --- a/cli/module_loader.rs +++ b/cli/module_loader.rs @@ -571,13 +571,13 @@ impl source, }) => { let js_source = match media_type { - MediaType::Cts => { + MediaType::Cts => Cow::Owned( self .emitter .emit_parsed_source(specifier, MediaType::Cts, source) - .await? - } - MediaType::Cjs => source.to_string(), + .await?, + ), + MediaType::Cjs => Cow::Borrowed(source.as_ref()), _ => unreachable!(), }; let text = self @@ -588,7 +588,7 @@ impl self.parsed_source_cache.free(specifier); Ok(Some(ModuleCodeStringSource { - code: ModuleSourceCode::String(text.into()), + code: ModuleSourceCode::String(text.into_owned().into()), found_url: specifier.clone(), media_type, })) diff --git a/cli/node.rs b/cli/node.rs index a9e9191f1d4c23..498d1c10671278 100644 --- a/cli/node.rs +++ b/cli/node.rs @@ -1,5 +1,6 @@ // Copyright 2018-2024 the Deno authors. All rights reserved. MIT license. +use std::borrow::Cow; use std::sync::Arc; use deno_ast::MediaType; @@ -156,11 +157,11 @@ impl CliCjsCodeAnalyzer { #[async_trait::async_trait(?Send)] impl CjsCodeAnalyzer for CliCjsCodeAnalyzer { - async fn analyze_cjs( + async fn analyze_cjs<'a>( &self, specifier: &ModuleSpecifier, - source: Option, - ) -> Result { + source: Option>, + ) -> Result, AnyError> { let source = match source { Some(source) => source, None => { @@ -168,7 +169,7 @@ impl CjsCodeAnalyzer for CliCjsCodeAnalyzer { if let Ok(source_from_file) = self.fs.read_text_file_lossy_async(path, None).await { - source_from_file + Cow::Owned(source_from_file) } else { return Ok(ExtNodeCjsAnalysis::Cjs(CjsAnalysisExports { exports: vec![], diff --git a/cli/resolver.rs b/cli/resolver.rs index 606c2c14669ef7..ac0c21789b1702 100644 --- a/cli/resolver.rs +++ b/cli/resolver.rs @@ -43,6 +43,7 @@ use node_resolver::NodeModuleKind; use node_resolver::NodeResolution; use node_resolver::NodeResolutionMode; use node_resolver::PackageJson; +use std::borrow::Cow; use std::path::Path; use std::path::PathBuf; use std::sync::Arc; @@ -418,8 +419,9 @@ impl NpmModuleLoader { ModuleSourceCode::String( self .node_code_translator - .translate_cjs_to_esm(specifier, Some(code)) + .translate_cjs_to_esm(specifier, Some(Cow::Owned(code))) .await? + .into_owned() .into(), ) } else { diff --git a/cli/standalone/mod.rs b/cli/standalone/mod.rs index 8ab173b3e80b2c..a98b455a99dadb 100644 --- a/cli/standalone/mod.rs +++ b/cli/standalone/mod.rs @@ -19,6 +19,7 @@ use deno_core::error::type_error; use deno_core::error::AnyError; use deno_core::futures::FutureExt; use deno_core::v8_set_flags; +use deno_core::FastString; use deno_core::FeatureChecker; use deno_core::ModuleLoader; use deno_core::ModuleSourceCode; @@ -44,6 +45,7 @@ use deno_semver::npm::NpmPackageReqReference; use import_map::parse_from_json; use node_resolver::analyze::NodeCodeTranslator; use node_resolver::NodeResolutionMode; +use serialization::DenoCompileModuleSource; use std::borrow::Cow; use std::rc::Rc; use std::sync::Arc; @@ -62,6 +64,7 @@ use crate::cache::NodeAnalysisCache; use crate::cache::RealDenoCacheEnv; use crate::http_util::HttpClientProvider; use crate::node::CliCjsCodeAnalyzer; +use crate::node::CliNodeCodeTranslator; use crate::npm::create_cli_npm_resolver; use crate::npm::CliByonmNpmResolverCreateOptions; use crate::npm::CliNodeRequireLoader; @@ -94,8 +97,10 @@ use self::binary::Metadata; use self::file_system::DenoCompileFileSystem; struct SharedModuleLoaderState { + cjs_resolutions: Arc, fs: Arc, modules: StandaloneModules, + node_code_translator: Arc, node_resolver: Arc, npm_module_loader: Arc, npm_resolver: Arc, @@ -329,16 +334,58 @@ impl ModuleLoader for EmbeddedModuleLoader { match self.shared.modules.read(original_specifier) { Ok(Some(module)) => { let (module_specifier, module_type, module_source) = - module.into_for_v8(); - deno_core::ModuleLoadResponse::Sync(Ok( - deno_core::ModuleSource::new_with_redirect( - module_type, - module_source, - original_specifier, - module_specifier, - None, - ), - )) + module.into_parts(); + if self.shared.cjs_resolutions.is_known_cjs(original_specifier) { + let original_specifier = original_specifier.clone(); + let module_specifier = module_specifier.clone(); + let shared = self.shared.clone(); + deno_core::ModuleLoadResponse::Async( + async move { + let source = match module_source { + DenoCompileModuleSource::String(string) => { + Cow::Borrowed(string) + } + DenoCompileModuleSource::Bytes(module_code_bytes) => { + match module_code_bytes { + Cow::Owned(bytes) => Cow::Owned( + crate::util::text_encoding::from_utf8_lossy_owned(bytes), + ), + Cow::Borrowed(bytes) => String::from_utf8_lossy(bytes), + } + } + }; + let source = shared + .node_code_translator + .translate_cjs_to_esm(&module_specifier, Some(source)) + .await?; + let module_source = match source { + Cow::Owned(source) => ModuleSourceCode::String(source.into()), + Cow::Borrowed(source) => { + ModuleSourceCode::String(FastString::from_static(source)) + } + }; + Ok(deno_core::ModuleSource::new_with_redirect( + module_type, + module_source, + &original_specifier, + &module_specifier, + None, + )) + } + .boxed_local(), + ) + } else { + let module_source = module_source.into_for_v8(); + deno_core::ModuleLoadResponse::Sync(Ok( + deno_core::ModuleSource::new_with_redirect( + module_type, + module_source, + original_specifier, + module_specifier, + None, + ), + )) + } } Ok(None) => deno_core::ModuleLoadResponse::Sync(Err(type_error( format!("{MODULE_NOT_FOUND}: {}", original_specifier), @@ -597,8 +644,10 @@ pub async fn run(data: StandaloneData) -> Result { }; let module_loader_factory = StandaloneModuleLoaderFactory { shared: Arc::new(SharedModuleLoaderState { + cjs_resolutions: cjs_resolutions.clone(), fs: fs.clone(), modules, + node_code_translator: node_code_translator.clone(), node_resolver: cli_node_resolver.clone(), npm_module_loader: Arc::new(NpmModuleLoader::new( cjs_resolutions.clone(), diff --git a/cli/standalone/serialization.rs b/cli/standalone/serialization.rs index 7b63c584e72790..2f2b4af9352150 100644 --- a/cli/standalone/serialization.rs +++ b/cli/standalone/serialization.rs @@ -214,14 +214,13 @@ impl RemoteModulesStoreBuilder { } } -pub struct DenoCompileModuleData<'a> { - pub specifier: &'a Url, - pub media_type: MediaType, - pub data: Cow<'static, [u8]>, +pub enum DenoCompileModuleSource { + String(&'static str), + Bytes(Cow<'static, [u8]>), } -impl<'a> DenoCompileModuleData<'a> { - pub fn into_for_v8(self) -> (&'a Url, ModuleType, ModuleSourceCode) { +impl DenoCompileModuleSource { + pub fn into_for_v8(self) -> ModuleSourceCode { fn into_bytes(data: Cow<'static, [u8]>) -> ModuleSourceCode { ModuleSourceCode::Bytes(match data { Cow::Borrowed(d) => d.into(), @@ -229,16 +228,31 @@ impl<'a> DenoCompileModuleData<'a> { }) } - fn into_string_unsafe(data: Cow<'static, [u8]>) -> ModuleSourceCode { + match self { // todo(https://github.com/denoland/deno_core/pull/943): store whether // the string is ascii or not ahead of time so we can avoid the is_ascii() // check in FastString::from_static + Self::String(s) => ModuleSourceCode::String(FastString::from_static(s)), + Self::Bytes(b) => into_bytes(b), + } + } +} + +pub struct DenoCompileModuleData<'a> { + pub specifier: &'a Url, + pub media_type: MediaType, + pub data: Cow<'static, [u8]>, +} + +impl<'a> DenoCompileModuleData<'a> { + pub fn into_parts(self) -> (&'a Url, ModuleType, DenoCompileModuleSource) { + fn into_string_unsafe(data: Cow<'static, [u8]>) -> DenoCompileModuleSource { match data { - Cow::Borrowed(d) => ModuleSourceCode::String( + Cow::Borrowed(d) => DenoCompileModuleSource::String( // SAFETY: we know this is a valid utf8 string - unsafe { FastString::from_static(std::str::from_utf8_unchecked(d)) }, + unsafe { std::str::from_utf8_unchecked(d) }, ), - Cow::Owned(d) => ModuleSourceCode::Bytes(d.into_boxed_slice().into()), + Cow::Owned(d) => DenoCompileModuleSource::Bytes(Cow::Owned(d)), } } @@ -257,11 +271,14 @@ impl<'a> DenoCompileModuleData<'a> { (ModuleType::JavaScript, into_string_unsafe(self.data)) } MediaType::Json => (ModuleType::Json, into_string_unsafe(self.data)), - MediaType::Wasm => (ModuleType::Wasm, into_bytes(self.data)), - // just assume javascript if we made it here - MediaType::TsBuildInfo | MediaType::SourceMap | MediaType::Unknown => { - (ModuleType::JavaScript, into_bytes(self.data)) + MediaType::Wasm => { + (ModuleType::Wasm, DenoCompileModuleSource::Bytes(self.data)) } + // just assume javascript if we made it here + MediaType::TsBuildInfo | MediaType::SourceMap | MediaType::Unknown => ( + ModuleType::JavaScript, + DenoCompileModuleSource::Bytes(self.data), + ), }; (self.specifier, media_type, source) } diff --git a/cli/util/logger.rs b/cli/util/logger.rs index cdc89411fe1b77..d93753dfd30391 100644 --- a/cli/util/logger.rs +++ b/cli/util/logger.rs @@ -65,6 +65,8 @@ pub fn init(maybe_level: Option) { .filter_module("swc_ecma_parser", log::LevelFilter::Error) // Suppress span lifecycle logs since they are too verbose .filter_module("tracing::span", log::LevelFilter::Off) + // for deno_compile, this is too verbose + .filter_module("editpe", log::LevelFilter::Error) .format(|buf, record| { let mut target = record.target().to_string(); if let Some(line_no) = record.line() { diff --git a/resolvers/node/analyze.rs b/resolvers/node/analyze.rs index 009296006afc8b..7b14b5b49bf554 100644 --- a/resolvers/node/analyze.rs +++ b/resolvers/node/analyze.rs @@ -27,10 +27,10 @@ use crate::NpmResolverRc; use crate::PathClean; #[derive(Debug, Clone)] -pub enum CjsAnalysis { +pub enum CjsAnalysis<'a> { /// File was found to be an ES module and the translator should /// load the code as ESM. - Esm(String), + Esm(Cow<'a, str>), Cjs(CjsAnalysisExports), } @@ -50,11 +50,11 @@ pub trait CjsCodeAnalyzer { /// already has it. If the source is needed by the implementation, /// then it can use the provided source, or otherwise load it if /// necessary. - async fn analyze_cjs( + async fn analyze_cjs<'a>( &self, specifier: &Url, - maybe_source: Option, - ) -> Result; + maybe_source: Option>, + ) -> Result, AnyError>; } pub struct NodeCodeTranslator< @@ -90,11 +90,11 @@ impl /// For all discovered reexports the analysis will be performed recursively. /// /// If successful a source code for equivalent ES module is returned. - pub async fn translate_cjs_to_esm( + pub async fn translate_cjs_to_esm<'a>( &self, entry_specifier: &Url, - source: Option, - ) -> Result { + source: Option>, + ) -> Result, AnyError> { let mut temp_var_count = 0; let analysis = self @@ -159,7 +159,7 @@ impl source.push("export default mod;".to_string()); let translated_source = source.join("\n"); - Ok(translated_source) + Ok(Cow::Owned(translated_source)) } async fn analyze_reexports<'a>( @@ -174,7 +174,7 @@ impl struct Analysis { reexport_specifier: url::Url, referrer: url::Url, - analysis: CjsAnalysis, + analysis: CjsAnalysis<'static>, } type AnalysisFuture<'a> = LocalBoxFuture<'a, Result>; From 39e928f6ef75c37690ed8affdd59cce3f22a7743 Mon Sep 17 00:00:00 2001 From: David Sherret Date: Mon, 28 Oct 2024 11:47:35 -0400 Subject: [PATCH 13/42] working test --- cli/factory.rs | 1 - cli/npm/byonm.rs | 1 - tests/specs/compile/cjs/__test__.jsonc | 4 ++-- tests/specs/compile/cjs/main.js | 4 +++- tests/specs/compile/cjs/multiply.cts | 4 ++++ tests/specs/compile/cjs/output.out | 1 + tests/specs/compile/cjs/{file.cjs => reexport.cjs} | 0 7 files changed, 10 insertions(+), 5 deletions(-) create mode 100644 tests/specs/compile/cjs/multiply.cts rename tests/specs/compile/cjs/{file.cjs => reexport.cjs} (100%) diff --git a/cli/factory.rs b/cli/factory.rs index 659c297fa3f67d..f1b8325506966f 100644 --- a/cli/factory.rs +++ b/cli/factory.rs @@ -34,7 +34,6 @@ use crate::node::CliCjsCodeAnalyzer; use crate::node::CliNodeCodeTranslator; use crate::npm::create_cli_npm_resolver; use crate::npm::CliByonmNpmResolverCreateOptions; -use crate::npm::CliNodeRequireLoader; use crate::npm::CliNpmResolver; use crate::npm::CliNpmResolverCreateOptions; use crate::npm::CliNpmResolverManagedCreateOptions; diff --git a/cli/npm/byonm.rs b/cli/npm/byonm.rs index 0fde6f9c5dc917..befa83c9967d5d 100644 --- a/cli/npm/byonm.rs +++ b/cli/npm/byonm.rs @@ -11,7 +11,6 @@ use deno_core::url::Url; use deno_resolver::npm::ByonmNpmResolver; use deno_resolver::npm::ByonmNpmResolverCreateOptions; use deno_runtime::deno_node::NodePermissions; -use deno_runtime::deno_node::NodeRequireLoader; use deno_runtime::ops::process::NpmProcessStateProvider; use deno_semver::package::PackageReq; use node_resolver::NpmResolver; diff --git a/tests/specs/compile/cjs/__test__.jsonc b/tests/specs/compile/cjs/__test__.jsonc index eda3a23d4fb1aa..00bb55e8db7a54 100644 --- a/tests/specs/compile/cjs/__test__.jsonc +++ b/tests/specs/compile/cjs/__test__.jsonc @@ -9,7 +9,7 @@ "commandName": "./main", "args": [], "output": "output.out", - "exitCode": 1 + "exitCode": 0 }, { "if": "windows", "args": "compile -A --output main.exe main.js", @@ -19,6 +19,6 @@ "commandName": "./main.exe", "args": [], "output": "output.out", - "exitCode": 1 + "exitCode": 0 }] } diff --git a/tests/specs/compile/cjs/main.js b/tests/specs/compile/cjs/main.js index ac78584fed0efa..c2a8c7be02ce2f 100644 --- a/tests/specs/compile/cjs/main.js +++ b/tests/specs/compile/cjs/main.js @@ -1,3 +1,5 @@ -import { add } from "./file.cjs"; +import { add } from "./reexport.cjs"; +import { multiply } from "./multiply.cts"; console.log(add(1, 2)); +console.log(multiply(2, 3)); diff --git a/tests/specs/compile/cjs/multiply.cts b/tests/specs/compile/cjs/multiply.cts new file mode 100644 index 00000000000000..1fed324c790130 --- /dev/null +++ b/tests/specs/compile/cjs/multiply.cts @@ -0,0 +1,4 @@ +/// +exports.multiply = function (a: number, b: number): number { + return a * b; +}; diff --git a/tests/specs/compile/cjs/output.out b/tests/specs/compile/cjs/output.out index 00750edc07d641..2559e5c49e72f9 100644 --- a/tests/specs/compile/cjs/output.out +++ b/tests/specs/compile/cjs/output.out @@ -1 +1,2 @@ 3 +6 diff --git a/tests/specs/compile/cjs/file.cjs b/tests/specs/compile/cjs/reexport.cjs similarity index 100% rename from tests/specs/compile/cjs/file.cjs rename to tests/specs/compile/cjs/reexport.cjs From bd824a644526bbe91ffff3dad3abd182a6a5efaa Mon Sep 17 00:00:00 2001 From: David Sherret Date: Mon, 28 Oct 2024 20:35:49 -0400 Subject: [PATCH 14/42] updates to pass around the module kind --- .dprint.json | 2 +- Cargo.lock | 26 ++-- Cargo.toml | 8 +- cli/Cargo.toml | 8 +- cli/cache/mod.rs | 82 ++---------- cli/cache/module_info.rs | 2 +- cli/cache/parsed_source.rs | 58 ++------- cli/emit.rs | 65 ++++++---- cli/factory.rs | 46 ++++--- cli/graph_util.rs | 13 +- cli/lsp/config.rs | 2 +- cli/lsp/language_server.rs | 2 +- cli/lsp/resolver.rs | 19 +-- cli/module_loader.rs | 122 +++++++++++------- cli/npm/mod.rs | 12 +- cli/resolver.rs | 95 +++++++++++--- cli/standalone/binary.rs | 29 +++-- cli/standalone/mod.rs | 31 +++-- cli/standalone/serialization.rs | 6 +- cli/tools/check.rs | 6 +- cli/tools/coverage/mod.rs | 13 +- cli/tools/doc.rs | 4 +- cli/tools/registry/tar.rs | 2 +- cli/tools/repl/session.rs | 5 +- cli/tools/run/hmr.rs | 28 ++-- cli/tsc/diagnostics.rs | 2 +- cli/tsc/mod.rs | 2 +- cli/util/text_encoding.rs | 1 + cli/worker.rs | 33 +---- resolvers/deno/sloppy_imports.rs | 2 +- runtime/shared.rs | 9 +- .../css_import/exists_run_with_check.out | 2 +- tests/specs/run/cjs_unprepared/__test__.jsonc | 4 + tests/specs/run/cjs_unprepared/file.cjs | 7 + tests/specs/run/cjs_unprepared/main.out | 1 + tests/specs/run/cjs_unprepared/main.ts | 7 + tests/specs/run/cjs_unprepared/output.cjs | 3 + 37 files changed, 408 insertions(+), 351 deletions(-) create mode 100644 tests/specs/run/cjs_unprepared/__test__.jsonc create mode 100644 tests/specs/run/cjs_unprepared/file.cjs create mode 100644 tests/specs/run/cjs_unprepared/main.out create mode 100644 tests/specs/run/cjs_unprepared/main.ts create mode 100644 tests/specs/run/cjs_unprepared/output.cjs diff --git a/.dprint.json b/.dprint.json index 7877f17647dc55..ea99f2aedafc05 100644 --- a/.dprint.json +++ b/.dprint.json @@ -68,7 +68,7 @@ "third_party" ], "plugins": [ - "https://plugins.dprint.dev/typescript-0.93.0.wasm", + "https://plugins.dprint.dev/typescript-0.93.1.wasm", "https://plugins.dprint.dev/json-0.19.4.wasm", "https://plugins.dprint.dev/markdown-0.17.8.wasm", "https://plugins.dprint.dev/toml-0.6.3.wasm", diff --git a/Cargo.lock b/Cargo.lock index 7e7e2fc404e6a8..a101afc404d401 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -1279,7 +1279,7 @@ dependencies = [ [[package]] name = "deno_ast" -version = "0.42.2" +version = "0.43.0" dependencies = [ "base64 0.21.7", "deno_media_type", @@ -1354,9 +1354,9 @@ dependencies = [ [[package]] name = "deno_cache_dir" -version = "0.13.1" +version = "0.13.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "693ca429aebf945de5fef30df232044f9f80be4cc5a5e7c8d767226c43880f5a" +checksum = "08c1f52170cd7715f8006da54cde1444863a0d6fbd9c11d037a737db2dec8e22" dependencies = [ "base32", "deno_media_type", @@ -1504,7 +1504,9 @@ dependencies = [ [[package]] name = "deno_doc" -version = "0.154.0" +version = "0.156.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2585b98d6ad76dae30bf2d7b6d71b8363cae041158b8780d14a2f4fe17590a61" dependencies = [ "anyhow", "cfg-if", @@ -1602,7 +1604,9 @@ dependencies = [ [[package]] name = "deno_graph" -version = "0.83.4" +version = "0.84.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c9d6c7edec696810563d8002e4330e87b63ecbdd48996a18a94946b9a0f4db50" dependencies = [ "anyhow", "async-trait", @@ -1720,9 +1724,9 @@ dependencies = [ [[package]] name = "deno_lint" -version = "0.67.0" +version = "0.68.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "871b60e32bfb6c110cbb9b0688dbf048f81e5d347fe4ce5a42239263de9dd938" +checksum = "bb994e6d1b18223df0a756c7948143b35682941d615edffef60d5b38822f38ac" dependencies = [ "anyhow", "deno_ast", @@ -1750,9 +1754,9 @@ dependencies = [ [[package]] name = "deno_media_type" -version = "0.1.4" +version = "0.2.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "a8978229b82552bf8457a0125aa20863f023619cfc21ebb007b1e571d68fd85b" +checksum = "7fcf552fbdedbe81c89705349d7d2485c7051382b000dfddbdbf7fc25931cf83" dependencies = [ "data-url", "serde", @@ -2602,9 +2606,9 @@ dependencies = [ [[package]] name = "dprint-plugin-typescript" -version = "0.93.0" +version = "0.93.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "e9308d98b923b7c0335c2ee1560199e3f2321b1be82803107b4ba4ed5dac46cc" +checksum = "5abfd78fe3cde4f5a6699d65f760c8d44da130cf446b6f80a7a9bc6580e156ab" dependencies = [ "anyhow", "deno_ast", diff --git a/Cargo.toml b/Cargo.toml index 7932808b0bb7a8..f84663bb8cf373 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -45,12 +45,12 @@ license = "MIT" repository = "https://github.com/denoland/deno" [workspace.dependencies] -deno_ast = { version = "=0.42.2", features = ["transpiling"] } +deno_ast = { version = "=0.43.0", features = ["transpiling"] } deno_core = { version = "0.314.2" } deno_bench_util = { version = "0.168.0", path = "./bench_util" } deno_lockfile = "=0.23.1" -deno_media_type = { version = "0.1.4", features = ["module_specifier"] } +deno_media_type = { version = "0.2.0", features = ["module_specifier"] } deno_npm = "=0.25.4" deno_path_util = "=0.2.1" deno_permissions = { version = "0.34.0", path = "./runtime/permissions" } @@ -111,7 +111,7 @@ console_static_text = "=0.8.1" dashmap = "5.5.3" data-encoding = "2.3.3" data-url = "=0.3.0" -deno_cache_dir = "=0.13.1" +deno_cache_dir = "=0.13.2" deno_package_json = { version = "0.1.2", default-features = false } dlopen2 = "0.6.1" ecb = "=0.1.2" @@ -328,6 +328,4 @@ opt-level = 3 opt-level = 3 [patch.crates-io] -deno_graph = { path = "../deno_graph" } deno_ast = { path = "../deno_ast" } -deno_doc = { path = "../deno_doc" } diff --git a/cli/Cargo.toml b/cli/Cargo.toml index 0065a2cbd0757e..792b372856ce29 100644 --- a/cli/Cargo.toml +++ b/cli/Cargo.toml @@ -72,9 +72,9 @@ deno_ast = { workspace = true, features = ["bundler", "cjs", "codegen", "proposa deno_cache_dir = { workspace = true } deno_config = { version = "=0.37.2", features = ["workspace", "sync"] } deno_core = { workspace = true, features = ["include_js_files_for_snapshotting"] } -deno_doc = { version = "0.154.0", default-features = false, features = ["rust", "html", "syntect"] } -deno_graph = { version = "=0.83.4" } -deno_lint = { version = "=0.67.0", features = ["docs"] } +deno_doc = { version = "0.156.0", default-features = false, features = ["rust", "html", "syntect"] } +deno_graph = { version = "=0.84.0" } +deno_lint = { version = "=0.68.0", features = ["docs"] } deno_lockfile.workspace = true deno_npm.workspace = true deno_package_json.workspace = true @@ -107,7 +107,7 @@ dotenvy = "0.15.7" dprint-plugin-json = "=0.19.4" dprint-plugin-jupyter = "=0.1.5" dprint-plugin-markdown = "=0.17.8" -dprint-plugin-typescript = "=0.93.0" +dprint-plugin-typescript = "=0.93.1" env_logger = "=0.10.0" fancy-regex = "=0.10.0" faster-hex.workspace = true diff --git a/cli/cache/mod.rs b/cli/cache/mod.rs index e150bbef543b07..5dc342509df7fb 100644 --- a/cli/cache/mod.rs +++ b/cli/cache/mod.rs @@ -9,13 +9,9 @@ use crate::file_fetcher::FetchPermissionsOptionRef; use crate::file_fetcher::FileFetcher; use crate::file_fetcher::FileOrRedirect; use crate::npm::CliNpmResolver; -use crate::resolver::CliNodeResolver; use crate::util::fs::atomic_write_file_with_retries; use crate::util::fs::atomic_write_file_with_retries_and_fs; use crate::util::fs::AtomicWriteFileFsAdapter; -use crate::util::path::specifier_has_extension; -use crate::util::text_encoding::arc_str_to_bytes; -use crate::util::text_encoding::from_utf8_lossy_owned; use deno_ast::MediaType; use deno_core::futures; @@ -60,7 +56,6 @@ pub use fast_check::FastCheckCache; pub use incremental::IncrementalCache; pub use module_info::ModuleInfoCache; pub use node::NodeAnalysisCache; -pub use parsed_source::EsmOrCjsChecker; pub use parsed_source::LazyGraphSourceParser; pub use parsed_source::ParsedSourceCache; @@ -181,46 +176,37 @@ pub struct FetchCacherOptions { pub permissions: PermissionsContainer, /// If we're publishing for `deno publish`. pub is_deno_publish: bool, - pub unstable_detect_cjs: bool, } /// A "wrapper" for the FileFetcher and DiskCache for the Deno CLI that provides /// a concise interface to the DENO_DIR when building module graphs. pub struct FetchCacher { pub file_header_overrides: HashMap>, - esm_or_cjs_checker: Arc, file_fetcher: Arc, global_http_cache: Arc, - node_resolver: Arc, npm_resolver: Arc, module_info_cache: Arc, permissions: PermissionsContainer, is_deno_publish: bool, - unstable_detect_cjs: bool, cache_info_enabled: bool, } impl FetchCacher { pub fn new( - esm_or_cjs_checker: Arc, file_fetcher: Arc, global_http_cache: Arc, - node_resolver: Arc, npm_resolver: Arc, module_info_cache: Arc, options: FetchCacherOptions, ) -> Self { Self { file_fetcher, - esm_or_cjs_checker, global_http_cache, - node_resolver, npm_resolver, module_info_cache, file_header_overrides: options.file_header_overrides, permissions: options.permissions, is_deno_publish: options.is_deno_publish, - unstable_detect_cjs: options.unstable_detect_cjs, cache_info_enabled: false, } } @@ -271,60 +257,20 @@ impl Loader for FetchCacher { ) -> LoadFuture { use deno_graph::source::CacheSetting as LoaderCacheSetting; - if specifier.scheme() == "file" { - if specifier.path().contains("/node_modules/") { - // The specifier might be in a completely different symlinked tree than - // what the node_modules url is in (ex. `/my-project-1/node_modules` - // symlinked to `/my-project-2/node_modules`), so first we checked if the path - // is in a node_modules dir to avoid needlessly canonicalizing, then now compare - // against the canonicalized specifier. - let specifier = - crate::node::resolve_specifier_into_node_modules(specifier); - if self.npm_resolver.in_npm_package(&specifier) { - return Box::pin(futures::future::ready(Ok(Some( - LoadResponse::External { specifier }, - )))); - } - } - - if self.unstable_detect_cjs && specifier_has_extension(specifier, "js") { - if let Ok(Some(pkg_json)) = - self.node_resolver.get_closest_package_json(specifier) - { - if pkg_json.typ == "commonjs" { - if let Ok(path) = specifier.to_file_path() { - if let Ok(bytes) = std::fs::read(&path) { - let text: Arc = from_utf8_lossy_owned(bytes).into(); - let is_es_module = match self.esm_or_cjs_checker.is_esm( - specifier, - text.clone(), - MediaType::JavaScript, - ) { - Ok(value) => value, - Err(err) => { - return Box::pin(futures::future::ready(Err(err.into()))); - } - }; - if !is_es_module { - self.node_resolver.mark_cjs_resolution(specifier.clone()); - return Box::pin(futures::future::ready(Ok(Some( - LoadResponse::External { - specifier: specifier.clone(), - }, - )))); - } else { - return Box::pin(futures::future::ready(Ok(Some( - LoadResponse::Module { - specifier: specifier.clone(), - content: arc_str_to_bytes(text), - maybe_headers: None, - }, - )))); - } - } - } - } - } + if specifier.scheme() == "file" + && specifier.path().contains("/node_modules/") + { + // The specifier might be in a completely different symlinked tree than + // what the node_modules url is in (ex. `/my-project-1/node_modules` + // symlinked to `/my-project-2/node_modules`), so first we checked if the path + // is in a node_modules dir to avoid needlessly canonicalizing, then now compare + // against the canonicalized specifier. + let specifier = + crate::node::resolve_specifier_into_node_modules(specifier); + if self.npm_resolver.in_npm_package(&specifier) { + return Box::pin(futures::future::ready(Ok(Some( + LoadResponse::External { specifier }, + )))); } } diff --git a/cli/cache/module_info.rs b/cli/cache/module_info.rs index 4dbb01c37b1bf1..86e7ed5b1c9709 100644 --- a/cli/cache/module_info.rs +++ b/cli/cache/module_info.rs @@ -202,7 +202,7 @@ fn serialize_media_type(media_type: MediaType) -> i64 { Tsx => 11, Json => 12, Wasm => 13, - TsBuildInfo => 14, + Css => 14, SourceMap => 15, Unknown => 16, } diff --git a/cli/cache/parsed_source.rs b/cli/cache/parsed_source.rs index 3104a0df3e4863..7e819ae9985ba3 100644 --- a/cli/cache/parsed_source.rs +++ b/cli/cache/parsed_source.rs @@ -5,12 +5,11 @@ use std::sync::Arc; use deno_ast::MediaType; use deno_ast::ModuleSpecifier; -use deno_ast::ParseDiagnostic; use deno_ast::ParsedSource; use deno_core::parking_lot::Mutex; -use deno_graph::CapturingModuleParser; -use deno_graph::DefaultModuleParser; -use deno_graph::ModuleParser; +use deno_graph::CapturingEsParser; +use deno_graph::DefaultEsParser; +use deno_graph::EsParser; use deno_graph::ParseOptions; use deno_graph::ParsedSourceStore; @@ -58,12 +57,11 @@ impl ParsedSourceCache { module: &deno_graph::JsModule, ) -> Result { let parser = self.as_capturing_parser(); - // this will conditionally parse because it's using a CapturingModuleParser - parser.parse_module(ParseOptions { + // this will conditionally parse because it's using a CapturingEsParser + parser.parse_program(ParseOptions { specifier: &module.specifier, source: module.source.clone(), media_type: module.media_type, - // don't bother enabling because this method is currently only used for vendoring scope_analysis: false, }) } @@ -87,10 +85,9 @@ impl ParsedSourceCache { specifier, source, media_type, - // don't bother enabling because this method is currently only used for emitting scope_analysis: false, }; - DefaultModuleParser.parse_module(options) + DefaultEsParser.parse_program(options) } /// Frees the parsed source from memory. @@ -100,8 +97,8 @@ impl ParsedSourceCache { /// Creates a parser that will reuse a ParsedSource from the store /// if it exists, or else parse. - pub fn as_capturing_parser(&self) -> CapturingModuleParser { - CapturingModuleParser::new(None, self) + pub fn as_capturing_parser(&self) -> CapturingEsParser { + CapturingEsParser::new(None, self) } } @@ -150,42 +147,3 @@ impl deno_graph::ParsedSourceStore for ParsedSourceCache { } } } - -pub struct EsmOrCjsChecker { - parsed_source_cache: Arc, -} - -impl EsmOrCjsChecker { - pub fn new(parsed_source_cache: Arc) -> Self { - Self { - parsed_source_cache, - } - } - - pub fn is_esm( - &self, - specifier: &ModuleSpecifier, - source: Arc, - media_type: MediaType, - ) -> Result { - // todo(dsherret): add a file cache here to avoid parsing with swc on each run - let source = match self.parsed_source_cache.get_parsed_source(specifier) { - Some(source) => source.clone(), - None => { - let source = deno_ast::parse_program(deno_ast::ParseParams { - specifier: specifier.clone(), - text: source, - media_type, - capture_tokens: true, // capture because it's used for cjs export analysis - scope_analysis: false, - maybe_syntax: None, - })?; - self - .parsed_source_cache - .set_parsed_source(specifier.clone(), source.clone()); - source - } - }; - Ok(source.is_module()) - } -} diff --git a/cli/emit.rs b/cli/emit.rs index 653d5afd08d3d0..c018588f244e4e 100644 --- a/cli/emit.rs +++ b/cli/emit.rs @@ -4,10 +4,12 @@ use crate::cache::EmitCache; use crate::cache::FastInsecureHasher; use crate::cache::ParsedSourceCache; +use deno_ast::ModuleKind; use deno_ast::SourceMapOption; use deno_ast::SourceRange; use deno_ast::SourceRanged; use deno_ast::SourceRangedForSpanned; +use deno_ast::TranspileModuleOptions; use deno_ast::TranspileResult; use deno_core::error::AnyError; use deno_core::futures::stream::FuturesUnordered; @@ -60,20 +62,13 @@ impl Emitter { continue; }; - // todo(https://github.com/denoland/deno_media_type/pull/12): use is_emittable() - let is_emittable = matches!( - module.media_type, - MediaType::TypeScript - | MediaType::Mts - | MediaType::Cts - | MediaType::Jsx - | MediaType::Tsx - ); - if is_emittable { + if module.media_type.is_emittable() { futures.push( self .emit_parsed_source( &module.specifier, + // todo(THIS PR): don't assume this is esm + ModuleKind::Esm, module.media_type, &module.source, ) @@ -93,21 +88,23 @@ impl Emitter { pub fn maybe_cached_emit( &self, specifier: &ModuleSpecifier, + module_kind: deno_ast::ModuleKind, source: &str, ) -> Option { - let source_hash = self.get_source_hash(source); + let source_hash = self.get_source_hash(module_kind, source); self.emit_cache.get_emit_code(specifier, source_hash) } pub async fn emit_parsed_source( &self, specifier: &ModuleSpecifier, + module_kind: deno_ast::ModuleKind, media_type: MediaType, source: &Arc, ) -> Result { // Note: keep this in sync with the sync version below let helper = EmitParsedSourceHelper(self); - match helper.pre_emit_parsed_source(specifier, source) { + match helper.pre_emit_parsed_source(specifier, module_kind, source) { PreEmitResult::Cached(emitted_text) => Ok(emitted_text), PreEmitResult::NotCached { source_hash } => { let parsed_source_cache = self.parsed_source_cache.clone(); @@ -120,8 +117,9 @@ impl Emitter { EmitParsedSourceHelper::transpile( &parsed_source_cache, &specifier, - source.clone(), + module_kind, media_type, + source.clone(), &transpile_and_emit_options.0, &transpile_and_emit_options.1, ) @@ -142,19 +140,21 @@ impl Emitter { pub fn emit_parsed_source_sync( &self, specifier: &ModuleSpecifier, + module_kind: deno_ast::ModuleKind, media_type: MediaType, source: &Arc, ) -> Result { // Note: keep this in sync with the async version above let helper = EmitParsedSourceHelper(self); - match helper.pre_emit_parsed_source(specifier, source) { + match helper.pre_emit_parsed_source(specifier, module_kind, source) { PreEmitResult::Cached(emitted_text) => Ok(emitted_text), PreEmitResult::NotCached { source_hash } => { let transpiled_source = EmitParsedSourceHelper::transpile( &self.parsed_source_cache, specifier, - source.clone(), + module_kind, media_type, + source.clone(), &self.transpile_and_emit_options.0, &self.transpile_and_emit_options.1, )?; @@ -172,6 +172,7 @@ impl Emitter { pub async fn load_and_emit_for_hmr( &self, specifier: &ModuleSpecifier, + module_kind: deno_ast::ModuleKind, ) -> Result { let media_type = MediaType::from_specifier(specifier); let source_code = tokio::fs::read_to_string( @@ -194,9 +195,14 @@ impl Emitter { let mut options = self.transpile_and_emit_options.1.clone(); options.source_map = SourceMapOption::None; let transpiled_source = parsed_source - .transpile(&self.transpile_and_emit_options.0, &options)? - .into_source() - .into_string()?; + .transpile( + &self.transpile_and_emit_options.0, + &deno_ast::TranspileModuleOptions { + module_kind: Some(module_kind), + }, + &options, + )? + .into_source(); Ok(transpiled_source.text) } MediaType::JavaScript @@ -207,7 +213,7 @@ impl Emitter { | MediaType::Dcts | MediaType::Json | MediaType::Wasm - | MediaType::TsBuildInfo + | MediaType::Css | MediaType::SourceMap | MediaType::Unknown => { // clear this specifier from the parsed source cache as it's now out of date @@ -220,10 +226,11 @@ impl Emitter { /// A hashing function that takes the source code and uses the global emit /// options then generates a string hash which can be stored to /// determine if the cached emit is valid or not. - fn get_source_hash(&self, source_text: &str) -> u64 { + fn get_source_hash(&self, module_kind: ModuleKind, source_text: &str) -> u64 { FastInsecureHasher::new_without_deno_version() // stored in the transpile_and_emit_options_hash .write_str(source_text) .write_u64(self.transpile_and_emit_options_hash) + .write_hashable(module_kind) .finish() } } @@ -240,9 +247,10 @@ impl<'a> EmitParsedSourceHelper<'a> { pub fn pre_emit_parsed_source( &self, specifier: &ModuleSpecifier, + module_kind: deno_ast::ModuleKind, source: &Arc, ) -> PreEmitResult { - let source_hash = self.0.get_source_hash(source); + let source_hash = self.0.get_source_hash(module_kind, source); if let Some(emit_code) = self.0.emit_cache.get_emit_code(specifier, source_hash) @@ -256,8 +264,9 @@ impl<'a> EmitParsedSourceHelper<'a> { pub fn transpile( parsed_source_cache: &ParsedSourceCache, specifier: &ModuleSpecifier, - source: Arc, + module_kind: deno_ast::ModuleKind, media_type: MediaType, + source: Arc, transpile_options: &deno_ast::TranspileOptions, emit_options: &deno_ast::EmitOptions, ) -> Result { @@ -266,8 +275,13 @@ impl<'a> EmitParsedSourceHelper<'a> { let parsed_source = parsed_source_cache .remove_or_parse_module(specifier, source, media_type)?; ensure_no_import_assertion(&parsed_source)?; - let transpile_result = - parsed_source.transpile(transpile_options, emit_options)?; + let transpile_result = parsed_source.transpile( + transpile_options, + &TranspileModuleOptions { + module_kind: Some(module_kind), + }, + emit_options, + )?; let transpiled_source = match transpile_result { TranspileResult::Owned(source) => source, TranspileResult::Cloned(source) => { @@ -276,8 +290,7 @@ impl<'a> EmitParsedSourceHelper<'a> { } }; debug_assert!(transpiled_source.source_map.is_none()); - let text = String::from_utf8(transpiled_source.source)?; - Ok(text) + Ok(transpiled_source.text) } pub fn post_emit_parsed_source( diff --git a/cli/factory.rs b/cli/factory.rs index f1b8325506966f..c59e2941082539 100644 --- a/cli/factory.rs +++ b/cli/factory.rs @@ -14,7 +14,6 @@ use crate::cache::CodeCache; use crate::cache::DenoDir; use crate::cache::DenoDirProvider; use crate::cache::EmitCache; -use crate::cache::EsmOrCjsChecker; use crate::cache::GlobalHttpCache; use crate::cache::HttpCache; use crate::cache::LocalHttpCache; @@ -38,7 +37,8 @@ use crate::npm::CliNpmResolver; use crate::npm::CliNpmResolverCreateOptions; use crate::npm::CliNpmResolverManagedCreateOptions; use crate::npm::CliNpmResolverManagedSnapshotOption; -use crate::resolver::CjsResolutionStore; +use crate::resolver::CjsTracker; +use crate::resolver::CjsTrackerOptions; use crate::resolver::CliDenoResolverFs; use crate::resolver::CliGraphResolver; use crate::resolver::CliGraphResolverOptions; @@ -172,7 +172,6 @@ struct CliFactoryServices { http_client_provider: Deferred>, emit_cache: Deferred>, emitter: Deferred>, - esm_or_cjs_checker: Deferred>, fs: Deferred>, main_graph_container: Deferred>, maybe_inspector_server: Deferred>>, @@ -193,7 +192,7 @@ struct CliFactoryServices { sloppy_imports_resolver: Deferred>>, text_only_progress_bar: Deferred, type_checker: Deferred>, - cjs_resolutions: Deferred>, + cjs_tracker: Deferred>, cli_node_resolver: Deferred>, feature_checker: Deferred>, code_cache: Deferred>, @@ -300,12 +299,6 @@ impl CliFactory { .get_or_init(|| ProgressBar::new(ProgressBarStyle::TextOnly)) } - pub fn esm_or_cjs_checker(&self) -> &Arc { - self.services.esm_or_cjs_checker.get_or_init(|| { - Arc::new(EsmOrCjsChecker::new(self.parsed_source_cache().clone())) - }) - } - pub fn global_http_cache(&self) -> Result<&Arc, AnyError> { self.services.global_http_cache.get_or_try_init(|| { Ok(Arc::new(GlobalHttpCache::new( @@ -628,10 +621,8 @@ impl CliFactory { Ok(Arc::new(ModuleGraphBuilder::new( cli_options.clone(), self.caches()?.clone(), - self.esm_or_cjs_checker().clone(), self.fs().clone(), self.resolver().await?.clone(), - self.cli_node_resolver().await?.clone(), self.npm_resolver().await?.clone(), self.module_info_cache()?.clone(), self.parsed_source_cache().clone(), @@ -710,8 +701,20 @@ impl CliFactory { .await } - pub fn cjs_resolutions(&self) -> &Arc { - self.services.cjs_resolutions.get_or_init(Default::default) + pub async fn cjs_tracker(&self) -> Result<&Arc, AnyError> { + self + .services + .cjs_tracker + .get_or_try_init_async(async { + let options = self.cli_options()?; + Ok(Arc::new(CjsTracker::new( + CjsTrackerOptions { + unstable_detect_cjs: options.unstable_detect_cjs(), + }, + self.node_resolver().await?.clone(), + ))) + }) + .await } pub async fn cli_node_resolver( @@ -722,7 +725,7 @@ impl CliFactory { .cli_node_resolver .get_or_try_init_async(async { Ok(Arc::new(CliNodeResolver::new( - self.cjs_resolutions().clone(), + self.cjs_tracker().await?.clone(), self.fs().clone(), self.node_resolver().await?.clone(), self.npm_resolver().await?.clone(), @@ -761,6 +764,7 @@ impl CliFactory { ) -> Result { let cli_options = self.cli_options()?; Ok(DenoCompileBinaryWriter::new( + self.cjs_tracker().await?, self.deno_dir()?, self.emitter()?, self.file_fetcher()?, @@ -802,10 +806,11 @@ impl CliFactory { None }; let node_code_translator = self.node_code_translator().await?; + let cjs_tracker = self.cjs_tracker().await?.clone(); Ok(CliMainWorkerFactory::new( self.blob_store().clone(), - self.cjs_resolutions().clone(), + cjs_tracker.clone(), if cli_options.code_cache_enabled() { Some(self.code_cache()?.clone()) } else { @@ -818,6 +823,7 @@ impl CliFactory { cli_options.maybe_lockfile().cloned(), Box::new(CliModuleLoaderFactory::new( cli_options, + cjs_tracker, if cli_options.code_cache_enabled() { Some(self.code_cache()?.clone()) } else { @@ -831,7 +837,7 @@ impl CliFactory { cli_node_resolver.clone(), cli_npm_resolver.clone(), NpmModuleLoader::new( - self.cjs_resolutions().clone(), + self.cjs_tracker().await?.clone(), node_code_translator.clone(), fs.clone(), cli_node_resolver.clone(), @@ -845,19 +851,21 @@ impl CliFactory { self.root_permissions_container()?.clone(), StorageKeyResolver::from_options(cli_options), cli_options.sub_command().clone(), - self.create_cli_main_worker_options()?, + self.create_cli_main_worker_options().await?, )) } - fn create_cli_main_worker_options( + async fn create_cli_main_worker_options( &self, ) -> Result { let cli_options = self.cli_options()?; let create_hmr_runner = if cli_options.has_hmr() { let watcher_communicator = self.watcher_communicator.clone().unwrap(); let emitter = self.emitter()?.clone(); + let cjs_tracker = self.cjs_tracker().await?.clone(); let fn_: crate::worker::CreateHmrRunnerCb = Box::new(move |session| { Box::new(HmrRunner::new( + cjs_tracker.clone(), emitter.clone(), session, watcher_communicator.clone(), diff --git a/cli/graph_util.rs b/cli/graph_util.rs index 2eaee228afbd92..291372b0c7ba9e 100644 --- a/cli/graph_util.rs +++ b/cli/graph_util.rs @@ -6,7 +6,6 @@ use crate::args::CliLockfile; use crate::args::CliOptions; use crate::args::DENO_DISABLE_PEDANTIC_NODE_WARNINGS; use crate::cache; -use crate::cache::EsmOrCjsChecker; use crate::cache::GlobalHttpCache; use crate::cache::ModuleInfoCache; use crate::cache::ParsedSourceCache; @@ -15,7 +14,6 @@ use crate::errors::get_error_class_name; use crate::file_fetcher::FileFetcher; use crate::npm::CliNpmResolver; use crate::resolver::CliGraphResolver; -use crate::resolver::CliNodeResolver; use crate::resolver::CliSloppyImportsResolver; use crate::resolver::SloppyImportsCachedFs; use crate::tools::check; @@ -381,10 +379,8 @@ pub struct BuildFastCheckGraphOptions<'a> { pub struct ModuleGraphBuilder { options: Arc, caches: Arc, - esm_or_cjs_checker: Arc, fs: Arc, resolver: Arc, - node_resolver: Arc, npm_resolver: Arc, module_info_cache: Arc, parsed_source_cache: Arc, @@ -400,10 +396,8 @@ impl ModuleGraphBuilder { pub fn new( options: Arc, caches: Arc, - esm_or_cjs_checker: Arc, fs: Arc, resolver: Arc, - node_resolver: Arc, npm_resolver: Arc, module_info_cache: Arc, parsed_source_cache: Arc, @@ -416,10 +410,8 @@ impl ModuleGraphBuilder { Self { options, caches, - esm_or_cjs_checker, fs, resolver, - node_resolver, npm_resolver, module_info_cache, parsed_source_cache, @@ -680,7 +672,7 @@ impl ModuleGraphBuilder { jsr_url_provider: &CliJsrUrlProvider, fast_check_cache: fast_check_cache.as_ref().map(|c| c as _), fast_check_dts: false, - module_parser: Some(&parser), + es_parser: Some(&parser), resolver: Some(graph_resolver), npm_resolver: Some(&graph_npm_resolver), workspace_fast_check: options.workspace_fast_check, @@ -699,10 +691,8 @@ impl ModuleGraphBuilder { permissions: PermissionsContainer, ) -> cache::FetchCacher { cache::FetchCacher::new( - self.esm_or_cjs_checker.clone(), self.file_fetcher.clone(), self.global_http_cache.clone(), - self.node_resolver.clone(), self.npm_resolver.clone(), self.module_info_cache.clone(), cache::FetchCacherOptions { @@ -712,7 +702,6 @@ impl ModuleGraphBuilder { self.options.sub_command(), crate::args::DenoSubcommand::Publish { .. } ), - unstable_detect_cjs: self.options.unstable_detect_cjs(), }, ) } diff --git a/cli/lsp/config.rs b/cli/lsp/config.rs index 3ffc9e657ad19d..abe471d5da57ab 100644 --- a/cli/lsp/config.rs +++ b/cli/lsp/config.rs @@ -984,7 +984,7 @@ impl Config { | MediaType::Tsx => Some(&workspace_settings.typescript), MediaType::Json | MediaType::Wasm - | MediaType::TsBuildInfo + | MediaType::Css | MediaType::SourceMap | MediaType::Unknown => None, } diff --git a/cli/lsp/language_server.rs b/cli/lsp/language_server.rs index 33ae539f8586d1..6c1e6d4ed0f1f6 100644 --- a/cli/lsp/language_server.rs +++ b/cli/lsp/language_server.rs @@ -904,7 +904,7 @@ impl Inner { | MediaType::Tsx => {} MediaType::Wasm | MediaType::SourceMap - | MediaType::TsBuildInfo + | MediaType::Css | MediaType::Unknown => { if path.extension().and_then(|s| s.to_str()) != Some("jsonc") { continue; diff --git a/cli/lsp/resolver.rs b/cli/lsp/resolver.rs index c89273147afb8a..979708c27f4b8c 100644 --- a/cli/lsp/resolver.rs +++ b/cli/lsp/resolver.rs @@ -48,7 +48,8 @@ use crate::npm::CliNpmResolverCreateOptions; use crate::npm::CliNpmResolverManagedCreateOptions; use crate::npm::CliNpmResolverManagedSnapshotOption; use crate::npm::ManagedCliNpmResolver; -use crate::resolver::CjsResolutionStore; +use crate::resolver::CjsTracker; +use crate::resolver::CjsTrackerOptions; use crate::resolver::CliDenoResolverFs; use crate::resolver::CliGraphResolver; use crate::resolver::CliGraphResolverOptions; @@ -509,21 +510,21 @@ async fn create_npm_resolver( fn create_node_resolver( npm_resolver: Option<&Arc>, ) -> Option> { - use once_cell::sync::Lazy; - - // it's not ideal to share this across all scopes and to - // never clear it, but it's fine for the time being - static CJS_RESOLUTIONS: Lazy> = - Lazy::new(Default::default); - let npm_resolver = npm_resolver?; let fs = Arc::new(deno_fs::RealFs); let node_resolver_inner = Arc::new(NodeResolver::new( deno_runtime::deno_node::DenoFsNodeResolverEnv::new(fs.clone()), npm_resolver.clone().into_npm_resolver(), )); + let cjs_tracker = Arc::new(CjsTracker::new( + CjsTrackerOptions { + // todo(dsherret): support in the lsp + unstable_detect_cjs: false, + }, + node_resolver_inner.clone(), + )); Some(Arc::new(CliNodeResolver::new( - CJS_RESOLUTIONS.clone(), + cjs_tracker, fs, node_resolver_inner, npm_resolver.clone(), diff --git a/cli/module_loader.rs b/cli/module_loader.rs index aa433467213931..fddf5d926b67e2 100644 --- a/cli/module_loader.rs +++ b/cli/module_loader.rs @@ -26,6 +26,7 @@ use crate::node; use crate::node::CliNodeCodeTranslator; use crate::npm::CliNodeRequireLoader; use crate::npm::CliNpmResolver; +use crate::resolver::CjsTracker; use crate::resolver::CliGraphResolver; use crate::resolver::CliNodeResolver; use crate::resolver::ModuleCodeStringSource; @@ -38,6 +39,7 @@ use crate::util::text_encoding::source_map_from_code; use crate::worker::CreateModuleLoaderResult; use crate::worker::ModuleLoaderFactory; use deno_ast::MediaType; +use deno_ast::ModuleKind; use deno_core::anyhow::anyhow; use deno_core::anyhow::bail; use deno_core::anyhow::Context; @@ -202,6 +204,7 @@ struct SharedCliModuleLoaderState { initial_cwd: PathBuf, is_inspecting: bool, is_repl: bool, + cjs_tracker: Arc, code_cache: Option>, emitter: Arc, fs: Arc, @@ -223,6 +226,7 @@ impl CliModuleLoaderFactory { #[allow(clippy::too_many_arguments)] pub fn new( options: &CliOptions, + cjs_tracker: Arc, code_cache: Option>, emitter: Arc, fs: Arc, @@ -246,6 +250,7 @@ impl CliModuleLoaderFactory { options.sub_command(), DenoSubcommand::Repl(_) | DenoSubcommand::Jupyter(_) ), + cjs_tracker, code_cache, emitter, fs, @@ -349,24 +354,7 @@ impl maybe_referrer: Option<&ModuleSpecifier>, requested_module_type: RequestedModuleType, ) -> Result { - let code_source = match self.load_prepared_module(specifier).await? { - Some(code_source) => code_source, - None => { - if self.shared.npm_module_loader.if_in_npm_package(specifier) { - self - .shared - .npm_module_loader - .load(specifier, maybe_referrer) - .await? - } else { - let mut msg = format!("Loading unprepared module: {specifier}"); - if let Some(referrer) = maybe_referrer { - msg = format!("{}, imported from: {}", msg, referrer.as_str()); - } - return Err(anyhow!(msg)); - } - } - }; + let code_source = self.load_code_source(specifier, maybe_referrer).await?; let code = if self.shared.is_inspecting { // we need the code with the source map in order for // it to work with --inspect or --inspect-brk @@ -420,6 +408,29 @@ impl )) } + async fn load_code_source( + &self, + specifier: &ModuleSpecifier, + maybe_referrer: Option<&ModuleSpecifier>, + ) -> Result { + if let Some(code_source) = self.load_prepared_module(specifier).await? { + return Ok(code_source); + } + if self.shared.npm_module_loader.if_in_npm_package(specifier) { + return self + .shared + .npm_module_loader + .load(specifier, maybe_referrer) + .await; + } + + let mut msg = format!("Loading unprepared module: {specifier}"); + if let Some(referrer) = maybe_referrer { + msg = format!("{}, imported from: {}", msg, referrer.as_str()); + } + Err(anyhow!(msg)) + } + fn resolve_referrer( &self, referrer: &str, @@ -552,7 +563,7 @@ impl }) => { let transpile_result = self .emitter - .emit_parsed_source(specifier, media_type, source) + .emit_parsed_source(specifier, ModuleKind::Esm, media_type, source) .await?; // at this point, we no longer need the parsed source in memory, so free it @@ -569,30 +580,7 @@ impl specifier, media_type, source, - }) => { - let js_source = match media_type { - MediaType::Cts => Cow::Owned( - self - .emitter - .emit_parsed_source(specifier, MediaType::Cts, source) - .await?, - ), - MediaType::Cjs => Cow::Borrowed(source.as_ref()), - _ => unreachable!(), - }; - let text = self - .node_code_translator - .translate_cjs_to_esm(specifier, Some(js_source)) - .await?; - // at this point, we no longer need the parsed source in memory, so free it - self.parsed_source_cache.free(specifier); - - Ok(Some(ModuleCodeStringSource { - code: ModuleSourceCode::String(text.into_owned().into()), - found_url: specifier.clone(), - media_type, - })) - } + }) => self.load_cjs(specifier, media_type, source).await.map(Some), None => Ok(None), } } @@ -610,9 +598,12 @@ impl media_type, source, }) => { - let transpile_result = self - .emitter - .emit_parsed_source_sync(specifier, media_type, source)?; + let transpile_result = self.emitter.emit_parsed_source_sync( + specifier, + ModuleKind::Esm, + media_type, + source, + )?; // at this point, we no longer need the parsed source in memory, so free it self.parsed_source_cache.free(specifier); @@ -661,8 +652,16 @@ impl source, media_type, specifier, + is_script, .. })) => { + if *is_script && self.shared.cjs_tracker.treat_as_cjs(specifier)? { + return Ok(Some(CodeOrDeferredEmit::Cjs { + specifier, + media_type: *media_type, + source, + })); + } let code: ModuleCodeString = match media_type { MediaType::JavaScript | MediaType::Unknown @@ -688,7 +687,7 @@ impl source, })); } - MediaType::TsBuildInfo | MediaType::Wasm | MediaType::SourceMap => { + MediaType::Css | MediaType::Wasm | MediaType::SourceMap => { panic!("Unexpected media type {media_type} for {specifier}") } }; @@ -710,6 +709,35 @@ impl | None => Ok(None), } } + + async fn load_cjs( + &self, + specifier: &ModuleSpecifier, + media_type: MediaType, + source: &Arc, + ) -> Result { + let js_source = if media_type.is_emittable() { + Cow::Owned( + self + .emitter + .emit_parsed_source(specifier, ModuleKind::Cjs, media_type, source) + .await?, + ) + } else { + Cow::Borrowed(source.as_ref()) + }; + let text = self + .node_code_translator + .translate_cjs_to_esm(specifier, Some(js_source)) + .await?; + // at this point, we no longer need the parsed source in memory, so free it + self.parsed_source_cache.free(specifier); + Ok(ModuleCodeStringSource { + code: ModuleSourceCode::String(text.into_owned().into()), + found_url: specifier.clone(), + media_type, + }) + } } enum CodeOrDeferredEmit<'a> { diff --git a/cli/npm/mod.rs b/cli/npm/mod.rs index 08362698b6d2ba..5d449068145b64 100644 --- a/cli/npm/mod.rs +++ b/cli/npm/mod.rs @@ -12,6 +12,7 @@ use std::sync::Arc; use common::maybe_auth_header_for_npm_registry; use dashmap::DashMap; use deno_ast::MediaType; +use deno_ast::ModuleKind; use deno_ast::ModuleSpecifier; use deno_core::error::AnyError; use deno_core::serde_json; @@ -232,13 +233,16 @@ impl NodeRequireLoader for CliNodeRequireLoader { fn load_text_file_lossy(&self, path: &Path) -> Result { // todo(dsherret): use the preloaded module from the graph if available - let specifier = deno_path_util::url_from_file_path(&path)?; + let specifier = deno_path_util::url_from_file_path(path)?; let media_type = MediaType::from_specifier(&specifier); let text = self.fs.read_text_file_lossy_sync(path, None)?; if media_type == MediaType::Cts { - self - .emitter - .emit_parsed_source_sync(&specifier, media_type, &text.into()) + self.emitter.emit_parsed_source_sync( + &specifier, + ModuleKind::Cjs, + media_type, + &text.into(), + ) } else { Ok(text) } diff --git a/cli/resolver.rs b/cli/resolver.rs index ac0c21789b1702..09f0e300cbd406 100644 --- a/cli/resolver.rs +++ b/cli/resolver.rs @@ -105,7 +105,7 @@ impl deno_resolver::fs::DenoResolverFs for CliDenoResolverFs { #[derive(Debug)] pub struct CliNodeResolver { - cjs_resolutions: Arc, + cjs_tracker: Arc, fs: Arc, node_resolver: Arc, npm_resolver: Arc, @@ -113,13 +113,13 @@ pub struct CliNodeResolver { impl CliNodeResolver { pub fn new( - cjs_resolutions: Arc, + cjs_tracker: Arc, fs: Arc, node_resolver: Arc, npm_resolver: Arc, ) -> Self { Self { - cjs_resolutions, + cjs_tracker, fs, node_resolver, npm_resolver, @@ -217,7 +217,11 @@ impl CliNodeResolver { referrer: &ModuleSpecifier, mode: NodeResolutionMode, ) -> Result { - let referrer_kind = if self.cjs_resolutions.is_known_cjs(referrer) { + let referrer_kind = if self + .cjs_tracker + .treat_as_cjs(referrer) + .map_err(|err| NodeResolveErrorKind::PackageResolve(err.into()))? + { NodeModuleKind::Cjs } else { NodeModuleKind::Esm @@ -339,14 +343,14 @@ impl CliNodeResolver { } pub fn mark_cjs_resolution(&self, specifier: ModuleSpecifier) { - self.cjs_resolutions.insert(specifier); + self.cjs_tracker.insert(specifier); } } // todo(dsherret): move to module_loader.rs #[derive(Clone)] pub struct NpmModuleLoader { - cjs_resolutions: Arc, + cjs_tracker: Arc, node_code_translator: Arc, fs: Arc, node_resolver: Arc, @@ -354,13 +358,13 @@ pub struct NpmModuleLoader { impl NpmModuleLoader { pub fn new( - cjs_resolutions: Arc, + cjs_tracker: Arc, node_code_translator: Arc, fs: Arc, node_resolver: Arc, ) -> Self { Self { - cjs_resolutions, + cjs_tracker, node_code_translator, fs, node_resolver, @@ -413,7 +417,7 @@ impl NpmModuleLoader { } })?; - let code = if self.cjs_resolutions.is_known_cjs(specifier) { + let code = if self.cjs_tracker.treat_as_cjs(specifier)? { // translate cjs to esm if it's cjs and inject node globals let code = from_utf8_lossy_owned(code); ModuleSourceCode::String( @@ -436,23 +440,78 @@ impl NpmModuleLoader { } } +pub struct CjsTrackerOptions { + pub unstable_detect_cjs: bool, +} + /// Keeps track of what module specifiers were resolved as CJS. -#[derive(Debug, Default)] -pub struct CjsResolutionStore(DashSet); +#[derive(Debug)] +pub struct CjsTracker { + unstable_detect_cjs: bool, + set: DashSet, + node_resolver: Arc, +} + +impl CjsTracker { + pub fn new( + options: CjsTrackerOptions, + node_resolver: Arc, + ) -> Self { + Self { + unstable_detect_cjs: options.unstable_detect_cjs, + set: DashSet::new(), + node_resolver, + } + } -impl CjsResolutionStore { - pub fn is_known_cjs(&self, specifier: &ModuleSpecifier) -> bool { + pub fn treat_as_cjs( + &self, + specifier: &ModuleSpecifier, + ) -> Result { if specifier.scheme() != "file" { - return false; + return Ok(false); + } + + let media_type = MediaType::from_specifier(specifier); + if matches!(media_type, MediaType::Cjs | MediaType::Cts) + || self.set.contains(specifier) + { + return Ok(true); + } + + if self.unstable_detect_cjs { + let is_pkg_json_type_affected = match media_type { + MediaType::JavaScript + | MediaType::Jsx + | MediaType::TypeScript + | MediaType::Tsx + | MediaType::Dts => true, + MediaType::Dmts + | MediaType::Dcts + | MediaType::Css + | MediaType::Json + | MediaType::Wasm + | MediaType::Mjs + | MediaType::Cjs + | MediaType::Mts + | MediaType::Cts + | MediaType::SourceMap + | MediaType::Unknown => false, + }; + if is_pkg_json_type_affected { + if let Some(pkg_json) = + self.node_resolver.get_closest_package_json(specifier)? + { + return Ok(pkg_json.typ == "commonjs"); + } + } } - specifier_has_extension(specifier, "cjs") - || specifier_has_extension(specifier, "cts") - || self.0.contains(specifier) + Ok(false) } pub fn insert(&self, specifier: ModuleSpecifier) { - self.0.insert(specifier); + self.set.insert(specifier); } } diff --git a/cli/standalone/binary.rs b/cli/standalone/binary.rs index f41f3003ff1b3e..59d746ff9b9d2a 100644 --- a/cli/standalone/binary.rs +++ b/cli/standalone/binary.rs @@ -21,6 +21,7 @@ use std::process::Command; use std::sync::Arc; use deno_ast::MediaType; +use deno_ast::ModuleKind; use deno_ast::ModuleSpecifier; use deno_config::workspace::PackageJsonDepResolution; use deno_config::workspace::ResolverWorkspaceJsrPackage; @@ -67,6 +68,7 @@ use crate::file_fetcher::FileFetcher; use crate::http_util::HttpClientProvider; use crate::npm::CliNpmResolver; use crate::npm::InnerCliNpmResolverRef; +use crate::resolver::CjsTracker; use crate::shared::ReleaseChannel; use crate::standalone::virtual_fs::VfsEntry; use crate::util::archive; @@ -353,6 +355,7 @@ pub fn extract_standalone( } pub struct DenoCompileBinaryWriter<'a> { + cjs_tracker: &'a CjsTracker, deno_dir: &'a DenoDir, emitter: &'a Emitter, file_fetcher: &'a FileFetcher, @@ -365,6 +368,7 @@ pub struct DenoCompileBinaryWriter<'a> { impl<'a> DenoCompileBinaryWriter<'a> { #[allow(clippy::too_many_arguments)] pub fn new( + cjs_tracker: &'a CjsTracker, deno_dir: &'a DenoDir, emitter: &'a Emitter, file_fetcher: &'a FileFetcher, @@ -374,6 +378,7 @@ impl<'a> DenoCompileBinaryWriter<'a> { npm_system_info: NpmSystemInfo, ) -> Self { Self { + cjs_tracker, deno_dir, emitter, file_fetcher, @@ -599,19 +604,21 @@ impl<'a> DenoCompileBinaryWriter<'a> { } let (maybe_source, media_type) = match module { deno_graph::Module::Js(m) => { - // todo(https://github.com/denoland/deno_media_type/pull/12): use is_emittable() - let is_emittable = matches!( - m.media_type, - MediaType::TypeScript - | MediaType::Mts - | MediaType::Cts - | MediaType::Jsx - | MediaType::Tsx - ); - let source = if is_emittable { + let source = if m.media_type.is_emittable() { + let module_kind = + if m.is_script && self.cjs_tracker.treat_as_cjs(&m.specifier)? { + ModuleKind::Cjs + } else { + ModuleKind::Esm + }; let source = self .emitter - .emit_parsed_source(&m.specifier, m.media_type, &m.source) + .emit_parsed_source( + &m.specifier, + module_kind, + m.media_type, + &m.source, + ) .await?; source.into_bytes() } else { diff --git a/cli/standalone/mod.rs b/cli/standalone/mod.rs index a98b455a99dadb..1c6a9478db8ac0 100644 --- a/cli/standalone/mod.rs +++ b/cli/standalone/mod.rs @@ -72,7 +72,8 @@ use crate::npm::CliNpmResolver; use crate::npm::CliNpmResolverCreateOptions; use crate::npm::CliNpmResolverManagedCreateOptions; use crate::npm::CliNpmResolverManagedSnapshotOption; -use crate::resolver::CjsResolutionStore; +use crate::resolver::CjsTracker; +use crate::resolver::CjsTrackerOptions; use crate::resolver::CliDenoResolverFs; use crate::resolver::CliNodeResolver; use crate::resolver::NpmModuleLoader; @@ -97,7 +98,7 @@ use self::binary::Metadata; use self::file_system::DenoCompileFileSystem; struct SharedModuleLoaderState { - cjs_resolutions: Arc, + cjs_tracker: Arc, fs: Arc, modules: StandaloneModules, node_code_translator: Arc, @@ -335,7 +336,16 @@ impl ModuleLoader for EmbeddedModuleLoader { Ok(Some(module)) => { let (module_specifier, module_type, module_source) = module.into_parts(); - if self.shared.cjs_resolutions.is_known_cjs(original_specifier) { + let is_cjs = + match self.shared.cjs_tracker.treat_as_cjs(original_specifier) { + Ok(is_cjs) => is_cjs, + Err(err) => { + return deno_core::ModuleLoadResponse::Sync(Err(type_error( + format!("{:?}", err), + ))); + } + }; + if is_cjs { let original_specifier = original_specifier.clone(); let module_specifier = module_specifier.clone(); let shared = self.shared.clone(); @@ -572,11 +582,16 @@ pub async fn run(data: StandaloneData) -> Result { deno_runtime::deno_node::DenoFsNodeResolverEnv::new(fs.clone()), npm_resolver.clone().into_npm_resolver(), )); - let cjs_resolutions = Arc::new(CjsResolutionStore::default()); + let cjs_tracker = Arc::new(CjsTracker::new( + CjsTrackerOptions { + unstable_detect_cjs: metadata.unstable_config.detect_cjs, + }, + node_resolver.clone(), + )); let cache_db = Caches::new(deno_dir_provider.clone()); let node_analysis_cache = NodeAnalysisCache::new(cache_db.node_analysis_db()); let cli_node_resolver = Arc::new(CliNodeResolver::new( - cjs_resolutions.clone(), + cjs_tracker.clone(), fs.clone(), node_resolver.clone(), npm_resolver.clone(), @@ -644,13 +659,13 @@ pub async fn run(data: StandaloneData) -> Result { }; let module_loader_factory = StandaloneModuleLoaderFactory { shared: Arc::new(SharedModuleLoaderState { - cjs_resolutions: cjs_resolutions.clone(), + cjs_tracker: cjs_tracker.clone(), fs: fs.clone(), modules, node_code_translator: node_code_translator.clone(), node_resolver: cli_node_resolver.clone(), npm_module_loader: Arc::new(NpmModuleLoader::new( - cjs_resolutions.clone(), + cjs_tracker.clone(), node_code_translator, fs.clone(), cli_node_resolver, @@ -695,7 +710,7 @@ pub async fn run(data: StandaloneData) -> Result { }); let worker_factory = CliMainWorkerFactory::new( Arc::new(BlobStore::default()), - cjs_resolutions, + cjs_tracker, // Code cache is not supported for standalone binary yet. None, feature_checker, diff --git a/cli/standalone/serialization.rs b/cli/standalone/serialization.rs index 2f2b4af9352150..a5eb649bfdfec7 100644 --- a/cli/standalone/serialization.rs +++ b/cli/standalone/serialization.rs @@ -275,7 +275,7 @@ impl<'a> DenoCompileModuleData<'a> { (ModuleType::Wasm, DenoCompileModuleSource::Bytes(self.data)) } // just assume javascript if we made it here - MediaType::TsBuildInfo | MediaType::SourceMap | MediaType::Unknown => ( + MediaType::Css | MediaType::SourceMap | MediaType::Unknown => ( ModuleType::JavaScript, DenoCompileModuleSource::Bytes(self.data), ), @@ -568,7 +568,7 @@ fn serialize_media_type(media_type: MediaType) -> u8 { MediaType::Tsx => 10, MediaType::Json => 11, MediaType::Wasm => 12, - MediaType::TsBuildInfo => 13, + MediaType::Css => 13, MediaType::SourceMap => 14, MediaType::Unknown => 15, } @@ -589,7 +589,7 @@ fn deserialize_media_type(value: u8) -> Result { 10 => Ok(MediaType::Tsx), 11 => Ok(MediaType::Json), 12 => Ok(MediaType::Wasm), - 13 => Ok(MediaType::TsBuildInfo), + 13 => Ok(MediaType::Css), 14 => Ok(MediaType::SourceMap), 15 => Ok(MediaType::Unknown), _ => bail!("Unknown media type value: {}", value), diff --git a/cli/tools/check.rs b/cli/tools/check.rs index 7edb392d48ba85..4baa6cfdf96b28 100644 --- a/cli/tools/check.rs +++ b/cli/tools/check.rs @@ -346,7 +346,7 @@ fn get_check_hash( } } MediaType::Json - | MediaType::TsBuildInfo + | MediaType::Css | MediaType::SourceMap | MediaType::Wasm | MediaType::Unknown => continue, @@ -428,7 +428,7 @@ fn get_tsc_roots( } MediaType::Json | MediaType::Wasm - | MediaType::TsBuildInfo + | MediaType::Css | MediaType::SourceMap | MediaType::Unknown => None, }, @@ -536,7 +536,7 @@ fn has_ts_check(media_type: MediaType, file_text: &str) -> bool { | MediaType::Tsx | MediaType::Json | MediaType::Wasm - | MediaType::TsBuildInfo + | MediaType::Css | MediaType::SourceMap | MediaType::Unknown => false, } diff --git a/cli/tools/coverage/mod.rs b/cli/tools/coverage/mod.rs index 3b08f2c77adc90..391718dad7c18d 100644 --- a/cli/tools/coverage/mod.rs +++ b/cli/tools/coverage/mod.rs @@ -12,6 +12,7 @@ use crate::tools::test::is_supported_test_path; use crate::util::text_encoding::source_map_from_code; use deno_ast::MediaType; +use deno_ast::ModuleKind; use deno_ast::ModuleSpecifier; use deno_config::glob::FileCollector; use deno_config::glob::FilePatterns; @@ -492,6 +493,7 @@ pub async fn cover_files( let npm_resolver = factory.npm_resolver().await?; let file_fetcher = factory.file_fetcher()?; let emitter = factory.emitter()?; + let cjs_tracker = factory.cjs_tracker().await?; assert!(!coverage_flags.files.include.is_empty()); @@ -568,6 +570,8 @@ pub async fn cover_files( let transpiled_code = match file.media_type { MediaType::JavaScript | MediaType::Unknown + | MediaType::Css + | MediaType::Wasm | MediaType::Cjs | MediaType::Mjs | MediaType::Json => None, @@ -577,7 +581,12 @@ pub async fn cover_files( | MediaType::Mts | MediaType::Cts | MediaType::Tsx => { - Some(match emitter.maybe_cached_emit(&file.specifier, &file.source) { + let module_kind = if cjs_tracker.treat_as_cjs(&file.specifier)? { + ModuleKind::Cjs + } else { + ModuleKind::Esm + }; + Some(match emitter.maybe_cached_emit(&file.specifier, module_kind, &file.source) { Some(code) => code, None => { return Err(anyhow!( @@ -588,7 +597,7 @@ pub async fn cover_files( } }) } - MediaType::Wasm | MediaType::TsBuildInfo | MediaType::SourceMap => { + MediaType::SourceMap => { unreachable!() } }; diff --git a/cli/tools/doc.rs b/cli/tools/doc.rs index 5e18546a287c52..8dcf4f324c0468 100644 --- a/cli/tools/doc.rs +++ b/cli/tools/doc.rs @@ -22,9 +22,9 @@ use deno_core::serde_json; use deno_doc as doc; use deno_doc::html::UrlResolveKind; use deno_graph::source::NullFileSystem; +use deno_graph::EsParser; use deno_graph::GraphKind; use deno_graph::ModuleAnalyzer; -use deno_graph::ModuleParser; use deno_graph::ModuleSpecifier; use doc::html::ShortPath; use doc::DocDiagnostic; @@ -37,7 +37,7 @@ const JSON_SCHEMA_VERSION: u8 = 1; async fn generate_doc_nodes_for_builtin_types( doc_flags: DocFlags, - parser: &dyn ModuleParser, + parser: &dyn EsParser, analyzer: &dyn ModuleAnalyzer, ) -> Result>, AnyError> { let source_file_specifier = diff --git a/cli/tools/registry/tar.rs b/cli/tools/registry/tar.rs index aca125e00b7708..6d1801ce6976fc 100644 --- a/cli/tools/registry/tar.rs +++ b/cli/tools/registry/tar.rs @@ -120,7 +120,7 @@ fn resolve_content_maybe_unfurling( | MediaType::Unknown | MediaType::Json | MediaType::Wasm - | MediaType::TsBuildInfo => { + | MediaType::Css => { // not unfurlable data return Ok(data); } diff --git a/cli/tools/repl/session.rs b/cli/tools/repl/session.rs index f306f0197478fc..23b0f11ac5266d 100644 --- a/cli/tools/repl/session.rs +++ b/cli/tools/repl/session.rs @@ -25,6 +25,7 @@ use deno_ast::swc::visit::noop_visit_type; use deno_ast::swc::visit::Visit; use deno_ast::swc::visit::VisitWith; use deno_ast::ImportsNotUsedAsValues; +use deno_ast::ModuleKind; use deno_ast::ModuleSpecifier; use deno_ast::ParseDiagnosticsError; use deno_ast::ParsedSource; @@ -643,6 +644,9 @@ impl ReplSession { var_decl_imports: true, verbatim_module_syntax: false, }, + &deno_ast::TranspileModuleOptions { + module_kind: Some(ModuleKind::Esm), + }, &deno_ast::EmitOptions { source_map: deno_ast::SourceMapOption::None, source_map_base: None, @@ -652,7 +656,6 @@ impl ReplSession { }, )? .into_source() - .into_string()? .text; let value = self diff --git a/cli/tools/run/hmr.rs b/cli/tools/run/hmr.rs index 6ccf8e344bc581..ad78d2c4160140 100644 --- a/cli/tools/run/hmr.rs +++ b/cli/tools/run/hmr.rs @@ -1,9 +1,10 @@ // Copyright 2018-2024 the Deno authors. All rights reserved. MIT license. -use crate::cdp; -use crate::emit::Emitter; -use crate::util::file_watcher::WatcherCommunicator; -use crate::util::file_watcher::WatcherRestartMode; +use std::collections::HashMap; +use std::path::PathBuf; +use std::sync::Arc; + +use deno_ast::ModuleKind; use deno_core::error::generic_error; use deno_core::error::AnyError; use deno_core::futures::StreamExt; @@ -12,11 +13,14 @@ use deno_core::serde_json::{self}; use deno_core::url::Url; use deno_core::LocalInspectorSession; use deno_terminal::colors; -use std::collections::HashMap; -use std::path::PathBuf; -use std::sync::Arc; use tokio::select; +use crate::cdp; +use crate::emit::Emitter; +use crate::resolver::CjsTracker; +use crate::util::file_watcher::WatcherCommunicator; +use crate::util::file_watcher::WatcherRestartMode; + fn explain(status: &cdp::Status) -> &'static str { match status { cdp::Status::Ok => "OK", @@ -58,6 +62,7 @@ pub struct HmrRunner { session: LocalInspectorSession, watcher_communicator: Arc, script_ids: HashMap, + cjs_tracker: Arc, emitter: Arc, } @@ -139,7 +144,12 @@ impl crate::worker::HmrRunner for HmrRunner { }; let source_code = self.emitter.load_and_emit_for_hmr( - &module_url + &module_url, + if self.cjs_tracker.treat_as_cjs(&module_url)? { + ModuleKind::Cjs + } else { + ModuleKind::Esm + }, ).await?; let mut tries = 1; @@ -172,12 +182,14 @@ impl crate::worker::HmrRunner for HmrRunner { impl HmrRunner { pub fn new( + cjs_tracker: Arc, emitter: Arc, session: LocalInspectorSession, watcher_communicator: Arc, ) -> Self { Self { session, + cjs_tracker, emitter, watcher_communicator, script_ids: HashMap::new(), diff --git a/cli/tsc/diagnostics.rs b/cli/tsc/diagnostics.rs index b0394ec177262a..7d276259f1c657 100644 --- a/cli/tsc/diagnostics.rs +++ b/cli/tsc/diagnostics.rs @@ -323,7 +323,7 @@ impl Diagnostics { // todo(dsherret): use a short lived cache to prevent parsing // source maps so often if let Ok(source_map) = - SourceMap::from_slice(&fast_check_module.source_map) + SourceMap::from_slice(&fast_check_module.source_map.as_bytes()) { if let Some(start) = d.start.as_mut() { let maybe_token = source_map diff --git a/cli/tsc/mod.rs b/cli/tsc/mod.rs index a55594460c34f0..7e9e2eb6d2674e 100644 --- a/cli/tsc/mod.rs +++ b/cli/tsc/mod.rs @@ -456,7 +456,7 @@ pub fn as_ts_script_kind(media_type: MediaType) -> i32 { MediaType::Tsx => 4, MediaType::Json => 6, MediaType::SourceMap - | MediaType::TsBuildInfo + | MediaType::Css | MediaType::Wasm | MediaType::Unknown => 0, } diff --git a/cli/util/text_encoding.rs b/cli/util/text_encoding.rs index df72cc2be63151..8524e63ebb32c2 100644 --- a/cli/util/text_encoding.rs +++ b/cli/util/text_encoding.rs @@ -97,6 +97,7 @@ fn find_source_map_range(code: &[u8]) -> Option> { } /// Converts an `Arc` to an `Arc<[u8]>`. +#[allow(dead_code)] pub fn arc_str_to_bytes(arc_str: Arc) -> Arc<[u8]> { let raw = Arc::into_raw(arc_str); // SAFETY: This is safe because they have the same memory layout. diff --git a/cli/worker.rs b/cli/worker.rs index fa2dbde94a2e7c..8a21e02e7a8c52 100644 --- a/cli/worker.rs +++ b/cli/worker.rs @@ -53,7 +53,7 @@ use crate::args::DenoSubcommand; use crate::args::StorageKeyResolver; use crate::errors; use crate::npm::CliNpmResolver; -use crate::resolver::CjsResolutionStore; +use crate::resolver::CjsTracker; use crate::util::checksum; use crate::util::file_watcher::WatcherCommunicator; use crate::util::file_watcher::WatcherRestartMode; @@ -131,7 +131,7 @@ pub struct CliMainWorkerOptions { struct SharedWorkerState { blob_store: Arc, broadcast_channel: InMemoryBroadcastChannel, - cjs_resolution_store: Arc, + cjs_tracker: Arc, code_cache: Option>, compiled_wasm_module_store: CompiledWasmModuleStore, feature_checker: Arc, @@ -156,7 +156,7 @@ impl SharedWorkerState { node_require_loader: NodeRequireLoaderRc, ) -> NodeExtInitServices { NodeExtInitServices { - node_require_loader: node_require_loader, + node_require_loader, node_resolver: self.node_resolver.clone(), npm_resolver: self.npm_resolver.clone().into_npm_resolver(), } @@ -432,7 +432,7 @@ impl CliMainWorkerFactory { #[allow(clippy::too_many_arguments)] pub fn new( blob_store: Arc, - cjs_resolution_store: Arc, + cjs_tracker: Arc, code_cache: Option>, feature_checker: Arc, fs: Arc, @@ -452,7 +452,7 @@ impl CliMainWorkerFactory { shared: Arc::new(SharedWorkerState { blob_store, broadcast_channel: Default::default(), - cjs_resolution_store, + cjs_tracker, code_cache, compiled_wasm_module_store: Default::default(), feature_checker, @@ -544,28 +544,7 @@ impl CliMainWorkerFactory { let is_main_cjs = matches!(node_resolution, NodeResolution::CommonJs(_)); (node_resolution.into_url(), is_main_cjs) } else { - let is_maybe_cjs_js_ext = self.shared.options.unstable_detect_cjs - && specifier_has_extension(&main_module, "js") - && self - .shared - .node_resolver - .get_closest_package_json(&main_module) - .ok() - .flatten() - .map(|pkg_json| pkg_json.typ == "commonjs") - .unwrap_or(false); - let is_cjs = if is_maybe_cjs_js_ext { - // fill the cjs resolution store by preparing the module load - module_loader - .prepare_load(&main_module, None, false) - .await?; - self.shared.cjs_resolution_store.is_known_cjs(&main_module) - } else { - main_module.scheme() == "file" - && (specifier_has_extension(&main_module, "cjs") - || specifier_has_extension(&main_module, "cts")) - }; - (main_module, is_cjs) + (main_module, false) }; let maybe_inspector_server = shared.maybe_inspector_server.clone(); diff --git a/resolvers/deno/sloppy_imports.rs b/resolvers/deno/sloppy_imports.rs index e215e876860bd9..7aba5b771a5ef9 100644 --- a/resolvers/deno/sloppy_imports.rs +++ b/resolvers/deno/sloppy_imports.rs @@ -232,7 +232,7 @@ impl SloppyImportsResolver { | MediaType::Tsx | MediaType::Json | MediaType::Wasm - | MediaType::TsBuildInfo + | MediaType::Css | MediaType::SourceMap => { return None; } diff --git a/runtime/shared.rs b/runtime/shared.rs index 02dfd18719d0b8..f7d76f67a760da 100644 --- a/runtime/shared.rs +++ b/runtime/shared.rs @@ -98,6 +98,7 @@ pub fn maybe_transpile_source( imports_not_used_as_values: deno_ast::ImportsNotUsedAsValues::Remove, ..Default::default() }, + &deno_ast::TranspileModuleOptions::default(), &deno_ast::EmitOptions { source_map: if cfg!(debug_assertions) { SourceMapOption::Separate @@ -109,9 +110,9 @@ pub fn maybe_transpile_source( )? .into_source(); - let maybe_source_map: Option = - transpiled_source.source_map.map(|sm| sm.into()); - let source_text = String::from_utf8(transpiled_source.source)?; - + let maybe_source_map: Option = transpiled_source + .source_map + .map(|sm| sm.into_bytes().into()); + let source_text = transpiled_source.text; Ok((source_text.into(), maybe_source_map)) } diff --git a/tests/specs/check/css_import/exists_run_with_check.out b/tests/specs/check/css_import/exists_run_with_check.out index 1a1dafeb74509c..315769e40cc6ae 100644 --- a/tests/specs/check/css_import/exists_run_with_check.out +++ b/tests/specs/check/css_import/exists_run_with_check.out @@ -1,3 +1,3 @@ -error: Expected a JavaScript or TypeScript module, but identified a Unknown module. Importing these types of modules is currently not supported. +error: Expected a JavaScript or TypeScript module, but identified a Css module. Importing these types of modules is currently not supported. Specifier: file:///[WILDLINE]/app.css at file:///[WILDLINE]/exists.ts:2:8 diff --git a/tests/specs/run/cjs_unprepared/__test__.jsonc b/tests/specs/run/cjs_unprepared/__test__.jsonc new file mode 100644 index 00000000000000..f816bad869762e --- /dev/null +++ b/tests/specs/run/cjs_unprepared/__test__.jsonc @@ -0,0 +1,4 @@ +{ + "args": "run -A main.ts", + "output": "main.out" +} diff --git a/tests/specs/run/cjs_unprepared/file.cjs b/tests/specs/run/cjs_unprepared/file.cjs new file mode 100644 index 00000000000000..ba0c6fa6ff0b48 --- /dev/null +++ b/tests/specs/run/cjs_unprepared/file.cjs @@ -0,0 +1,7 @@ +// non-analyzable +const moduleName = "./output.cjs"; +function getModuleName() { + return moduleName; +} + +require(getModuleName()); diff --git a/tests/specs/run/cjs_unprepared/main.out b/tests/specs/run/cjs_unprepared/main.out new file mode 100644 index 00000000000000..e965047ad7c578 --- /dev/null +++ b/tests/specs/run/cjs_unprepared/main.out @@ -0,0 +1 @@ +Hello diff --git a/tests/specs/run/cjs_unprepared/main.ts b/tests/specs/run/cjs_unprepared/main.ts new file mode 100644 index 00000000000000..5630b4fbc3c913 --- /dev/null +++ b/tests/specs/run/cjs_unprepared/main.ts @@ -0,0 +1,7 @@ +// non-analyzable +const moduleName = "./output.cjs"; +function getModuleName() { + return moduleName; +} + +await import(getModuleName()); diff --git a/tests/specs/run/cjs_unprepared/output.cjs b/tests/specs/run/cjs_unprepared/output.cjs new file mode 100644 index 00000000000000..54ed3702f516d6 --- /dev/null +++ b/tests/specs/run/cjs_unprepared/output.cjs @@ -0,0 +1,3 @@ +console.log("Hello"); + +module.exports = 1; From 01c773f494f69d316115710cf1abd9fa002e6b30 Mon Sep 17 00:00:00 2001 From: David Sherret Date: Mon, 28 Oct 2024 21:43:06 -0400 Subject: [PATCH 15/42] Create PackageJsonResolver --- cli/factory.rs | 59 ++++++++++--------- cli/lsp/resolver.rs | 44 ++++++++++---- cli/node.rs | 93 +++++++++++++---------------- cli/resolver.rs | 18 ++---- cli/standalone/mod.rs | 14 +++-- cli/tools/coverage/mod.rs | 2 +- cli/util/path.rs | 30 ---------- cli/worker.rs | 11 ++-- ext/node/lib.rs | 8 +++ ext/node/ops/require.rs | 22 ++++--- resolvers/node/lib.rs | 2 + resolvers/node/package_json.rs | 84 +++++++++++++++++++++++++- resolvers/node/resolution.rs | 104 +++++++++------------------------ 13 files changed, 261 insertions(+), 230 deletions(-) diff --git a/cli/factory.rs b/cli/factory.rs index c59e2941082539..2a0495abc74a74 100644 --- a/cli/factory.rs +++ b/cli/factory.rs @@ -68,6 +68,7 @@ use deno_core::FeatureChecker; use deno_runtime::deno_fs; use deno_runtime::deno_node::DenoFsNodeResolverEnv; use deno_runtime::deno_node::NodeResolver; +use deno_runtime::deno_node::PackageJsonResolver; use deno_runtime::deno_permissions::Permissions; use deno_runtime::deno_permissions::PermissionsContainer; use deno_runtime::deno_tls::rustls::RootCertStore; @@ -188,6 +189,7 @@ struct CliFactoryServices { node_resolver: Deferred>, npm_resolver: Deferred>, permission_desc_parser: Deferred>, + pkg_json_resolver: Deferred>, root_permissions_container: Deferred, sloppy_imports_resolver: Deferred>>, text_only_progress_bar: Deferred, @@ -558,6 +560,7 @@ impl CliFactory { Ok(Arc::new(NodeResolver::new( DenoFsNodeResolverEnv::new(self.fs().clone()), self.npm_resolver().await?.clone().into_npm_resolver(), + self.pkg_json_resolver().clone(), ))) } .boxed_local(), @@ -575,24 +578,32 @@ impl CliFactory { let caches = self.caches()?; let node_analysis_cache = NodeAnalysisCache::new(caches.node_analysis_db()); - let node_resolver = self.cli_node_resolver().await?.clone(); + let node_resolver = self.node_resolver().await?.clone(); let cjs_esm_analyzer = CliCjsCodeAnalyzer::new( node_analysis_cache, + self.cjs_tracker()?.clone(), self.fs().clone(), - node_resolver, Some(self.parsed_source_cache().clone()), ); Ok(Arc::new(NodeCodeTranslator::new( cjs_esm_analyzer, DenoFsNodeResolverEnv::new(self.fs().clone()), - self.node_resolver().await?.clone(), + node_resolver, self.npm_resolver().await?.clone().into_npm_resolver(), ))) }) .await } + pub fn pkg_json_resolver(&self) -> &Arc { + self.services.pkg_json_resolver.get_or_init(|| { + Arc::new(PackageJsonResolver::new(DenoFsNodeResolverEnv::new( + self.fs().clone(), + ))) + }) + } + pub async fn type_checker(&self) -> Result<&Arc, AnyError> { self .services @@ -701,20 +712,16 @@ impl CliFactory { .await } - pub async fn cjs_tracker(&self) -> Result<&Arc, AnyError> { - self - .services - .cjs_tracker - .get_or_try_init_async(async { - let options = self.cli_options()?; - Ok(Arc::new(CjsTracker::new( - CjsTrackerOptions { - unstable_detect_cjs: options.unstable_detect_cjs(), - }, - self.node_resolver().await?.clone(), - ))) - }) - .await + pub fn cjs_tracker(&self) -> Result<&Arc, AnyError> { + self.services.cjs_tracker.get_or_try_init(|| { + let options = self.cli_options()?; + Ok(Arc::new(CjsTracker::new( + CjsTrackerOptions { + unstable_detect_cjs: options.unstable_detect_cjs(), + }, + self.pkg_json_resolver().clone(), + ))) + }) } pub async fn cli_node_resolver( @@ -725,7 +732,7 @@ impl CliFactory { .cli_node_resolver .get_or_try_init_async(async { Ok(Arc::new(CliNodeResolver::new( - self.cjs_tracker().await?.clone(), + self.cjs_tracker()?.clone(), self.fs().clone(), self.node_resolver().await?.clone(), self.npm_resolver().await?.clone(), @@ -764,7 +771,7 @@ impl CliFactory { ) -> Result { let cli_options = self.cli_options()?; Ok(DenoCompileBinaryWriter::new( - self.cjs_tracker().await?, + self.cjs_tracker()?, self.deno_dir()?, self.emitter()?, self.file_fetcher()?, @@ -806,11 +813,11 @@ impl CliFactory { None }; let node_code_translator = self.node_code_translator().await?; - let cjs_tracker = self.cjs_tracker().await?.clone(); + let cjs_tracker = self.cjs_tracker()?.clone(); + let pkg_json_resolver = self.pkg_json_resolver().clone(); Ok(CliMainWorkerFactory::new( self.blob_store().clone(), - cjs_tracker.clone(), if cli_options.code_cache_enabled() { Some(self.code_cache()?.clone()) } else { @@ -837,7 +844,7 @@ impl CliFactory { cli_node_resolver.clone(), cli_npm_resolver.clone(), NpmModuleLoader::new( - self.cjs_tracker().await?.clone(), + self.cjs_tracker()?.clone(), node_code_translator.clone(), fs.clone(), cli_node_resolver.clone(), @@ -847,22 +854,23 @@ impl CliFactory { )), node_resolver.clone(), npm_resolver.clone(), + pkg_json_resolver, self.root_cert_store_provider().clone(), self.root_permissions_container()?.clone(), StorageKeyResolver::from_options(cli_options), cli_options.sub_command().clone(), - self.create_cli_main_worker_options().await?, + self.create_cli_main_worker_options()?, )) } - async fn create_cli_main_worker_options( + fn create_cli_main_worker_options( &self, ) -> Result { let cli_options = self.cli_options()?; let create_hmr_runner = if cli_options.has_hmr() { let watcher_communicator = self.watcher_communicator.clone().unwrap(); let emitter = self.emitter()?.clone(); - let cjs_tracker = self.cjs_tracker().await?.clone(); + let cjs_tracker = self.cjs_tracker()?.clone(); let fn_: crate::worker::CreateHmrRunnerCb = Box::new(move |session| { Box::new(HmrRunner::new( cjs_tracker.clone(), @@ -920,7 +928,6 @@ impl CliFactory { node_ipc: cli_options.node_ipc_fd(), serve_port: cli_options.serve_port(), serve_host: cli_options.serve_host(), - unstable_detect_cjs: cli_options.unstable_detect_cjs(), }) } } diff --git a/cli/lsp/resolver.rs b/cli/lsp/resolver.rs index 979708c27f4b8c..83979608bec2cd 100644 --- a/cli/lsp/resolver.rs +++ b/cli/lsp/resolver.rs @@ -14,6 +14,7 @@ use deno_path_util::url_to_file_path; use deno_runtime::deno_fs; use deno_runtime::deno_node::NodeResolver; use deno_runtime::deno_node::PackageJson; +use deno_runtime::deno_node::PackageJsonResolver; use deno_semver::jsr::JsrPackageReqReference; use deno_semver::npm::NpmPackageReqReference; use deno_semver::package::PackageNv; @@ -64,6 +65,7 @@ struct LspScopeResolver { jsr_resolver: Option>, npm_resolver: Option>, node_resolver: Option>, + pkg_json_resolver: Option>, redirect_resolver: Option>, graph_imports: Arc>, config_data: Option>, @@ -76,6 +78,7 @@ impl Default for LspScopeResolver { jsr_resolver: None, npm_resolver: None, node_resolver: None, + pkg_json_resolver: None, redirect_resolver: None, graph_imports: Default::default(), config_data: None, @@ -91,6 +94,10 @@ impl LspScopeResolver { ) -> Self { let mut npm_resolver = None; let mut node_resolver = None; + let fs = Arc::new(deno_fs::RealFs); + let pkg_json_resolver = Arc::new(PackageJsonResolver::new( + deno_runtime::deno_node::DenoFsNodeResolverEnv::new(fs.clone()), + )); if let Some(http_client) = http_client_provider { npm_resolver = create_npm_resolver( config_data.map(|d| d.as_ref()), @@ -98,7 +105,13 @@ impl LspScopeResolver { http_client, ) .await; - node_resolver = create_node_resolver(npm_resolver.as_ref()); + node_resolver = npm_resolver.as_ref().map(|npm_resolver| { + create_node_resolver( + fs.clone(), + npm_resolver, + pkg_json_resolver.clone(), + ) + }); } let graph_resolver = create_graph_resolver( config_data.map(|d| d.as_ref()), @@ -139,6 +152,7 @@ impl LspScopeResolver { jsr_resolver, npm_resolver, node_resolver, + pkg_json_resolver: Some(pkg_json_resolver), redirect_resolver, graph_imports, config_data: config_data.cloned(), @@ -148,7 +162,13 @@ impl LspScopeResolver { fn snapshot(&self) -> Arc { let npm_resolver = self.npm_resolver.as_ref().map(|r| r.clone_snapshotted()); - let node_resolver = create_node_resolver(npm_resolver.as_ref()); + let fs = Arc::new(deno_fs::RealFs); + let pkg_json_resolver = Arc::new(PackageJsonResolver::new( + deno_runtime::deno_node::DenoFsNodeResolverEnv::new(fs.clone()), + )); + let node_resolver = npm_resolver.as_ref().map(|npm_resolver| { + create_node_resolver(fs, npm_resolver, pkg_json_resolver.clone()) + }); let graph_resolver = create_graph_resolver( self.config_data.as_deref(), npm_resolver.as_ref(), @@ -160,6 +180,7 @@ impl LspScopeResolver { npm_resolver, node_resolver, redirect_resolver: self.redirect_resolver.clone(), + pkg_json_resolver: Some(pkg_json_resolver), graph_imports: self.graph_imports.clone(), config_data: self.config_data.clone(), }) @@ -399,10 +420,10 @@ impl LspResolver { referrer: &ModuleSpecifier, ) -> Result>, ClosestPkgJsonError> { let resolver = self.get_scope_resolver(Some(referrer)); - let Some(node_resolver) = resolver.node_resolver.as_ref() else { + let Some(pkg_json_resolver) = resolver.pkg_json_resolver.as_ref() else { return Ok(None); }; - node_resolver.get_closest_package_json(referrer) + pkg_json_resolver.get_closest_package_json(referrer) } pub fn resolve_redirects( @@ -508,27 +529,28 @@ async fn create_npm_resolver( } fn create_node_resolver( - npm_resolver: Option<&Arc>, -) -> Option> { - let npm_resolver = npm_resolver?; - let fs = Arc::new(deno_fs::RealFs); + fs: Arc, + npm_resolver: &Arc, + pkg_json_resolver: Arc, +) -> Arc { let node_resolver_inner = Arc::new(NodeResolver::new( deno_runtime::deno_node::DenoFsNodeResolverEnv::new(fs.clone()), npm_resolver.clone().into_npm_resolver(), + pkg_json_resolver.clone(), )); let cjs_tracker = Arc::new(CjsTracker::new( CjsTrackerOptions { // todo(dsherret): support in the lsp unstable_detect_cjs: false, }, - node_resolver_inner.clone(), + pkg_json_resolver, )); - Some(Arc::new(CliNodeResolver::new( + Arc::new(CliNodeResolver::new( cjs_tracker, fs, node_resolver_inner, npm_resolver.clone(), - ))) + )) } fn create_graph_resolver( diff --git a/cli/node.rs b/cli/node.rs index 498d1c10671278..9bc97d0751b8aa 100644 --- a/cli/node.rs +++ b/cli/node.rs @@ -19,7 +19,7 @@ use serde::Serialize; use crate::cache::CacheDBHash; use crate::cache::NodeAnalysisCache; use crate::cache::ParsedSourceCache; -use crate::resolver::CliNodeResolver; +use crate::resolver::CjsTracker; use crate::util::fs::canonicalize_path_maybe_not_exists; pub type CliNodeCodeTranslator = @@ -57,22 +57,22 @@ pub enum CliCjsAnalysis { pub struct CliCjsCodeAnalyzer { cache: NodeAnalysisCache, + cjs_tracker: Arc, fs: deno_fs::FileSystemRc, - node_resolver: Arc, parsed_source_cache: Option>, } impl CliCjsCodeAnalyzer { pub fn new( cache: NodeAnalysisCache, + cjs_tracker: Arc, fs: deno_fs::FileSystemRc, - node_resolver: Arc, parsed_source_cache: Option>, ) -> Self { Self { cache, + cjs_tracker, fs, - node_resolver, parsed_source_cache, } } @@ -89,7 +89,7 @@ impl CliCjsCodeAnalyzer { return Ok(analysis); } - let mut media_type = MediaType::from_specifier(specifier); + let media_type = MediaType::from_specifier(specifier); if media_type == MediaType::Json { return Ok(CliCjsAnalysis::Cjs { exports: vec![], @@ -97,55 +97,44 @@ impl CliCjsCodeAnalyzer { }); } - if media_type == MediaType::JavaScript { - if let Some(package_json) = - self.node_resolver.get_closest_package_json(specifier)? - { - match package_json.typ.as_str() { - "commonjs" => { - media_type = MediaType::Cjs; - } - "module" => { - media_type = MediaType::Mjs; - } - _ => {} - } - } - } - - let maybe_parsed_source = self - .parsed_source_cache - .as_ref() - .and_then(|c| c.remove_parsed_source(specifier)); - - let analysis = deno_core::unsync::spawn_blocking({ - let specifier = specifier.clone(); - let source: Arc = source.into(); - move || -> Result<_, deno_ast::ParseDiagnostic> { - let parsed_source = - maybe_parsed_source.map(Ok).unwrap_or_else(|| { - deno_ast::parse_program(deno_ast::ParseParams { - specifier, - text: source, - media_type, - capture_tokens: true, - scope_analysis: false, - maybe_syntax: None, + let treat_as_cjs = self.cjs_tracker.treat_as_cjs(&specifier)?; + let analysis = if treat_as_cjs { + let maybe_parsed_source = self + .parsed_source_cache + .as_ref() + .and_then(|c| c.remove_parsed_source(specifier)); + + deno_core::unsync::spawn_blocking({ + let specifier = specifier.clone(); + let source: Arc = source.into(); + move || -> Result<_, AnyError> { + let parsed_source = + maybe_parsed_source.map(Ok).unwrap_or_else(|| { + deno_ast::parse_program(deno_ast::ParseParams { + specifier, + text: source, + media_type, + capture_tokens: true, + scope_analysis: false, + maybe_syntax: None, + }) + })?; + if parsed_source.is_script() && treat_as_cjs { + let analysis = parsed_source.analyze_cjs(); + Ok(CliCjsAnalysis::Cjs { + exports: analysis.exports, + reexports: analysis.reexports, }) - })?; - if parsed_source.is_script() || media_type == MediaType::Cjs { - let analysis = parsed_source.analyze_cjs(); - Ok(CliCjsAnalysis::Cjs { - exports: analysis.exports, - reexports: analysis.reexports, - }) - } else { - Ok(CliCjsAnalysis::Esm) + } else { + Ok(CliCjsAnalysis::Esm) + } } - } - }) - .await - .unwrap()?; + }) + .await + .unwrap()? + } else { + CliCjsAnalysis::Esm + }; self .cache diff --git a/cli/resolver.rs b/cli/resolver.rs index 09f0e300cbd406..7d8003e0f9631c 100644 --- a/cli/resolver.rs +++ b/cli/resolver.rs @@ -29,6 +29,7 @@ use deno_runtime::deno_fs; use deno_runtime::deno_fs::FileSystem; use deno_runtime::deno_node::is_builtin_node_module; use deno_runtime::deno_node::NodeResolver; +use deno_runtime::deno_node::PackageJsonResolver; use deno_semver::npm::NpmPackageReqReference; use deno_semver::package::PackageReq; use node_resolver::errors::ClosestPkgJsonError; @@ -42,7 +43,6 @@ use node_resolver::errors::UrlToNodeResolutionError; use node_resolver::NodeModuleKind; use node_resolver::NodeResolution; use node_resolver::NodeResolutionMode; -use node_resolver::PackageJson; use std::borrow::Cow; use std::path::Path; use std::path::PathBuf; @@ -53,7 +53,6 @@ use crate::args::DENO_DISABLE_PEDANTIC_NODE_WARNINGS; use crate::node::CliNodeCodeTranslator; use crate::npm::CliNpmResolver; use crate::npm::InnerCliNpmResolverRef; -use crate::util::path::specifier_has_extension; use crate::util::sync::AtomicFlag; use crate::util::text_encoding::from_utf8_lossy_owned; @@ -130,13 +129,6 @@ impl CliNodeResolver { self.npm_resolver.in_npm_package(specifier) } - pub fn get_closest_package_json( - &self, - referrer: &ModuleSpecifier, - ) -> Result>, ClosestPkgJsonError> { - self.node_resolver.get_closest_package_json(referrer) - } - pub fn resolve_if_for_npm_pkg( &self, specifier: &str, @@ -449,18 +441,18 @@ pub struct CjsTrackerOptions { pub struct CjsTracker { unstable_detect_cjs: bool, set: DashSet, - node_resolver: Arc, + pkg_json_resolver: Arc, } impl CjsTracker { pub fn new( options: CjsTrackerOptions, - node_resolver: Arc, + pkg_json_resolver: Arc, ) -> Self { Self { unstable_detect_cjs: options.unstable_detect_cjs, set: DashSet::new(), - node_resolver, + pkg_json_resolver, } } @@ -500,7 +492,7 @@ impl CjsTracker { }; if is_pkg_json_type_affected { if let Some(pkg_json) = - self.node_resolver.get_closest_package_json(specifier)? + self.pkg_json_resolver.get_closest_package_json(specifier)? { return Ok(pkg_json.typ == "commonjs"); } diff --git a/cli/standalone/mod.rs b/cli/standalone/mod.rs index 1c6a9478db8ac0..67b525cba5e396 100644 --- a/cli/standalone/mod.rs +++ b/cli/standalone/mod.rs @@ -33,6 +33,7 @@ use deno_runtime::deno_fs; use deno_runtime::deno_node::create_host_defined_options; use deno_runtime::deno_node::NodeRequireLoader; use deno_runtime::deno_node::NodeResolver; +use deno_runtime::deno_node::PackageJsonResolver; use deno_runtime::deno_permissions::Permissions; use deno_runtime::deno_permissions::PermissionsContainer; use deno_runtime::deno_tls::rustls::RootCertStore; @@ -578,15 +579,19 @@ pub async fn run(data: StandaloneData) -> Result { }; let has_node_modules_dir = npm_resolver.root_node_modules_path().is_some(); + let pkg_json_resolver = Arc::new(PackageJsonResolver::new( + deno_runtime::deno_node::DenoFsNodeResolverEnv::new(fs.clone()), + )); let node_resolver = Arc::new(NodeResolver::new( deno_runtime::deno_node::DenoFsNodeResolverEnv::new(fs.clone()), npm_resolver.clone().into_npm_resolver(), + pkg_json_resolver.clone(), )); let cjs_tracker = Arc::new(CjsTracker::new( CjsTrackerOptions { unstable_detect_cjs: metadata.unstable_config.detect_cjs, }, - node_resolver.clone(), + pkg_json_resolver.clone(), )); let cache_db = Caches::new(deno_dir_provider.clone()); let node_analysis_cache = NodeAnalysisCache::new(cache_db.node_analysis_db()); @@ -598,8 +603,8 @@ pub async fn run(data: StandaloneData) -> Result { )); let cjs_esm_code_analyzer = CliCjsCodeAnalyzer::new( node_analysis_cache, + cjs_tracker.clone(), fs.clone(), - cli_node_resolver.clone(), None, ); let node_code_translator = Arc::new(NodeCodeTranslator::new( @@ -665,7 +670,7 @@ pub async fn run(data: StandaloneData) -> Result { node_code_translator: node_code_translator.clone(), node_resolver: cli_node_resolver.clone(), npm_module_loader: Arc::new(NpmModuleLoader::new( - cjs_tracker.clone(), + cjs_tracker, node_code_translator, fs.clone(), cli_node_resolver, @@ -710,7 +715,6 @@ pub async fn run(data: StandaloneData) -> Result { }); let worker_factory = CliMainWorkerFactory::new( Arc::new(BlobStore::default()), - cjs_tracker, // Code cache is not supported for standalone binary yet. None, feature_checker, @@ -721,6 +725,7 @@ pub async fn run(data: StandaloneData) -> Result { Box::new(module_loader_factory), node_resolver, npm_resolver, + pkg_json_resolver, root_cert_store_provider, permissions, StorageKeyResolver::empty(), @@ -753,7 +758,6 @@ pub async fn run(data: StandaloneData) -> Result { node_ipc: None, serve_port: None, serve_host: None, - unstable_detect_cjs: metadata.unstable_config.detect_cjs, }, ); diff --git a/cli/tools/coverage/mod.rs b/cli/tools/coverage/mod.rs index 391718dad7c18d..2db94963776318 100644 --- a/cli/tools/coverage/mod.rs +++ b/cli/tools/coverage/mod.rs @@ -493,7 +493,7 @@ pub async fn cover_files( let npm_resolver = factory.npm_resolver().await?; let file_fetcher = factory.file_fetcher()?; let emitter = factory.emitter()?; - let cjs_tracker = factory.cjs_tracker().await?; + let cjs_tracker = factory.cjs_tracker()?; assert!(!coverage_flags.files.include.is_empty()); diff --git a/cli/util/path.rs b/cli/util/path.rs index b68e1361b7a0e6..58bed664f9ba98 100644 --- a/cli/util/path.rs +++ b/cli/util/path.rs @@ -42,24 +42,6 @@ pub fn get_extension(file_path: &Path) -> Option { .map(|e| e.to_lowercase()); } -// todo(dsherret): replace with https://github.com/denoland/deno_path_util/pull/4 -pub fn specifier_has_extension( - specifier: &ModuleSpecifier, - searching_ext: &str, -) -> bool { - let searching_ext = searching_ext.strip_prefix('.').unwrap_or(searching_ext); - debug_assert!(!searching_ext.contains('.')); // exts like .d.ts are not implemented here - let path = specifier.path(); - if path.len() < searching_ext.len() { - return false; - } - let ext_pos = path.len() - searching_ext.len(); - let (start_path, end_path) = path.split_at(ext_pos); - end_path.eq_ignore_ascii_case(searching_ext) - && start_path.ends_with('.') - && !start_path.ends_with("/.") -} - pub fn get_atomic_dir_path(file_path: &Path) -> PathBuf { let rand = gen_rand_path_component(); let new_file_name = format!( @@ -353,18 +335,6 @@ mod test { } } - #[test] - fn test_specifier_has_extension() { - fn get(specifier: &str, ext: &str) -> bool { - specifier_has_extension(&ModuleSpecifier::parse(specifier).unwrap(), ext) - } - - assert!(get("file:///a/b/c.ts", "ts")); - assert!(get("file:///a/b/c.ts", ".ts")); - assert!(!get("file:///a/b/c.ts", ".cts")); - assert!(get("file:///a/b/c.CtS", ".cts")); - } - #[test] fn test_to_percent_decoded_str() { let str = to_percent_decoded_str("%F0%9F%A6%95"); diff --git a/cli/worker.rs b/cli/worker.rs index 8a21e02e7a8c52..f150242b3fa262 100644 --- a/cli/worker.rs +++ b/cli/worker.rs @@ -26,6 +26,7 @@ use deno_runtime::deno_node::NodeExtInitServices; use deno_runtime::deno_node::NodeRequireLoader; use deno_runtime::deno_node::NodeRequireLoaderRc; use deno_runtime::deno_node::NodeResolver; +use deno_runtime::deno_node::PackageJsonResolver; use deno_runtime::deno_permissions::PermissionsContainer; use deno_runtime::deno_tls::RootCertStoreProvider; use deno_runtime::deno_web::BlobStore; @@ -53,11 +54,9 @@ use crate::args::DenoSubcommand; use crate::args::StorageKeyResolver; use crate::errors; use crate::npm::CliNpmResolver; -use crate::resolver::CjsTracker; use crate::util::checksum; use crate::util::file_watcher::WatcherCommunicator; use crate::util::file_watcher::WatcherRestartMode; -use crate::util::path::specifier_has_extension; use crate::version; pub struct CreateModuleLoaderResult { @@ -125,13 +124,11 @@ pub struct CliMainWorkerOptions { pub node_ipc: Option, pub serve_port: Option, pub serve_host: Option, - pub unstable_detect_cjs: bool, } struct SharedWorkerState { blob_store: Arc, broadcast_channel: InMemoryBroadcastChannel, - cjs_tracker: Arc, code_cache: Option>, compiled_wasm_module_store: CompiledWasmModuleStore, feature_checker: Arc, @@ -142,6 +139,7 @@ struct SharedWorkerState { module_loader_factory: Box, node_resolver: Arc, npm_resolver: Arc, + pkg_json_resolver: Arc, root_cert_store_provider: Arc, root_permissions: PermissionsContainer, shared_array_buffer_store: SharedArrayBufferStore, @@ -159,6 +157,7 @@ impl SharedWorkerState { node_require_loader, node_resolver: self.node_resolver.clone(), npm_resolver: self.npm_resolver.clone().into_npm_resolver(), + pkg_json_resolver: self.pkg_json_resolver.clone(), } } @@ -432,7 +431,6 @@ impl CliMainWorkerFactory { #[allow(clippy::too_many_arguments)] pub fn new( blob_store: Arc, - cjs_tracker: Arc, code_cache: Option>, feature_checker: Arc, fs: Arc, @@ -442,6 +440,7 @@ impl CliMainWorkerFactory { module_loader_factory: Box, node_resolver: Arc, npm_resolver: Arc, + pkg_json_resolver: Arc, root_cert_store_provider: Arc, root_permissions: PermissionsContainer, storage_key_resolver: StorageKeyResolver, @@ -452,7 +451,6 @@ impl CliMainWorkerFactory { shared: Arc::new(SharedWorkerState { blob_store, broadcast_channel: Default::default(), - cjs_tracker, code_cache, compiled_wasm_module_store: Default::default(), feature_checker, @@ -463,6 +461,7 @@ impl CliMainWorkerFactory { module_loader_factory, node_resolver, npm_resolver, + pkg_json_resolver, root_cert_store_provider, root_permissions, shared_array_buffer_store: Default::default(), diff --git a/ext/node/lib.rs b/ext/node/lib.rs index 4352c0fe56ec33..7750390dc7f5d0 100644 --- a/ext/node/lib.rs +++ b/ext/node/lib.rs @@ -158,6 +158,7 @@ pub struct NodeExtInitServices { pub node_require_loader: NodeRequireLoaderRc, pub node_resolver: NodeResolverRc, pub npm_resolver: NpmResolverRc, + pub pkg_json_resolver: PackageJsonResolverRc, } deno_core::extension!(deno_node, @@ -644,6 +645,7 @@ deno_core::extension!(deno_node, state.put(init.node_require_loader.clone()); state.put(init.node_resolver.clone()); state.put(init.npm_resolver.clone()); + state.put(init.pkg_json_resolver.clone()); } }, global_template_middleware = global_template_middleware, @@ -790,6 +792,12 @@ pub type NodeResolver = node_resolver::NodeResolver; #[allow(clippy::disallowed_types)] pub type NodeResolverRc = deno_fs::sync::MaybeArc>; +pub type PackageJsonResolver = + node_resolver::PackageJsonResolver; +#[allow(clippy::disallowed_types)] +pub type PackageJsonResolverRc = deno_fs::sync::MaybeArc< + node_resolver::PackageJsonResolver, +>; #[derive(Debug)] pub struct DenoFsNodeResolverEnv { diff --git a/ext/node/ops/require.rs b/ext/node/ops/require.rs index 0338723a4f4de4..fca44b42ab32cf 100644 --- a/ext/node/ops/require.rs +++ b/ext/node/ops/require.rs @@ -23,6 +23,7 @@ use crate::NodePermissions; use crate::NodeRequireLoaderRc; use crate::NodeResolverRc; use crate::NpmResolverRc; +use crate::PackageJsonResolverRc; #[must_use = "the resolved return value to mitigate time-of-check to time-of-use issues"] fn ensure_read_permission<'a, P>( @@ -417,8 +418,8 @@ where return Ok(None); } - let node_resolver = state.borrow::(); - let pkg = node_resolver + let pkg_json_resolver = state.borrow::(); + let pkg = pkg_json_resolver .get_closest_package_json_from_path(&PathBuf::from(parent_path.unwrap())) .ok() .flatten(); @@ -447,6 +448,7 @@ where let referrer = deno_core::url::Url::from_file_path(&pkg.path).unwrap(); if let Some(exports) = &pkg.exports { + let node_resolver = state.borrow::(); let r = node_resolver.package_exports_resolve( &pkg.path, &expansion, @@ -514,6 +516,7 @@ where let fs = state.borrow::(); let npm_resolver = state.borrow::(); let node_resolver = state.borrow::(); + let pkg_json_resolver = state.borrow::(); let modules_path = PathBuf::from(&modules_path_str); let pkg_path = if npm_resolver.in_npm_package_at_file_path(&modules_path) @@ -530,7 +533,7 @@ where } }; let Some(pkg) = - node_resolver.load_package_json(&pkg_path.join("package.json"))? + pkg_json_resolver.load_package_json(&pkg_path.join("package.json"))? else { return Ok(None); }; @@ -566,8 +569,8 @@ where { let filename = PathBuf::from(filename); // permissions: allow reading the closest package.json files - let node_resolver = state.borrow::().clone(); - node_resolver.get_closest_package_json_from_path(&filename) + let pkg_json_resolver = state.borrow::(); + pkg_json_resolver.get_closest_package_json_from_path(&filename) } #[op2] @@ -579,13 +582,13 @@ pub fn op_require_read_package_scope

( where P: NodePermissions + 'static, { - let node_resolver = state.borrow::().clone(); + let pkg_json_resolver = state.borrow::(); let package_json_path = PathBuf::from(package_json_path); if package_json_path.file_name() != Some("package.json".as_ref()) { // permissions: do not allow reading a non-package.json file return None; } - node_resolver + pkg_json_resolver .load_package_json(&package_json_path) .ok() .flatten() @@ -604,14 +607,15 @@ where let referrer_path = PathBuf::from(&referrer_filename); let referrer_path = ensure_read_permission::

(state, &referrer_path) .map_err(RequireError::Permission)?; - let node_resolver = state.borrow::(); + let pkg_json_resolver = state.borrow::(); let Some(pkg) = - node_resolver.get_closest_package_json_from_path(&referrer_path)? + pkg_json_resolver.get_closest_package_json_from_path(&referrer_path)? else { return Ok(None); }; if pkg.imports.is_some() { + let node_resolver = state.borrow::(); let referrer_url = Url::from_file_path(&referrer_filename).unwrap(); let url = node_resolver.package_imports_resolve( &request, diff --git a/resolvers/node/lib.rs b/resolvers/node/lib.rs index f03f7704869fe5..66c0520edbe5d9 100644 --- a/resolvers/node/lib.rs +++ b/resolvers/node/lib.rs @@ -16,6 +16,8 @@ pub use deno_package_json::PackageJson; pub use npm::NpmResolver; pub use npm::NpmResolverRc; pub use package_json::load_pkg_json; +pub use package_json::PackageJsonResolver; +pub use package_json::PackageJsonResolverRc; pub use package_json::PackageJsonThreadLocalCache; pub use path::PathClean; pub use resolution::parse_npm_pkg_name; diff --git a/resolvers/node/package_json.rs b/resolvers/node/package_json.rs index de750f1d7e3db3..1d87254eeb3ded 100644 --- a/resolvers/node/package_json.rs +++ b/resolvers/node/package_json.rs @@ -2,15 +2,21 @@ use deno_package_json::PackageJson; use deno_package_json::PackageJsonRc; +use deno_path_util::strip_unc_prefix; use std::cell::RefCell; use std::collections::HashMap; use std::io::ErrorKind; use std::path::Path; use std::path::PathBuf; +use url::Url; +use crate::env::NodeResolverEnv; +use crate::errors::CanonicalizingPkgJsonDirError; +use crate::errors::ClosestPkgJsonError; use crate::errors::PackageJsonLoadError; -// use a thread local cache so that workers have their own distinct cache +// todo(dsherret): this isn't exactly correct and we should change it to instead +// be created per worker and passed down as a ctor arg to the pkg json resolver thread_local! { static CACHE: RefCell> = RefCell::new(HashMap::new()); } @@ -33,6 +39,82 @@ impl deno_package_json::PackageJsonCache for PackageJsonThreadLocalCache { } } +#[allow(clippy::disallowed_types)] +pub type PackageJsonResolverRc = + crate::sync::MaybeArc>; + +#[derive(Debug)] +pub struct PackageJsonResolver { + env: TEnv, +} + +impl PackageJsonResolver { + pub fn new(env: TEnv) -> Self { + Self { env } + } + + pub fn get_closest_package_json( + &self, + url: &Url, + ) -> Result, ClosestPkgJsonError> { + let Ok(file_path) = deno_path_util::url_to_file_path(url) else { + return Ok(None); + }; + self.get_closest_package_json_from_path(&file_path) + } + + pub fn get_closest_package_json_from_path( + &self, + file_path: &Path, + ) -> Result, ClosestPkgJsonError> { + // we use this for deno compile using byonm because the script paths + // won't be in virtual file system, but the package.json paths will be + fn canonicalize_first_ancestor_exists( + dir_path: &Path, + env: &TEnv, + ) -> Result, std::io::Error> { + for ancestor in dir_path.ancestors() { + match env.realpath_sync(ancestor) { + Ok(dir_path) => return Ok(Some(dir_path)), + Err(err) if err.kind() == std::io::ErrorKind::NotFound => { + // keep searching + } + Err(err) => return Err(err), + } + } + Ok(None) + } + + let parent_dir = file_path.parent().unwrap(); + let Some(start_dir) = canonicalize_first_ancestor_exists( + parent_dir, &self.env, + ) + .map_err(|source| CanonicalizingPkgJsonDirError { + dir_path: parent_dir.to_path_buf(), + source, + })? + else { + return Ok(None); + }; + let start_dir = strip_unc_prefix(start_dir); + for current_dir in start_dir.ancestors() { + let package_json_path = current_dir.join("package.json"); + if let Some(pkg_json) = self.load_package_json(&package_json_path)? { + return Ok(Some(pkg_json)); + } + } + + Ok(None) + } + + pub fn load_package_json( + &self, + package_json_path: &Path, + ) -> Result, PackageJsonLoadError> { + load_pkg_json(self.env.pkg_json_fs(), package_json_path) + } +} + /// Helper to load a package.json file using the thread local cache /// in node_resolver. pub fn load_pkg_json( diff --git a/resolvers/node/resolution.rs b/resolvers/node/resolution.rs index 811583a5ee0474..2d4132f21d2fb2 100644 --- a/resolvers/node/resolution.rs +++ b/resolvers/node/resolution.rs @@ -7,8 +7,6 @@ use std::path::PathBuf; use anyhow::bail; use anyhow::Error as AnyError; use deno_media_type::MediaType; -use deno_package_json::PackageJsonRc; -use deno_path_util::strip_unc_prefix; use deno_path_util::url_from_file_path; use serde_json::Map; use serde_json::Value; @@ -16,8 +14,6 @@ use url::Url; use crate::env::NodeResolverEnv; use crate::errors; -use crate::errors::CanonicalizingPkgJsonDirError; -use crate::errors::ClosestPkgJsonError; use crate::errors::DataUrlReferrerError; use crate::errors::FinalizeResolutionError; use crate::errors::InvalidModuleSpecifierError; @@ -32,7 +28,6 @@ use crate::errors::PackageExportsResolveError; use crate::errors::PackageImportNotDefinedError; use crate::errors::PackageImportsResolveError; use crate::errors::PackageImportsResolveErrorKind; -use crate::errors::PackageJsonLoadError; use crate::errors::PackagePathNotExportedError; use crate::errors::PackageResolveError; use crate::errors::PackageSubpathResolveError; @@ -50,6 +45,7 @@ use crate::errors::UnsupportedDirImportError; use crate::errors::UnsupportedEsmUrlSchemeError; use crate::errors::UrlToNodeResolutionError; use crate::NpmResolverRc; +use crate::PackageJsonResolverRc; use crate::PathClean; use deno_package_json::PackageJson; @@ -137,11 +133,20 @@ pub type NodeResolverRc = crate::sync::MaybeArc>; pub struct NodeResolver { env: TEnv, npm_resolver: NpmResolverRc, + pkg_json_resolver: PackageJsonResolverRc, } impl NodeResolver { - pub fn new(env: TEnv, npm_resolver: NpmResolverRc) -> Self { - Self { env, npm_resolver } + pub fn new( + env: TEnv, + npm_resolver: NpmResolverRc, + pkg_json_resolver: PackageJsonResolverRc, + ) -> Self { + Self { + env, + npm_resolver, + pkg_json_resolver, + } } pub fn in_npm_package(&self, specifier: &Url) -> bool { @@ -236,6 +241,7 @@ impl NodeResolver { })?) } else if specifier.starts_with('#') { let pkg_config = self + .pkg_json_resolver .get_closest_package_json(referrer) .map_err(PackageImportsResolveErrorKind::ClosestPkgJson) .map_err(|err| PackageImportsResolveError(Box::new(err)))?; @@ -356,7 +362,9 @@ impl NodeResolver { package_folder: &Path, ) -> Result, ResolveBinaryCommandsError> { let pkg_json_path = package_folder.join("package.json"); - let Some(package_json) = self.load_package_json(&pkg_json_path)? else { + let Some(package_json) = + self.pkg_json_resolver.load_package_json(&pkg_json_path)? + else { return Ok(Vec::new()); }; @@ -383,7 +391,9 @@ impl NodeResolver { sub_path: Option<&str>, ) -> Result { let pkg_json_path = package_folder.join("package.json"); - let Some(package_json) = self.load_package_json(&pkg_json_path)? else { + let Some(package_json) = + self.pkg_json_resolver.load_package_json(&pkg_json_path)? + else { return Err(ResolvePkgJsonBinExportError::MissingPkgJson { pkg_json_path, }); @@ -410,7 +420,8 @@ impl NodeResolver { if url_str.starts_with("http") || url_str.ends_with(".json") { Ok(NodeResolution::Esm(url)) } else if url_str.ends_with(".js") || url_str.ends_with(".d.ts") { - let maybe_package_config = self.get_closest_package_json(&url)?; + let maybe_package_config = + self.pkg_json_resolver.get_closest_package_json(&url)?; match maybe_package_config { Some(c) if c.typ == "module" => Ok(NodeResolution::Esm(url)), Some(_) => Ok(NodeResolution::CommonJs(url)), @@ -1101,7 +1112,9 @@ impl NodeResolver { let (package_name, package_subpath, _is_scoped) = parse_npm_pkg_name(specifier, referrer)?; - if let Some(package_config) = self.get_closest_package_json(referrer)? { + if let Some(package_config) = + self.pkg_json_resolver.get_closest_package_json(referrer)? + { // ResolveSelf if package_config.name.as_ref() == Some(&package_name) { if let Some(exports) = &package_config.exports { @@ -1216,7 +1229,10 @@ impl NodeResolver { mode: NodeResolutionMode, ) -> Result { let package_json_path = package_dir_path.join("package.json"); - match self.load_package_json(&package_json_path)? { + match self + .pkg_json_resolver + .load_package_json(&package_json_path)? + { Some(pkg_json) => self.resolve_package_subpath( &pkg_json, package_subpath, @@ -1337,70 +1353,6 @@ impl NodeResolver { } } - pub fn get_closest_package_json( - &self, - url: &Url, - ) -> Result, ClosestPkgJsonError> { - let Ok(file_path) = deno_path_util::url_to_file_path(url) else { - return Ok(None); - }; - self.get_closest_package_json_from_path(&file_path) - } - - pub fn get_closest_package_json_from_path( - &self, - file_path: &Path, - ) -> Result, ClosestPkgJsonError> { - // we use this for deno compile using byonm because the script paths - // won't be in virtual file system, but the package.json paths will be - fn canonicalize_first_ancestor_exists( - dir_path: &Path, - env: &dyn NodeResolverEnv, - ) -> Result, std::io::Error> { - for ancestor in dir_path.ancestors() { - match env.realpath_sync(ancestor) { - Ok(dir_path) => return Ok(Some(dir_path)), - Err(err) if err.kind() == std::io::ErrorKind::NotFound => { - // keep searching - } - Err(err) => return Err(err), - } - } - Ok(None) - } - - let parent_dir = file_path.parent().unwrap(); - let Some(start_dir) = canonicalize_first_ancestor_exists( - parent_dir, &self.env, - ) - .map_err(|source| CanonicalizingPkgJsonDirError { - dir_path: parent_dir.to_path_buf(), - source, - })? - else { - return Ok(None); - }; - let start_dir = strip_unc_prefix(start_dir); - for current_dir in start_dir.ancestors() { - let package_json_path = current_dir.join("package.json"); - if let Some(pkg_json) = self.load_package_json(&package_json_path)? { - return Ok(Some(pkg_json)); - } - } - - Ok(None) - } - - pub fn load_package_json( - &self, - package_json_path: &Path, - ) -> Result, PackageJsonLoadError> { - crate::package_json::load_pkg_json( - self.env.pkg_json_fs(), - package_json_path, - ) - } - pub(super) fn legacy_main_resolve( &self, package_json: &PackageJson, From 63acd1a5335f39546afb664ccf18af674ea6b1f8 Mon Sep 17 00:00:00 2001 From: David Sherret Date: Tue, 29 Oct 2024 10:22:20 -0400 Subject: [PATCH 16/42] Use fs more --- cli/cache/mod.rs | 10 +++++-- cli/graph_util.rs | 1 + cli/module_loader.rs | 5 +++- cli/node.rs | 12 +++++---- cli/resolver.rs | 63 +++++++++++++++++++++----------------------- cli/tsc/mod.rs | 13 ++++++--- 6 files changed, 59 insertions(+), 45 deletions(-) diff --git a/cli/cache/mod.rs b/cli/cache/mod.rs index 5dc342509df7fb..27e77b121fe3b6 100644 --- a/cli/cache/mod.rs +++ b/cli/cache/mod.rs @@ -21,6 +21,7 @@ use deno_graph::source::CacheInfo; use deno_graph::source::LoadFuture; use deno_graph::source::LoadResponse; use deno_graph::source::Loader; +use deno_runtime::deno_fs; use deno_runtime::deno_permissions::PermissionsContainer; use std::collections::HashMap; use std::path::Path; @@ -183,6 +184,7 @@ pub struct FetchCacherOptions { pub struct FetchCacher { pub file_header_overrides: HashMap>, file_fetcher: Arc, + fs: Arc, global_http_cache: Arc, npm_resolver: Arc, module_info_cache: Arc, @@ -194,6 +196,7 @@ pub struct FetchCacher { impl FetchCacher { pub fn new( file_fetcher: Arc, + fs: Arc, global_http_cache: Arc, npm_resolver: Arc, module_info_cache: Arc, @@ -201,6 +204,7 @@ impl FetchCacher { ) -> Self { Self { file_fetcher, + fs, global_http_cache, npm_resolver, module_info_cache, @@ -265,8 +269,10 @@ impl Loader for FetchCacher { // symlinked to `/my-project-2/node_modules`), so first we checked if the path // is in a node_modules dir to avoid needlessly canonicalizing, then now compare // against the canonicalized specifier. - let specifier = - crate::node::resolve_specifier_into_node_modules(specifier); + let specifier = crate::node::resolve_specifier_into_node_modules( + specifier, + self.fs.as_ref(), + ); if self.npm_resolver.in_npm_package(&specifier) { return Box::pin(futures::future::ready(Ok(Some( LoadResponse::External { specifier }, diff --git a/cli/graph_util.rs b/cli/graph_util.rs index 291372b0c7ba9e..1b9e176695e80a 100644 --- a/cli/graph_util.rs +++ b/cli/graph_util.rs @@ -692,6 +692,7 @@ impl ModuleGraphBuilder { ) -> cache::FetchCacher { cache::FetchCacher::new( self.file_fetcher.clone(), + self.fs.clone(), self.global_http_cache.clone(), self.npm_resolver.clone(), self.module_info_cache.clone(), diff --git a/cli/module_loader.rs b/cli/module_loader.rs index fddf5d926b67e2..290de4c9786f55 100644 --- a/cli/module_loader.rs +++ b/cli/module_loader.rs @@ -541,7 +541,10 @@ impl Some(Module::Js(module)) => module.specifier.clone(), Some(Module::Json(module)) => module.specifier.clone(), Some(Module::External(module)) => { - node::resolve_specifier_into_node_modules(&module.specifier) + node::resolve_specifier_into_node_modules( + &module.specifier, + self.shared.fs.as_ref(), + ) } None => specifier.into_owned(), }; diff --git a/cli/node.rs b/cli/node.rs index 9bc97d0751b8aa..9dd17a636bfe5a 100644 --- a/cli/node.rs +++ b/cli/node.rs @@ -7,6 +7,8 @@ use deno_ast::MediaType; use deno_ast::ModuleSpecifier; use deno_core::error::AnyError; use deno_graph::ParsedSourceStore; +use deno_path_util::url_from_file_path; +use deno_path_util::url_to_file_path; use deno_runtime::deno_fs; use deno_runtime::deno_node::DenoFsNodeResolverEnv; use node_resolver::analyze::CjsAnalysis as ExtNodeCjsAnalysis; @@ -20,7 +22,7 @@ use crate::cache::CacheDBHash; use crate::cache::NodeAnalysisCache; use crate::cache::ParsedSourceCache; use crate::resolver::CjsTracker; -use crate::util::fs::canonicalize_path_maybe_not_exists; +use crate::util::fs::canonicalize_path_maybe_not_exists_with_fs; pub type CliNodeCodeTranslator = NodeCodeTranslator; @@ -33,14 +35,14 @@ pub type CliNodeCodeTranslator = /// because the node_modules folder might not exist at that time. pub fn resolve_specifier_into_node_modules( specifier: &ModuleSpecifier, + fs: &dyn deno_fs::FileSystem, ) -> ModuleSpecifier { - specifier - .to_file_path() + url_to_file_path(specifier) .ok() // this path might not exist at the time the graph is being created // because the node_modules folder might not yet exist - .and_then(|path| canonicalize_path_maybe_not_exists(&path).ok()) - .and_then(|path| ModuleSpecifier::from_file_path(path).ok()) + .and_then(|path| canonicalize_path_maybe_not_exists_with_fs(&path, fs).ok()) + .and_then(|path| url_from_file_path(&path).ok()) .unwrap_or_else(|| specifier.clone()) } diff --git a/cli/resolver.rs b/cli/resolver.rs index 7d8003e0f9631c..a0505476d87985 100644 --- a/cli/resolver.rs +++ b/cli/resolver.rs @@ -303,8 +303,10 @@ impl CliNodeResolver { // so canoncalize then check if it's in the node_modules directory. // If so, check if we need to store this specifier as being a CJS // resolution. - let specifier = - crate::node::resolve_specifier_into_node_modules(specifier); + let specifier = crate::node::resolve_specifier_into_node_modules( + specifier, + self.fs.as_ref(), + ); if self.in_npm_package(&specifier) { let resolution = self.node_resolver.url_to_node_resolution(specifier)?; @@ -465,41 +467,36 @@ impl CjsTracker { } let media_type = MediaType::from_specifier(specifier); - if matches!(media_type, MediaType::Cjs | MediaType::Cts) - || self.set.contains(specifier) - { - return Ok(true); - } - if self.unstable_detect_cjs { - let is_pkg_json_type_affected = match media_type { - MediaType::JavaScript - | MediaType::Jsx - | MediaType::TypeScript - | MediaType::Tsx - | MediaType::Dts => true, - MediaType::Dmts - | MediaType::Dcts - | MediaType::Css - | MediaType::Json - | MediaType::Wasm - | MediaType::Mjs - | MediaType::Cjs - | MediaType::Mts - | MediaType::Cts - | MediaType::SourceMap - | MediaType::Unknown => false, - }; - if is_pkg_json_type_affected { - if let Some(pkg_json) = - self.pkg_json_resolver.get_closest_package_json(specifier)? - { - return Ok(pkg_json.typ == "commonjs"); + match media_type { + MediaType::Mts | MediaType::Mjs | MediaType::Dmts => Ok(false), + MediaType::Cjs | MediaType::Cts | MediaType::Dcts => Ok(true), + MediaType::JavaScript + | MediaType::Jsx + | MediaType::TypeScript + | MediaType::Tsx + | MediaType::Dts + // treat these as unknown + | MediaType::Json + | MediaType::Css + | MediaType::Wasm + | MediaType::SourceMap + | MediaType::Unknown => { + if self.set.contains(specifier) { + return Ok(true); } + + if self.unstable_detect_cjs { + if let Some(pkg_json) = + self.pkg_json_resolver.get_closest_package_json(specifier)? + { + return Ok(pkg_json.typ == "commonjs"); + } + } + + Ok(false) } } - - Ok(false) } pub fn insert(&self, specifier: ModuleSpecifier) { diff --git a/cli/tsc/mod.rs b/cli/tsc/mod.rs index 7e9e2eb6d2674e..69849032f969fc 100644 --- a/cli/tsc/mod.rs +++ b/cli/tsc/mod.rs @@ -32,6 +32,7 @@ use deno_graph::GraphKind; use deno_graph::Module; use deno_graph::ModuleGraph; use deno_graph::ResolutionResolved; +use deno_runtime::deno_fs; use deno_runtime::deno_node::NodeResolver; use deno_semver::npm::NpmPackageReqReference; use node_resolver::errors::NodeJsErrorCode; @@ -576,8 +577,10 @@ fn op_load_inner( Module::Npm(_) | Module::Node(_) => None, Module::External(module) => { // means it's Deno code importing an npm module - let specifier = - node::resolve_specifier_into_node_modules(&module.specifier); + let specifier = node::resolve_specifier_into_node_modules( + &module.specifier, + &deno_fs::RealFs, + ); Some(Cow::Owned(load_from_node_modules( &specifier, state.maybe_npm.as_ref().map(|n| n.node_resolver.as_ref()), @@ -831,8 +834,10 @@ fn resolve_graph_specifier_types( Some(Module::External(module)) => { // we currently only use "External" for when the module is in an npm package Ok(state.maybe_npm.as_ref().map(|npm| { - let specifier = - node::resolve_specifier_into_node_modules(&module.specifier); + let specifier = node::resolve_specifier_into_node_modules( + &module.specifier, + &deno_fs::RealFs, + ); NodeResolution::into_specifier_and_media_type( npm.node_resolver.url_to_node_resolution(specifier).ok(), ) From 84f1f7083d726ec300306dfcd9ce71bbfa1a181d Mon Sep 17 00:00:00 2001 From: David Sherret Date: Tue, 29 Oct 2024 11:13:45 -0400 Subject: [PATCH 17/42] better cjs/esm tracking --- cli/factory.rs | 3 +-- cli/module_loader.rs | 9 ++++++--- cli/node.rs | 11 +++++++++-- cli/resolver.rs | 44 +++++++++++++++++++++---------------------- cli/standalone/mod.rs | 3 +-- 5 files changed, 39 insertions(+), 31 deletions(-) diff --git a/cli/factory.rs b/cli/factory.rs index 2a0495abc74a74..fcad09058e91be 100644 --- a/cli/factory.rs +++ b/cli/factory.rs @@ -845,9 +845,8 @@ impl CliFactory { cli_npm_resolver.clone(), NpmModuleLoader::new( self.cjs_tracker()?.clone(), - node_code_translator.clone(), fs.clone(), - cli_node_resolver.clone(), + node_code_translator.clone(), ), self.parsed_source_cache().clone(), self.resolver().await?.clone(), diff --git a/cli/module_loader.rs b/cli/module_loader.rs index 290de4c9786f55..fb47eed4ad4c4d 100644 --- a/cli/module_loader.rs +++ b/cli/module_loader.rs @@ -416,7 +416,7 @@ impl if let Some(code_source) = self.load_prepared_module(specifier).await? { return Ok(code_source); } - if self.shared.npm_module_loader.if_in_npm_package(specifier) { + if self.shared.node_resolver.in_npm_package(specifier) { return self .shared .npm_module_loader @@ -583,7 +583,10 @@ impl specifier, media_type, source, - }) => self.load_cjs(specifier, media_type, source).await.map(Some), + }) => self + .load_maybe_cjs(specifier, media_type, source) + .await + .map(Some), None => Ok(None), } } @@ -713,7 +716,7 @@ impl } } - async fn load_cjs( + async fn load_maybe_cjs( &self, specifier: &ModuleSpecifier, media_type: MediaType, diff --git a/cli/node.rs b/cli/node.rs index 9dd17a636bfe5a..c0bfe9533906c8 100644 --- a/cli/node.rs +++ b/cli/node.rs @@ -4,6 +4,7 @@ use std::borrow::Cow; use std::sync::Arc; use deno_ast::MediaType; +use deno_ast::ModuleKind; use deno_ast::ModuleSpecifier; use deno_core::error::AnyError; use deno_graph::ParsedSourceStore; @@ -99,7 +100,8 @@ impl CliCjsCodeAnalyzer { }); } - let treat_as_cjs = self.cjs_tracker.treat_as_cjs(&specifier)?; + let cjs_tracker = self.cjs_tracker.clone(); + let treat_as_cjs = cjs_tracker.treat_as_cjs(&specifier)?; let analysis = if treat_as_cjs { let maybe_parsed_source = self .parsed_source_cache @@ -121,7 +123,12 @@ impl CliCjsCodeAnalyzer { maybe_syntax: None, }) })?; - if parsed_source.is_script() && treat_as_cjs { + let is_cjs = parsed_source.is_script() && treat_as_cjs; + cjs_tracker.mark_kind( + parsed_source.specifier().clone(), + ModuleKind::from_is_cjs(is_cjs), + ); + if is_cjs { let analysis = parsed_source.analyze_cjs(); Ok(CliCjsAnalysis::Cjs { exports: analysis.exports, diff --git a/cli/resolver.rs b/cli/resolver.rs index a0505476d87985..22b1040d8fe4b0 100644 --- a/cli/resolver.rs +++ b/cli/resolver.rs @@ -4,6 +4,7 @@ use async_trait::async_trait; use dashmap::DashMap; use dashmap::DashSet; use deno_ast::MediaType; +use deno_ast::ModuleKind; use deno_config::workspace::MappedResolution; use deno_config::workspace::MappedResolutionDiagnostic; use deno_config::workspace::MappedResolutionError; @@ -329,46 +330,45 @@ impl CliNodeResolver { &self, resolution: NodeResolution, ) -> NodeResolution { - if let NodeResolution::CommonJs(specifier) = &resolution { - // remember that this was a common js resolution - self.mark_cjs_resolution(specifier.clone()); + match &resolution { + NodeResolution::CommonJs(specifier) => { + // remember that this was a common js resolution + self + .cjs_tracker + .mark_kind(specifier.clone(), ModuleKind::Cjs); + } + NodeResolution::Esm(specifier) => { + self + .cjs_tracker + .mark_kind(specifier.clone(), ModuleKind::Esm); + } + NodeResolution::BuiltIn(_) => {} } resolution } - - pub fn mark_cjs_resolution(&self, specifier: ModuleSpecifier) { - self.cjs_tracker.insert(specifier); - } } // todo(dsherret): move to module_loader.rs #[derive(Clone)] pub struct NpmModuleLoader { cjs_tracker: Arc, - node_code_translator: Arc, fs: Arc, - node_resolver: Arc, + node_code_translator: Arc, } impl NpmModuleLoader { pub fn new( cjs_tracker: Arc, - node_code_translator: Arc, fs: Arc, - node_resolver: Arc, + node_code_translator: Arc, ) -> Self { Self { cjs_tracker, node_code_translator, fs, - node_resolver, } } - pub fn if_in_npm_package(&self, specifier: &ModuleSpecifier) -> bool { - self.node_resolver.in_npm_package(specifier) - } - pub async fn load( &self, specifier: &ModuleSpecifier, @@ -442,7 +442,7 @@ pub struct CjsTrackerOptions { #[derive(Debug)] pub struct CjsTracker { unstable_detect_cjs: bool, - set: DashSet, + known: DashMap, pkg_json_resolver: Arc, } @@ -453,7 +453,7 @@ impl CjsTracker { ) -> Self { Self { unstable_detect_cjs: options.unstable_detect_cjs, - set: DashSet::new(), + known: Default::default(), pkg_json_resolver, } } @@ -482,8 +482,8 @@ impl CjsTracker { | MediaType::Wasm | MediaType::SourceMap | MediaType::Unknown => { - if self.set.contains(specifier) { - return Ok(true); + if let Some(value) = self.known.get(specifier) { + return Ok(value.is_cjs()); } if self.unstable_detect_cjs { @@ -499,8 +499,8 @@ impl CjsTracker { } } - pub fn insert(&self, specifier: ModuleSpecifier) { - self.set.insert(specifier); + pub fn mark_kind(&self, specifier: ModuleSpecifier, kind: ModuleKind) { + self.known.insert(specifier, kind); } } diff --git a/cli/standalone/mod.rs b/cli/standalone/mod.rs index 67b525cba5e396..cc8bd2c0f67811 100644 --- a/cli/standalone/mod.rs +++ b/cli/standalone/mod.rs @@ -671,9 +671,8 @@ pub async fn run(data: StandaloneData) -> Result { node_resolver: cli_node_resolver.clone(), npm_module_loader: Arc::new(NpmModuleLoader::new( cjs_tracker, - node_code_translator, fs.clone(), - cli_node_resolver, + node_code_translator, )), npm_resolver: npm_resolver.clone(), workspace_resolver, From ebaa01f35f77327a7992ff8a6e6073b1e901026e Mon Sep 17 00:00:00 2001 From: David Sherret Date: Tue, 29 Oct 2024 13:57:24 -0400 Subject: [PATCH 18/42] compiling large refactor --- cli/args/mod.rs | 9 + cli/cache/mod.rs | 9 +- cli/factory.rs | 198 ++++++++++++------ cli/graph_util.rs | 60 +++--- cli/lsp/analysis.rs | 7 +- cli/lsp/resolver.rs | 64 ++++-- cli/module_loader.rs | 6 +- cli/node.rs | 14 +- cli/npm/byonm.rs | 6 +- cli/npm/managed/cache/mod.rs | 12 +- cli/npm/managed/mod.rs | 68 ++++-- cli/npm/managed/resolvers/common.rs | 4 - cli/npm/managed/resolvers/global.rs | 6 +- cli/npm/managed/resolvers/local.rs | 4 - cli/npm/mod.rs | 26 ++- cli/resolver.rs | 52 ++++- cli/standalone/binary.rs | 15 +- cli/standalone/mod.rs | 167 +++++++++------ cli/tools/coverage/mod.rs | 16 +- cli/tools/run/hmr.rs | 6 +- ext/node/ops/require.rs | 13 +- resolvers/deno/fs.rs | 10 - resolvers/deno/npm/byonm.rs | 41 ++-- resolvers/deno/npm/mod.rs | 1 + resolvers/node/analyze.rs | 21 +- resolvers/node/errors.rs | 15 ++ resolvers/node/lib.rs | 3 +- resolvers/node/npm.rs | 6 + resolvers/node/package_json.rs | 34 ++- resolvers/node/resolution.rs | 64 ++++-- runtime/errors.rs | 1 + .../@denotest/type-commonjs/1.0.0/index.js | 6 +- .../npm/require_type_commonjs/__test__.jsonc | 3 +- .../specs/npm/require_type_commonjs/main.out | 5 +- tests/specs/npm/require_type_commonjs/main.ts | 3 +- 35 files changed, 638 insertions(+), 337 deletions(-) diff --git a/cli/args/mod.rs b/cli/args/mod.rs index 9c874d0974ce22..bf966f794bea1c 100644 --- a/cli/args/mod.rs +++ b/cli/args/mod.rs @@ -1597,6 +1597,15 @@ impl CliOptions { } pub fn use_byonm(&self) -> bool { + if matches!( + self.sub_command(), + DenoSubcommand::Install(_) + | DenoSubcommand::Add(_) + | DenoSubcommand::Remove(_) + ) { + // For `deno install/add/remove` we want to force the managed resolver so it can set up `node_modules/` directory. + return false; + } if self.node_modules_dir().ok().flatten().is_none() && self.maybe_node_modules_folder.is_some() && self diff --git a/cli/cache/mod.rs b/cli/cache/mod.rs index 27e77b121fe3b6..87892fb9080d41 100644 --- a/cli/cache/mod.rs +++ b/cli/cache/mod.rs @@ -23,6 +23,7 @@ use deno_graph::source::LoadResponse; use deno_graph::source::Loader; use deno_runtime::deno_fs; use deno_runtime::deno_permissions::PermissionsContainer; +use node_resolver::InNpmPackageChecker; use std::collections::HashMap; use std::path::Path; use std::path::PathBuf; @@ -186,7 +187,7 @@ pub struct FetchCacher { file_fetcher: Arc, fs: Arc, global_http_cache: Arc, - npm_resolver: Arc, + in_npm_pkg_checker: Arc, module_info_cache: Arc, permissions: PermissionsContainer, is_deno_publish: bool, @@ -198,7 +199,7 @@ impl FetchCacher { file_fetcher: Arc, fs: Arc, global_http_cache: Arc, - npm_resolver: Arc, + in_npm_pkg_checker: Arc, module_info_cache: Arc, options: FetchCacherOptions, ) -> Self { @@ -206,7 +207,7 @@ impl FetchCacher { file_fetcher, fs, global_http_cache, - npm_resolver, + in_npm_pkg_checker, module_info_cache, file_header_overrides: options.file_header_overrides, permissions: options.permissions, @@ -273,7 +274,7 @@ impl Loader for FetchCacher { specifier, self.fs.as_ref(), ); - if self.npm_resolver.in_npm_package(&specifier) { + if self.in_npm_pkg_checker.in_npm_package(&specifier) { return Box::pin(futures::future::ready(Ok(Some( LoadResponse::External { specifier }, )))); diff --git a/cli/factory.rs b/cli/factory.rs index fcad09058e91be..6cfedf97cb2d91 100644 --- a/cli/factory.rs +++ b/cli/factory.rs @@ -11,6 +11,7 @@ use crate::args::StorageKeyResolver; use crate::args::TsConfigType; use crate::cache::Caches; use crate::cache::CodeCache; +use crate::cache::DenoCacheEnvFsAdapter; use crate::cache::DenoDir; use crate::cache::DenoDirProvider; use crate::cache::EmitCache; @@ -32,11 +33,14 @@ use crate::module_loader::ModuleLoadPreparer; use crate::node::CliCjsCodeAnalyzer; use crate::node::CliNodeCodeTranslator; use crate::npm::create_cli_npm_resolver; +use crate::npm::create_in_npm_pkg_checker; use crate::npm::CliByonmNpmResolverCreateOptions; +use crate::npm::CliManagedInNpmPkgCheckerCreateOptions; +use crate::npm::CliManagedNpmResolverCreateOptions; use crate::npm::CliNpmResolver; use crate::npm::CliNpmResolverCreateOptions; -use crate::npm::CliNpmResolverManagedCreateOptions; use crate::npm::CliNpmResolverManagedSnapshotOption; +use crate::npm::CreateInNpmPkgCheckerOptions; use crate::resolver::CjsTracker; use crate::resolver::CjsTrackerOptions; use crate::resolver::CliDenoResolverFs; @@ -59,12 +63,14 @@ use crate::worker::CliMainWorkerFactory; use crate::worker::CliMainWorkerOptions; use std::path::PathBuf; +use deno_cache_dir::npm::NpmCacheDir; use deno_config::workspace::PackageJsonDepResolution; use deno_config::workspace::WorkspaceResolver; use deno_core::error::AnyError; use deno_core::futures::FutureExt; use deno_core::FeatureChecker; +use deno_resolver::npm::ByonmInNpmPackageChecker; use deno_runtime::deno_fs; use deno_runtime::deno_node::DenoFsNodeResolverEnv; use deno_runtime::deno_node::NodeResolver; @@ -78,6 +84,7 @@ use deno_runtime::inspector_server::InspectorServer; use deno_runtime::permissions::RuntimePermissionDescriptorParser; use log::warn; use node_resolver::analyze::NodeCodeTranslator; +use node_resolver::InNpmPackageChecker; use once_cell::sync::OnceCell; use std::future::Future; use std::sync::Arc; @@ -165,39 +172,41 @@ impl Deferred { #[derive(Default)] struct CliFactoryServices { - cli_options: Deferred>, + blob_store: Deferred>, caches: Deferred>, + cjs_tracker: Deferred>, + cli_node_resolver: Deferred>, + cli_options: Deferred>, + code_cache: Deferred>, + emit_cache: Deferred>, + emitter: Deferred>, + feature_checker: Deferred>, file_fetcher: Deferred>, + fs: Deferred>, global_http_cache: Deferred>, http_cache: Deferred>, http_client_provider: Deferred>, - emit_cache: Deferred>, - emitter: Deferred>, - fs: Deferred>, + in_npm_pkg_checker: Deferred>, main_graph_container: Deferred>, - maybe_inspector_server: Deferred>>, - root_cert_store_provider: Deferred>, - blob_store: Deferred>, - module_info_cache: Deferred>, - parsed_source_cache: Deferred>, - resolver: Deferred>, maybe_file_watcher_reporter: Deferred>, + maybe_inspector_server: Deferred>>, module_graph_builder: Deferred>, module_graph_creator: Deferred>, + module_info_cache: Deferred>, module_load_preparer: Deferred>, node_code_translator: Deferred>, node_resolver: Deferred>, + npm_cache_dir: Deferred>, npm_resolver: Deferred>, + parsed_source_cache: Deferred>, permission_desc_parser: Deferred>, pkg_json_resolver: Deferred>, + resolver: Deferred>, + root_cert_store_provider: Deferred>, root_permissions_container: Deferred, sloppy_imports_resolver: Deferred>>, text_only_progress_bar: Deferred, type_checker: Deferred>, - cjs_tracker: Deferred>, - cli_node_resolver: Deferred>, - feature_checker: Deferred>, - code_cache: Deferred>, workspace_resolver: Deferred>, } @@ -354,56 +363,112 @@ impl CliFactory { self.services.fs.get_or_init(|| Arc::new(deno_fs::RealFs)) } + pub fn in_npm_pkg_checker( + &self, + ) -> Result<&Arc, AnyError> { + self.services.in_npm_pkg_checker.get_or_try_init(|| { + let cli_options = self.cli_options()?; + let options = if cli_options.use_byonm() { + CreateInNpmPkgCheckerOptions::Byonm + } else { + CreateInNpmPkgCheckerOptions::Managed( + CliManagedInNpmPkgCheckerCreateOptions { + root_cache_dir_url: self.npm_cache_dir()?.root_dir_url(), + maybe_node_modules_path: cli_options + .node_modules_dir_path() + .map(|p| p.as_path()), + }, + ) + }; + Ok(create_in_npm_pkg_checker(options)) + }) + } + + pub fn npm_cache_dir(&self) -> Result<&Arc, AnyError> { + self.services.npm_cache_dir.get_or_try_init(|| { + let fs = self.fs(); + let global_path = self.deno_dir()?.npm_folder_path(); + let cli_options = self.cli_options()?; + Ok(Arc::new(NpmCacheDir::new( + &DenoCacheEnvFsAdapter(fs.as_ref()), + global_path, + cli_options.npmrc().get_all_known_registries_urls(), + ))) + }) + } + pub async fn npm_resolver( &self, ) -> Result<&Arc, AnyError> { self .services .npm_resolver - .get_or_try_init_async(async { - let fs = self.fs(); - let cli_options = self.cli_options()?; - // For `deno install` we want to force the managed resolver so it can set up `node_modules/` directory. - create_cli_npm_resolver(if cli_options.use_byonm() && !matches!(cli_options.sub_command(), DenoSubcommand::Install(_) | DenoSubcommand::Add(_) | DenoSubcommand::Remove(_)) { - CliNpmResolverCreateOptions::Byonm(CliByonmNpmResolverCreateOptions { - fs: CliDenoResolverFs(fs.clone()), - root_node_modules_dir: Some(match cli_options.node_modules_dir_path() { - Some(node_modules_path) => node_modules_path.to_path_buf(), - // path needs to be canonicalized for node resolution - // (node_modules_dir_path above is already canonicalized) - None => canonicalize_path_maybe_not_exists(cli_options.initial_cwd())? - .join("node_modules"), - }), - }) - } else { - CliNpmResolverCreateOptions::Managed(CliNpmResolverManagedCreateOptions { - snapshot: match cli_options.resolve_npm_resolution_snapshot()? { - Some(snapshot) => { - CliNpmResolverManagedSnapshotOption::Specified(Some(snapshot)) - } - None => match cli_options.maybe_lockfile() { - Some(lockfile) => { - CliNpmResolverManagedSnapshotOption::ResolveFromLockfile( - lockfile.clone(), - ) - } - None => CliNpmResolverManagedSnapshotOption::Specified(None), + .get_or_try_init_async( + async { + let fs = self.fs(); + let cli_options = self.cli_options()?; + create_cli_npm_resolver(if cli_options.use_byonm() { + CliNpmResolverCreateOptions::Byonm( + CliByonmNpmResolverCreateOptions { + fs: CliDenoResolverFs(fs.clone()), + pkg_json_resolver: self.pkg_json_resolver().clone(), + root_node_modules_dir: Some( + match cli_options.node_modules_dir_path() { + Some(node_modules_path) => node_modules_path.to_path_buf(), + // path needs to be canonicalized for node resolution + // (node_modules_dir_path above is already canonicalized) + None => canonicalize_path_maybe_not_exists( + cli_options.initial_cwd(), + )? + .join("node_modules"), + }, + ), }, - }, - maybe_lockfile: cli_options.maybe_lockfile().cloned(), - fs: fs.clone(), - http_client_provider: self.http_client_provider().clone(), - npm_global_cache_dir: self.deno_dir()?.npm_folder_path(), - cache_setting: cli_options.cache_setting(), - text_only_progress_bar: self.text_only_progress_bar().clone(), - maybe_node_modules_path: cli_options.node_modules_dir_path().cloned(), - npm_install_deps_provider: Arc::new(NpmInstallDepsProvider::from_workspace(cli_options.workspace())), - npm_system_info: cli_options.npm_system_info(), - npmrc: cli_options.npmrc().clone(), - lifecycle_scripts: cli_options.lifecycle_scripts_config(), + ) + } else { + CliNpmResolverCreateOptions::Managed( + CliManagedNpmResolverCreateOptions { + snapshot: match cli_options.resolve_npm_resolution_snapshot()? { + Some(snapshot) => { + CliNpmResolverManagedSnapshotOption::Specified(Some( + snapshot, + )) + } + None => match cli_options.maybe_lockfile() { + Some(lockfile) => { + CliNpmResolverManagedSnapshotOption::ResolveFromLockfile( + lockfile.clone(), + ) + } + None => { + CliNpmResolverManagedSnapshotOption::Specified(None) + } + }, + }, + maybe_lockfile: cli_options.maybe_lockfile().cloned(), + fs: fs.clone(), + http_client_provider: self.http_client_provider().clone(), + npm_cache_dir: self.npm_cache_dir()?.clone(), + cache_setting: cli_options.cache_setting(), + text_only_progress_bar: self.text_only_progress_bar().clone(), + maybe_node_modules_path: cli_options + .node_modules_dir_path() + .cloned(), + npm_install_deps_provider: Arc::new( + NpmInstallDepsProvider::from_workspace( + cli_options.workspace(), + ), + ), + npm_system_info: cli_options.npm_system_info(), + npmrc: cli_options.npmrc().clone(), + lifecycle_scripts: cli_options.lifecycle_scripts_config(), + }, + ) }) - }).await - }.boxed_local()) + .await + } + .boxed_local(), + ) .await } @@ -559,6 +624,7 @@ impl CliFactory { async { Ok(Arc::new(NodeResolver::new( DenoFsNodeResolverEnv::new(self.fs().clone()), + self.in_npm_pkg_checker()?.clone(), self.npm_resolver().await?.clone().into_npm_resolver(), self.pkg_json_resolver().clone(), ))) @@ -589,8 +655,10 @@ impl CliFactory { Ok(Arc::new(NodeCodeTranslator::new( cjs_esm_analyzer, DenoFsNodeResolverEnv::new(self.fs().clone()), + self.in_npm_pkg_checker()?.clone(), node_resolver, self.npm_resolver().await?.clone().into_npm_resolver(), + self.pkg_json_resolver().clone(), ))) }) .await @@ -630,17 +698,18 @@ impl CliFactory { .get_or_try_init_async(async { let cli_options = self.cli_options()?; Ok(Arc::new(ModuleGraphBuilder::new( - cli_options.clone(), self.caches()?.clone(), + cli_options.clone(), + self.file_fetcher()?.clone(), self.fs().clone(), - self.resolver().await?.clone(), - self.npm_resolver().await?.clone(), - self.module_info_cache()?.clone(), - self.parsed_source_cache().clone(), + self.global_http_cache()?.clone(), + self.in_npm_pkg_checker()?.clone(), cli_options.maybe_lockfile().cloned(), self.maybe_file_watcher_reporter().clone(), - self.file_fetcher()?.clone(), - self.global_http_cache()?.clone(), + self.module_info_cache()?.clone(), + self.npm_resolver().await?.clone(), + self.parsed_source_cache().clone(), + self.resolver().await?.clone(), self.root_permissions_container()?.clone(), ))) }) @@ -734,6 +803,7 @@ impl CliFactory { Ok(Arc::new(CliNodeResolver::new( self.cjs_tracker()?.clone(), self.fs().clone(), + self.in_npm_pkg_checker()?.clone(), self.node_resolver().await?.clone(), self.npm_resolver().await?.clone(), ))) diff --git a/cli/graph_util.rs b/cli/graph_util.rs index 1b9e176695e80a..0df22b680e1125 100644 --- a/cli/graph_util.rs +++ b/cli/graph_util.rs @@ -48,6 +48,7 @@ use deno_runtime::deno_permissions::PermissionsContainer; use deno_semver::jsr::JsrDepPackageReq; use deno_semver::package::PackageNv; use import_map::ImportMapError; +use node_resolver::InNpmPackageChecker; use std::collections::HashSet; use std::error::Error; use std::ops::Deref; @@ -377,48 +378,51 @@ pub struct BuildFastCheckGraphOptions<'a> { } pub struct ModuleGraphBuilder { - options: Arc, caches: Arc, + cli_options: Arc, + file_fetcher: Arc, fs: Arc, - resolver: Arc, - npm_resolver: Arc, - module_info_cache: Arc, - parsed_source_cache: Arc, + global_http_cache: Arc, + in_npm_pkg_checker: Arc, lockfile: Option>, maybe_file_watcher_reporter: Option, - file_fetcher: Arc, - global_http_cache: Arc, + module_info_cache: Arc, + npm_resolver: Arc, + parsed_source_cache: Arc, + resolver: Arc, root_permissions_container: PermissionsContainer, } impl ModuleGraphBuilder { #[allow(clippy::too_many_arguments)] pub fn new( - options: Arc, caches: Arc, + cli_options: Arc, + file_fetcher: Arc, fs: Arc, - resolver: Arc, - npm_resolver: Arc, - module_info_cache: Arc, - parsed_source_cache: Arc, + global_http_cache: Arc, + in_npm_pkg_checker: Arc, lockfile: Option>, maybe_file_watcher_reporter: Option, - file_fetcher: Arc, - global_http_cache: Arc, + module_info_cache: Arc, + npm_resolver: Arc, + parsed_source_cache: Arc, + resolver: Arc, root_permissions_container: PermissionsContainer, ) -> Self { Self { - options, caches, + cli_options, + file_fetcher, fs, - resolver, - npm_resolver, - module_info_cache, - parsed_source_cache, + global_http_cache, + in_npm_pkg_checker, lockfile, maybe_file_watcher_reporter, - file_fetcher, - global_http_cache, + module_info_cache, + npm_resolver, + parsed_source_cache, + resolver, root_permissions_container, } } @@ -504,7 +508,7 @@ impl ModuleGraphBuilder { } let maybe_imports = if options.graph_kind.include_types() { - self.options.to_compiler_option_types()? + self.cli_options.to_compiler_option_types()? } else { Vec::new() }; @@ -558,7 +562,7 @@ impl ModuleGraphBuilder { // ensure an "npm install" is done if the user has explicitly // opted into using a node_modules directory if self - .options + .cli_options .node_modules_dir()? .map(|m| m.uses_node_modules_dir()) .unwrap_or(false) @@ -694,13 +698,13 @@ impl ModuleGraphBuilder { self.file_fetcher.clone(), self.fs.clone(), self.global_http_cache.clone(), - self.npm_resolver.clone(), + self.in_npm_pkg_checker.clone(), self.module_info_cache.clone(), cache::FetchCacherOptions { - file_header_overrides: self.options.resolve_file_header_overrides(), + file_header_overrides: self.cli_options.resolve_file_header_overrides(), permissions, is_deno_publish: matches!( - self.options.sub_command(), + self.cli_options.sub_command(), crate::args::DenoSubcommand::Publish { .. } ), }, @@ -727,12 +731,12 @@ impl ModuleGraphBuilder { &self.fs, roots, GraphValidOptions { - kind: if self.options.type_check_mode().is_true() { + kind: if self.cli_options.type_check_mode().is_true() { GraphKind::All } else { GraphKind::CodeOnly }, - check_js: self.options.check_js(), + check_js: self.cli_options.check_js(), exit_integrity_errors: true, }, ) diff --git a/cli/lsp/analysis.rs b/cli/lsp/analysis.rs index d16b427246a3fe..ab5b3ca1ac64d3 100644 --- a/cli/lsp/analysis.rs +++ b/cli/lsp/analysis.rs @@ -336,7 +336,12 @@ impl<'a> TsResponseImportMapper<'a> { .resolver .maybe_managed_npm_resolver(Some(&self.file_referrer)) { - if npm_resolver.in_npm_package(specifier) { + let in_npm_pkg = self + .resolver + .maybe_node_resolver(Some(&self.file_referrer)) + .map(|n| n.in_npm_package(specifier)) + .unwrap_or(false); + if in_npm_pkg { if let Ok(Some(pkg_id)) = npm_resolver.resolve_pkg_id_from_specifier(specifier) { diff --git a/cli/lsp/resolver.rs b/cli/lsp/resolver.rs index 83979608bec2cd..06cfd2b791b4d2 100644 --- a/cli/lsp/resolver.rs +++ b/cli/lsp/resolver.rs @@ -2,6 +2,7 @@ use dashmap::DashMap; use deno_ast::MediaType; +use deno_cache_dir::npm::NpmCacheDir; use deno_cache_dir::HttpCache; use deno_config::workspace::PackageJsonDepResolution; use deno_config::workspace::WorkspaceResolver; @@ -21,9 +22,9 @@ use deno_semver::package::PackageNv; use deno_semver::package::PackageReq; use indexmap::IndexMap; use node_resolver::errors::ClosestPkgJsonError; +use node_resolver::InNpmPackageChecker; use node_resolver::NodeResolution; use node_resolver::NodeResolutionMode; -use node_resolver::NpmResolver; use std::borrow::Cow; use std::collections::BTreeMap; use std::collections::BTreeSet; @@ -37,17 +38,21 @@ use crate::args::create_default_npmrc; use crate::args::CacheSetting; use crate::args::CliLockfile; use crate::args::NpmInstallDepsProvider; +use crate::cache::DenoCacheEnvFsAdapter; use crate::graph_util::CliJsrUrlProvider; use crate::http_util::HttpClientProvider; use crate::lsp::config::Config; use crate::lsp::config::ConfigData; use crate::lsp::logging::lsp_warn; use crate::npm::create_cli_npm_resolver_for_lsp; +use crate::npm::create_in_npm_pkg_checker; use crate::npm::CliByonmNpmResolverCreateOptions; +use crate::npm::CliManagedInNpmPkgCheckerCreateOptions; +use crate::npm::CliManagedNpmResolverCreateOptions; use crate::npm::CliNpmResolver; use crate::npm::CliNpmResolverCreateOptions; -use crate::npm::CliNpmResolverManagedCreateOptions; use crate::npm::CliNpmResolverManagedSnapshotOption; +use crate::npm::CreateInNpmPkgCheckerOptions; use crate::npm::ManagedCliNpmResolver; use crate::resolver::CjsTracker; use crate::resolver::CjsTrackerOptions; @@ -103,6 +108,7 @@ impl LspScopeResolver { config_data.map(|d| d.as_ref()), cache, http_client, + &pkg_json_resolver, ) .await; node_resolver = npm_resolver.as_ref().map(|npm_resolver| { @@ -283,6 +289,14 @@ impl LspResolver { resolver.graph_resolver.create_graph_npm_resolver() } + pub fn maybe_node_resolver( + &self, + file_referrer: Option<&ModuleSpecifier>, + ) -> Option<&Arc> { + let resolver = self.get_scope_resolver(file_referrer); + resolver.node_resolver.as_ref() + } + pub fn maybe_managed_npm_resolver( &self, file_referrer: Option<&ModuleSpecifier>, @@ -368,14 +382,10 @@ impl LspResolver { .contains("/node_modules/") } - let global_npm_resolver = self - .get_scope_resolver(Some(specifier)) - .npm_resolver - .as_ref() - .and_then(|npm_resolver| npm_resolver.as_managed()) - .filter(|r| r.root_node_modules_path().is_none()); - if let Some(npm_resolver) = &global_npm_resolver { - if npm_resolver.in_npm_package(specifier) { + if let Some(node_resolver) = + &self.get_scope_resolver(Some(specifier)).node_resolver + { + if node_resolver.in_npm_package(specifier) { return true; } } @@ -479,11 +489,13 @@ async fn create_npm_resolver( config_data: Option<&ConfigData>, cache: &LspCache, http_client_provider: &Arc, + pkg_json_resolver: &Arc, ) -> Option> { let enable_byonm = config_data.map(|d| d.byonm).unwrap_or(false); let options = if enable_byonm { CliNpmResolverCreateOptions::Byonm(CliByonmNpmResolverCreateOptions { fs: CliDenoResolverFs(Arc::new(deno_fs::RealFs)), + pkg_json_resolver: pkg_json_resolver.clone(), root_node_modules_dir: config_data.and_then(|config_data| { config_data.node_modules_dir.clone().or_else(|| { url_to_file_path(&config_data.scope) @@ -493,7 +505,15 @@ async fn create_npm_resolver( }), }) } else { - CliNpmResolverCreateOptions::Managed(CliNpmResolverManagedCreateOptions { + let npmrc = config_data + .and_then(|d| d.npmrc.clone()) + .unwrap_or_else(create_default_npmrc); + let npm_cache_dir = Arc::new(NpmCacheDir::new( + &DenoCacheEnvFsAdapter(&deno_fs::RealFs), + cache.deno_dir().npm_folder_path(), + npmrc.get_all_known_registries_urls(), + )); + CliNpmResolverCreateOptions::Managed(CliManagedNpmResolverCreateOptions { http_client_provider: http_client_provider.clone(), snapshot: match config_data.and_then(|d| d.lockfile.as_ref()) { Some(lockfile) => { @@ -507,7 +527,7 @@ async fn create_npm_resolver( // updating it. Only the cache request should update the lockfile. maybe_lockfile: None, fs: Arc::new(deno_fs::RealFs), - npm_global_cache_dir: cache.deno_dir().npm_folder_path(), + npm_cache_dir, // Use an "only" cache setting in order to make the // user do an explicit "cache" command and prevent // the cache from being filled with lots of packages while @@ -518,9 +538,7 @@ async fn create_npm_resolver( .and_then(|d| d.node_modules_dir.clone()), // only used for top level install, so we can ignore this npm_install_deps_provider: Arc::new(NpmInstallDepsProvider::empty()), - npmrc: config_data - .and_then(|d| d.npmrc.clone()) - .unwrap_or_else(create_default_npmrc), + npmrc, npm_system_info: NpmSystemInfo::default(), lifecycle_scripts: Default::default(), }) @@ -533,8 +551,23 @@ fn create_node_resolver( npm_resolver: &Arc, pkg_json_resolver: Arc, ) -> Arc { + let in_npm_pkg_checker = + create_in_npm_pkg_checker(match npm_resolver.as_inner() { + crate::npm::InnerCliNpmResolverRef::Byonm(_) => { + CreateInNpmPkgCheckerOptions::Byonm + } + crate::npm::InnerCliNpmResolverRef::Managed(m) => { + CreateInNpmPkgCheckerOptions::Managed( + CliManagedInNpmPkgCheckerCreateOptions { + root_cache_dir_url: m.global_cache_root_url(), + maybe_node_modules_path: m.maybe_node_modules_path(), + }, + ) + } + }); let node_resolver_inner = Arc::new(NodeResolver::new( deno_runtime::deno_node::DenoFsNodeResolverEnv::new(fs.clone()), + in_npm_pkg_checker.clone(), npm_resolver.clone().into_npm_resolver(), pkg_json_resolver.clone(), )); @@ -548,6 +581,7 @@ fn create_node_resolver( Arc::new(CliNodeResolver::new( cjs_tracker, fs, + in_npm_pkg_checker, node_resolver_inner, npm_resolver.clone(), )) diff --git a/cli/module_loader.rs b/cli/module_loader.rs index fb47eed4ad4c4d..d55f048e3eb0be 100644 --- a/cli/module_loader.rs +++ b/cli/module_loader.rs @@ -661,7 +661,11 @@ impl is_script, .. })) => { - if *is_script && self.shared.cjs_tracker.treat_as_cjs(specifier)? { + if self + .shared + .cjs_tracker + .is_cjs_with_known_is_script(specifier, *is_script)? + { return Ok(Some(CodeOrDeferredEmit::Cjs { specifier, media_type: *media_type, diff --git a/cli/node.rs b/cli/node.rs index c0bfe9533906c8..1eea82a8b23ce1 100644 --- a/cli/node.rs +++ b/cli/node.rs @@ -4,7 +4,6 @@ use std::borrow::Cow; use std::sync::Arc; use deno_ast::MediaType; -use deno_ast::ModuleKind; use deno_ast::ModuleSpecifier; use deno_core::error::AnyError; use deno_graph::ParsedSourceStore; @@ -101,8 +100,8 @@ impl CliCjsCodeAnalyzer { } let cjs_tracker = self.cjs_tracker.clone(); - let treat_as_cjs = cjs_tracker.treat_as_cjs(&specifier)?; - let analysis = if treat_as_cjs { + let is_maybe_cjs = cjs_tracker.is_maybe_cjs(&specifier)?; + let analysis = if is_maybe_cjs { let maybe_parsed_source = self .parsed_source_cache .as_ref() @@ -123,11 +122,10 @@ impl CliCjsCodeAnalyzer { maybe_syntax: None, }) })?; - let is_cjs = parsed_source.is_script() && treat_as_cjs; - cjs_tracker.mark_kind( - parsed_source.specifier().clone(), - ModuleKind::from_is_cjs(is_cjs), - ); + let is_cjs = cjs_tracker.is_cjs_with_known_is_script( + parsed_source.specifier(), + parsed_source.is_script(), + )?; if is_cjs { let analysis = parsed_source.analyze_cjs(); Ok(CliCjsAnalysis::Cjs { diff --git a/cli/npm/byonm.rs b/cli/npm/byonm.rs index befa83c9967d5d..4535b07fc530a0 100644 --- a/cli/npm/byonm.rs +++ b/cli/npm/byonm.rs @@ -10,6 +10,7 @@ use deno_core::serde_json; use deno_core::url::Url; use deno_resolver::npm::ByonmNpmResolver; use deno_resolver::npm::ByonmNpmResolverCreateOptions; +use deno_runtime::deno_node::DenoFsNodeResolverEnv; use deno_runtime::deno_node::NodePermissions; use deno_runtime::ops::process::NpmProcessStateProvider; use deno_semver::package::PackageReq; @@ -24,8 +25,9 @@ use super::InnerCliNpmResolverRef; use super::ResolvePkgFolderFromDenoReqError; pub type CliByonmNpmResolverCreateOptions = - ByonmNpmResolverCreateOptions; -pub type CliByonmNpmResolver = ByonmNpmResolver; + ByonmNpmResolverCreateOptions; +pub type CliByonmNpmResolver = + ByonmNpmResolver; // todo(dsherret): the services hanging off `CliNpmResolver` doesn't seem ideal. We should probably decouple. #[derive(Debug)] diff --git a/cli/npm/managed/cache/mod.rs b/cli/npm/managed/cache/mod.rs index aaec4866812521..8ae99f41e014b5 100644 --- a/cli/npm/managed/cache/mod.rs +++ b/cli/npm/managed/cache/mod.rs @@ -36,7 +36,7 @@ pub use tarball::TarballCache; /// Stores a single copy of npm packages in a cache. #[derive(Debug)] pub struct NpmCache { - cache_dir: NpmCacheDir, + cache_dir: Arc, cache_setting: CacheSetting, npmrc: Arc, /// ensures a package is only downloaded once per run @@ -45,7 +45,7 @@ pub struct NpmCache { impl NpmCache { pub fn new( - cache_dir: NpmCacheDir, + cache_dir: Arc, cache_setting: CacheSetting, npmrc: Arc, ) -> Self { @@ -61,6 +61,10 @@ impl NpmCache { &self.cache_setting } + pub fn root_dir_path(&self) -> &Path { + self.cache_dir.root_dir() + } + pub fn root_dir_url(&self) -> &Url { self.cache_dir.root_dir_url() } @@ -152,10 +156,6 @@ impl NpmCache { self.cache_dir.package_name_folder(name, registry_url) } - pub fn root_folder(&self) -> PathBuf { - self.cache_dir.root_dir().to_owned() - } - pub fn resolve_package_folder_id_from_specifier( &self, specifier: &ModuleSpecifier, diff --git a/cli/npm/managed/mod.rs b/cli/npm/managed/mod.rs index 4096b78009ed7e..e85a9280696152 100644 --- a/cli/npm/managed/mod.rs +++ b/cli/npm/managed/mod.rs @@ -12,6 +12,7 @@ use deno_cache_dir::npm::NpmCacheDir; use deno_core::anyhow::Context; use deno_core::error::AnyError; use deno_core::serde_json; +use deno_core::url::Url; use deno_npm::npm_rc::ResolvedNpmRc; use deno_npm::registry::NpmPackageInfo; use deno_npm::registry::NpmRegistryApi; @@ -29,6 +30,7 @@ use deno_semver::package::PackageNv; use deno_semver::package::PackageReq; use node_resolver::errors::PackageFolderResolveError; use node_resolver::errors::PackageFolderResolveIoError; +use node_resolver::InNpmPackageChecker; use node_resolver::NpmResolver; use resolution::AddPkgReqsResult; @@ -37,7 +39,6 @@ use crate::args::LifecycleScriptsConfig; use crate::args::NpmInstallDepsProvider; use crate::args::NpmProcessState; use crate::args::NpmProcessStateKind; -use crate::cache::DenoCacheEnvFsAdapter; use crate::cache::FastInsecureHasher; use crate::http_util::HttpClientProvider; use crate::util::fs::canonicalize_path_maybe_not_exists_with_fs; @@ -64,12 +65,12 @@ pub enum CliNpmResolverManagedSnapshotOption { Specified(Option), } -pub struct CliNpmResolverManagedCreateOptions { +pub struct CliManagedNpmResolverCreateOptions { pub snapshot: CliNpmResolverManagedSnapshotOption, pub maybe_lockfile: Option>, pub fs: Arc, pub http_client_provider: Arc, - pub npm_global_cache_dir: PathBuf, + pub npm_cache_dir: Arc, pub cache_setting: crate::args::CacheSetting, pub text_only_progress_bar: crate::util::progress_bar::ProgressBar, pub maybe_node_modules_path: Option, @@ -80,7 +81,7 @@ pub struct CliNpmResolverManagedCreateOptions { } pub async fn create_managed_npm_resolver_for_lsp( - options: CliNpmResolverManagedCreateOptions, + options: CliManagedNpmResolverCreateOptions, ) -> Arc { let npm_cache = create_cache(&options); let npm_api = create_api(&options, npm_cache.clone()); @@ -113,7 +114,7 @@ pub async fn create_managed_npm_resolver_for_lsp( } pub async fn create_managed_npm_resolver( - options: CliNpmResolverManagedCreateOptions, + options: CliManagedNpmResolverCreateOptions, ) -> Result, AnyError> { let npm_cache = create_cache(&options); let npm_api = create_api(&options, npm_cache.clone()); @@ -187,20 +188,16 @@ fn create_inner( )) } -fn create_cache(options: &CliNpmResolverManagedCreateOptions) -> Arc { +fn create_cache(options: &CliManagedNpmResolverCreateOptions) -> Arc { Arc::new(NpmCache::new( - NpmCacheDir::new( - &DenoCacheEnvFsAdapter(options.fs.as_ref()), - options.npm_global_cache_dir.clone(), - options.npmrc.get_all_known_registries_urls(), - ), + options.npm_cache_dir.clone(), options.cache_setting.clone(), options.npmrc.clone(), )) } fn create_api( - options: &CliNpmResolverManagedCreateOptions, + options: &CliManagedNpmResolverCreateOptions, npm_cache: Arc, ) -> Arc { Arc::new(CliNpmRegistryApi::new( @@ -257,6 +254,35 @@ async fn snapshot_from_lockfile( Ok(snapshot) } +#[derive(Debug)] +struct ManagedInNpmPackageChecker { + root_dir: Url, +} + +impl InNpmPackageChecker for ManagedInNpmPackageChecker { + fn in_npm_package(&self, specifier: &Url) -> bool { + specifier.as_ref().starts_with(self.root_dir.as_str()) + } +} + +pub struct CliManagedInNpmPkgCheckerCreateOptions<'a> { + pub root_cache_dir_url: &'a Url, + pub maybe_node_modules_path: Option<&'a Path>, +} + +pub fn create_managed_in_npm_pkg_checker( + options: CliManagedInNpmPkgCheckerCreateOptions, +) -> Arc { + let root_dir = match options.maybe_node_modules_path { + Some(node_modules_folder) => { + deno_path_util::url_from_directory_path(node_modules_folder).unwrap() + } + None => options.root_cache_dir_url.clone(), + }; + debug_assert!(root_dir.as_str().ends_with('/')); + Arc::new(ManagedInNpmPackageChecker { root_dir }) +} + /// An npm resolver where the resolution is managed by Deno rather than /// the user bringing their own node_modules (BYONM) on the file system. pub struct ManagedCliNpmResolver { @@ -548,8 +574,16 @@ impl ManagedCliNpmResolver { .map_err(|err| err.into()) } - pub fn global_cache_root_folder(&self) -> PathBuf { - self.npm_cache.root_folder() + pub fn maybe_node_modules_path(&self) -> Option<&Path> { + self.fs_resolver.node_modules_path() + } + + pub fn global_cache_root_path(&self) -> &Path { + self.npm_cache.root_dir_path() + } + + pub fn global_cache_root_url(&self) -> &Url { + self.npm_cache.root_dir_url() } } @@ -584,12 +618,6 @@ impl NpmResolver for ManagedCliNpmResolver { log::debug!("Resolved {} from {} to {}", name, referrer, path.display()); Ok(path) } - - fn in_npm_package(&self, specifier: &ModuleSpecifier) -> bool { - let root_dir_url = self.fs_resolver.root_dir_url(); - debug_assert!(root_dir_url.as_str().ends_with('/')); - specifier.as_ref().starts_with(root_dir_url.as_str()) - } } impl NpmProcessStateProvider for ManagedCliNpmResolver { diff --git a/cli/npm/managed/resolvers/common.rs b/cli/npm/managed/resolvers/common.rs index 867bb4168a1de7..590f8fb25c6c0f 100644 --- a/cli/npm/managed/resolvers/common.rs +++ b/cli/npm/managed/resolvers/common.rs @@ -17,7 +17,6 @@ use deno_core::anyhow::Context; use deno_core::error::AnyError; use deno_core::futures; use deno_core::futures::StreamExt; -use deno_core::url::Url; use deno_npm::NpmPackageCacheFolderId; use deno_npm::NpmPackageId; use deno_npm::NpmResolutionPackage; @@ -30,9 +29,6 @@ use crate::npm::managed::cache::TarballCache; /// Part of the resolution that interacts with the file system. #[async_trait(?Send)] pub trait NpmPackageFsResolver: Send + Sync { - /// Specifier for the root directory. - fn root_dir_url(&self) -> &Url; - /// The local node_modules folder if it is applicable to the implementation. fn node_modules_path(&self) -> Option<&Path>; diff --git a/cli/npm/managed/resolvers/global.rs b/cli/npm/managed/resolvers/global.rs index 5be315e9925d67..d4c9d13f7cc6fe 100644 --- a/cli/npm/managed/resolvers/global.rs +++ b/cli/npm/managed/resolvers/global.rs @@ -56,7 +56,7 @@ impl GlobalNpmPackageResolver { Self { registry_read_permission_checker: RegistryReadPermissionChecker::new( fs, - cache.root_folder(), + cache.root_dir_path().to_path_buf(), ), cache, tarball_cache, @@ -69,10 +69,6 @@ impl GlobalNpmPackageResolver { #[async_trait(?Send)] impl NpmPackageFsResolver for GlobalNpmPackageResolver { - fn root_dir_url(&self) -> &Url { - self.cache.root_dir_url() - } - fn node_modules_path(&self) -> Option<&Path> { None } diff --git a/cli/npm/managed/resolvers/local.rs b/cli/npm/managed/resolvers/local.rs index 54f7576ade6a4d..0968be8a7e4d07 100644 --- a/cli/npm/managed/resolvers/local.rs +++ b/cli/npm/managed/resolvers/local.rs @@ -155,10 +155,6 @@ impl LocalNpmPackageResolver { #[async_trait(?Send)] impl NpmPackageFsResolver for LocalNpmPackageResolver { - fn root_dir_url(&self) -> &Url { - &self.root_node_modules_url - } - fn node_modules_path(&self) -> Option<&Path> { Some(self.root_node_modules_path.as_ref()) } diff --git a/cli/npm/mod.rs b/cli/npm/mod.rs index 5d449068145b64..ce88e14ddc6561 100644 --- a/cli/npm/mod.rs +++ b/cli/npm/mod.rs @@ -18,6 +18,7 @@ use deno_core::error::AnyError; use deno_core::serde_json; use deno_npm::npm_rc::ResolvedNpmRc; use deno_npm::registry::NpmPackageInfo; +use deno_resolver::npm::ByonmInNpmPackageChecker; use deno_resolver::npm::ByonmNpmResolver; use deno_resolver::npm::ByonmResolvePkgFolderFromDenoReqError; use deno_runtime::deno_fs::FileSystem; @@ -27,6 +28,8 @@ use deno_runtime::ops::process::NpmProcessStateProvider; use deno_semver::package::PackageNv; use deno_semver::package::PackageReq; use managed::cache::registry_info::get_package_url; +use managed::create_managed_in_npm_pkg_checker; +use node_resolver::InNpmPackageChecker; use node_resolver::NpmResolver; use thiserror::Error; @@ -35,7 +38,8 @@ use crate::file_fetcher::FileFetcher; pub use self::byonm::CliByonmNpmResolver; pub use self::byonm::CliByonmNpmResolverCreateOptions; -pub use self::managed::CliNpmResolverManagedCreateOptions; +pub use self::managed::CliManagedInNpmPkgCheckerCreateOptions; +pub use self::managed::CliManagedNpmResolverCreateOptions; pub use self::managed::CliNpmResolverManagedSnapshotOption; pub use self::managed::ManagedCliNpmResolver; @@ -48,7 +52,7 @@ pub enum ResolvePkgFolderFromDenoReqError { } pub enum CliNpmResolverCreateOptions { - Managed(CliNpmResolverManagedCreateOptions), + Managed(CliManagedNpmResolverCreateOptions), Byonm(CliByonmNpmResolverCreateOptions), } @@ -74,6 +78,24 @@ pub async fn create_cli_npm_resolver( } } +pub enum CreateInNpmPkgCheckerOptions<'a> { + Managed(CliManagedInNpmPkgCheckerCreateOptions<'a>), + Byonm, +} + +pub fn create_in_npm_pkg_checker( + options: CreateInNpmPkgCheckerOptions, +) -> Arc { + match options { + CreateInNpmPkgCheckerOptions::Managed(options) => { + create_managed_in_npm_pkg_checker(options) + } + CreateInNpmPkgCheckerOptions::Byonm => { + Arc::new(ByonmInNpmPackageChecker::default()) + } + } +} + pub enum InnerCliNpmResolverRef<'a> { Managed(&'a ManagedCliNpmResolver), #[allow(dead_code)] diff --git a/cli/resolver.rs b/cli/resolver.rs index 22b1040d8fe4b0..e9bbf392e7bf05 100644 --- a/cli/resolver.rs +++ b/cli/resolver.rs @@ -41,6 +41,7 @@ use node_resolver::errors::PackageFolderResolveIoError; use node_resolver::errors::PackageNotFoundError; use node_resolver::errors::PackageResolveErrorKind; use node_resolver::errors::UrlToNodeResolutionError; +use node_resolver::InNpmPackageChecker; use node_resolver::NodeModuleKind; use node_resolver::NodeResolution; use node_resolver::NodeResolutionMode; @@ -107,6 +108,7 @@ impl deno_resolver::fs::DenoResolverFs for CliDenoResolverFs { pub struct CliNodeResolver { cjs_tracker: Arc, fs: Arc, + in_npm_pkg_checker: Arc, node_resolver: Arc, npm_resolver: Arc, } @@ -115,19 +117,21 @@ impl CliNodeResolver { pub fn new( cjs_tracker: Arc, fs: Arc, + in_npm_pkg_checker: Arc, node_resolver: Arc, npm_resolver: Arc, ) -> Self { Self { cjs_tracker, fs, + in_npm_pkg_checker, node_resolver, npm_resolver, } } pub fn in_npm_package(&self, specifier: &ModuleSpecifier) -> bool { - self.npm_resolver.in_npm_package(specifier) + self.in_npm_pkg_checker.in_npm_package(specifier) } pub fn resolve_if_for_npm_pkg( @@ -212,7 +216,7 @@ impl CliNodeResolver { ) -> Result { let referrer_kind = if self .cjs_tracker - .treat_as_cjs(referrer) + .is_maybe_cjs(referrer) .map_err(|err| NodeResolveErrorKind::PackageResolve(err.into()))? { NodeModuleKind::Cjs @@ -411,7 +415,7 @@ impl NpmModuleLoader { } })?; - let code = if self.cjs_tracker.treat_as_cjs(specifier)? { + let code = if self.cjs_tracker.is_maybe_cjs(specifier)? { // translate cjs to esm if it's cjs and inject node globals let code = from_utf8_lossy_owned(code); ModuleSourceCode::String( @@ -458,9 +462,25 @@ impl CjsTracker { } } - pub fn treat_as_cjs( + pub fn is_cjs_with_known_is_script( &self, specifier: &ModuleSpecifier, + is_script: bool, + ) -> Result { + self.treat_as_cjs_with_is_script(specifier, Some(is_script)) + } + + pub fn is_maybe_cjs( + &self, + specifier: &ModuleSpecifier, + ) -> Result { + self.treat_as_cjs_with_is_script(specifier, None) + } + + fn treat_as_cjs_with_is_script( + &self, + specifier: &ModuleSpecifier, + is_script: Option, ) -> Result { if specifier.scheme() != "file" { return Ok(false); @@ -482,15 +502,29 @@ impl CjsTracker { | MediaType::Wasm | MediaType::SourceMap | MediaType::Unknown => { - if let Some(value) = self.known.get(specifier) { - return Ok(value.is_cjs()); + if let Some(value) = self.get_known_kind(specifier) { + let is_known_cjs = value.is_cjs(); + if is_known_cjs && is_script == Some(false) { + // we now know this is actually esm + self.known.insert(specifier.clone(), ModuleKind::Esm); + return Ok(false); + } else { + return Ok(is_known_cjs); + } } if self.unstable_detect_cjs { if let Some(pkg_json) = self.pkg_json_resolver.get_closest_package_json(specifier)? { - return Ok(pkg_json.typ == "commonjs"); + let is_cjs_type = pkg_json.typ == "commonjs"; + if let Some(is_script) = is_script { + let module_kind = ModuleKind::from_is_cjs(is_script && is_cjs_type); + self.known.insert(specifier.clone(), module_kind); + return Ok(module_kind.is_cjs()); + } else { + return Ok(is_cjs_type); + } } } @@ -502,6 +536,10 @@ impl CjsTracker { pub fn mark_kind(&self, specifier: ModuleSpecifier, kind: ModuleKind) { self.known.insert(specifier, kind); } + + fn get_known_kind(&self, specifier: &ModuleSpecifier) -> Option { + self.known.get(specifier).map(|v| v.clone()) + } } pub type CliSloppyImportsResolver = diff --git a/cli/standalone/binary.rs b/cli/standalone/binary.rs index 59d746ff9b9d2a..e3253165fb3422 100644 --- a/cli/standalone/binary.rs +++ b/cli/standalone/binary.rs @@ -605,12 +605,10 @@ impl<'a> DenoCompileBinaryWriter<'a> { let (maybe_source, media_type) = match module { deno_graph::Module::Js(m) => { let source = if m.media_type.is_emittable() { - let module_kind = - if m.is_script && self.cjs_tracker.treat_as_cjs(&m.specifier)? { - ModuleKind::Cjs - } else { - ModuleKind::Esm - }; + let is_cjs = self + .cjs_tracker + .is_cjs_with_known_is_script(&m.specifier, m.is_script)?; + let module_kind = ModuleKind::from_is_cjs(is_cjs); let source = self .emitter .emit_parsed_source( @@ -752,8 +750,9 @@ impl<'a> DenoCompileBinaryWriter<'a> { } else { // DO NOT include the user's registry url as it may contain credentials, // but also don't make this dependent on the registry url - let global_cache_root_path = npm_resolver.global_cache_root_folder(); - let mut builder = VfsBuilder::new(global_cache_root_path)?; + let global_cache_root_path = npm_resolver.global_cache_root_path(); + let mut builder = + VfsBuilder::new(global_cache_root_path.to_path_buf())?; let mut packages = npm_resolver.all_system_packages(&self.npm_system_info); packages.sort_by(|a, b| a.id.cmp(&b.id)); // determinism diff --git a/cli/standalone/mod.rs b/cli/standalone/mod.rs index cc8bd2c0f67811..d3d81eb781f441 100644 --- a/cli/standalone/mod.rs +++ b/cli/standalone/mod.rs @@ -67,12 +67,15 @@ use crate::http_util::HttpClientProvider; use crate::node::CliCjsCodeAnalyzer; use crate::node::CliNodeCodeTranslator; use crate::npm::create_cli_npm_resolver; +use crate::npm::create_in_npm_pkg_checker; use crate::npm::CliByonmNpmResolverCreateOptions; +use crate::npm::CliManagedInNpmPkgCheckerCreateOptions; +use crate::npm::CliManagedNpmResolverCreateOptions; use crate::npm::CliNodeRequireLoader; use crate::npm::CliNpmResolver; use crate::npm::CliNpmResolverCreateOptions; -use crate::npm::CliNpmResolverManagedCreateOptions; use crate::npm::CliNpmResolverManagedSnapshotOption; +use crate::npm::CreateInNpmPkgCheckerOptions; use crate::resolver::CjsTracker; use crate::resolver::CjsTrackerOptions; use crate::resolver::CliDenoResolverFs; @@ -338,7 +341,7 @@ impl ModuleLoader for EmbeddedModuleLoader { let (module_specifier, module_type, module_source) = module.into_parts(); let is_cjs = - match self.shared.cjs_tracker.treat_as_cjs(original_specifier) { + match self.shared.cjs_tracker.is_maybe_cjs(original_specifier) { Ok(is_cjs) => is_cjs, Err(err) => { return deno_core::ModuleLoadResponse::Sync(Err(type_error( @@ -503,87 +506,122 @@ pub async fn run(data: StandaloneData) -> Result { let main_module = root_dir_url.join(&metadata.entrypoint_key).unwrap(); let npm_global_cache_dir = root_path.join(".deno_compile_node_modules"); let cache_setting = CacheSetting::Only; - let npm_resolver = match metadata.node_modules { + let pkg_json_resolver = Arc::new(PackageJsonResolver::new( + deno_runtime::deno_node::DenoFsNodeResolverEnv::new(fs.clone()), + )); + let (in_npm_pkg_checker, npm_resolver) = match metadata.node_modules { Some(binary::NodeModules::Managed { node_modules_dir }) => { + // create an npmrc that uses the fake npm_registry_url to resolve packages + let npmrc = Arc::new(ResolvedNpmRc { + default_config: deno_npm::npm_rc::RegistryConfigWithUrl { + registry_url: npm_registry_url.clone(), + config: Default::default(), + }, + scopes: Default::default(), + registry_configs: Default::default(), + }); + let npm_cache_dir = Arc::new(NpmCacheDir::new( + &DenoCacheEnvFsAdapter(fs.as_ref()), + npm_global_cache_dir, + npmrc.get_all_known_registries_urls(), + )); let snapshot = npm_snapshot.unwrap(); let maybe_node_modules_path = node_modules_dir .map(|node_modules_dir| root_path.join(node_modules_dir)); - create_cli_npm_resolver(CliNpmResolverCreateOptions::Managed( - CliNpmResolverManagedCreateOptions { - snapshot: CliNpmResolverManagedSnapshotOption::Specified(Some( - snapshot, - )), - maybe_lockfile: None, - fs: fs.clone(), - http_client_provider: http_client_provider.clone(), - npm_global_cache_dir, - cache_setting, - text_only_progress_bar: progress_bar, - maybe_node_modules_path, - npm_system_info: Default::default(), - npm_install_deps_provider: Arc::new( - // this is only used for installing packages, which isn't necessary with deno compile - NpmInstallDepsProvider::empty(), - ), - // create an npmrc that uses the fake npm_registry_url to resolve packages - npmrc: Arc::new(ResolvedNpmRc { - default_config: deno_npm::npm_rc::RegistryConfigWithUrl { - registry_url: npm_registry_url.clone(), - config: Default::default(), - }, - scopes: Default::default(), - registry_configs: Default::default(), - }), - lifecycle_scripts: Default::default(), - }, - )) - .await? + let in_npm_pkg_checker = + create_in_npm_pkg_checker(CreateInNpmPkgCheckerOptions::Managed( + CliManagedInNpmPkgCheckerCreateOptions { + root_cache_dir_url: npm_cache_dir.root_dir_url(), + maybe_node_modules_path: maybe_node_modules_path.as_deref(), + }, + )); + let npm_resolver = + create_cli_npm_resolver(CliNpmResolverCreateOptions::Managed( + CliManagedNpmResolverCreateOptions { + snapshot: CliNpmResolverManagedSnapshotOption::Specified(Some( + snapshot, + )), + maybe_lockfile: None, + fs: fs.clone(), + http_client_provider: http_client_provider.clone(), + npm_cache_dir, + cache_setting, + text_only_progress_bar: progress_bar, + maybe_node_modules_path, + npm_system_info: Default::default(), + npm_install_deps_provider: Arc::new( + // this is only used for installing packages, which isn't necessary with deno compile + NpmInstallDepsProvider::empty(), + ), + npmrc, + lifecycle_scripts: Default::default(), + }, + )) + .await?; + (in_npm_pkg_checker, npm_resolver) } Some(binary::NodeModules::Byonm { root_node_modules_dir, }) => { let root_node_modules_dir = root_node_modules_dir.map(|p| vfs.root().join(p)); - create_cli_npm_resolver(CliNpmResolverCreateOptions::Byonm( - CliByonmNpmResolverCreateOptions { + let in_npm_pkg_checker = + create_in_npm_pkg_checker(CreateInNpmPkgCheckerOptions::Byonm); + let npm_resolver = create_cli_npm_resolver( + CliNpmResolverCreateOptions::Byonm(CliByonmNpmResolverCreateOptions { fs: CliDenoResolverFs(fs.clone()), + pkg_json_resolver: pkg_json_resolver.clone(), root_node_modules_dir, - }, - )) - .await? + }), + ) + .await?; + (in_npm_pkg_checker, npm_resolver) } None => { - create_cli_npm_resolver(CliNpmResolverCreateOptions::Managed( - CliNpmResolverManagedCreateOptions { - snapshot: CliNpmResolverManagedSnapshotOption::Specified(None), - maybe_lockfile: None, - fs: fs.clone(), - http_client_provider: http_client_provider.clone(), - npm_global_cache_dir, - cache_setting, - text_only_progress_bar: progress_bar, - maybe_node_modules_path: None, - npm_system_info: Default::default(), - npm_install_deps_provider: Arc::new( - // this is only used for installing packages, which isn't necessary with deno compile - NpmInstallDepsProvider::empty(), - ), - // Packages from different registries are already inlined in the binary, - // so no need to create actual `.npmrc` configuration. - npmrc: create_default_npmrc(), - lifecycle_scripts: Default::default(), - }, - )) - .await? + // Packages from different registries are already inlined in the binary, + // so no need to create actual `.npmrc` configuration. + let npmrc = create_default_npmrc(); + let npm_cache_dir = Arc::new(NpmCacheDir::new( + &DenoCacheEnvFsAdapter(fs.as_ref()), + npm_global_cache_dir, + npmrc.get_all_known_registries_urls(), + )); + let in_npm_pkg_checker = + create_in_npm_pkg_checker(CreateInNpmPkgCheckerOptions::Managed( + CliManagedInNpmPkgCheckerCreateOptions { + root_cache_dir_url: npm_cache_dir.root_dir_url(), + maybe_node_modules_path: None, + }, + )); + let npm_resolver = + create_cli_npm_resolver(CliNpmResolverCreateOptions::Managed( + CliManagedNpmResolverCreateOptions { + snapshot: CliNpmResolverManagedSnapshotOption::Specified(None), + maybe_lockfile: None, + fs: fs.clone(), + http_client_provider: http_client_provider.clone(), + npm_cache_dir, + cache_setting, + text_only_progress_bar: progress_bar, + maybe_node_modules_path: None, + npm_system_info: Default::default(), + npm_install_deps_provider: Arc::new( + // this is only used for installing packages, which isn't necessary with deno compile + NpmInstallDepsProvider::empty(), + ), + npmrc: create_default_npmrc(), + lifecycle_scripts: Default::default(), + }, + )) + .await?; + (in_npm_pkg_checker, npm_resolver) } }; let has_node_modules_dir = npm_resolver.root_node_modules_path().is_some(); - let pkg_json_resolver = Arc::new(PackageJsonResolver::new( - deno_runtime::deno_node::DenoFsNodeResolverEnv::new(fs.clone()), - )); let node_resolver = Arc::new(NodeResolver::new( deno_runtime::deno_node::DenoFsNodeResolverEnv::new(fs.clone()), + in_npm_pkg_checker.clone(), npm_resolver.clone().into_npm_resolver(), pkg_json_resolver.clone(), )); @@ -598,6 +636,7 @@ pub async fn run(data: StandaloneData) -> Result { let cli_node_resolver = Arc::new(CliNodeResolver::new( cjs_tracker.clone(), fs.clone(), + in_npm_pkg_checker.clone(), node_resolver.clone(), npm_resolver.clone(), )); @@ -610,8 +649,10 @@ pub async fn run(data: StandaloneData) -> Result { let node_code_translator = Arc::new(NodeCodeTranslator::new( cjs_esm_code_analyzer, deno_runtime::deno_node::DenoFsNodeResolverEnv::new(fs.clone()), + in_npm_pkg_checker, node_resolver.clone(), npm_resolver.clone().into_npm_resolver(), + pkg_json_resolver.clone(), )); let workspace_resolver = { let import_map = match metadata.workspace_resolver.import_map { diff --git a/cli/tools/coverage/mod.rs b/cli/tools/coverage/mod.rs index 2db94963776318..5fe0e7ac3f87f5 100644 --- a/cli/tools/coverage/mod.rs +++ b/cli/tools/coverage/mod.rs @@ -26,6 +26,7 @@ use deno_core::serde_json; use deno_core::sourcemap::SourceMap; use deno_core::url::Url; use deno_core::LocalInspectorSession; +use node_resolver::InNpmPackageChecker; use regex::Regex; use std::fs; use std::fs::File; @@ -445,7 +446,7 @@ fn filter_coverages( coverages: Vec, include: Vec, exclude: Vec, - npm_resolver: &dyn CliNpmResolver, + in_npm_pkg_checker: &dyn InNpmPackageChecker, ) -> Vec { let include: Vec = include.iter().map(|e| Regex::new(e).unwrap()).collect(); @@ -469,7 +470,7 @@ fn filter_coverages( || doc_test_re.is_match(e.url.as_str()) || Url::parse(&e.url) .ok() - .map(|url| npm_resolver.in_npm_package(&url)) + .map(|url| in_npm_pkg_checker.in_npm_package(&url)) .unwrap_or(false); let is_included = include.iter().any(|p| p.is_match(&e.url)); @@ -490,7 +491,7 @@ pub async fn cover_files( let factory = CliFactory::from_flags(flags); let cli_options = factory.cli_options()?; - let npm_resolver = factory.npm_resolver().await?; + let in_npm_pkg_checker = factory.in_npm_pkg_checker()?; let file_fetcher = factory.file_fetcher()?; let emitter = factory.emitter()?; let cjs_tracker = factory.cjs_tracker()?; @@ -513,7 +514,7 @@ pub async fn cover_files( script_coverages, coverage_flags.include, coverage_flags.exclude, - npm_resolver.as_ref(), + in_npm_pkg_checker.as_ref(), ); if script_coverages.is_empty() { return Err(generic_error("No covered files included in the report")); @@ -581,11 +582,8 @@ pub async fn cover_files( | MediaType::Mts | MediaType::Cts | MediaType::Tsx => { - let module_kind = if cjs_tracker.treat_as_cjs(&file.specifier)? { - ModuleKind::Cjs - } else { - ModuleKind::Esm - }; + let module_kind = + ModuleKind::from_is_cjs(cjs_tracker.is_maybe_cjs(&file.specifier)?); Some(match emitter.maybe_cached_emit(&file.specifier, module_kind, &file.source) { Some(code) => code, None => { diff --git a/cli/tools/run/hmr.rs b/cli/tools/run/hmr.rs index ad78d2c4160140..4172573d19d9f2 100644 --- a/cli/tools/run/hmr.rs +++ b/cli/tools/run/hmr.rs @@ -145,11 +145,7 @@ impl crate::worker::HmrRunner for HmrRunner { let source_code = self.emitter.load_and_emit_for_hmr( &module_url, - if self.cjs_tracker.treat_as_cjs(&module_url)? { - ModuleKind::Cjs - } else { - ModuleKind::Esm - }, + ModuleKind::from_is_cjs(self.cjs_tracker.is_maybe_cjs(&module_url)?), ).await?; let mut tries = 1; diff --git a/ext/node/ops/require.rs b/ext/node/ops/require.rs index fca44b42ab32cf..61b5c4e3b75ba6 100644 --- a/ext/node/ops/require.rs +++ b/ext/node/ops/require.rs @@ -59,6 +59,8 @@ pub enum RequireError { #[error(transparent)] FilePathConversion(#[from] deno_path_util::UrlToFilePathError), #[error(transparent)] + UrlConversion(#[from] deno_path_util::PathToUrlError), + #[error(transparent)] Fs(#[from] deno_io::fs::FsError), #[error(transparent)] ReadModule(deno_core::error::AnyError), @@ -235,9 +237,10 @@ pub fn op_require_resolve_deno_dir( pub fn op_require_is_deno_dir_package( state: &mut OpState, #[string] path: String, -) -> bool { - let resolver = state.borrow::(); - resolver.in_npm_package_at_file_path(&PathBuf::from(path)) +) -> Result { + let resolver = state.borrow::(); + let specifier = deno_path_util::url_from_file_path(&PathBuf::from(path))?; + Ok(resolver.in_npm_package(&specifier)) } #[op2] @@ -514,12 +517,12 @@ where P: NodePermissions + 'static, { let fs = state.borrow::(); - let npm_resolver = state.borrow::(); let node_resolver = state.borrow::(); let pkg_json_resolver = state.borrow::(); let modules_path = PathBuf::from(&modules_path_str); - let pkg_path = if npm_resolver.in_npm_package_at_file_path(&modules_path) + let modules_specifier = deno_path_util::url_from_file_path(&modules_path)?; + let pkg_path = if node_resolver.in_npm_package(&modules_specifier) && !uses_local_node_modules_dir { modules_path diff --git a/resolvers/deno/fs.rs b/resolvers/deno/fs.rs index b08be37982fc75..44495fa7c2474a 100644 --- a/resolvers/deno/fs.rs +++ b/resolvers/deno/fs.rs @@ -15,13 +15,3 @@ pub trait DenoResolverFs { fn is_dir_sync(&self, path: &Path) -> bool; fn read_dir_sync(&self, dir_path: &Path) -> std::io::Result>; } - -pub(crate) struct DenoPkgJsonFsAdapter<'a, Fs: DenoResolverFs>(pub &'a Fs); - -impl<'a, Fs: DenoResolverFs> deno_package_json::fs::DenoPkgJsonFs - for DenoPkgJsonFsAdapter<'a, Fs> -{ - fn read_to_string_lossy(&self, path: &Path) -> std::io::Result { - self.0.read_to_string_lossy(path) - } -} diff --git a/resolvers/deno/npm/byonm.rs b/resolvers/deno/npm/byonm.rs index 3394b3e50180d0..e15a721dc60e90 100644 --- a/resolvers/deno/npm/byonm.rs +++ b/resolvers/deno/npm/byonm.rs @@ -10,16 +10,17 @@ use deno_package_json::PackageJsonDepValue; use deno_path_util::url_to_file_path; use deno_semver::package::PackageReq; use deno_semver::Version; +use node_resolver::env::NodeResolverEnv; use node_resolver::errors::PackageFolderResolveError; use node_resolver::errors::PackageFolderResolveIoError; use node_resolver::errors::PackageJsonLoadError; use node_resolver::errors::PackageNotFoundError; -use node_resolver::load_pkg_json; +use node_resolver::InNpmPackageChecker; use node_resolver::NpmResolver; +use node_resolver::PackageJsonResolverRc; use thiserror::Error; use url::Url; -use crate::fs::DenoPkgJsonFsAdapter; use crate::fs::DenoResolverFs; use super::local::normalize_pkg_name_for_node_modules_deno_folder; @@ -36,32 +37,41 @@ pub enum ByonmResolvePkgFolderFromDenoReqError { Io(#[from] std::io::Error), } -pub struct ByonmNpmResolverCreateOptions { - pub fs: Fs, +pub struct ByonmNpmResolverCreateOptions< + Fs: DenoResolverFs, + TEnv: NodeResolverEnv, +> { // todo(dsherret): investigate removing this pub root_node_modules_dir: Option, + pub fs: Fs, + pub pkg_json_resolver: PackageJsonResolverRc, } #[derive(Debug)] -pub struct ByonmNpmResolver { +pub struct ByonmNpmResolver { fs: Fs, + pkg_json_resolver: PackageJsonResolverRc, root_node_modules_dir: Option, } -impl Clone for ByonmNpmResolver { +impl Clone + for ByonmNpmResolver +{ fn clone(&self) -> Self { Self { fs: self.fs.clone(), + pkg_json_resolver: self.pkg_json_resolver.clone(), root_node_modules_dir: self.root_node_modules_dir.clone(), } } } -impl ByonmNpmResolver { - pub fn new(options: ByonmNpmResolverCreateOptions) -> Self { +impl ByonmNpmResolver { + pub fn new(options: ByonmNpmResolverCreateOptions) -> Self { Self { - fs: options.fs, root_node_modules_dir: options.root_node_modules_dir, + fs: options.fs, + pkg_json_resolver: options.pkg_json_resolver, } } @@ -73,7 +83,7 @@ impl ByonmNpmResolver { &self, path: &Path, ) -> Result>, PackageJsonLoadError> { - load_pkg_json(&DenoPkgJsonFsAdapter(&self.fs), path) + self.pkg_json_resolver.load_package_json(path) } /// Finds the ancestor package.json that contains the specified dependency. @@ -290,8 +300,10 @@ impl ByonmNpmResolver { } } -impl NpmResolver - for ByonmNpmResolver +impl< + Fs: DenoResolverFs + Send + Sync + std::fmt::Debug, + TEnv: NodeResolverEnv, + > NpmResolver for ByonmNpmResolver { fn resolve_package_folder_from_package( &self, @@ -342,7 +354,12 @@ impl NpmResolver .into() }) } +} + +#[derive(Default, Debug)] +pub struct ByonmInNpmPackageChecker; +impl InNpmPackageChecker for ByonmInNpmPackageChecker { fn in_npm_package(&self, specifier: &Url) -> bool { specifier.scheme() == "file" && specifier diff --git a/resolvers/deno/npm/mod.rs b/resolvers/deno/npm/mod.rs index 9d885cad31ef8c..45e2341c78b25f 100644 --- a/resolvers/deno/npm/mod.rs +++ b/resolvers/deno/npm/mod.rs @@ -3,6 +3,7 @@ mod byonm; mod local; +pub use byonm::ByonmInNpmPackageChecker; pub use byonm::ByonmNpmResolver; pub use byonm::ByonmNpmResolverCreateOptions; pub use byonm::ByonmResolvePkgFolderFromDenoReqError; diff --git a/resolvers/node/analyze.rs b/resolvers/node/analyze.rs index 7b14b5b49bf554..d369f39a68d5ec 100644 --- a/resolvers/node/analyze.rs +++ b/resolvers/node/analyze.rs @@ -19,11 +19,12 @@ use anyhow::Error as AnyError; use url::Url; use crate::env::NodeResolverEnv; -use crate::package_json::load_pkg_json; +use crate::npm::InNpmPackageCheckerRc; use crate::resolution::NodeResolverRc; use crate::NodeModuleKind; use crate::NodeResolutionMode; use crate::NpmResolverRc; +use crate::PackageJsonResolverRc; use crate::PathClean; #[derive(Debug, Clone)] @@ -63,8 +64,10 @@ pub struct NodeCodeTranslator< > { cjs_code_analyzer: TCjsCodeAnalyzer, env: TNodeResolverEnv, + in_npm_pkg_checker: InNpmPackageCheckerRc, node_resolver: NodeResolverRc, npm_resolver: NpmResolverRc, + pkg_json_resolver: PackageJsonResolverRc, } impl @@ -73,14 +76,18 @@ impl pub fn new( cjs_code_analyzer: TCjsCodeAnalyzer, env: TNodeResolverEnv, + in_npm_pkg_checker: InNpmPackageCheckerRc, node_resolver: NodeResolverRc, npm_resolver: NpmResolverRc, + pkg_json_resolver: PackageJsonResolverRc, ) -> Self { Self { cjs_code_analyzer, env, + in_npm_pkg_checker, node_resolver, npm_resolver, + pkg_json_resolver, } } @@ -329,8 +336,9 @@ impl }?; let package_json_path = module_dir.join("package.json"); - let maybe_package_json = - load_pkg_json(self.env.pkg_json_fs(), &package_json_path)?; + let maybe_package_json = self + .pkg_json_resolver + .load_package_json(&package_json_path)?; if let Some(package_json) = maybe_package_json { if let Some(exports) = &package_json.exports { return Some( @@ -356,8 +364,9 @@ impl if self.env.is_dir_sync(&d) { // subdir might have a package.json that specifies the entrypoint let package_json_path = d.join("package.json"); - let maybe_package_json = - load_pkg_json(self.env.pkg_json_fs(), &package_json_path)?; + let maybe_package_json = self + .pkg_json_resolver + .load_package_json(&package_json_path)?; if let Some(package_json) = maybe_package_json { if let Some(main) = package_json.main(NodeModuleKind::Cjs) { return Ok(Some(url_from_file_path(&d.join(main).clean())?)); @@ -382,7 +391,7 @@ impl // as a fallback, attempt to resolve it via the ancestor directories let mut last = referrer_path.as_path(); while let Some(parent) = last.parent() { - if !self.npm_resolver.in_npm_package_at_dir_path(parent) { + if !self.in_npm_pkg_checker.in_npm_package_at_dir_path(parent) { break; } let path = if parent.ends_with("node_modules") { diff --git a/resolvers/node/errors.rs b/resolvers/node/errors.rs index 4ba829eda54e9a..a35d232d62ff1a 100644 --- a/resolvers/node/errors.rs +++ b/resolvers/node/errors.rs @@ -394,6 +394,18 @@ impl NodeJsErrorCoded for CanonicalizingPkgJsonDirError { } } +#[derive(Debug, Error)] +#[error("JSX files are not supported in npm packages: {specifier}")] +pub struct JsxNotSupportedInNpmError { + pub specifier: Url, +} + +impl NodeJsErrorCoded for JsxNotSupportedInNpmError { + fn code(&self) -> NodeJsErrorCode { + NodeJsErrorCode::ERR_UNKNOWN_FILE_EXTENSION + } +} + #[derive(Debug, Error)] #[error("TypeScript files are not supported in npm packages: {specifier}")] pub struct TypeScriptNotSupportedInNpmError { @@ -411,6 +423,7 @@ kinded_err!(UrlToNodeResolutionError, UrlToNodeResolutionErrorKind); impl NodeJsErrorCoded for UrlToNodeResolutionError { fn code(&self) -> NodeJsErrorCode { match self.as_kind() { + UrlToNodeResolutionErrorKind::JsxNotSupported(e) => e.code(), UrlToNodeResolutionErrorKind::TypeScriptNotSupported(e) => e.code(), UrlToNodeResolutionErrorKind::ClosestPkgJson(e) => e.code(), } @@ -419,6 +432,8 @@ impl NodeJsErrorCoded for UrlToNodeResolutionError { #[derive(Debug, Error)] pub enum UrlToNodeResolutionErrorKind { + #[error(transparent)] + JsxNotSupported(#[from] JsxNotSupportedInNpmError), #[error(transparent)] TypeScriptNotSupported(#[from] TypeScriptNotSupportedInNpmError), #[error(transparent)] diff --git a/resolvers/node/lib.rs b/resolvers/node/lib.rs index 66c0520edbe5d9..18b0a85363f456 100644 --- a/resolvers/node/lib.rs +++ b/resolvers/node/lib.rs @@ -13,9 +13,10 @@ mod resolution; mod sync; pub use deno_package_json::PackageJson; +pub use npm::InNpmPackageChecker; +pub use npm::InNpmPackageCheckerRc; pub use npm::NpmResolver; pub use npm::NpmResolverRc; -pub use package_json::load_pkg_json; pub use package_json::PackageJsonResolver; pub use package_json::PackageJsonResolverRc; pub use package_json::PackageJsonThreadLocalCache; diff --git a/resolvers/node/npm.rs b/resolvers/node/npm.rs index 6b5f21db62b643..2132f0b545939d 100644 --- a/resolvers/node/npm.rs +++ b/resolvers/node/npm.rs @@ -22,7 +22,13 @@ pub trait NpmResolver: std::fmt::Debug + MaybeSend + MaybeSync { specifier: &str, referrer: &Url, ) -> Result; +} + +#[allow(clippy::disallowed_types)] +pub type InNpmPackageCheckerRc = crate::sync::MaybeArc; +/// Checks if a provided specifier is in an npm package. +pub trait InNpmPackageChecker: std::fmt::Debug + MaybeSend + MaybeSync { fn in_npm_package(&self, specifier: &Url) -> bool; fn in_npm_package_at_dir_path(&self, path: &Path) -> bool { diff --git a/resolvers/node/package_json.rs b/resolvers/node/package_json.rs index 1d87254eeb3ded..6967779e5d9113 100644 --- a/resolvers/node/package_json.rs +++ b/resolvers/node/package_json.rs @@ -109,27 +109,21 @@ impl PackageJsonResolver { pub fn load_package_json( &self, - package_json_path: &Path, + path: &Path, ) -> Result, PackageJsonLoadError> { - load_pkg_json(self.env.pkg_json_fs(), package_json_path) - } -} - -/// Helper to load a package.json file using the thread local cache -/// in node_resolver. -pub fn load_pkg_json( - fs: &dyn deno_package_json::fs::DenoPkgJsonFs, - path: &Path, -) -> Result, PackageJsonLoadError> { - let result = - PackageJson::load_from_path(path, fs, Some(&PackageJsonThreadLocalCache)); - match result { - Ok(pkg_json) => Ok(Some(pkg_json)), - Err(deno_package_json::PackageJsonLoadError::Io { source, .. }) - if source.kind() == ErrorKind::NotFound => - { - Ok(None) + let result = PackageJson::load_from_path( + path, + self.env.pkg_json_fs(), + Some(&PackageJsonThreadLocalCache), + ); + match result { + Ok(pkg_json) => Ok(Some(pkg_json)), + Err(deno_package_json::PackageJsonLoadError::Io { source, .. }) + if source.kind() == ErrorKind::NotFound => + { + Ok(None) + } + Err(err) => Err(PackageJsonLoadError(err)), } - Err(err) => Err(PackageJsonLoadError(err)), } } diff --git a/resolvers/node/resolution.rs b/resolvers/node/resolution.rs index 2d4132f21d2fb2..6ad508a3b4135e 100644 --- a/resolvers/node/resolution.rs +++ b/resolvers/node/resolution.rs @@ -18,6 +18,7 @@ use crate::errors::DataUrlReferrerError; use crate::errors::FinalizeResolutionError; use crate::errors::InvalidModuleSpecifierError; use crate::errors::InvalidPackageTargetError; +use crate::errors::JsxNotSupportedInNpmError; use crate::errors::LegacyResolveError; use crate::errors::ModuleNotFoundError; use crate::errors::NodeJsErrorCode; @@ -44,6 +45,7 @@ use crate::errors::TypesNotFoundErrorData; use crate::errors::UnsupportedDirImportError; use crate::errors::UnsupportedEsmUrlSchemeError; use crate::errors::UrlToNodeResolutionError; +use crate::npm::InNpmPackageCheckerRc; use crate::NpmResolverRc; use crate::PackageJsonResolverRc; use crate::PathClean; @@ -132,6 +134,7 @@ pub type NodeResolverRc = crate::sync::MaybeArc>; #[derive(Debug)] pub struct NodeResolver { env: TEnv, + in_npm_pkg_checker: InNpmPackageCheckerRc, npm_resolver: NpmResolverRc, pkg_json_resolver: PackageJsonResolverRc, } @@ -139,18 +142,20 @@ pub struct NodeResolver { impl NodeResolver { pub fn new( env: TEnv, + in_npm_pkg_checker: InNpmPackageCheckerRc, npm_resolver: NpmResolverRc, pkg_json_resolver: PackageJsonResolverRc, ) -> Self { Self { env, + in_npm_pkg_checker, npm_resolver, pkg_json_resolver, } } pub fn in_npm_package(&self, specifier: &Url) -> bool { - self.npm_resolver.in_npm_package(specifier) + self.in_npm_pkg_checker.in_npm_package(specifier) } /// This function is an implementation of `defaultResolve` in @@ -416,27 +421,48 @@ impl NodeResolver { &self, url: Url, ) -> Result { - let url_str = url.as_str().to_lowercase(); - if url_str.starts_with("http") || url_str.ends_with(".json") { - Ok(NodeResolution::Esm(url)) - } else if url_str.ends_with(".js") || url_str.ends_with(".d.ts") { - let maybe_package_config = - self.pkg_json_resolver.get_closest_package_json(&url)?; - match maybe_package_config { - Some(c) if c.typ == "module" => Ok(NodeResolution::Esm(url)), - Some(_) => Ok(NodeResolution::CommonJs(url)), - None => Ok(NodeResolution::Esm(url)), + if url.scheme() != "file" { + return Ok(NodeResolution::Esm(url)); + } + let media_type = MediaType::from_specifier(&url); + match media_type { + MediaType::JavaScript | MediaType::Dts => { + let maybe_package_config = + self.pkg_json_resolver.get_closest_package_json(&url)?; + match maybe_package_config { + Some(c) if c.typ == "module" => Ok(NodeResolution::Esm(url)), + Some(_) => Ok(NodeResolution::CommonJs(url)), + None => Ok(NodeResolution::Esm(url)), + } } - } else if url_str.ends_with(".mjs") || url_str.ends_with(".d.mts") { - Ok(NodeResolution::Esm(url)) - } else if url_str.ends_with(".ts") || url_str.ends_with(".mts") { - if self.in_npm_package(&url) { - Err(TypeScriptNotSupportedInNpmError { specifier: url }.into()) - } else { + MediaType::Json | MediaType::Wasm | MediaType::Mjs | MediaType::Dmts => { Ok(NodeResolution::Esm(url)) } - } else { - Ok(NodeResolution::CommonJs(url)) + MediaType::Cjs | MediaType::Dcts => Ok(NodeResolution::CommonJs(url)), + MediaType::TypeScript | MediaType::Mts => { + if self.in_npm_pkg_checker.in_npm_package(&url) { + Err(TypeScriptNotSupportedInNpmError { specifier: url }.into()) + } else { + Ok(NodeResolution::Esm(url)) + } + } + MediaType::Cts => { + if self.in_npm_pkg_checker.in_npm_package(&url) { + Err(TypeScriptNotSupportedInNpmError { specifier: url }.into()) + } else { + Ok(NodeResolution::CommonJs(url)) + } + } + MediaType::Jsx | MediaType::Tsx => { + if self.in_npm_pkg_checker.in_npm_package(&url) { + Err(JsxNotSupportedInNpmError { specifier: url }.into()) + } else { + Ok(NodeResolution::Esm(url)) + } + } + MediaType::Css | MediaType::SourceMap | MediaType::Unknown => { + Ok(NodeResolution::CommonJs(url)) + } } } diff --git a/runtime/errors.rs b/runtime/errors.rs index 325990c7831065..b4796c92083074 100644 --- a/runtime/errors.rs +++ b/runtime/errors.rs @@ -1062,6 +1062,7 @@ mod node { | RequireError::PackageJsonLoad(_) | RequireError::ClosestPkgJson(_) | RequireError::FilePathConversion(_) + | RequireError::UrlConversion(_) | RequireError::ReadModule(_) | RequireError::PackageImportsResolve(_) => "Error", RequireError::Fs(e) | RequireError::UnableToGetCwd(e) => { diff --git a/tests/registry/npm/@denotest/type-commonjs/1.0.0/index.js b/tests/registry/npm/@denotest/type-commonjs/1.0.0/index.js index cb0ff5c3b541f6..d3f80a049543c6 100644 --- a/tests/registry/npm/@denotest/type-commonjs/1.0.0/index.js +++ b/tests/registry/npm/@denotest/type-commonjs/1.0.0/index.js @@ -1 +1,5 @@ -export {}; +// this module is declared as CommonJS, but during loading we'll +// discover it's ESM and load it fine +export function add(a, b) { + return a + b; +} diff --git a/tests/specs/npm/require_type_commonjs/__test__.jsonc b/tests/specs/npm/require_type_commonjs/__test__.jsonc index c9ba97ff522a78..a71173ce37b911 100644 --- a/tests/specs/npm/require_type_commonjs/__test__.jsonc +++ b/tests/specs/npm/require_type_commonjs/__test__.jsonc @@ -1,5 +1,4 @@ { "args": "run --allow-read --quiet main.ts", - "output": "main.out", - "exitCode": 1 + "output": "main.out" } diff --git a/tests/specs/npm/require_type_commonjs/main.out b/tests/specs/npm/require_type_commonjs/main.out index d715db8a94e686..00750edc07d641 100644 --- a/tests/specs/npm/require_type_commonjs/main.out +++ b/tests/specs/npm/require_type_commonjs/main.out @@ -1,4 +1 @@ -error: 'import', and 'export' cannot be used outside of module code at file://[WILDCARD]/@denotest/type-commonjs/1.0.0/index.js:1:1 - - export {}; - ~~~~~~ +3 diff --git a/tests/specs/npm/require_type_commonjs/main.ts b/tests/specs/npm/require_type_commonjs/main.ts index 243eb216e47bf6..95ec6099ec4f0d 100644 --- a/tests/specs/npm/require_type_commonjs/main.ts +++ b/tests/specs/npm/require_type_commonjs/main.ts @@ -1 +1,2 @@ -import "npm:@denotest/type-commonjs"; +import { add } from "npm:@denotest/type-commonjs"; +console.log(add(1, 2)); From cf9fd71c8228d92aec4cb4d0a78ae8cee3442a1f Mon Sep 17 00:00:00 2001 From: David Sherret Date: Tue, 29 Oct 2024 14:04:53 -0400 Subject: [PATCH 19/42] fix bug --- ext/node/ops/require.rs | 8 +++++--- 1 file changed, 5 insertions(+), 3 deletions(-) diff --git a/ext/node/ops/require.rs b/ext/node/ops/require.rs index 61b5c4e3b75ba6..c0204947daa42c 100644 --- a/ext/node/ops/require.rs +++ b/ext/node/ops/require.rs @@ -237,10 +237,12 @@ pub fn op_require_resolve_deno_dir( pub fn op_require_is_deno_dir_package( state: &mut OpState, #[string] path: String, -) -> Result { +) -> bool { let resolver = state.borrow::(); - let specifier = deno_path_util::url_from_file_path(&PathBuf::from(path))?; - Ok(resolver.in_npm_package(&specifier)) + match deno_path_util::url_from_file_path(&PathBuf::from(path)) { + Ok(specifier) => resolver.in_npm_package(&specifier), + Err(_) => false, + } } #[op2] From e0b0fb5ed7bbb2d0a3fcae560fd2d46a112c473f Mon Sep 17 00:00:00 2001 From: David Sherret Date: Tue, 29 Oct 2024 14:13:52 -0400 Subject: [PATCH 20/42] passing tests --- cli/factory.rs | 3 ++- cli/lsp/resolver.rs | 3 ++- cli/resolver.rs | 31 +++++++++++++++++++++++++++---- cli/standalone/mod.rs | 3 ++- 4 files changed, 33 insertions(+), 7 deletions(-) diff --git a/cli/factory.rs b/cli/factory.rs index 6cfedf97cb2d91..d266bf73ab7194 100644 --- a/cli/factory.rs +++ b/cli/factory.rs @@ -785,10 +785,11 @@ impl CliFactory { self.services.cjs_tracker.get_or_try_init(|| { let options = self.cli_options()?; Ok(Arc::new(CjsTracker::new( + self.in_npm_pkg_checker()?.clone(), + self.pkg_json_resolver().clone(), CjsTrackerOptions { unstable_detect_cjs: options.unstable_detect_cjs(), }, - self.pkg_json_resolver().clone(), ))) }) } diff --git a/cli/lsp/resolver.rs b/cli/lsp/resolver.rs index 06cfd2b791b4d2..4d5e15951ca627 100644 --- a/cli/lsp/resolver.rs +++ b/cli/lsp/resolver.rs @@ -572,11 +572,12 @@ fn create_node_resolver( pkg_json_resolver.clone(), )); let cjs_tracker = Arc::new(CjsTracker::new( + in_npm_pkg_checker.clone(), + pkg_json_resolver, CjsTrackerOptions { // todo(dsherret): support in the lsp unstable_detect_cjs: false, }, - pkg_json_resolver, )); Arc::new(CliNodeResolver::new( cjs_tracker, diff --git a/cli/resolver.rs b/cli/resolver.rs index e9bbf392e7bf05..55e7461096038a 100644 --- a/cli/resolver.rs +++ b/cli/resolver.rs @@ -445,20 +445,23 @@ pub struct CjsTrackerOptions { /// Keeps track of what module specifiers were resolved as CJS. #[derive(Debug)] pub struct CjsTracker { + in_npm_pkg_checker: Arc, + pkg_json_resolver: Arc, unstable_detect_cjs: bool, known: DashMap, - pkg_json_resolver: Arc, } impl CjsTracker { pub fn new( - options: CjsTrackerOptions, + in_npm_pkg_checker: Arc, pkg_json_resolver: Arc, + options: CjsTrackerOptions, ) -> Self { Self { + in_npm_pkg_checker, + pkg_json_resolver, unstable_detect_cjs: options.unstable_detect_cjs, known: Default::default(), - pkg_json_resolver, } } @@ -513,7 +516,27 @@ impl CjsTracker { } } - if self.unstable_detect_cjs { + if let Some(is_script) = is_script { + if !is_script { + self.known.insert(specifier.clone(), ModuleKind::Esm); + return Ok(false); + } + } + + if self.in_npm_pkg_checker.in_npm_package(specifier) { + if let Some(pkg_json) = + self.pkg_json_resolver.get_closest_package_json(specifier)? + { + let is_file_location_cjs = pkg_json.typ != "module"; + if let Some(is_script) = is_script { + let module_kind = ModuleKind::from_is_cjs(is_script && is_file_location_cjs); + self.known.insert(specifier.clone(), module_kind); + return Ok(module_kind.is_cjs()); + } else { + return Ok(is_file_location_cjs); + } + } + } else if self.unstable_detect_cjs { if let Some(pkg_json) = self.pkg_json_resolver.get_closest_package_json(specifier)? { diff --git a/cli/standalone/mod.rs b/cli/standalone/mod.rs index d3d81eb781f441..795b1b78f46555 100644 --- a/cli/standalone/mod.rs +++ b/cli/standalone/mod.rs @@ -626,10 +626,11 @@ pub async fn run(data: StandaloneData) -> Result { pkg_json_resolver.clone(), )); let cjs_tracker = Arc::new(CjsTracker::new( + in_npm_pkg_checker.clone(), + pkg_json_resolver.clone(), CjsTrackerOptions { unstable_detect_cjs: metadata.unstable_config.detect_cjs, }, - pkg_json_resolver.clone(), )); let cache_db = Caches::new(deno_dir_provider.clone()); let node_analysis_cache = NodeAnalysisCache::new(cache_db.node_analysis_db()); From c8361e8b693c22bca7a12a2da47ca074f53acb3f Mon Sep 17 00:00:00 2001 From: David Sherret Date: Tue, 29 Oct 2024 17:19:39 -0400 Subject: [PATCH 21/42] Start working on removing `NodeResolution` --- cli/args/mod.rs | 8 -- cli/cache/cache_db.rs | 6 +- cli/cache/module_info.rs | 128 +++++++++++++++++++------ cli/factory.rs | 8 +- cli/graph_util.rs | 4 +- cli/module_loader.rs | 73 ++++++++++++++- cli/npm/mod.rs | 48 ---------- cli/resolver.rs | 165 +++++++++++++++++++++------------ cli/standalone/mod.rs | 5 +- cli/tools/check.rs | 10 ++ cli/tools/doc.rs | 2 +- cli/tsc/mod.rs | 50 +++++++--- cli/worker.rs | 62 +++---------- ext/node/lib.rs | 25 ----- ext/node/ops/worker_threads.rs | 34 +------ runtime/errors.rs | 2 - 16 files changed, 356 insertions(+), 274 deletions(-) diff --git a/cli/args/mod.rs b/cli/args/mod.rs index bf966f794bea1c..e8ace69b244bd0 100644 --- a/cli/args/mod.rs +++ b/cli/args/mod.rs @@ -1206,14 +1206,6 @@ impl CliOptions { } } - // If the main module should be treated as being in an npm package. - // This is triggered via a secret environment variable which is used - // for functionality like child_process.fork. Users should NOT depend - // on this functionality. - pub fn is_npm_main(&self) -> bool { - NPM_PROCESS_STATE.is_some() - } - pub fn has_node_modules_dir(&self) -> bool { self.maybe_node_modules_folder.is_some() } diff --git a/cli/cache/cache_db.rs b/cli/cache/cache_db.rs index b24078f29bd6fd..329ed2d9704dff 100644 --- a/cli/cache/cache_db.rs +++ b/cli/cache/cache_db.rs @@ -57,7 +57,7 @@ impl rusqlite::types::FromSql for CacheDBHash { } /// What should the cache should do on failure? -#[derive(Default)] +#[derive(Debug, Default)] pub enum CacheFailure { /// Return errors if failure mode otherwise unspecified. #[default] @@ -69,6 +69,7 @@ pub enum CacheFailure { } /// Configuration SQL and other parameters for a [`CacheDB`]. +#[derive(Debug)] pub struct CacheDBConfiguration { /// SQL to run for a new database. pub table_initializer: &'static str, @@ -98,6 +99,7 @@ impl CacheDBConfiguration { } } +#[derive(Debug)] enum ConnectionState { Connected(Connection), Blackhole, @@ -106,7 +108,7 @@ enum ConnectionState { /// A cache database that eagerly initializes itself off-thread, preventing initialization operations /// from blocking the main thread. -#[derive(Clone)] +#[derive(Debug, Clone)] pub struct CacheDB { // TODO(mmastrac): We can probably simplify our thread-safe implementation here conn: Arc>>, diff --git a/cli/cache/module_info.rs b/cli/cache/module_info.rs index 86e7ed5b1c9709..2c6dc54f3e2d17 100644 --- a/cli/cache/module_info.rs +++ b/cli/cache/module_info.rs @@ -44,18 +44,32 @@ pub static MODULE_INFO_CACHE_DB: CacheDBConfiguration = CacheDBConfiguration { /// A cache of `deno_graph::ModuleInfo` objects. Using this leads to a considerable /// performance improvement because when it exists we can skip parsing a module for /// deno_graph. +#[derive(Debug)] pub struct ModuleInfoCache { conn: CacheDB, + parsed_source_cache: Arc, } impl ModuleInfoCache { #[cfg(test)] - pub fn new_in_memory(version: &'static str) -> Self { - Self::new(CacheDB::in_memory(&MODULE_INFO_CACHE_DB, version)) + pub fn new_in_memory( + version: &'static str, + parsed_source_cache: Arc, + ) -> Self { + Self::new( + CacheDB::in_memory(&MODULE_INFO_CACHE_DB, version), + parsed_source_cache, + ) } - pub fn new(conn: CacheDB) -> Self { - Self { conn } + pub fn new( + conn: CacheDB, + parsed_source_cache: Arc, + ) -> Self { + Self { + conn, + parsed_source_cache, + } } /// Useful for testing: re-create this cache DB with a different current version. @@ -63,6 +77,7 @@ impl ModuleInfoCache { pub(crate) fn recreate_with_version(self, version: &'static str) -> Self { Self { conn: self.conn.recreate_with_version(version), + parsed_source_cache: self.parsed_source_cache, } } @@ -113,13 +128,10 @@ impl ModuleInfoCache { Ok(()) } - pub fn as_module_analyzer<'a>( - &'a self, - parsed_source_cache: &'a Arc, - ) -> ModuleInfoCacheModuleAnalyzer<'a> { + pub fn as_module_analyzer<'a>(&'a self) -> ModuleInfoCacheModuleAnalyzer<'a> { ModuleInfoCacheModuleAnalyzer { module_info_cache: self, - parsed_source_cache, + parsed_source_cache: &self.parsed_source_cache, } } } @@ -129,31 +141,99 @@ pub struct ModuleInfoCacheModuleAnalyzer<'a> { parsed_source_cache: &'a Arc, } -#[async_trait::async_trait(?Send)] -impl<'a> deno_graph::ModuleAnalyzer for ModuleInfoCacheModuleAnalyzer<'a> { - async fn analyze( +impl<'a> ModuleInfoCacheModuleAnalyzer<'a> { + fn load_cached_module_info( &self, specifier: &ModuleSpecifier, - source: Arc, media_type: MediaType, - ) -> Result { - // attempt to load from the cache - let source_hash = CacheDBHash::from_source(&source); + source_hash: CacheDBHash, + ) -> Option { match self.module_info_cache.get_module_info( specifier, media_type, source_hash, ) { - Ok(Some(info)) => return Ok(info), - Ok(None) => {} + Ok(Some(info)) => Some(info), + Ok(None) => None, Err(err) => { log::debug!( "Error loading module cache info for {}. {:#}", specifier, err ); + None } } + } + + fn save_module_info_to_cache( + &self, + specifier: &ModuleSpecifier, + media_type: MediaType, + source_hash: CacheDBHash, + module_info: &ModuleInfo, + ) { + if let Err(err) = self.module_info_cache.set_module_info( + specifier, + media_type, + source_hash, + &module_info, + ) { + log::debug!( + "Error saving module cache info for {}. {:#}", + specifier, + err + ); + } + } + + pub fn analyze_sync( + &self, + specifier: &ModuleSpecifier, + source: &str, + media_type: MediaType, + ) -> Result { + // attempt to load from the cache + let source_hash = CacheDBHash::from_source(&source); + if let Some(info) = + self.load_cached_module_info(specifier, media_type, source_hash) + { + return Ok(info); + } + + // otherwise, get the module info from the parsed source cache + let parser = self.parsed_source_cache.as_capturing_parser(); + let analyzer = ParserModuleAnalyzer::new(&parser); + let module_info = + analyzer.analyze_sync(&specifier, Arc::from(source), media_type)?; + + // then attempt to cache it + self.save_module_info_to_cache( + specifier, + media_type, + source_hash, + &module_info, + ); + + Ok(module_info) + } +} + +#[async_trait::async_trait(?Send)] +impl<'a> deno_graph::ModuleAnalyzer for ModuleInfoCacheModuleAnalyzer<'a> { + async fn analyze( + &self, + specifier: &ModuleSpecifier, + source: Arc, + media_type: MediaType, + ) -> Result { + // attempt to load from the cache + let source_hash = CacheDBHash::from_source(&source); + if let Some(info) = + self.load_cached_module_info(specifier, media_type, source_hash) + { + return Ok(info); + } // otherwise, get the module info from the parsed source cache let module_info = deno_core::unsync::spawn_blocking({ @@ -169,18 +249,12 @@ impl<'a> deno_graph::ModuleAnalyzer for ModuleInfoCacheModuleAnalyzer<'a> { .unwrap()?; // then attempt to cache it - if let Err(err) = self.module_info_cache.set_module_info( + self.save_module_info_to_cache( specifier, media_type, source_hash, &module_info, - ) { - log::debug!( - "Error saving module cache info for {}. {:#}", - specifier, - err - ); - } + ); Ok(module_info) } @@ -217,7 +291,7 @@ mod test { #[test] pub fn module_info_cache_general_use() { - let cache = ModuleInfoCache::new_in_memory("1.0.0"); + let cache = ModuleInfoCache::new_in_memory("1.0.0", Default::default()); let specifier1 = ModuleSpecifier::parse("https://localhost/mod.ts").unwrap(); let specifier2 = diff --git a/cli/factory.rs b/cli/factory.rs index d266bf73ab7194..560b71bc5eab4e 100644 --- a/cli/factory.rs +++ b/cli/factory.rs @@ -70,7 +70,6 @@ use deno_core::error::AnyError; use deno_core::futures::FutureExt; use deno_core::FeatureChecker; -use deno_resolver::npm::ByonmInNpmPackageChecker; use deno_runtime::deno_fs; use deno_runtime::deno_node::DenoFsNodeResolverEnv; use deno_runtime::deno_node::NodeResolver; @@ -573,6 +572,7 @@ impl CliFactory { self.services.module_info_cache.get_or_try_init(|| { Ok(Arc::new(ModuleInfoCache::new( self.caches()?.dep_analysis_db(), + self.parsed_source_cache().clone(), ))) }) } @@ -680,8 +680,10 @@ impl CliFactory { let cli_options = self.cli_options()?; Ok(Arc::new(TypeChecker::new( self.caches()?.clone(), + self.cjs_tracker()?.clone(), cli_options.clone(), self.module_graph_builder().await?.clone(), + self.module_info_cache()?.clone(), self.node_resolver().await?.clone(), self.npm_resolver().await?.clone(), ))) @@ -878,6 +880,7 @@ impl CliFactory { let npm_resolver = self.npm_resolver().await?; let cli_node_resolver = self.cli_node_resolver().await?; let cli_npm_resolver = self.npm_resolver().await?.clone(); + let in_npm_pkg_checker = self.in_npm_pkg_checker()?; let maybe_file_watcher_communicator = if cli_options.has_hmr() { Some(self.watcher_communicator.clone().unwrap()) } else { @@ -889,6 +892,7 @@ impl CliFactory { Ok(CliMainWorkerFactory::new( self.blob_store().clone(), + cjs_tracker.clone(), if cli_options.code_cache_enabled() { Some(self.code_cache()?.clone()) } else { @@ -909,6 +913,7 @@ impl CliFactory { }, self.emitter()?.clone(), fs.clone(), + in_npm_pkg_checker.clone(), self.main_module_graph_container().await?.clone(), self.module_load_preparer().await?.clone(), node_code_translator.clone(), @@ -980,7 +985,6 @@ impl CliFactory { inspect_wait: cli_options.inspect_wait().is_some(), strace_ops: cli_options.strace_ops().clone(), is_inspecting: cli_options.is_inspecting(), - is_npm_main: cli_options.is_npm_main(), location: cli_options.location_flag().clone(), // if the user ran a binary command, we'll need to set process.argv[0] // to be the name of the binary command instead of deno diff --git a/cli/graph_util.rs b/cli/graph_util.rs index 0df22b680e1125..7501048ba9dbfa 100644 --- a/cli/graph_util.rs +++ b/cli/graph_util.rs @@ -512,9 +512,7 @@ impl ModuleGraphBuilder { } else { Vec::new() }; - let analyzer = self - .module_info_cache - .as_module_analyzer(&self.parsed_source_cache); + let analyzer = self.module_info_cache.as_module_analyzer(); let mut loader = match options.loader { Some(loader) => MutLoaderRef::Borrowed(loader), None => MutLoaderRef::Owned(self.create_graph_loader()), diff --git a/cli/module_loader.rs b/cli/module_loader.rs index d55f048e3eb0be..3050aa2df5c5fc 100644 --- a/cli/module_loader.rs +++ b/cli/module_loader.rs @@ -2,6 +2,7 @@ use std::borrow::Cow; use std::cell::RefCell; +use std::path::Path; use std::path::PathBuf; use std::pin::Pin; use std::rc::Rc; @@ -24,17 +25,18 @@ use crate::graph_util::CreateGraphOptions; use crate::graph_util::ModuleGraphBuilder; use crate::node; use crate::node::CliNodeCodeTranslator; -use crate::npm::CliNodeRequireLoader; use crate::npm::CliNpmResolver; use crate::resolver::CjsTracker; use crate::resolver::CliGraphResolver; use crate::resolver::CliNodeResolver; use crate::resolver::ModuleCodeStringSource; +use crate::resolver::NotSupportedKindInNpmError; use crate::resolver::NpmModuleLoader; use crate::tools::check; use crate::tools::check::TypeChecker; use crate::util::progress_bar::ProgressBar; use crate::util::text_encoding::code_without_source_map; +use crate::util::text_encoding::from_utf8_lossy_owned; use crate::util::text_encoding::source_map_from_code; use crate::worker::CreateModuleLoaderResult; use crate::worker::ModuleLoaderFactory; @@ -49,6 +51,7 @@ use deno_core::error::AnyError; use deno_core::futures::future::FutureExt; use deno_core::futures::Future; use deno_core::resolve_url; +use deno_core::url::Url; use deno_core::ModuleCodeString; use deno_core::ModuleLoader; use deno_core::ModuleSource; @@ -67,11 +70,15 @@ use deno_graph::Module; use deno_graph::ModuleGraph; use deno_graph::Resolution; use deno_runtime::code_cache; +use deno_runtime::deno_fs; use deno_runtime::deno_fs::FileSystem; use deno_runtime::deno_node::create_host_defined_options; +use deno_runtime::deno_node::NodeRequireLoader; use deno_runtime::deno_permissions::PermissionsContainer; use deno_semver::npm::NpmPackageReqReference; +use node_resolver::InNpmPackageChecker; use node_resolver::NodeResolutionMode; +use thiserror::Error; pub struct ModuleLoadPreparer { options: Arc, @@ -208,6 +215,7 @@ struct SharedCliModuleLoaderState { code_cache: Option>, emitter: Arc, fs: Arc, + in_npm_pkg_checker: Arc, main_module_graph_container: Arc, module_load_preparer: Arc, node_code_translator: Arc, @@ -230,6 +238,7 @@ impl CliModuleLoaderFactory { code_cache: Option>, emitter: Arc, fs: Arc, + in_npm_pkg_checker: Arc, main_module_graph_container: Arc, module_load_preparer: Arc, node_code_translator: Arc, @@ -254,6 +263,7 @@ impl CliModuleLoaderFactory { code_cache, emitter, fs, + in_npm_pkg_checker, main_module_graph_container, module_load_preparer, node_code_translator, @@ -289,6 +299,7 @@ impl CliModuleLoaderFactory { let node_require_loader = Rc::new(CliNodeRequireLoader::new( self.shared.emitter.clone(), self.shared.fs.clone(), + self.shared.in_npm_pkg_checker.clone(), self.shared.npm_resolver.clone(), )); CreateModuleLoaderResult { @@ -1005,3 +1016,63 @@ impl ModuleGraphUpdatePermit for WorkerModuleGraphUpdatePermit { drop(self.permit); // explicit drop for clarity } } + +#[derive(Debug)] +struct CliNodeRequireLoader { + emitter: Arc, + fs: Arc, + in_npm_pkg_checker: Arc, + npm_resolver: Arc, +} + +impl CliNodeRequireLoader { + pub fn new( + emitter: Arc, + fs: Arc, + in_npm_pkg_checker: Arc, + npm_resolver: Arc, + ) -> Self { + Self { + emitter, + fs, + in_npm_pkg_checker, + npm_resolver, + } + } +} + +impl NodeRequireLoader for CliNodeRequireLoader { + fn ensure_read_permission<'a>( + &self, + permissions: &mut dyn deno_runtime::deno_node::NodePermissions, + path: &'a Path, + ) -> Result, AnyError> { + self.npm_resolver.ensure_read_permission(permissions, path) + } + + fn load_text_file_lossy(&self, path: &Path) -> Result { + // todo(dsherret): use the preloaded module from the graph if available + let specifier = deno_path_util::url_from_file_path(path)?; + let media_type = MediaType::from_specifier(&specifier); + let text = self.fs.read_text_file_lossy_sync(path, None)?; + if media_type.is_emittable() { + if self.in_npm_pkg_checker.in_npm_package(&specifier) { + return Err( + NotSupportedKindInNpmError { + media_type, + specifier, + } + .into(), + ); + } + self.emitter.emit_parsed_source_sync( + &specifier, + ModuleKind::Cjs, + media_type, + &text.into(), + ) + } else { + Ok(text) + } + } +} diff --git a/cli/npm/mod.rs b/cli/npm/mod.rs index ce88e14ddc6561..249be1be313fb2 100644 --- a/cli/npm/mod.rs +++ b/cli/npm/mod.rs @@ -222,51 +222,3 @@ impl NpmFetchResolver { info } } - -#[derive(Debug)] -pub struct CliNodeRequireLoader { - emitter: Arc, - fs: Arc, - npm_resolver: Arc, -} - -impl CliNodeRequireLoader { - pub fn new( - emitter: Arc, - fs: Arc, - npm_resolver: Arc, - ) -> Self { - Self { - emitter, - fs, - npm_resolver, - } - } -} - -impl NodeRequireLoader for CliNodeRequireLoader { - fn ensure_read_permission<'a>( - &self, - permissions: &mut dyn deno_runtime::deno_node::NodePermissions, - path: &'a Path, - ) -> Result, AnyError> { - self.npm_resolver.ensure_read_permission(permissions, path) - } - - fn load_text_file_lossy(&self, path: &Path) -> Result { - // todo(dsherret): use the preloaded module from the graph if available - let specifier = deno_path_util::url_from_file_path(path)?; - let media_type = MediaType::from_specifier(&specifier); - let text = self.fs.read_text_file_lossy_sync(path, None)?; - if media_type == MediaType::Cts { - self.emitter.emit_parsed_source_sync( - &specifier, - ModuleKind::Cjs, - media_type, - &text.into(), - ) - } else { - Ok(text) - } - } -} diff --git a/cli/resolver.rs b/cli/resolver.rs index 55e7461096038a..8ee85158b50b9d 100644 --- a/cli/resolver.rs +++ b/cli/resolver.rs @@ -12,6 +12,7 @@ use deno_config::workspace::WorkspaceResolver; use deno_core::anyhow::anyhow; use deno_core::anyhow::Context; use deno_core::error::AnyError; +use deno_core::url::Url; use deno_core::ModuleSourceCode; use deno_core::ModuleSpecifier; use deno_graph::source::ResolutionMode; @@ -29,6 +30,7 @@ use deno_runtime::colors; use deno_runtime::deno_fs; use deno_runtime::deno_fs::FileSystem; use deno_runtime::deno_node::is_builtin_node_module; +use deno_runtime::deno_node::NodeRequireLoader; use deno_runtime::deno_node::NodeResolver; use deno_runtime::deno_node::PackageJsonResolver; use deno_semver::npm::NpmPackageReqReference; @@ -49,9 +51,11 @@ use std::borrow::Cow; use std::path::Path; use std::path::PathBuf; use std::sync::Arc; +use thiserror::Error; use crate::args::JsxImportSourceConfig; use crate::args::DENO_DISABLE_PEDANTIC_NODE_WARNINGS; +use crate::emit::Emitter; use crate::node::CliNodeCodeTranslator; use crate::npm::CliNpmResolver; use crate::npm::InnerCliNpmResolverRef; @@ -312,12 +316,7 @@ impl CliNodeResolver { specifier, self.fs.as_ref(), ); - if self.in_npm_package(&specifier) { - let resolution = - self.node_resolver.url_to_node_resolution(specifier)?; - let resolution = self.handle_node_resolution(resolution); - return Ok(Some(resolution.into_url())); - } + return Ok(Some(specifier)); } Ok(None) @@ -352,7 +351,14 @@ impl CliNodeResolver { } } -// todo(dsherret): move to module_loader.rs +#[derive(Debug, Error)] +#[error("{media_type} files are not supported in npm packages: {specifier}")] +pub struct NotSupportedKindInNpmError { + pub media_type: MediaType, + pub specifier: Url, +} + +// todo(dsherret): move to module_loader.rs (it seems to be here due to use in standalone) #[derive(Clone)] pub struct NpmModuleLoader { cjs_tracker: Arc, @@ -415,6 +421,14 @@ impl NpmModuleLoader { } })?; + let media_type = MediaType::from_specifier(specifier); + if media_type.is_emittable() { + return Err(AnyError::from(NotSupportedKindInNpmError { + media_type, + specifier: specifier.clone(), + })); + } + let code = if self.cjs_tracker.is_maybe_cjs(specifier)? { // translate cjs to esm if it's cjs and inject node globals let code = from_utf8_lossy_owned(code); @@ -430,6 +444,7 @@ impl NpmModuleLoader { // esm and json code is untouched ModuleSourceCode::Bytes(code.into_boxed_slice().into()) }; + Ok(ModuleCodeStringSource { code, found_url: specifier.clone(), @@ -465,6 +480,21 @@ impl CjsTracker { } } + /// Checks whether the file might be treated as CJS, but it's not for sure + /// yet because the source hasn't been loaded to see whether it contains + /// imports or exports. + pub fn is_maybe_cjs( + &self, + specifier: &ModuleSpecifier, + ) -> Result { + self.treat_as_cjs_with_is_script(specifier, None) + } + + /// Gets whether the file is CJS. If true, this is for sure + /// cjs because `is_script` is provided. + /// + /// `is_script` should be `true` when the contents of the file at the + /// provided specifier are known to be a script and not an ES module. pub fn is_cjs_with_known_is_script( &self, specifier: &ModuleSpecifier, @@ -473,96 +503,111 @@ impl CjsTracker { self.treat_as_cjs_with_is_script(specifier, Some(is_script)) } - pub fn is_maybe_cjs( + fn treat_as_cjs_with_is_script( &self, specifier: &ModuleSpecifier, + is_script: Option, ) -> Result { - self.treat_as_cjs_with_is_script(specifier, None) + let kind = match self.get_known_kind_with_is_script(specifier, is_script) { + Some(kind) => kind, + None => self.check_based_on_pkg_json(specifier)?, + }; + Ok(kind.is_cjs()) } - fn treat_as_cjs_with_is_script( + pub fn get_known_kind( + &self, + specifier: &ModuleSpecifier, + ) -> Option { + self.get_known_kind_with_is_script(specifier, None) + } + + fn get_known_kind_with_is_script( &self, specifier: &ModuleSpecifier, is_script: Option, - ) -> Result { + ) -> Option { if specifier.scheme() != "file" { - return Ok(false); + return Some(ModuleKind::Esm); } let media_type = MediaType::from_specifier(specifier); match media_type { - MediaType::Mts | MediaType::Mjs | MediaType::Dmts => Ok(false), - MediaType::Cjs | MediaType::Cts | MediaType::Dcts => Ok(true), + MediaType::Mts | MediaType::Mjs | MediaType::Dmts => Some(ModuleKind::Esm), + MediaType::Cjs | MediaType::Cts | MediaType::Dcts => Some(ModuleKind::Cjs), + MediaType::Dts => { + // dts files are always determined based on the package.json because + // they contain imports/exports even when considered CJS + if let Some(value) = self.known.get(specifier).map(|v| v.clone()) { + Some(value) + } else { + let value = self.check_based_on_pkg_json(specifier).ok(); + if let Some(value) = value { + self.known.insert(specifier.clone(), value.clone()); + } + Some(value.unwrap_or(ModuleKind::Esm)) + } + } MediaType::JavaScript | MediaType::Jsx | MediaType::TypeScript | MediaType::Tsx - | MediaType::Dts // treat these as unknown | MediaType::Json | MediaType::Css | MediaType::Wasm | MediaType::SourceMap | MediaType::Unknown => { - if let Some(value) = self.get_known_kind(specifier) { - let is_known_cjs = value.is_cjs(); - if is_known_cjs && is_script == Some(false) { + if let Some(value) = self.known.get(specifier).map(|v| v.clone()) { + if value.is_cjs() && is_script == Some(false) { // we now know this is actually esm self.known.insert(specifier.clone(), ModuleKind::Esm); - return Ok(false); + Some(ModuleKind::Esm) } else { - return Ok(is_known_cjs); - } - } - - if let Some(is_script) = is_script { - if !is_script { - self.known.insert(specifier.clone(), ModuleKind::Esm); - return Ok(false); - } - } - - if self.in_npm_pkg_checker.in_npm_package(specifier) { - if let Some(pkg_json) = - self.pkg_json_resolver.get_closest_package_json(specifier)? - { - let is_file_location_cjs = pkg_json.typ != "module"; - if let Some(is_script) = is_script { - let module_kind = ModuleKind::from_is_cjs(is_script && is_file_location_cjs); - self.known.insert(specifier.clone(), module_kind); - return Ok(module_kind.is_cjs()); - } else { - return Ok(is_file_location_cjs); - } - } - } else if self.unstable_detect_cjs { - if let Some(pkg_json) = - self.pkg_json_resolver.get_closest_package_json(specifier)? - { - let is_cjs_type = pkg_json.typ == "commonjs"; - if let Some(is_script) = is_script { - let module_kind = ModuleKind::from_is_cjs(is_script && is_cjs_type); - self.known.insert(specifier.clone(), module_kind); - return Ok(module_kind.is_cjs()); - } else { - return Ok(is_cjs_type); - } + Some(value) } + } else if is_script == Some(false) { + // we know this is esm + self.known.insert(specifier.clone(), ModuleKind::Esm); + Some(ModuleKind::Esm) + } else { + None } + } + } + } - Ok(false) + fn check_based_on_pkg_json( + &self, + specifier: &ModuleSpecifier, + ) -> Result { + if self.in_npm_pkg_checker.in_npm_package(specifier) { + if let Some(pkg_json) = + self.pkg_json_resolver.get_closest_package_json(specifier)? + { + let is_file_location_cjs = pkg_json.typ != "module"; + Ok(ModuleKind::from_is_cjs(is_file_location_cjs)) + } else { + Ok(ModuleKind::Cjs) + } + } else if self.unstable_detect_cjs { + if let Some(pkg_json) = + self.pkg_json_resolver.get_closest_package_json(specifier)? + { + let is_cjs_type = pkg_json.typ == "commonjs"; + Ok(ModuleKind::from_is_cjs(is_cjs_type)) + } else { + Ok(ModuleKind::Esm) } + } else { + Ok(ModuleKind::Esm) } } pub fn mark_kind(&self, specifier: ModuleSpecifier, kind: ModuleKind) { self.known.insert(specifier, kind); } - - fn get_known_kind(&self, specifier: &ModuleSpecifier) -> Option { - self.known.get(specifier).map(|v| v.clone()) - } } pub type CliSloppyImportsResolver = diff --git a/cli/standalone/mod.rs b/cli/standalone/mod.rs index 795b1b78f46555..5ff4240d4f9a52 100644 --- a/cli/standalone/mod.rs +++ b/cli/standalone/mod.rs @@ -71,7 +71,6 @@ use crate::npm::create_in_npm_pkg_checker; use crate::npm::CliByonmNpmResolverCreateOptions; use crate::npm::CliManagedInNpmPkgCheckerCreateOptions; use crate::npm::CliManagedNpmResolverCreateOptions; -use crate::npm::CliNodeRequireLoader; use crate::npm::CliNpmResolver; use crate::npm::CliNpmResolverCreateOptions; use crate::npm::CliNpmResolverManagedSnapshotOption; @@ -712,7 +711,7 @@ pub async fn run(data: StandaloneData) -> Result { node_code_translator: node_code_translator.clone(), node_resolver: cli_node_resolver.clone(), npm_module_loader: Arc::new(NpmModuleLoader::new( - cjs_tracker, + cjs_tracker.clone(), fs.clone(), node_code_translator, )), @@ -756,6 +755,7 @@ pub async fn run(data: StandaloneData) -> Result { }); let worker_factory = CliMainWorkerFactory::new( Arc::new(BlobStore::default()), + cjs_tracker, // Code cache is not supported for standalone binary yet. None, feature_checker, @@ -782,7 +782,6 @@ pub async fn run(data: StandaloneData) -> Result { inspect_wait: false, strace_ops: None, is_inspecting: false, - is_npm_main: main_module.scheme() == "npm", skip_op_registration: true, location: metadata.location, argv0: NpmPackageReqReference::from_specifier(&main_module) diff --git a/cli/tools/check.rs b/cli/tools/check.rs index 4baa6cfdf96b28..682857a83f936a 100644 --- a/cli/tools/check.rs +++ b/cli/tools/check.rs @@ -25,11 +25,13 @@ use crate::args::TypeCheckMode; use crate::cache::CacheDBHash; use crate::cache::Caches; use crate::cache::FastInsecureHasher; +use crate::cache::ModuleInfoCache; use crate::cache::TypeCheckCache; use crate::factory::CliFactory; use crate::graph_util::BuildFastCheckGraphOptions; use crate::graph_util::ModuleGraphBuilder; use crate::npm::CliNpmResolver; +use crate::resolver::CjsTracker; use crate::tsc; use crate::tsc::Diagnostics; use crate::util::extract; @@ -99,8 +101,10 @@ pub struct CheckOptions { pub struct TypeChecker { caches: Arc, + cjs_tracker: Arc, cli_options: Arc, module_graph_builder: Arc, + module_info_cache: Arc, node_resolver: Arc, npm_resolver: Arc, } @@ -108,15 +112,19 @@ pub struct TypeChecker { impl TypeChecker { pub fn new( caches: Arc, + cjs_tracker: Arc, cli_options: Arc, module_graph_builder: Arc, + module_info_cache: Arc, node_resolver: Arc, npm_resolver: Arc, ) -> Self { Self { caches, + cjs_tracker, cli_options, module_graph_builder, + module_info_cache, node_resolver, npm_resolver, } @@ -244,6 +252,8 @@ impl TypeChecker { graph: graph.clone(), hash_data, maybe_npm: Some(tsc::RequestNpmState { + cjs_tracker: self.cjs_tracker.clone(), + module_info_cache: self.module_info_cache.clone(), node_resolver: self.node_resolver.clone(), npm_resolver: self.npm_resolver.clone(), }), diff --git a/cli/tools/doc.rs b/cli/tools/doc.rs index 8dcf4f324c0468..e33da4efb294ac 100644 --- a/cli/tools/doc.rs +++ b/cli/tools/doc.rs @@ -96,7 +96,7 @@ pub async fn doc( let module_info_cache = factory.module_info_cache()?; let parsed_source_cache = factory.parsed_source_cache(); let capturing_parser = parsed_source_cache.as_capturing_parser(); - let analyzer = module_info_cache.as_module_analyzer(parsed_source_cache); + let analyzer = module_info_cache.as_module_analyzer(); let doc_nodes_by_url = match doc_flags.source_files { DocSourceFileFlag::Builtin => { diff --git a/cli/tsc/mod.rs b/cli/tsc/mod.rs index 69849032f969fc..f17b346c58c781 100644 --- a/cli/tsc/mod.rs +++ b/cli/tsc/mod.rs @@ -3,9 +3,11 @@ use crate::args::TsConfig; use crate::args::TypeCheckMode; use crate::cache::FastInsecureHasher; +use crate::cache::ModuleInfoCache; use crate::node; use crate::npm::CliNpmResolver; use crate::npm::ResolvePkgFolderFromDenoReqError; +use crate::resolver::CjsTracker; use crate::util::checksum; use crate::util::path::mapped_specifier_for_tsc; @@ -305,6 +307,9 @@ pub struct EmittedFile { #[derive(Debug)] pub struct RequestNpmState { + pub cjs_tracker: Arc, + /// Used for telling if the source of a module is a script. + pub module_info_cache: Arc, pub node_resolver: Arc, pub npm_resolver: Arc, } @@ -490,24 +495,43 @@ fn op_load_inner( ) -> Result, AnyError> { fn load_from_node_modules( specifier: &ModuleSpecifier, - node_resolver: Option<&NodeResolver>, + npm_state: Option<&RequestNpmState>, media_type: &mut MediaType, is_cjs: &mut bool, ) -> Result { *media_type = MediaType::from_specifier(specifier); - *is_cjs = node_resolver - .map(|node_resolver| { - match node_resolver.url_to_node_resolution(specifier.clone()) { - Ok(NodeResolution::CommonJs(_)) => true, - Ok(NodeResolution::Esm(_)) - | Ok(NodeResolution::BuiltIn(_)) - | Err(_) => false, - } - }) - .unwrap_or(false); let file_path = specifier.to_file_path().unwrap(); let code = std::fs::read_to_string(&file_path) .with_context(|| format!("Unable to load {}", file_path.display()))?; + *is_cjs = npm_state + .map(|npm_state| { + if let Some(module_kind) = + npm_state.cjs_tracker.get_known_kind(specifier) + { + return module_kind.is_cjs(); + } else { + let maybe_is_script = npm_state + .module_info_cache + .as_module_analyzer() + .analyze_sync(specifier, &code, *media_type) + .ok() + .map(|info| info.is_script); + maybe_is_script + .and_then(|is_script| { + npm_state + .cjs_tracker + .is_cjs_with_known_is_script(specifier, is_script) + .ok() + }) + .unwrap_or_else(|| { + npm_state + .cjs_tracker + .is_maybe_cjs(specifier) + .unwrap_or(false) + }) + } + }) + .unwrap_or(false); Ok(code) } @@ -583,7 +607,7 @@ fn op_load_inner( ); Some(Cow::Owned(load_from_node_modules( &specifier, - state.maybe_npm.as_ref().map(|n| n.node_resolver.as_ref()), + state.maybe_npm.as_ref(), &mut media_type, &mut is_cjs, )?)) @@ -596,7 +620,7 @@ fn op_load_inner( { Some(Cow::Owned(load_from_node_modules( specifier, - Some(npm.node_resolver.as_ref()), + Some(npm), &mut media_type, &mut is_cjs, )?)) diff --git a/cli/worker.rs b/cli/worker.rs index f150242b3fa262..7107cb18bd33cc 100644 --- a/cli/worker.rs +++ b/cli/worker.rs @@ -54,6 +54,7 @@ use crate::args::DenoSubcommand; use crate::args::StorageKeyResolver; use crate::errors; use crate::npm::CliNpmResolver; +use crate::resolver::CjsTracker; use crate::util::checksum; use crate::util::file_watcher::WatcherCommunicator; use crate::util::file_watcher::WatcherRestartMode; @@ -111,7 +112,6 @@ pub struct CliMainWorkerOptions { pub inspect_wait: bool, pub strace_ops: Option>, pub is_inspecting: bool, - pub is_npm_main: bool, pub location: Option, pub argv0: Option, pub node_debug: Option, @@ -129,6 +129,7 @@ pub struct CliMainWorkerOptions { struct SharedWorkerState { blob_store: Arc, broadcast_channel: InMemoryBroadcastChannel, + cjs_tracker: Arc, code_cache: Option>, compiled_wasm_module_store: CompiledWasmModuleStore, feature_checker: Arc, @@ -168,7 +169,6 @@ impl SharedWorkerState { pub struct CliMainWorker { main_module: ModuleSpecifier, - is_main_cjs: bool, worker: MainWorker, shared: Arc, } @@ -190,17 +190,7 @@ impl CliMainWorker { log::debug!("main_module {}", self.main_module); - if self.is_main_cjs { - deno_node::load_cjs_module( - &mut self.worker.js_runtime, - &self.main_module.to_file_path().unwrap().to_string_lossy(), - true, - self.shared.options.inspect_brk, - )?; - } else { - self.execute_main_module_possibly_with_npm().await?; - } - + self.execute_main_module_possibly_with_npm().await?; self.worker.dispatch_load_event()?; loop { @@ -288,22 +278,7 @@ impl CliMainWorker { /// Execute the given main module emitting load and unload events before and after execution /// respectively. pub async fn execute(&mut self) -> Result<(), AnyError> { - if self.inner.is_main_cjs { - deno_node::load_cjs_module( - &mut self.inner.worker.js_runtime, - &self - .inner - .main_module - .to_file_path() - .unwrap() - .to_string_lossy(), - true, - self.inner.shared.options.inspect_brk, - )?; - } else { - self.inner.execute_main_module_possibly_with_npm().await?; - } - + self.inner.execute_main_module_possibly_with_npm().await?; self.inner.worker.dispatch_load_event()?; self.pending_unload = true; @@ -431,6 +406,7 @@ impl CliMainWorkerFactory { #[allow(clippy::too_many_arguments)] pub fn new( blob_store: Arc, + cjs_tracker: Arc, code_cache: Option>, feature_checker: Arc, fs: Arc, @@ -451,6 +427,7 @@ impl CliMainWorkerFactory { shared: Arc::new(SharedWorkerState { blob_store, broadcast_channel: Default::default(), + cjs_tracker, code_cache, compiled_wasm_module_store: Default::default(), feature_checker, @@ -503,7 +480,7 @@ impl CliMainWorkerFactory { } = shared .module_loader_factory .create_for_main(permissions.clone()); - let (main_module, is_main_cjs) = if let Ok(package_ref) = + let main_module = if let Ok(package_ref) = NpmPackageReqReference::from_specifier(&main_module) { if let Some(npm_resolver) = shared.npm_resolver.as_managed() { @@ -523,9 +500,8 @@ impl CliMainWorkerFactory { package_ref.req(), &referrer, )?; - let node_resolution = self + let main_module = self .resolve_binary_entrypoint(&package_folder, package_ref.sub_path())?; - let is_main_cjs = matches!(node_resolution, NodeResolution::CommonJs(_)); if let Some(lockfile) = &shared.maybe_lockfile { // For npm binary commands, ensure that the lockfile gets updated @@ -534,16 +510,9 @@ impl CliMainWorkerFactory { lockfile.write_if_changed()?; } - (node_resolution.into_url(), is_main_cjs) - } else if shared.options.is_npm_main - || shared.node_resolver.in_npm_package(&main_module) - { - let node_resolution = - shared.node_resolver.url_to_node_resolution(main_module)?; - let is_main_cjs = matches!(node_resolution, NodeResolution::CommonJs(_)); - (node_resolution.into_url(), is_main_cjs) + main_module } else { - (main_module, false) + main_module }; let maybe_inspector_server = shared.maybe_inspector_server.clone(); @@ -672,7 +641,6 @@ impl CliMainWorkerFactory { Ok(CliMainWorker { main_module, - is_main_cjs, worker, shared: shared.clone(), }) @@ -682,19 +650,19 @@ impl CliMainWorkerFactory { &self, package_folder: &Path, sub_path: Option<&str>, - ) -> Result { + ) -> Result { match self .shared .node_resolver .resolve_binary_export(package_folder, sub_path) { - Ok(node_resolution) => Ok(node_resolution), + Ok(specifier) => Ok(specifier.into_url()), Err(original_err) => { // if the binary entrypoint was not found, fallback to regular node resolution let result = self.resolve_binary_entrypoint_fallback(package_folder, sub_path); match result { - Ok(Some(resolution)) => Ok(resolution), + Ok(Some(specifier)) => Ok(specifier), Ok(None) => Err(original_err.into()), Err(fallback_err) => { bail!("{:#}\n\nFallback failed: {:#}", original_err, fallback_err) @@ -709,7 +677,7 @@ impl CliMainWorkerFactory { &self, package_folder: &Path, sub_path: Option<&str>, - ) -> Result, AnyError> { + ) -> Result, AnyError> { // only fallback if the user specified a sub path if sub_path.is_none() { // it's confusing to users if the package doesn't have any binary @@ -735,7 +703,7 @@ impl CliMainWorkerFactory { .map(|p| p.exists()) .unwrap_or(false) { - Ok(Some(resolution)) + Ok(Some(resolution.into_url())) } else { bail!("Cannot find module '{}'", specifier) } diff --git a/ext/node/lib.rs b/ext/node/lib.rs index 7750390dc7f5d0..085208e721c40a 100644 --- a/ext/node/lib.rs +++ b/ext/node/lib.rs @@ -9,13 +9,11 @@ use std::path::Path; use std::path::PathBuf; use deno_core::error::AnyError; -use deno_core::located_script_name; use deno_core::op2; use deno_core::url::Url; #[allow(unused_imports)] use deno_core::v8; use deno_core::v8::ExternalReference; -use deno_core::JsRuntime; use deno_fs::sync::MaybeSend; use deno_fs::sync::MaybeSync; use node_resolver::NpmResolverRc; @@ -765,29 +763,6 @@ deno_core::extension!(deno_node, }, ); -pub fn load_cjs_module( - js_runtime: &mut JsRuntime, - module: &str, - main: bool, - inspect_brk: bool, -) -> Result<(), AnyError> { - fn escape_for_single_quote_string(text: &str) -> String { - text.replace('\\', r"\\").replace('\'', r"\'") - } - - let source_code = format!( - r#"(function loadCjsModule(moduleName, isMain, inspectBrk) {{ - Deno[Deno.internal].node.loadCjsModule(moduleName, isMain, inspectBrk); - }})('{module}', {main}, {inspect_brk});"#, - main = main, - module = escape_for_single_quote_string(module), - inspect_brk = inspect_brk, - ); - - js_runtime.execute_script(located_script_name!(), source_code)?; - Ok(()) -} - pub type NodeResolver = node_resolver::NodeResolver; #[allow(clippy::disallowed_types)] pub type NodeResolverRc = diff --git a/ext/node/ops/worker_threads.rs b/ext/node/ops/worker_threads.rs index 53a256b973e520..d2e57588265360 100644 --- a/ext/node/ops/worker_threads.rs +++ b/ext/node/ops/worker_threads.rs @@ -4,14 +4,12 @@ use deno_core::op2; use deno_core::url::Url; use deno_core::OpState; use deno_fs::FileSystemRc; -use node_resolver::NodeResolution; use std::borrow::Cow; use std::path::Path; use std::path::PathBuf; use crate::NodePermissions; use crate::NodeRequireLoaderRc; -use crate::NodeResolverRc; #[must_use = "the resolved return value to mitigate time-of-check to time-of-use issues"] fn ensure_read_permission<'a, P>( @@ -42,14 +40,11 @@ pub enum WorkerThreadsFilenameError { UrlToPath, #[error("File not found [{0:?}]")] FileNotFound(PathBuf), - #[error("Neither ESM nor CJS")] - NeitherEsmNorCjs, - #[error("{0}")] - UrlToNodeResolution(node_resolver::errors::UrlToNodeResolutionError), #[error(transparent)] Fs(#[from] deno_io::fs::FsError), } +// todo(dsherret): we should remove this and do all this work inside op_create_worker #[op2] #[string] pub fn op_worker_threads_filename

( @@ -88,30 +83,5 @@ where url_path.to_path_buf(), )); } - let node_resolver = state.borrow::(); - match node_resolver - .url_to_node_resolution(url) - .map_err(WorkerThreadsFilenameError::UrlToNodeResolution)? - { - NodeResolution::Esm(u) => Ok(u.to_string()), - NodeResolution::CommonJs(u) => wrap_cjs(u), - NodeResolution::BuiltIn(_) => { - Err(WorkerThreadsFilenameError::NeitherEsmNorCjs) - } - } -} - -/// -/// Wrap a CJS file-URL and the required setup in a stringified `data:`-URL -/// -fn wrap_cjs(url: Url) -> Result { - let path = url - .to_file_path() - .map_err(|_| WorkerThreadsFilenameError::UrlToPath)?; - let filename = path.file_name().unwrap().to_string_lossy(); - Ok(format!( - "data:text/javascript,import {{ createRequire }} from \"node:module\";\ - const require = createRequire(\"{}\"); require(\"./{}\");", - url, filename, - )) + Ok(url.to_string()) } diff --git a/runtime/errors.rs b/runtime/errors.rs index b4796c92083074..b05d4a1d15c08a 100644 --- a/runtime/errors.rs +++ b/runtime/errors.rs @@ -1048,8 +1048,6 @@ mod node { WorkerThreadsFilenameError::UrlToPathString => "Error", WorkerThreadsFilenameError::UrlToPath => "Error", WorkerThreadsFilenameError::FileNotFound(_) => "Error", - WorkerThreadsFilenameError::NeitherEsmNorCjs => "Error", - WorkerThreadsFilenameError::UrlToNodeResolution(_) => "Error", WorkerThreadsFilenameError::Fs(e) => super::get_fs_error(e), } } From e513f6a8ab9e9a3c3e96575932cee47442716c8d Mon Sep 17 00:00:00 2001 From: David Sherret Date: Tue, 29 Oct 2024 17:35:13 -0400 Subject: [PATCH 22/42] commit and going to revert to simplify --- cli/cache/module_info.rs | 2 +- cli/factory.rs | 7 ++-- cli/tools/check.rs | 11 ++---- cli/tsc/mod.rs | 76 +++++++++++++++++++++++++--------------- 4 files changed, 57 insertions(+), 39 deletions(-) diff --git a/cli/cache/module_info.rs b/cli/cache/module_info.rs index 2c6dc54f3e2d17..92528d500dd296 100644 --- a/cli/cache/module_info.rs +++ b/cli/cache/module_info.rs @@ -190,8 +190,8 @@ impl<'a> ModuleInfoCacheModuleAnalyzer<'a> { pub fn analyze_sync( &self, specifier: &ModuleSpecifier, - source: &str, media_type: MediaType, + source: &str, ) -> Result { // attempt to load from the cache let source_hash = CacheDBHash::from_source(&source); diff --git a/cli/factory.rs b/cli/factory.rs index 560b71bc5eab4e..509044db88032a 100644 --- a/cli/factory.rs +++ b/cli/factory.rs @@ -55,6 +55,7 @@ use crate::tools::check::TypeChecker; use crate::tools::coverage::CoverageCollector; use crate::tools::lint::LintRuleProvider; use crate::tools::run::hmr::HmrRunner; +use crate::tsc::TypeCheckingCjsTracker; use crate::util::file_watcher::WatcherCommunicator; use crate::util::fs::canonicalize_path_maybe_not_exists; use crate::util::progress_bar::ProgressBar; @@ -680,10 +681,12 @@ impl CliFactory { let cli_options = self.cli_options()?; Ok(Arc::new(TypeChecker::new( self.caches()?.clone(), - self.cjs_tracker()?.clone(), + Arc::new(TypeCheckingCjsTracker::new( + self.cjs_tracker()?.clone(), + self.module_info_cache()?.clone(), + )), cli_options.clone(), self.module_graph_builder().await?.clone(), - self.module_info_cache()?.clone(), self.node_resolver().await?.clone(), self.npm_resolver().await?.clone(), ))) diff --git a/cli/tools/check.rs b/cli/tools/check.rs index 682857a83f936a..d880278884457c 100644 --- a/cli/tools/check.rs +++ b/cli/tools/check.rs @@ -25,15 +25,14 @@ use crate::args::TypeCheckMode; use crate::cache::CacheDBHash; use crate::cache::Caches; use crate::cache::FastInsecureHasher; -use crate::cache::ModuleInfoCache; use crate::cache::TypeCheckCache; use crate::factory::CliFactory; use crate::graph_util::BuildFastCheckGraphOptions; use crate::graph_util::ModuleGraphBuilder; use crate::npm::CliNpmResolver; -use crate::resolver::CjsTracker; use crate::tsc; use crate::tsc::Diagnostics; +use crate::tsc::TypeCheckingCjsTracker; use crate::util::extract; use crate::util::path::to_percent_decoded_str; @@ -101,10 +100,9 @@ pub struct CheckOptions { pub struct TypeChecker { caches: Arc, - cjs_tracker: Arc, + cjs_tracker: Arc, cli_options: Arc, module_graph_builder: Arc, - module_info_cache: Arc, node_resolver: Arc, npm_resolver: Arc, } @@ -112,10 +110,9 @@ pub struct TypeChecker { impl TypeChecker { pub fn new( caches: Arc, - cjs_tracker: Arc, + cjs_tracker: Arc, cli_options: Arc, module_graph_builder: Arc, - module_info_cache: Arc, node_resolver: Arc, npm_resolver: Arc, ) -> Self { @@ -124,7 +121,6 @@ impl TypeChecker { cjs_tracker, cli_options, module_graph_builder, - module_info_cache, node_resolver, npm_resolver, } @@ -253,7 +249,6 @@ impl TypeChecker { hash_data, maybe_npm: Some(tsc::RequestNpmState { cjs_tracker: self.cjs_tracker.clone(), - module_info_cache: self.module_info_cache.clone(), node_resolver: self.node_resolver.clone(), npm_resolver: self.npm_resolver.clone(), }), diff --git a/cli/tsc/mod.rs b/cli/tsc/mod.rs index f17b346c58c781..17cf3f853938e4 100644 --- a/cli/tsc/mod.rs +++ b/cli/tsc/mod.rs @@ -305,11 +305,55 @@ pub struct EmittedFile { pub media_type: MediaType, } +#[derive(Debug)] +pub struct TypeCheckingCjsTracker { + cjs_tracker: Arc, + module_info_cache: Arc, +} + +impl TypeCheckingCjsTracker { + pub fn new( + cjs_tracker: Arc, + module_info_cache: Arc, + ) -> Self { + Self { + cjs_tracker, + module_info_cache, + } + } + + pub fn is_cjs( + &self, + specifier: &ModuleSpecifier, + media_type: MediaType, + code: &str, + ) -> bool { + if let Some(module_kind) = self.cjs_tracker.get_known_kind(specifier) { + module_kind.is_cjs() + } else { + let maybe_is_script = self + .module_info_cache + .as_module_analyzer() + .analyze_sync(specifier, media_type, &code) + .ok() + .map(|info| info.is_script); + maybe_is_script + .and_then(|is_script| { + self + .cjs_tracker + .is_cjs_with_known_is_script(specifier, is_script) + .ok() + }) + .unwrap_or_else(|| { + self.cjs_tracker.is_maybe_cjs(specifier).unwrap_or(false) + }) + } + } +} + #[derive(Debug)] pub struct RequestNpmState { - pub cjs_tracker: Arc, - /// Used for telling if the source of a module is a script. - pub module_info_cache: Arc, + pub cjs_tracker: Arc, pub node_resolver: Arc, pub npm_resolver: Arc, } @@ -505,31 +549,7 @@ fn op_load_inner( .with_context(|| format!("Unable to load {}", file_path.display()))?; *is_cjs = npm_state .map(|npm_state| { - if let Some(module_kind) = - npm_state.cjs_tracker.get_known_kind(specifier) - { - return module_kind.is_cjs(); - } else { - let maybe_is_script = npm_state - .module_info_cache - .as_module_analyzer() - .analyze_sync(specifier, &code, *media_type) - .ok() - .map(|info| info.is_script); - maybe_is_script - .and_then(|is_script| { - npm_state - .cjs_tracker - .is_cjs_with_known_is_script(specifier, is_script) - .ok() - }) - .unwrap_or_else(|| { - npm_state - .cjs_tracker - .is_maybe_cjs(specifier) - .unwrap_or(false) - }) - } + npm_state.cjs_tracker.is_cjs(specifier, *media_type, &code) }) .unwrap_or(false); Ok(code) From e7e129906fd22bb5cd8e545a8e9fd10651f71813 Mon Sep 17 00:00:00 2001 From: David Sherret Date: Tue, 29 Oct 2024 18:21:00 -0400 Subject: [PATCH 23/42] More passing tests --- cli/tools/compile.rs | 10 ----- cli/tsc/99_main_compiler.js | 2 + cli/tsc/mod.rs | 45 +++++++++++++------ resolvers/node/resolution.rs | 2 + tests/specs/compile/detect_cjs/__test__.jsonc | 4 +- tests/specs/compile/detect_cjs/compile.out | 1 - tests/specs/compile/detect_cjs/output.out | 3 +- tests/specs/run/require_esm/main.out | 2 +- 8 files changed, 40 insertions(+), 29 deletions(-) diff --git a/cli/tools/compile.rs b/cli/tools/compile.rs index 5a4a938bb90870..b3e9993379e0e7 100644 --- a/cli/tools/compile.rs +++ b/cli/tools/compile.rs @@ -53,16 +53,6 @@ pub async fn compile( ); } - if cli_options.unstable_detect_cjs() { - log::warn!( - concat!( - "{} --unstable-detect-cjs is not properly supported in deno compile. ", - "The compiled executable may encounter runtime errors.", - ), - crate::colors::yellow("Warning"), - ); - } - let output_path = resolve_compile_executable_output_path( http_client, &compile_flags, diff --git a/cli/tsc/99_main_compiler.js b/cli/tsc/99_main_compiler.js index 6011dece76b6c6..bdc4340e330f7d 100644 --- a/cli/tsc/99_main_compiler.js +++ b/cli/tsc/99_main_compiler.js @@ -845,6 +845,8 @@ delete Object.prototype.__proto__; jqueryMessage, "Cannot_find_name_0_Do_you_need_to_install_type_definitions_for_jQuery_Try_npm_i_save_dev_types_Slash_2592": jqueryMessage, + "Module_0_was_resolved_to_1_but_allowArbitraryExtensions_is_not_set_6263": + "Module '{0}' was resolved to '{1}', but importing these modules is not supported.", }; })()); diff --git a/cli/tsc/mod.rs b/cli/tsc/mod.rs index 17cf3f853938e4..7c56869900f54c 100644 --- a/cli/tsc/mod.rs +++ b/cli/tsc/mod.rs @@ -305,6 +305,22 @@ pub struct EmittedFile { pub media_type: MediaType, } +pub fn into_specifier_and_media_type( + specifier: Option, +) -> (ModuleSpecifier, MediaType) { + match specifier { + Some(specifier) => { + let media_type = MediaType::from_specifier(&specifier); + + (specifier, media_type) + } + None => ( + Url::parse("internal:///missing_dependency.d.ts").unwrap(), + MediaType::Dts, + ), + } +} + #[derive(Debug)] pub struct TypeCheckingCjsTracker { cjs_tracker: Arc, @@ -789,7 +805,13 @@ fn op_resolve_inner( } } }; - (specifier_str, media_type.as_ts_extension()) + ( + specifier_str, + match media_type { + MediaType::Css => ".js", + media_type => media_type.as_ts_extension(), + }, + ) } None => ( MISSING_DEPENDENCY_SPECIFIER.to_string(), @@ -861,30 +883,26 @@ fn resolve_graph_specifier_types( NodeResolutionMode::Types, ); let maybe_resolution = match res_result { - Ok(res) => Some(res), + Ok(res) => Some(res.into_url()), Err(err) => match err.code() { NodeJsErrorCode::ERR_TYPES_NOT_FOUND | NodeJsErrorCode::ERR_MODULE_NOT_FOUND => None, _ => return Err(err.into()), }, }; - Ok(Some(NodeResolution::into_specifier_and_media_type( - maybe_resolution, - ))) + Ok(Some(into_specifier_and_media_type(maybe_resolution))) } else { Ok(None) } } Some(Module::External(module)) => { // we currently only use "External" for when the module is in an npm package - Ok(state.maybe_npm.as_ref().map(|npm| { + Ok(state.maybe_npm.as_ref().map(|_| { let specifier = node::resolve_specifier_into_node_modules( &module.specifier, &deno_fs::RealFs, ); - NodeResolution::into_specifier_and_media_type( - npm.node_resolver.url_to_node_resolution(specifier).ok(), - ) + into_specifier_and_media_type(Some(specifier)) })) } Some(Module::Node(_)) | None => Ok(None), @@ -915,7 +933,7 @@ fn resolve_non_graph_specifier_types( let node_resolver = &npm.node_resolver; if node_resolver.in_npm_package(referrer) { // we're in an npm package, so use node resolution - Ok(Some(NodeResolution::into_specifier_and_media_type( + Ok(Some(into_specifier_and_media_type( node_resolver .resolve( raw_specifier, @@ -923,7 +941,8 @@ fn resolve_non_graph_specifier_types( referrer_kind, NodeResolutionMode::Types, ) - .ok(), + .ok() + .map(|res| res.into_url()), ))) } else if let Ok(npm_req_ref) = NpmPackageReqReference::from_str(raw_specifier) @@ -950,8 +969,8 @@ fn resolve_non_graph_specifier_types( _ => return Err(err.into()), }, }; - Ok(Some(NodeResolution::into_specifier_and_media_type( - maybe_resolution, + Ok(Some(into_specifier_and_media_type( + maybe_resolution.map(|res| res.into_url()), ))) } else { Ok(None) diff --git a/resolvers/node/resolution.rs b/resolvers/node/resolution.rs index 6ad508a3b4135e..c10fe4a3c7de68 100644 --- a/resolvers/node/resolution.rs +++ b/resolvers/node/resolution.rs @@ -69,6 +69,7 @@ impl NodeResolutionMode { } } +// todo(THIS PR): REMOVE THIS #[derive(Debug)] pub enum NodeResolution { Esm(Url), @@ -91,6 +92,7 @@ impl NodeResolution { } } + // todo(THIS PR): REMOVE THIS pub fn into_specifier_and_media_type( resolution: Option, ) -> (Url, MediaType) { diff --git a/tests/specs/compile/detect_cjs/__test__.jsonc b/tests/specs/compile/detect_cjs/__test__.jsonc index 32bebb7a57b8b0..6d4368899b9201 100644 --- a/tests/specs/compile/detect_cjs/__test__.jsonc +++ b/tests/specs/compile/detect_cjs/__test__.jsonc @@ -9,7 +9,7 @@ "commandName": "./main", "args": [], "output": "output.out", - "exitCode": 1 + "exitCode": 0 }, { "if": "windows", "args": "compile --allow-read --output main.exe main.js", @@ -19,6 +19,6 @@ "commandName": "./main.exe", "args": [], "output": "output.out", - "exitCode": 1 + "exitCode": 0 }] } diff --git a/tests/specs/compile/detect_cjs/compile.out b/tests/specs/compile/detect_cjs/compile.out index 6509b7f29c6596..913e363c3ebdbf 100644 --- a/tests/specs/compile/detect_cjs/compile.out +++ b/tests/specs/compile/detect_cjs/compile.out @@ -1,3 +1,2 @@ -Warning --unstable-detect-cjs is not properly supported in deno compile. The compiled executable may encounter runtime errors. Check file:///[WILDLINE]/main.js Compile file:///[WILDLINE] diff --git a/tests/specs/compile/detect_cjs/output.out b/tests/specs/compile/detect_cjs/output.out index e1c27b8dcf4d2c..00750edc07d641 100644 --- a/tests/specs/compile/detect_cjs/output.out +++ b/tests/specs/compile/detect_cjs/output.out @@ -1,2 +1 @@ -error: Uncaught SyntaxError: The requested module './add.js' does not provide an export named 'add' - at (file:///[WILDLINE]) +3 diff --git a/tests/specs/run/require_esm/main.out b/tests/specs/run/require_esm/main.out index d17b1ead55680a..57b842b345f791 100644 --- a/tests/specs/run/require_esm/main.out +++ b/tests/specs/run/require_esm/main.out @@ -1,6 +1,6 @@ [Module: null prototype] { sync_js: 1 } [Module: null prototype] { sync_mjs: 1 } -error: Uncaught Error: Top-level await is not allowed in synchronous evaluation +error: Uncaught (in promise) Error: Top-level await is not allowed in synchronous evaluation at loadESMFromCJS (node:module:[WILDCARD]) at Module._compile (node:module:[WILDCARD]) at Object.Module._extensions..js (node:module:[WILDCARD]) From fd26e78e90baabb32d22f5b15e2b796782b0fbbe Mon Sep 17 00:00:00 2001 From: David Sherret Date: Tue, 29 Oct 2024 19:03:33 -0400 Subject: [PATCH 24/42] going to revert --- cli/cache/module_info.rs | 6 +- cli/lsp/documents.rs | 2 +- cli/lsp/language_server.rs | 1 + cli/lsp/resolver.rs | 120 +++++++++++++++++++++++++---------- cli/lsp/tsc.rs | 19 +++--- cli/module_loader.rs | 1 - cli/resolver.rs | 9 +++ cli/tsc/mod.rs | 58 +++++++++++++---- resolvers/node/resolution.rs | 37 ----------- 9 files changed, 159 insertions(+), 94 deletions(-) diff --git a/cli/cache/module_info.rs b/cli/cache/module_info.rs index 92528d500dd296..fb2e36181ffd20 100644 --- a/cli/cache/module_info.rs +++ b/cli/cache/module_info.rs @@ -191,10 +191,10 @@ impl<'a> ModuleInfoCacheModuleAnalyzer<'a> { &self, specifier: &ModuleSpecifier, media_type: MediaType, - source: &str, + source: &Arc, ) -> Result { // attempt to load from the cache - let source_hash = CacheDBHash::from_source(&source); + let source_hash = CacheDBHash::from_source(source); if let Some(info) = self.load_cached_module_info(specifier, media_type, source_hash) { @@ -205,7 +205,7 @@ impl<'a> ModuleInfoCacheModuleAnalyzer<'a> { let parser = self.parsed_source_cache.as_capturing_parser(); let analyzer = ParserModuleAnalyzer::new(&parser); let module_info = - analyzer.analyze_sync(&specifier, Arc::from(source), media_type)?; + analyzer.analyze_sync(&specifier, source.clone(), media_type)?; // then attempt to cache it self.save_module_info_to_cache( diff --git a/cli/lsp/documents.rs b/cli/lsp/documents.rs index b7578e9614d428..5953faad2278ba 100644 --- a/cli/lsp/documents.rs +++ b/cli/lsp/documents.rs @@ -1561,7 +1561,7 @@ fn parse_source( text: Arc, media_type: MediaType, ) -> ParsedSourceResult { - deno_ast::parse_module(deno_ast::ParseParams { + deno_ast::parse_program(deno_ast::ParseParams { specifier, text, media_type, diff --git a/cli/lsp/language_server.rs b/cli/lsp/language_server.rs index 6c1e6d4ed0f1f6..e2e11bd128d95d 100644 --- a/cli/lsp/language_server.rs +++ b/cli/lsp/language_server.rs @@ -110,6 +110,7 @@ use crate::tools::fmt::format_file; use crate::tools::fmt::format_parsed_source; use crate::tools::upgrade::check_for_upgrades_for_lsp; use crate::tools::upgrade::upgrade_check_enabled; +use crate::tsc::TypeCheckingCjsTracker; use crate::util::fs::remove_dir_all_if_exists; use crate::util::path::is_importable_ext; use crate::util::path::to_percent_decoded_str; diff --git a/cli/lsp/resolver.rs b/cli/lsp/resolver.rs index 4d5e15951ca627..adc36a62fd3ef3 100644 --- a/cli/lsp/resolver.rs +++ b/cli/lsp/resolver.rs @@ -45,7 +45,6 @@ use crate::lsp::config::Config; use crate::lsp::config::ConfigData; use crate::lsp::logging::lsp_warn; use crate::npm::create_cli_npm_resolver_for_lsp; -use crate::npm::create_in_npm_pkg_checker; use crate::npm::CliByonmNpmResolverCreateOptions; use crate::npm::CliManagedInNpmPkgCheckerCreateOptions; use crate::npm::CliManagedNpmResolverCreateOptions; @@ -61,11 +60,14 @@ use crate::resolver::CliGraphResolver; use crate::resolver::CliGraphResolverOptions; use crate::resolver::CliNodeResolver; use crate::resolver::WorkerCliNpmGraphResolver; +use crate::tsc::into_specifier_and_media_type; +use crate::tsc::TypeCheckingCjsTracker; use crate::util::progress_bar::ProgressBar; use crate::util::progress_bar::ProgressBarStyle; #[derive(Debug, Clone)] struct LspScopeResolver { + cjs_tracker: Option>, graph_resolver: Arc, jsr_resolver: Option>, npm_resolver: Option>, @@ -79,6 +81,7 @@ struct LspScopeResolver { impl Default for LspScopeResolver { fn default() -> Self { Self { + cjs_tracker: None, graph_resolver: create_graph_resolver(None, None, None), jsr_resolver: None, npm_resolver: None, @@ -99,6 +102,7 @@ impl LspScopeResolver { ) -> Self { let mut npm_resolver = None; let mut node_resolver = None; + let mut type_checking_cjs_tracker = None; let fs = Arc::new(deno_fs::RealFs); let pkg_json_resolver = Arc::new(PackageJsonResolver::new( deno_runtime::deno_node::DenoFsNodeResolverEnv::new(fs.clone()), @@ -111,13 +115,23 @@ impl LspScopeResolver { &pkg_json_resolver, ) .await; - node_resolver = npm_resolver.as_ref().map(|npm_resolver| { - create_node_resolver( + if let Some(npm_resolver) = &npm_resolver { + let in_npm_pkg_checker = create_in_npm_pkg_checker(npm_resolver); + let cjs_tracker = create_cjs_tracker( + in_npm_pkg_checker.clone(), + pkg_json_resolver.clone(), + ); + type_checking_cjs_tracker = Some(Arc::new( + TypeCheckingCjsTracker::new_for_lsp(cjs_tracker.clone()), + )); + node_resolver = Some(create_node_resolver( + cjs_tracker, fs.clone(), + in_npm_pkg_checker, npm_resolver, pkg_json_resolver.clone(), - ) - }); + )); + } } let graph_resolver = create_graph_resolver( config_data.map(|d| d.as_ref()), @@ -154,6 +168,7 @@ impl LspScopeResolver { }) .unwrap_or_default(); Self { + cjs_tracker: type_checking_cjs_tracker, graph_resolver, jsr_resolver, npm_resolver, @@ -172,15 +187,32 @@ impl LspScopeResolver { let pkg_json_resolver = Arc::new(PackageJsonResolver::new( deno_runtime::deno_node::DenoFsNodeResolverEnv::new(fs.clone()), )); - let node_resolver = npm_resolver.as_ref().map(|npm_resolver| { - create_node_resolver(fs, npm_resolver, pkg_json_resolver.clone()) - }); + let mut node_resolver = None; + let mut type_checking_cjs_tracker = None; + if let Some(npm_resolver) = &npm_resolver { + let in_npm_pkg_checker = create_in_npm_pkg_checker(npm_resolver); + let cjs_tracker = create_cjs_tracker( + in_npm_pkg_checker.clone(), + pkg_json_resolver.clone(), + ); + type_checking_cjs_tracker = Some(Arc::new( + TypeCheckingCjsTracker::new_for_lsp(cjs_tracker.clone()), + )); + node_resolver = Some(create_node_resolver( + cjs_tracker, + fs, + in_npm_pkg_checker, + npm_resolver, + pkg_json_resolver.clone(), + )); + } let graph_resolver = create_graph_resolver( self.config_data.as_deref(), npm_resolver.as_ref(), node_resolver.as_ref(), ); Arc::new(Self { + cjs_tracker: type_checking_cjs_tracker, graph_resolver, jsr_resolver: self.jsr_resolver.clone(), npm_resolver, @@ -289,6 +321,14 @@ impl LspResolver { resolver.graph_resolver.create_graph_npm_resolver() } + pub fn maybe_cjs_tracker( + &self, + file_referrer: Option<&ModuleSpecifier>, + ) -> Option<&Arc> { + let resolver = self.get_scope_resolver(file_referrer); + resolver.cjs_tracker.as_ref() + } + pub fn maybe_node_resolver( &self, file_referrer: Option<&ModuleSpecifier>, @@ -364,9 +404,10 @@ impl LspResolver { ) -> Option<(ModuleSpecifier, MediaType)> { let resolver = self.get_scope_resolver(file_referrer); let node_resolver = resolver.node_resolver.as_ref()?; - Some(NodeResolution::into_specifier_and_media_type(Some( + Some(into_specifier_and_media_type(Some( node_resolver .resolve_req_reference(req_ref, referrer, NodeResolutionMode::Types) + .map(|res| res.into_url()) .ok()?, ))) } @@ -402,7 +443,7 @@ impl LspResolver { let resolution = node_resolver .url_to_node_resolution(specifier.clone()) .ok()?; - Some(NodeResolution::into_specifier_and_media_type(Some(resolution)).1) + Some(into_specifier_and_media_type(Some(resolution.into_url())).1) } pub fn is_bare_package_json_dep( @@ -546,41 +587,54 @@ async fn create_npm_resolver( Some(create_cli_npm_resolver_for_lsp(options).await) } +fn create_cjs_tracker( + in_npm_pkg_checker: Arc, + pkg_json_resolver: Arc, +) -> Arc { + Arc::new(CjsTracker::new( + in_npm_pkg_checker, + pkg_json_resolver, + CjsTrackerOptions { + // todo(dsherret): support in the lsp by stabilizing the feature + // so that we don't have to pipe the config in here + unstable_detect_cjs: false, + }, + )) +} + +fn create_in_npm_pkg_checker( + npm_resolver: &Arc, +) -> Arc { + crate::npm::create_in_npm_pkg_checker(match npm_resolver.as_inner() { + crate::npm::InnerCliNpmResolverRef::Byonm(_) => { + CreateInNpmPkgCheckerOptions::Byonm + } + crate::npm::InnerCliNpmResolverRef::Managed(m) => { + CreateInNpmPkgCheckerOptions::Managed( + CliManagedInNpmPkgCheckerCreateOptions { + root_cache_dir_url: m.global_cache_root_url(), + maybe_node_modules_path: m.maybe_node_modules_path(), + }, + ) + } + }) +} + fn create_node_resolver( + cjs_tracker: Arc, fs: Arc, + in_npm_pkg_checker: Arc, npm_resolver: &Arc, pkg_json_resolver: Arc, ) -> Arc { - let in_npm_pkg_checker = - create_in_npm_pkg_checker(match npm_resolver.as_inner() { - crate::npm::InnerCliNpmResolverRef::Byonm(_) => { - CreateInNpmPkgCheckerOptions::Byonm - } - crate::npm::InnerCliNpmResolverRef::Managed(m) => { - CreateInNpmPkgCheckerOptions::Managed( - CliManagedInNpmPkgCheckerCreateOptions { - root_cache_dir_url: m.global_cache_root_url(), - maybe_node_modules_path: m.maybe_node_modules_path(), - }, - ) - } - }); let node_resolver_inner = Arc::new(NodeResolver::new( deno_runtime::deno_node::DenoFsNodeResolverEnv::new(fs.clone()), in_npm_pkg_checker.clone(), npm_resolver.clone().into_npm_resolver(), pkg_json_resolver.clone(), )); - let cjs_tracker = Arc::new(CjsTracker::new( - in_npm_pkg_checker.clone(), - pkg_json_resolver, - CjsTrackerOptions { - // todo(dsherret): support in the lsp - unstable_detect_cjs: false, - }, - )); Arc::new(CliNodeResolver::new( - cjs_tracker, + cjs_tracker.clone(), fs, in_npm_pkg_checker, node_resolver_inner, diff --git a/cli/lsp/tsc.rs b/cli/lsp/tsc.rs index 0f31d7dd3b0ae6..4b71ba66bb5337 100644 --- a/cli/lsp/tsc.rs +++ b/cli/lsp/tsc.rs @@ -4362,14 +4362,17 @@ fn op_load<'s>( None } else { let asset_or_document = state.get_asset_or_document(&specifier); - asset_or_document.map(|doc| LoadResponse { - data: doc.text(), - script_kind: crate::tsc::as_ts_script_kind(doc.media_type()), - version: state.script_version(&specifier), - is_cjs: matches!( - doc.media_type(), - MediaType::Cjs | MediaType::Cts | MediaType::Dcts - ), + asset_or_document.map(|doc| { + let maybe_cjs_tracker = + state.state_snapshot.resolver.maybe_cjs_tracker(specifier); + LoadResponse { + data: doc.text(), + script_kind: crate::tsc::as_ts_script_kind(doc.media_type()), + version: state.script_version(&specifier), + is_cjs: maybe_cjs_tracker + .map(|t| t.is_cjs(&specifier, doc.media_type(), &doc.text())) + .unwrap_or(false), + } }) }; diff --git a/cli/module_loader.rs b/cli/module_loader.rs index 3050aa2df5c5fc..1f70c6287b0032 100644 --- a/cli/module_loader.rs +++ b/cli/module_loader.rs @@ -36,7 +36,6 @@ use crate::tools::check; use crate::tools::check::TypeChecker; use crate::util::progress_bar::ProgressBar; use crate::util::text_encoding::code_without_source_map; -use crate::util::text_encoding::from_utf8_lossy_owned; use crate::util::text_encoding::source_map_from_code; use crate::worker::CreateModuleLoaderResult; use crate::worker::ModuleLoaderFactory; diff --git a/cli/resolver.rs b/cli/resolver.rs index 8ee85158b50b9d..e14050ab972f83 100644 --- a/cli/resolver.rs +++ b/cli/resolver.rs @@ -608,6 +608,15 @@ impl CjsTracker { pub fn mark_kind(&self, specifier: ModuleSpecifier, kind: ModuleKind) { self.known.insert(specifier, kind); } + + pub fn snapshot(&self) -> CjsTracker { + CjsTracker { + in_npm_pkg_checker: self.in_npm_pkg_checker.clone(), + pkg_json_resolver: self.pkg_json_resolver.clone(), + unstable_detect_cjs: self.unstable_detect_cjs, + known: self.known.clone(), + } + } } pub type CliSloppyImportsResolver = diff --git a/cli/tsc/mod.rs b/cli/tsc/mod.rs index 7c56869900f54c..6439b5754f616a 100644 --- a/cli/tsc/mod.rs +++ b/cli/tsc/mod.rs @@ -321,10 +321,34 @@ pub fn into_specifier_and_media_type( } } +trait IsScriptModuleAnalyzer: std::fmt::Debug + Send + Sync { + fn analyze_sync( + &self, + specifier: &ModuleSpecifier, + media_type: MediaType, + code: &Arc, + ) -> Option; +} + +impl IsScriptModuleAnalyzer for ModuleInfoCache { + fn analyze_sync( + &self, + specifier: &ModuleSpecifier, + media_type: MediaType, + code: &Arc, + ) -> Option { + self + .as_module_analyzer() + .analyze_sync(specifier, media_type, code) + .map(|info| info.is_script) + .ok() + } +} + #[derive(Debug)] pub struct TypeCheckingCjsTracker { cjs_tracker: Arc, - module_info_cache: Arc, + analyzer: Arc, } impl TypeCheckingCjsTracker { @@ -334,7 +358,14 @@ impl TypeCheckingCjsTracker { ) -> Self { Self { cjs_tracker, - module_info_cache, + analyzer: module_info_cache, + } + } + + pub fn new_for_lsp(cjs_tracker: Arc) -> Self { + Self { + cjs_tracker, + analyzer: module_info_cache, } } @@ -342,17 +373,13 @@ impl TypeCheckingCjsTracker { &self, specifier: &ModuleSpecifier, media_type: MediaType, - code: &str, + code: &Arc, ) -> bool { if let Some(module_kind) = self.cjs_tracker.get_known_kind(specifier) { module_kind.is_cjs() } else { - let maybe_is_script = self - .module_info_cache - .as_module_analyzer() - .analyze_sync(specifier, media_type, &code) - .ok() - .map(|info| info.is_script); + let maybe_is_script = + self.analyzer.analyze_sync(specifier, media_type, code); maybe_is_script .and_then(|is_script| { self @@ -365,6 +392,13 @@ impl TypeCheckingCjsTracker { }) } } + + pub fn snapshot(&self) -> TypeCheckingCjsTracker { + Self { + cjs_tracker: Arc::new(self.cjs_tracker.snapshot()), + analyzer: self.analyzer.clone(), + } + } } #[derive(Debug)] @@ -563,12 +597,14 @@ fn op_load_inner( let file_path = specifier.to_file_path().unwrap(); let code = std::fs::read_to_string(&file_path) .with_context(|| format!("Unable to load {}", file_path.display()))?; + let code: Arc = code.into(); *is_cjs = npm_state .map(|npm_state| { npm_state.cjs_tracker.is_cjs(specifier, *media_type, &code) }) .unwrap_or(false); - Ok(code) + // todo(dsherret): how to avoid cloning here? + Ok(code.to_string()) } let state = state.borrow_mut::(); @@ -808,7 +844,7 @@ fn op_resolve_inner( ( specifier_str, match media_type { - MediaType::Css => ".js", + MediaType::Css => ".js", // surface these as .js for typescript media_type => media_type.as_ts_extension(), }, ) diff --git a/resolvers/node/resolution.rs b/resolvers/node/resolution.rs index c10fe4a3c7de68..0c6cce41e4513e 100644 --- a/resolvers/node/resolution.rs +++ b/resolvers/node/resolution.rs @@ -91,43 +91,6 @@ impl NodeResolution { } } } - - // todo(THIS PR): REMOVE THIS - pub fn into_specifier_and_media_type( - resolution: Option, - ) -> (Url, MediaType) { - match resolution { - Some(NodeResolution::CommonJs(specifier)) => { - let media_type = MediaType::from_specifier(&specifier); - ( - specifier, - match media_type { - MediaType::JavaScript | MediaType::Jsx => MediaType::Cjs, - MediaType::TypeScript | MediaType::Tsx => MediaType::Cts, - MediaType::Dts => MediaType::Dcts, - _ => media_type, - }, - ) - } - Some(NodeResolution::Esm(specifier)) => { - let media_type = MediaType::from_specifier(&specifier); - ( - specifier, - match media_type { - MediaType::JavaScript | MediaType::Jsx => MediaType::Mjs, - MediaType::TypeScript | MediaType::Tsx => MediaType::Mts, - MediaType::Dts => MediaType::Dmts, - _ => media_type, - }, - ) - } - Some(resolution) => (resolution.into_url(), MediaType::Dts), - None => ( - Url::parse("internal:///missing_dependency.d.ts").unwrap(), - MediaType::Dts, - ), - } - } } #[allow(clippy::disallowed_types)] From d23a91a16d997b5339e0f90a011012eff0ba4f07 Mon Sep 17 00:00:00 2001 From: David Sherret Date: Tue, 29 Oct 2024 19:07:57 -0400 Subject: [PATCH 25/42] Add LspCjsTracker --- cli/lsp/resolver.rs | 57 ++++++++++++++++++++++++++++++++++----------- cli/lsp/tsc.rs | 14 ++++++++--- cli/tsc/mod.rs | 50 +++++++-------------------------------- 3 files changed, 63 insertions(+), 58 deletions(-) diff --git a/cli/lsp/resolver.rs b/cli/lsp/resolver.rs index adc36a62fd3ef3..c0c71ab70202a3 100644 --- a/cli/lsp/resolver.rs +++ b/cli/lsp/resolver.rs @@ -2,6 +2,7 @@ use dashmap::DashMap; use deno_ast::MediaType; +use deno_ast::ParsedSource; use deno_cache_dir::npm::NpmCacheDir; use deno_cache_dir::HttpCache; use deno_config::workspace::PackageJsonDepResolution; @@ -61,13 +62,12 @@ use crate::resolver::CliGraphResolverOptions; use crate::resolver::CliNodeResolver; use crate::resolver::WorkerCliNpmGraphResolver; use crate::tsc::into_specifier_and_media_type; -use crate::tsc::TypeCheckingCjsTracker; use crate::util::progress_bar::ProgressBar; use crate::util::progress_bar::ProgressBarStyle; #[derive(Debug, Clone)] struct LspScopeResolver { - cjs_tracker: Option>, + cjs_tracker: Option>, graph_resolver: Arc, jsr_resolver: Option>, npm_resolver: Option>, @@ -102,7 +102,7 @@ impl LspScopeResolver { ) -> Self { let mut npm_resolver = None; let mut node_resolver = None; - let mut type_checking_cjs_tracker = None; + let mut lsp_cjs_tracker = None; let fs = Arc::new(deno_fs::RealFs); let pkg_json_resolver = Arc::new(PackageJsonResolver::new( deno_runtime::deno_node::DenoFsNodeResolverEnv::new(fs.clone()), @@ -121,9 +121,8 @@ impl LspScopeResolver { in_npm_pkg_checker.clone(), pkg_json_resolver.clone(), ); - type_checking_cjs_tracker = Some(Arc::new( - TypeCheckingCjsTracker::new_for_lsp(cjs_tracker.clone()), - )); + lsp_cjs_tracker = + Some(Arc::new(LspCjsTracker::new(cjs_tracker.clone()))); node_resolver = Some(create_node_resolver( cjs_tracker, fs.clone(), @@ -168,7 +167,7 @@ impl LspScopeResolver { }) .unwrap_or_default(); Self { - cjs_tracker: type_checking_cjs_tracker, + cjs_tracker: lsp_cjs_tracker, graph_resolver, jsr_resolver, npm_resolver, @@ -188,16 +187,14 @@ impl LspScopeResolver { deno_runtime::deno_node::DenoFsNodeResolverEnv::new(fs.clone()), )); let mut node_resolver = None; - let mut type_checking_cjs_tracker = None; + let mut lsp_cjs_tracker = None; if let Some(npm_resolver) = &npm_resolver { let in_npm_pkg_checker = create_in_npm_pkg_checker(npm_resolver); let cjs_tracker = create_cjs_tracker( in_npm_pkg_checker.clone(), pkg_json_resolver.clone(), ); - type_checking_cjs_tracker = Some(Arc::new( - TypeCheckingCjsTracker::new_for_lsp(cjs_tracker.clone()), - )); + lsp_cjs_tracker = Some(Arc::new(LspCjsTracker::new(cjs_tracker.clone()))); node_resolver = Some(create_node_resolver( cjs_tracker, fs, @@ -212,7 +209,7 @@ impl LspScopeResolver { node_resolver.as_ref(), ); Arc::new(Self { - cjs_tracker: type_checking_cjs_tracker, + cjs_tracker: lsp_cjs_tracker, graph_resolver, jsr_resolver: self.jsr_resolver.clone(), npm_resolver, @@ -324,7 +321,7 @@ impl LspResolver { pub fn maybe_cjs_tracker( &self, file_referrer: Option<&ModuleSpecifier>, - ) -> Option<&Arc> { + ) -> Option<&Arc> { let resolver = self.get_scope_resolver(file_referrer); resolver.cjs_tracker.as_ref() } @@ -814,6 +811,40 @@ impl RedirectResolver { } } +#[derive(Debug)] +pub struct LspCjsTracker { + cjs_tracker: Arc, +} + +impl LspCjsTracker { + pub fn new(cjs_tracker: Arc) -> Self { + Self { cjs_tracker } + } + + pub fn is_cjs( + &self, + specifier: &ModuleSpecifier, + media_type: MediaType, + maybe_parsed_source: Option<&ParsedSource>, + ) -> bool { + if let Some(module_kind) = self.cjs_tracker.get_known_kind(specifier) { + module_kind.is_cjs() + } else { + let maybe_is_script = maybe_parsed_source.map(|p| p.is_script()); + maybe_is_script + .and_then(|is_script| { + self + .cjs_tracker + .is_cjs_with_known_is_script(specifier, is_script) + .ok() + }) + .unwrap_or_else(|| { + self.cjs_tracker.is_maybe_cjs(specifier).unwrap_or(false) + }) + } + } +} + #[cfg(test)] mod tests { use super::*; diff --git a/cli/lsp/tsc.rs b/cli/lsp/tsc.rs index 4b71ba66bb5337..5fcdb3575a5528 100644 --- a/cli/lsp/tsc.rs +++ b/cli/lsp/tsc.rs @@ -4363,14 +4363,22 @@ fn op_load<'s>( } else { let asset_or_document = state.get_asset_or_document(&specifier); asset_or_document.map(|doc| { - let maybe_cjs_tracker = - state.state_snapshot.resolver.maybe_cjs_tracker(specifier); + let maybe_cjs_tracker = state + .state_snapshot + .resolver + .maybe_cjs_tracker(Some(&specifier)); LoadResponse { data: doc.text(), script_kind: crate::tsc::as_ts_script_kind(doc.media_type()), version: state.script_version(&specifier), is_cjs: maybe_cjs_tracker - .map(|t| t.is_cjs(&specifier, doc.media_type(), &doc.text())) + .map(|t| { + t.is_cjs( + &specifier, + doc.media_type(), + doc.maybe_parsed_source().and_then(|p| p.as_ref().ok()), + ) + }) .unwrap_or(false), } }) diff --git a/cli/tsc/mod.rs b/cli/tsc/mod.rs index 6439b5754f616a..1823e275d68946 100644 --- a/cli/tsc/mod.rs +++ b/cli/tsc/mod.rs @@ -321,34 +321,10 @@ pub fn into_specifier_and_media_type( } } -trait IsScriptModuleAnalyzer: std::fmt::Debug + Send + Sync { - fn analyze_sync( - &self, - specifier: &ModuleSpecifier, - media_type: MediaType, - code: &Arc, - ) -> Option; -} - -impl IsScriptModuleAnalyzer for ModuleInfoCache { - fn analyze_sync( - &self, - specifier: &ModuleSpecifier, - media_type: MediaType, - code: &Arc, - ) -> Option { - self - .as_module_analyzer() - .analyze_sync(specifier, media_type, code) - .map(|info| info.is_script) - .ok() - } -} - #[derive(Debug)] pub struct TypeCheckingCjsTracker { cjs_tracker: Arc, - analyzer: Arc, + module_info_cache: Arc, } impl TypeCheckingCjsTracker { @@ -358,14 +334,7 @@ impl TypeCheckingCjsTracker { ) -> Self { Self { cjs_tracker, - analyzer: module_info_cache, - } - } - - pub fn new_for_lsp(cjs_tracker: Arc) -> Self { - Self { - cjs_tracker, - analyzer: module_info_cache, + module_info_cache, } } @@ -378,8 +347,12 @@ impl TypeCheckingCjsTracker { if let Some(module_kind) = self.cjs_tracker.get_known_kind(specifier) { module_kind.is_cjs() } else { - let maybe_is_script = - self.analyzer.analyze_sync(specifier, media_type, code); + let maybe_is_script = self + .module_info_cache + .as_module_analyzer() + .analyze_sync(specifier, media_type, code) + .ok() + .map(|info| info.is_script); maybe_is_script .and_then(|is_script| { self @@ -392,13 +365,6 @@ impl TypeCheckingCjsTracker { }) } } - - pub fn snapshot(&self) -> TypeCheckingCjsTracker { - Self { - cjs_tracker: Arc::new(self.cjs_tracker.snapshot()), - analyzer: self.analyzer.clone(), - } - } } #[derive(Debug)] From d56c766b61089fcf2c947cfa1db796d07ac6fcdb Mon Sep 17 00:00:00 2001 From: David Sherret Date: Tue, 29 Oct 2024 19:14:06 -0400 Subject: [PATCH 26/42] remove resolving media type in lsp based on node resolution --- cli/lsp/documents.rs | 16 ++-------------- cli/lsp/resolver.rs | 12 ------------ 2 files changed, 2 insertions(+), 26 deletions(-) diff --git a/cli/lsp/documents.rs b/cli/lsp/documents.rs index 5953faad2278ba..8609aed05e8196 100644 --- a/cli/lsp/documents.rs +++ b/cli/lsp/documents.rs @@ -332,12 +332,8 @@ impl Document { .filter(|s| cache.is_valid_file_referrer(s)) .cloned() .or(file_referrer); - let media_type = resolve_media_type( - &specifier, - maybe_headers.as_ref(), - maybe_language_id, - &resolver, - ); + let media_type = + resolve_media_type(&specifier, maybe_headers.as_ref(), maybe_language_id); let (maybe_parsed_source, maybe_module) = if media_type_is_diagnosable(media_type) { parse_and_analyze_module( @@ -399,7 +395,6 @@ impl Document { &self.specifier, self.maybe_headers.as_ref(), self.maybe_language_id, - &resolver, ); let dependencies; let maybe_types_dependency; @@ -764,14 +759,7 @@ fn resolve_media_type( specifier: &ModuleSpecifier, maybe_headers: Option<&HashMap>, maybe_language_id: Option, - resolver: &LspResolver, ) -> MediaType { - if resolver.in_node_modules(specifier) { - if let Some(media_type) = resolver.node_media_type(specifier) { - return media_type; - } - } - if let Some(language_id) = maybe_language_id { return MediaType::from_specifier_and_content_type( specifier, diff --git a/cli/lsp/resolver.rs b/cli/lsp/resolver.rs index c0c71ab70202a3..104c4579fe8a8d 100644 --- a/cli/lsp/resolver.rs +++ b/cli/lsp/resolver.rs @@ -431,18 +431,6 @@ impl LspResolver { has_node_modules_dir(specifier) } - pub fn node_media_type( - &self, - specifier: &ModuleSpecifier, - ) -> Option { - let resolver = self.get_scope_resolver(Some(specifier)); - let node_resolver = resolver.node_resolver.as_ref()?; - let resolution = node_resolver - .url_to_node_resolution(specifier.clone()) - .ok()?; - Some(into_specifier_and_media_type(Some(resolution.into_url())).1) - } - pub fn is_bare_package_json_dep( &self, specifier_text: &str, From 9e7d89659571713ff4bd4ef3aacacb7de20d2374 Mon Sep 17 00:00:00 2001 From: David Sherret Date: Tue, 29 Oct 2024 19:27:57 -0400 Subject: [PATCH 27/42] fix json imports --- cli/resolver.rs | 36 ++++-------------------------------- 1 file changed, 4 insertions(+), 32 deletions(-) diff --git a/cli/resolver.rs b/cli/resolver.rs index e14050ab972f83..1a495975dad451 100644 --- a/cli/resolver.rs +++ b/cli/resolver.rs @@ -232,7 +232,7 @@ impl CliNodeResolver { self .node_resolver .resolve(specifier, referrer, referrer_kind, mode)?; - Ok(self.handle_node_resolution(res)) + Ok(res) } pub fn resolve_req_reference( @@ -297,7 +297,7 @@ impl CliNodeResolver { maybe_referrer, mode, )?; - Ok(self.handle_node_resolution(res)) + Ok(res) } pub fn handle_if_in_node_modules( @@ -321,34 +321,6 @@ impl CliNodeResolver { Ok(None) } - - pub fn url_to_node_resolution( - &self, - specifier: ModuleSpecifier, - ) -> Result { - self.node_resolver.url_to_node_resolution(specifier) - } - - fn handle_node_resolution( - &self, - resolution: NodeResolution, - ) -> NodeResolution { - match &resolution { - NodeResolution::CommonJs(specifier) => { - // remember that this was a common js resolution - self - .cjs_tracker - .mark_kind(specifier.clone(), ModuleKind::Cjs); - } - NodeResolution::Esm(specifier) => { - self - .cjs_tracker - .mark_kind(specifier.clone(), ModuleKind::Esm); - } - NodeResolution::BuiltIn(_) => {} - } - resolution - } } #[derive(Debug, Error)] @@ -549,14 +521,14 @@ impl CjsTracker { Some(value.unwrap_or(ModuleKind::Esm)) } } + MediaType::Wasm | + MediaType::Json => Some(ModuleKind::Esm), MediaType::JavaScript | MediaType::Jsx | MediaType::TypeScript | MediaType::Tsx // treat these as unknown - | MediaType::Json | MediaType::Css - | MediaType::Wasm | MediaType::SourceMap | MediaType::Unknown => { if let Some(value) = self.known.get(specifier).map(|v| v.clone()) { From 4d3097a8c2a9c869240a9a0eee40c0d3d0ee8bf8 Mon Sep 17 00:00:00 2001 From: David Sherret Date: Tue, 29 Oct 2024 20:01:25 -0400 Subject: [PATCH 28/42] Remove url_to_node_resolution --- cli/lsp/resolver.rs | 12 ++- cli/module_loader.rs | 25 +++--- cli/node.rs | 3 +- cli/resolver.rs | 67 ++++++++-------- cli/standalone/binary.rs | 8 +- cli/standalone/mod.rs | 50 ++++++------ cli/tools/coverage/mod.rs | 5 +- cli/tools/run/hmr.rs | 3 +- cli/tsc/mod.rs | 29 +++---- cli/worker.rs | 25 +++--- resolvers/node/errors.rs | 73 ------------------ resolvers/node/resolution.rs | 77 +++---------------- tests/integration/run_tests.rs | 2 +- .../npm/deno_run_cowsay_no_permissions.out | 2 +- 14 files changed, 122 insertions(+), 259 deletions(-) diff --git a/cli/lsp/resolver.rs b/cli/lsp/resolver.rs index 104c4579fe8a8d..9e914f2392e55e 100644 --- a/cli/lsp/resolver.rs +++ b/cli/lsp/resolver.rs @@ -404,7 +404,6 @@ impl LspResolver { Some(into_specifier_and_media_type(Some( node_resolver .resolve_req_reference(req_ref, referrer, NodeResolutionMode::Types) - .map(|res| res.into_url()) .ok()?, ))) } @@ -815,7 +814,9 @@ impl LspCjsTracker { media_type: MediaType, maybe_parsed_source: Option<&ParsedSource>, ) -> bool { - if let Some(module_kind) = self.cjs_tracker.get_known_kind(specifier) { + if let Some(module_kind) = + self.cjs_tracker.get_known_kind(specifier, media_type) + { module_kind.is_cjs() } else { let maybe_is_script = maybe_parsed_source.map(|p| p.is_script()); @@ -823,11 +824,14 @@ impl LspCjsTracker { .and_then(|is_script| { self .cjs_tracker - .is_cjs_with_known_is_script(specifier, is_script) + .is_cjs_with_known_is_script(specifier, media_type, is_script) .ok() }) .unwrap_or_else(|| { - self.cjs_tracker.is_maybe_cjs(specifier).unwrap_or(false) + self + .cjs_tracker + .is_maybe_cjs(specifier, media_type) + .unwrap_or(false) }) } } diff --git a/cli/module_loader.rs b/cli/module_loader.rs index 1f70c6287b0032..905d6df955ec00 100644 --- a/cli/module_loader.rs +++ b/cli/module_loader.rs @@ -513,15 +513,11 @@ impl if self.shared.is_repl { if let Ok(reference) = NpmPackageReqReference::from_specifier(&specifier) { - return self - .shared - .node_resolver - .resolve_req_reference( - &reference, - referrer, - NodeResolutionMode::Execution, - ) - .map(|res| res.into_url()); + return self.shared.node_resolver.resolve_req_reference( + &reference, + referrer, + NodeResolutionMode::Execution, + ); } } @@ -545,7 +541,6 @@ impl .with_context(|| { format!("Could not resolve '{}'.", module.nv_reference) })? - .into_url() } Some(Module::Node(module)) => module.specifier.clone(), Some(Module::Js(module)) => module.specifier.clone(), @@ -671,11 +666,11 @@ impl is_script, .. })) => { - if self - .shared - .cjs_tracker - .is_cjs_with_known_is_script(specifier, *is_script)? - { + if self.shared.cjs_tracker.is_cjs_with_known_is_script( + specifier, + *media_type, + *is_script, + )? { return Ok(Some(CodeOrDeferredEmit::Cjs { specifier, media_type: *media_type, diff --git a/cli/node.rs b/cli/node.rs index 1eea82a8b23ce1..d911d17fc2df81 100644 --- a/cli/node.rs +++ b/cli/node.rs @@ -100,7 +100,7 @@ impl CliCjsCodeAnalyzer { } let cjs_tracker = self.cjs_tracker.clone(); - let is_maybe_cjs = cjs_tracker.is_maybe_cjs(&specifier)?; + let is_maybe_cjs = cjs_tracker.is_maybe_cjs(&specifier, media_type)?; let analysis = if is_maybe_cjs { let maybe_parsed_source = self .parsed_source_cache @@ -124,6 +124,7 @@ impl CliCjsCodeAnalyzer { })?; let is_cjs = cjs_tracker.is_cjs_with_known_is_script( parsed_source.specifier(), + media_type, parsed_source.is_script(), )?; if is_cjs { diff --git a/cli/resolver.rs b/cli/resolver.rs index 1a495975dad451..fe0a87d2a4bf7e 100644 --- a/cli/resolver.rs +++ b/cli/resolver.rs @@ -42,7 +42,7 @@ use node_resolver::errors::PackageFolderResolveErrorKind; use node_resolver::errors::PackageFolderResolveIoError; use node_resolver::errors::PackageNotFoundError; use node_resolver::errors::PackageResolveErrorKind; -use node_resolver::errors::UrlToNodeResolutionError; +use node_resolver::errors::PackageSubpathResolveError; use node_resolver::InNpmPackageChecker; use node_resolver::NodeModuleKind; use node_resolver::NodeResolution; @@ -155,8 +155,7 @@ impl CliNodeResolver { | NodeResolveErrorKind::UnsupportedEsmUrlScheme(_) | NodeResolveErrorKind::DataUrlReferrer(_) | NodeResolveErrorKind::TypesNotFound(_) - | NodeResolveErrorKind::FinalizeResolution(_) - | NodeResolveErrorKind::UrlToNodeResolution(_) => Err(err.into()), + | NodeResolveErrorKind::FinalizeResolution(_) => Err(err.into()), NodeResolveErrorKind::PackageResolve(err) => { let err = err.into_kind(); match err { @@ -220,7 +219,7 @@ impl CliNodeResolver { ) -> Result { let referrer_kind = if self .cjs_tracker - .is_maybe_cjs(referrer) + .is_maybe_cjs(referrer, MediaType::from_specifier(referrer)) .map_err(|err| NodeResolveErrorKind::PackageResolve(err.into()))? { NodeModuleKind::Cjs @@ -240,7 +239,7 @@ impl CliNodeResolver { req_ref: &NpmPackageReqReference, referrer: &ModuleSpecifier, mode: NodeResolutionMode, - ) -> Result { + ) -> Result { self.resolve_req_with_sub_path( req_ref.req(), req_ref.sub_path(), @@ -255,7 +254,7 @@ impl CliNodeResolver { sub_path: Option<&str>, referrer: &ModuleSpecifier, mode: NodeResolutionMode, - ) -> Result { + ) -> Result { let package_folder = self .npm_resolver .resolve_pkg_folder_from_deno_module_req(req, referrer)?; @@ -266,7 +265,7 @@ impl CliNodeResolver { mode, ); match resolution_result { - Ok(resolution) => Ok(resolution), + Ok(url) => Ok(url), Err(err) => { if self.npm_resolver.as_byonm().is_some() { let package_json_path = package_folder.join("package.json"); @@ -277,7 +276,7 @@ impl CliNodeResolver { )); } } - Err(err) + Err(err.into()) } } } @@ -288,16 +287,13 @@ impl CliNodeResolver { sub_path: Option<&str>, maybe_referrer: Option<&ModuleSpecifier>, mode: NodeResolutionMode, - ) -> Result { - let res = self - .node_resolver - .resolve_package_subpath_from_deno_module( - package_folder, - sub_path, - maybe_referrer, - mode, - )?; - Ok(res) + ) -> Result { + self.node_resolver.resolve_package_subpath_from_deno_module( + package_folder, + sub_path, + maybe_referrer, + mode, + ) } pub fn handle_if_in_node_modules( @@ -401,7 +397,7 @@ impl NpmModuleLoader { })); } - let code = if self.cjs_tracker.is_maybe_cjs(specifier)? { + let code = if self.cjs_tracker.is_maybe_cjs(specifier, media_type)? { // translate cjs to esm if it's cjs and inject node globals let code = from_utf8_lossy_owned(code); ModuleSourceCode::String( @@ -458,8 +454,9 @@ impl CjsTracker { pub fn is_maybe_cjs( &self, specifier: &ModuleSpecifier, + media_type: MediaType, ) -> Result { - self.treat_as_cjs_with_is_script(specifier, None) + self.treat_as_cjs_with_is_script(specifier, media_type, None) } /// Gets whether the file is CJS. If true, this is for sure @@ -470,17 +467,21 @@ impl CjsTracker { pub fn is_cjs_with_known_is_script( &self, specifier: &ModuleSpecifier, + media_type: MediaType, is_script: bool, ) -> Result { - self.treat_as_cjs_with_is_script(specifier, Some(is_script)) + self.treat_as_cjs_with_is_script(specifier, media_type, Some(is_script)) } fn treat_as_cjs_with_is_script( &self, specifier: &ModuleSpecifier, + media_type: MediaType, is_script: Option, ) -> Result { - let kind = match self.get_known_kind_with_is_script(specifier, is_script) { + let kind = match self + .get_known_kind_with_is_script(specifier, media_type, is_script) + { Some(kind) => kind, None => self.check_based_on_pkg_json(specifier)?, }; @@ -490,21 +491,21 @@ impl CjsTracker { pub fn get_known_kind( &self, specifier: &ModuleSpecifier, + media_type: MediaType, ) -> Option { - self.get_known_kind_with_is_script(specifier, None) + self.get_known_kind_with_is_script(specifier, media_type, None) } fn get_known_kind_with_is_script( &self, specifier: &ModuleSpecifier, + media_type: MediaType, is_script: Option, ) -> Option { if specifier.scheme() != "file" { return Some(ModuleKind::Esm); } - let media_type = MediaType::from_specifier(specifier); - match media_type { MediaType::Mts | MediaType::Mjs | MediaType::Dmts => Some(ModuleKind::Esm), MediaType::Cjs | MediaType::Cts | MediaType::Dcts => Some(ModuleKind::Cjs), @@ -771,8 +772,7 @@ impl Resolver for CliGraphResolver { Some(referrer), to_node_mode(mode), ) - .map_err(ResolveError::Other) - .map(|res| res.into_url()), + .map_err(|e| ResolveError::Other(e.into())), MappedResolution::PackageJson { dep_result, alias, @@ -813,8 +813,8 @@ impl Resolver for CliGraphResolver { sub_path.as_deref(), Some(referrer), to_node_mode(mode), - )? - .into_url(), + ) + .map_err(|e| ResolveError::Other(e.into()))?, ) }), }) @@ -862,8 +862,8 @@ impl Resolver for CliGraphResolver { npm_req_ref.sub_path(), Some(referrer), to_node_mode(mode), - )? - .into_url(), + ) + .map_err(|e| ResolveError::Other(e.into()))?, ); } @@ -871,7 +871,6 @@ impl Resolver for CliGraphResolver { if is_byonm { return node_resolver .resolve_req_reference(&npm_req_ref, referrer, to_node_mode(mode)) - .map(|res| res.into_url()) .map_err(|err| err.into()); } } @@ -889,9 +888,7 @@ impl Resolver for CliGraphResolver { .map_err(ResolveError::Other)?; if let Some(res) = maybe_resolution { match res { - NodeResolution::Esm(url) | NodeResolution::CommonJs(url) => { - return Ok(url) - } + NodeResolution::Module(url) => return Ok(url), NodeResolution::BuiltIn(_) => { // don't resolve bare specifiers for built-in modules via node resolution } diff --git a/cli/standalone/binary.rs b/cli/standalone/binary.rs index e3253165fb3422..fdc57b7af1ecab 100644 --- a/cli/standalone/binary.rs +++ b/cli/standalone/binary.rs @@ -605,9 +605,11 @@ impl<'a> DenoCompileBinaryWriter<'a> { let (maybe_source, media_type) = match module { deno_graph::Module::Js(m) => { let source = if m.media_type.is_emittable() { - let is_cjs = self - .cjs_tracker - .is_cjs_with_known_is_script(&m.specifier, m.is_script)?; + let is_cjs = self.cjs_tracker.is_cjs_with_known_is_script( + &m.specifier, + m.media_type, + m.is_script, + )?; let module_kind = ModuleKind::from_is_cjs(is_cjs); let source = self .emitter diff --git a/cli/standalone/mod.rs b/cli/standalone/mod.rs index 5ff4240d4f9a52..ef64aa1283f630 100644 --- a/cli/standalone/mod.rs +++ b/cli/standalone/mod.rs @@ -179,8 +179,7 @@ impl ModuleLoader for EmbeddedModuleLoader { sub_path.as_deref(), Some(&referrer), NodeResolutionMode::Execution, - )? - .into_url(), + )?, ), Ok(MappedResolution::PackageJson { dep_result, @@ -188,16 +187,14 @@ impl ModuleLoader for EmbeddedModuleLoader { alias, .. }) => match dep_result.as_ref().map_err(|e| AnyError::from(e.clone()))? { - PackageJsonDepValue::Req(req) => self - .shared - .node_resolver - .resolve_req_with_sub_path( + PackageJsonDepValue::Req(req) => { + self.shared.node_resolver.resolve_req_with_sub_path( req, sub_path.as_deref(), &referrer, NodeResolutionMode::Execution, ) - .map(|res| res.into_url()), + } PackageJsonDepValue::Workspace(version_req) => { let pkg_folder = self .shared @@ -215,8 +212,7 @@ impl ModuleLoader for EmbeddedModuleLoader { sub_path.as_deref(), Some(&referrer), NodeResolutionMode::Execution, - )? - .into_url(), + )?, ) } }, @@ -225,15 +221,11 @@ impl ModuleLoader for EmbeddedModuleLoader { if let Ok(reference) = NpmPackageReqReference::from_specifier(&specifier) { - return self - .shared - .node_resolver - .resolve_req_reference( - &reference, - &referrer, - NodeResolutionMode::Execution, - ) - .map(|res| res.into_url()); + return self.shared.node_resolver.resolve_req_reference( + &reference, + &referrer, + NodeResolutionMode::Execution, + ); } if specifier.scheme() == "jsr" { @@ -337,17 +329,21 @@ impl ModuleLoader for EmbeddedModuleLoader { match self.shared.modules.read(original_specifier) { Ok(Some(module)) => { + let media_type = module.media_type; let (module_specifier, module_type, module_source) = module.into_parts(); - let is_cjs = - match self.shared.cjs_tracker.is_maybe_cjs(original_specifier) { - Ok(is_cjs) => is_cjs, - Err(err) => { - return deno_core::ModuleLoadResponse::Sync(Err(type_error( - format!("{:?}", err), - ))); - } - }; + let is_cjs = match self + .shared + .cjs_tracker + .is_maybe_cjs(original_specifier, media_type) + { + Ok(is_cjs) => is_cjs, + Err(err) => { + return deno_core::ModuleLoadResponse::Sync(Err(type_error( + format!("{:?}", err), + ))); + } + }; if is_cjs { let original_specifier = original_specifier.clone(); let module_specifier = module_specifier.clone(); diff --git a/cli/tools/coverage/mod.rs b/cli/tools/coverage/mod.rs index 5fe0e7ac3f87f5..1a5b11d17285c7 100644 --- a/cli/tools/coverage/mod.rs +++ b/cli/tools/coverage/mod.rs @@ -582,8 +582,9 @@ pub async fn cover_files( | MediaType::Mts | MediaType::Cts | MediaType::Tsx => { - let module_kind = - ModuleKind::from_is_cjs(cjs_tracker.is_maybe_cjs(&file.specifier)?); + let module_kind = ModuleKind::from_is_cjs( + cjs_tracker.is_maybe_cjs(&file.specifier, file.media_type)?, + ); Some(match emitter.maybe_cached_emit(&file.specifier, module_kind, &file.source) { Some(code) => code, None => { diff --git a/cli/tools/run/hmr.rs b/cli/tools/run/hmr.rs index 4172573d19d9f2..6cebedd0127c5f 100644 --- a/cli/tools/run/hmr.rs +++ b/cli/tools/run/hmr.rs @@ -4,6 +4,7 @@ use std::collections::HashMap; use std::path::PathBuf; use std::sync::Arc; +use deno_ast::MediaType; use deno_ast::ModuleKind; use deno_core::error::generic_error; use deno_core::error::AnyError; @@ -145,7 +146,7 @@ impl crate::worker::HmrRunner for HmrRunner { let source_code = self.emitter.load_and_emit_for_hmr( &module_url, - ModuleKind::from_is_cjs(self.cjs_tracker.is_maybe_cjs(&module_url)?), + ModuleKind::from_is_cjs(self.cjs_tracker.is_maybe_cjs(&module_url, MediaType::from_specifier(&module_url))?), ).await?; let mut tries = 1; diff --git a/cli/tsc/mod.rs b/cli/tsc/mod.rs index 1823e275d68946..ddcd0eee960949 100644 --- a/cli/tsc/mod.rs +++ b/cli/tsc/mod.rs @@ -39,7 +39,7 @@ use deno_runtime::deno_node::NodeResolver; use deno_semver::npm::NpmPackageReqReference; use node_resolver::errors::NodeJsErrorCode; use node_resolver::errors::NodeJsErrorCoded; -use node_resolver::errors::ResolvePkgSubpathFromDenoModuleError; +use node_resolver::errors::PackageSubpathResolveError; use node_resolver::NodeModuleKind; use node_resolver::NodeResolution; use node_resolver::NodeResolutionMode; @@ -344,7 +344,9 @@ impl TypeCheckingCjsTracker { media_type: MediaType, code: &Arc, ) -> bool { - if let Some(module_kind) = self.cjs_tracker.get_known_kind(specifier) { + if let Some(module_kind) = + self.cjs_tracker.get_known_kind(specifier, media_type) + { module_kind.is_cjs() } else { let maybe_is_script = self @@ -357,11 +359,14 @@ impl TypeCheckingCjsTracker { .and_then(|is_script| { self .cjs_tracker - .is_cjs_with_known_is_script(specifier, is_script) + .is_cjs_with_known_is_script(specifier, media_type, is_script) .ok() }) .unwrap_or_else(|| { - self.cjs_tracker.is_maybe_cjs(specifier).unwrap_or(false) + self + .cjs_tracker + .is_maybe_cjs(specifier, media_type) + .unwrap_or(false) }) } } @@ -884,15 +889,15 @@ fn resolve_graph_specifier_types( Some(referrer), NodeResolutionMode::Types, ); - let maybe_resolution = match res_result { - Ok(res) => Some(res.into_url()), + let maybe_url = match res_result { + Ok(url) => Some(url), Err(err) => match err.code() { NodeJsErrorCode::ERR_TYPES_NOT_FOUND | NodeJsErrorCode::ERR_MODULE_NOT_FOUND => None, _ => return Err(err.into()), }, }; - Ok(Some(into_specifier_and_media_type(maybe_resolution))) + Ok(Some(into_specifier_and_media_type(maybe_url))) } else { Ok(None) } @@ -916,7 +921,7 @@ enum ResolveNonGraphSpecifierTypesError { #[error(transparent)] ResolvePkgFolderFromDenoReq(#[from] ResolvePkgFolderFromDenoReqError), #[error(transparent)] - ResolvePkgSubpathFromDenoModule(#[from] ResolvePkgSubpathFromDenoModuleError), + PackageSubpathResolve(#[from] PackageSubpathResolveError), } fn resolve_non_graph_specifier_types( @@ -963,17 +968,15 @@ fn resolve_non_graph_specifier_types( Some(referrer), NodeResolutionMode::Types, ); - let maybe_resolution = match res_result { - Ok(res) => Some(res), + let maybe_url = match res_result { + Ok(url) => Some(url), Err(err) => match err.code() { NodeJsErrorCode::ERR_TYPES_NOT_FOUND | NodeJsErrorCode::ERR_MODULE_NOT_FOUND => None, _ => return Err(err.into()), }, }; - Ok(Some(into_specifier_and_media_type( - maybe_resolution.map(|res| res.into_url()), - ))) + Ok(Some(into_specifier_and_media_type(maybe_url))) } else { Ok(None) } diff --git a/cli/worker.rs b/cli/worker.rs index 7107cb18bd33cc..3e859da23e5f9c 100644 --- a/cli/worker.rs +++ b/cli/worker.rs @@ -656,7 +656,7 @@ impl CliMainWorkerFactory { .node_resolver .resolve_binary_export(package_folder, sub_path) { - Ok(specifier) => Ok(specifier.into_url()), + Ok(specifier) => Ok(specifier), Err(original_err) => { // if the binary entrypoint was not found, fallback to regular node resolution let result = @@ -686,7 +686,7 @@ impl CliMainWorkerFactory { return Ok(None); } - let resolution = self + let specifier = self .shared .node_resolver .resolve_package_subpath_from_deno_module( @@ -695,19 +695,14 @@ impl CliMainWorkerFactory { /* referrer */ None, NodeResolutionMode::Execution, )?; - match &resolution { - NodeResolution::BuiltIn(_) => Ok(None), - NodeResolution::CommonJs(specifier) | NodeResolution::Esm(specifier) => { - if specifier - .to_file_path() - .map(|p| p.exists()) - .unwrap_or(false) - { - Ok(Some(resolution.into_url())) - } else { - bail!("Cannot find module '{}'", specifier) - } - } + if specifier + .to_file_path() + .map(|p| p.exists()) + .unwrap_or(false) + { + Ok(Some(specifier)) + } else { + bail!("Cannot find module '{}'", specifier) } } } diff --git a/resolvers/node/errors.rs b/resolvers/node/errors.rs index a35d232d62ff1a..aacbecefb4b638 100644 --- a/resolvers/node/errors.rs +++ b/resolvers/node/errors.rs @@ -81,29 +81,6 @@ pub trait NodeJsErrorCoded { fn code(&self) -> NodeJsErrorCode; } -kinded_err!( - ResolvePkgSubpathFromDenoModuleError, - ResolvePkgSubpathFromDenoModuleErrorKind -); - -impl NodeJsErrorCoded for ResolvePkgSubpathFromDenoModuleError { - fn code(&self) -> NodeJsErrorCode { - use ResolvePkgSubpathFromDenoModuleErrorKind::*; - match self.as_kind() { - PackageSubpathResolve(e) => e.code(), - UrlToNodeResolution(e) => e.code(), - } - } -} - -#[derive(Debug, Error)] -pub enum ResolvePkgSubpathFromDenoModuleErrorKind { - #[error(transparent)] - PackageSubpathResolve(#[from] PackageSubpathResolveError), - #[error(transparent)] - UrlToNodeResolution(#[from] UrlToNodeResolutionError), -} - // todo(https://github.com/denoland/deno_core/issues/810): make this a TypeError #[derive(Debug, Clone, Error)] #[error( @@ -394,52 +371,6 @@ impl NodeJsErrorCoded for CanonicalizingPkgJsonDirError { } } -#[derive(Debug, Error)] -#[error("JSX files are not supported in npm packages: {specifier}")] -pub struct JsxNotSupportedInNpmError { - pub specifier: Url, -} - -impl NodeJsErrorCoded for JsxNotSupportedInNpmError { - fn code(&self) -> NodeJsErrorCode { - NodeJsErrorCode::ERR_UNKNOWN_FILE_EXTENSION - } -} - -#[derive(Debug, Error)] -#[error("TypeScript files are not supported in npm packages: {specifier}")] -pub struct TypeScriptNotSupportedInNpmError { - pub specifier: Url, -} - -impl NodeJsErrorCoded for TypeScriptNotSupportedInNpmError { - fn code(&self) -> NodeJsErrorCode { - NodeJsErrorCode::ERR_UNKNOWN_FILE_EXTENSION - } -} - -kinded_err!(UrlToNodeResolutionError, UrlToNodeResolutionErrorKind); - -impl NodeJsErrorCoded for UrlToNodeResolutionError { - fn code(&self) -> NodeJsErrorCode { - match self.as_kind() { - UrlToNodeResolutionErrorKind::JsxNotSupported(e) => e.code(), - UrlToNodeResolutionErrorKind::TypeScriptNotSupported(e) => e.code(), - UrlToNodeResolutionErrorKind::ClosestPkgJson(e) => e.code(), - } - } -} - -#[derive(Debug, Error)] -pub enum UrlToNodeResolutionErrorKind { - #[error(transparent)] - JsxNotSupported(#[from] JsxNotSupportedInNpmError), - #[error(transparent)] - TypeScriptNotSupported(#[from] TypeScriptNotSupportedInNpmError), - #[error(transparent)] - ClosestPkgJson(#[from] ClosestPkgJsonError), -} - // todo(https://github.com/denoland/deno_core/issues/810): make this a TypeError #[derive(Debug, Error)] #[error( @@ -548,8 +479,6 @@ pub enum NodeResolveErrorKind { TypesNotFound(#[from] TypesNotFoundError), #[error(transparent)] FinalizeResolution(#[from] FinalizeResolutionError), - #[error(transparent)] - UrlToNodeResolution(#[from] UrlToNodeResolutionError), } kinded_err!(FinalizeResolutionError, FinalizeResolutionErrorKind); @@ -743,8 +672,6 @@ pub enum ResolvePkgJsonBinExportError { MissingPkgJson { pkg_json_path: PathBuf }, #[error("Failed resolving binary export. {message}")] InvalidBinProperty { message: String }, - #[error(transparent)] - UrlToNodeResolution(#[from] UrlToNodeResolutionError), } #[derive(Debug, Error)] diff --git a/resolvers/node/resolution.rs b/resolvers/node/resolution.rs index 0c6cce41e4513e..d44539e9786ad9 100644 --- a/resolvers/node/resolution.rs +++ b/resolvers/node/resolution.rs @@ -6,7 +6,6 @@ use std::path::PathBuf; use anyhow::bail; use anyhow::Error as AnyError; -use deno_media_type::MediaType; use deno_path_util::url_from_file_path; use serde_json::Map; use serde_json::Value; @@ -18,7 +17,6 @@ use crate::errors::DataUrlReferrerError; use crate::errors::FinalizeResolutionError; use crate::errors::InvalidModuleSpecifierError; use crate::errors::InvalidPackageTargetError; -use crate::errors::JsxNotSupportedInNpmError; use crate::errors::LegacyResolveError; use crate::errors::ModuleNotFoundError; use crate::errors::NodeJsErrorCode; @@ -38,13 +36,10 @@ use crate::errors::PackageTargetResolveError; use crate::errors::PackageTargetResolveErrorKind; use crate::errors::ResolveBinaryCommandsError; use crate::errors::ResolvePkgJsonBinExportError; -use crate::errors::ResolvePkgSubpathFromDenoModuleError; -use crate::errors::TypeScriptNotSupportedInNpmError; use crate::errors::TypesNotFoundError; use crate::errors::TypesNotFoundErrorData; use crate::errors::UnsupportedDirImportError; use crate::errors::UnsupportedEsmUrlSchemeError; -use crate::errors::UrlToNodeResolutionError; use crate::npm::InNpmPackageCheckerRc; use crate::NpmResolverRc; use crate::PackageJsonResolverRc; @@ -69,19 +64,16 @@ impl NodeResolutionMode { } } -// todo(THIS PR): REMOVE THIS #[derive(Debug)] pub enum NodeResolution { - Esm(Url), - CommonJs(Url), + Module(Url), BuiltIn(String), } impl NodeResolution { pub fn into_url(self) -> Url { match self { - Self::Esm(u) => u, - Self::CommonJs(u) => u, + Self::Module(u) => u, Self::BuiltIn(specifier) => { if specifier.starts_with("node:") { Url::parse(&specifier).unwrap() @@ -141,7 +133,7 @@ impl NodeResolver { if let Ok(url) = Url::parse(specifier) { if url.scheme() == "data" { - return Ok(NodeResolution::Esm(url)); + return Ok(NodeResolution::Module(url)); } if let Some(module_name) = @@ -166,7 +158,7 @@ impl NodeResolver { let url = referrer .join(specifier) .map_err(|source| DataUrlReferrerError { source })?; - return Ok(NodeResolution::Esm(url)); + return Ok(NodeResolution::Module(url)); } } @@ -187,7 +179,7 @@ impl NodeResolver { }; let url = self.finalize_resolution(url, Some(referrer))?; - let resolve_response = self.url_to_node_resolution(url)?; + let resolve_response = NodeResolution::Module(url); // TODO(bartlomieju): skipped checking errors for commonJS resolution and // "preserveSymlinksMain"/"preserveSymlinks" options. Ok(resolve_response) @@ -308,7 +300,7 @@ impl NodeResolver { package_subpath: Option<&str>, maybe_referrer: Option<&Url>, mode: NodeResolutionMode, - ) -> Result { + ) -> Result { let node_module_kind = NodeModuleKind::Esm; let package_subpath = package_subpath .map(|s| format!("./{s}")) @@ -321,10 +313,9 @@ impl NodeResolver { DEFAULT_CONDITIONS, mode, )?; - let resolve_response = self.url_to_node_resolution(resolved_url)?; // TODO(bartlomieju): skipped checking errors for commonJS resolution and // "preserveSymlinksMain"/"preserveSymlinks" options. - Ok(resolve_response) + Ok(resolved_url) } pub fn resolve_binary_commands( @@ -359,7 +350,7 @@ impl NodeResolver { &self, package_folder: &Path, sub_path: Option<&str>, - ) -> Result { + ) -> Result { let pkg_json_path = package_folder.join("package.json"); let Some(package_json) = self.pkg_json_resolver.load_package_json(&pkg_json_path)? @@ -376,59 +367,9 @@ impl NodeResolver { })?; let url = url_from_file_path(&package_folder.join(bin_entry)).unwrap(); - let resolve_response = self.url_to_node_resolution(url)?; // TODO(bartlomieju): skipped checking errors for commonJS resolution and // "preserveSymlinksMain"/"preserveSymlinks" options. - Ok(resolve_response) - } - - pub fn url_to_node_resolution( - &self, - url: Url, - ) -> Result { - if url.scheme() != "file" { - return Ok(NodeResolution::Esm(url)); - } - let media_type = MediaType::from_specifier(&url); - match media_type { - MediaType::JavaScript | MediaType::Dts => { - let maybe_package_config = - self.pkg_json_resolver.get_closest_package_json(&url)?; - match maybe_package_config { - Some(c) if c.typ == "module" => Ok(NodeResolution::Esm(url)), - Some(_) => Ok(NodeResolution::CommonJs(url)), - None => Ok(NodeResolution::Esm(url)), - } - } - MediaType::Json | MediaType::Wasm | MediaType::Mjs | MediaType::Dmts => { - Ok(NodeResolution::Esm(url)) - } - MediaType::Cjs | MediaType::Dcts => Ok(NodeResolution::CommonJs(url)), - MediaType::TypeScript | MediaType::Mts => { - if self.in_npm_pkg_checker.in_npm_package(&url) { - Err(TypeScriptNotSupportedInNpmError { specifier: url }.into()) - } else { - Ok(NodeResolution::Esm(url)) - } - } - MediaType::Cts => { - if self.in_npm_pkg_checker.in_npm_package(&url) { - Err(TypeScriptNotSupportedInNpmError { specifier: url }.into()) - } else { - Ok(NodeResolution::CommonJs(url)) - } - } - MediaType::Jsx | MediaType::Tsx => { - if self.in_npm_pkg_checker.in_npm_package(&url) { - Err(JsxNotSupportedInNpmError { specifier: url }.into()) - } else { - Ok(NodeResolution::Esm(url)) - } - } - MediaType::Css | MediaType::SourceMap | MediaType::Unknown => { - Ok(NodeResolution::CommonJs(url)) - } - } + Ok(url) } /// Checks if the resolved file has a corresponding declaration file. diff --git a/tests/integration/run_tests.rs b/tests/integration/run_tests.rs index db9f79556ef9fb..d6a2078d617b80 100644 --- a/tests/integration/run_tests.rs +++ b/tests/integration/run_tests.rs @@ -3605,7 +3605,7 @@ fn running_declaration_files() { temp_dir.write(file, ""); context .new_command() - .args_vec(["run", file]) + .args_vec(["run", "--allow-read", file]) .run() .skip_output_check() .assert_exit_code(0); diff --git a/tests/testdata/npm/deno_run_cowsay_no_permissions.out b/tests/testdata/npm/deno_run_cowsay_no_permissions.out index 6434620e2e58c6..25b79d9a7dce23 100644 --- a/tests/testdata/npm/deno_run_cowsay_no_permissions.out +++ b/tests/testdata/npm/deno_run_cowsay_no_permissions.out @@ -1,2 +1,2 @@ -error: Uncaught NotCapable: Requires read access to , specify the required permissions during compilation using `deno compile --allow-read` +error: Uncaught (in promise) NotCapable: Requires read access to , specify the required permissions during compilation using `deno compile --allow-read` [WILDCARD] From 157dfd508f525e24c8883e9ae32a3f4b05245e67 Mon Sep 17 00:00:00 2001 From: David Sherret Date: Tue, 29 Oct 2024 20:04:04 -0400 Subject: [PATCH 29/42] update test --- tests/specs/npm/typescript_file_in_package/__test__.jsonc | 4 ++-- tests/specs/npm/typescript_file_in_package/main.out | 3 +++ .../{typescript_file_in_package => }/main.ts | 0 .../typescript_file_in_package/main.out | 6 ------ 4 files changed, 5 insertions(+), 8 deletions(-) create mode 100644 tests/specs/npm/typescript_file_in_package/main.out rename tests/specs/npm/typescript_file_in_package/{typescript_file_in_package => }/main.ts (100%) delete mode 100644 tests/specs/npm/typescript_file_in_package/typescript_file_in_package/main.out diff --git a/tests/specs/npm/typescript_file_in_package/__test__.jsonc b/tests/specs/npm/typescript_file_in_package/__test__.jsonc index 08979ed257838e..7b5c5e1b67d05f 100644 --- a/tests/specs/npm/typescript_file_in_package/__test__.jsonc +++ b/tests/specs/npm/typescript_file_in_package/__test__.jsonc @@ -1,5 +1,5 @@ { - "args": "run typescript_file_in_package/main.ts", - "output": "typescript_file_in_package/main.out", + "args": "run main.ts", + "output": "main.out", "exitCode": 1 } diff --git a/tests/specs/npm/typescript_file_in_package/main.out b/tests/specs/npm/typescript_file_in_package/main.out new file mode 100644 index 00000000000000..58290a87377280 --- /dev/null +++ b/tests/specs/npm/typescript_file_in_package/main.out @@ -0,0 +1,3 @@ +Download http://localhost:4260/@denotest%2ftypescript-file +Download http://localhost:4260/@denotest/typescript-file/1.0.0.tgz +error: TypeScript files are not supported in npm packages: file:///[WILDCARD]/@denotest/typescript-file/1.0.0/index.ts diff --git a/tests/specs/npm/typescript_file_in_package/typescript_file_in_package/main.ts b/tests/specs/npm/typescript_file_in_package/main.ts similarity index 100% rename from tests/specs/npm/typescript_file_in_package/typescript_file_in_package/main.ts rename to tests/specs/npm/typescript_file_in_package/main.ts diff --git a/tests/specs/npm/typescript_file_in_package/typescript_file_in_package/main.out b/tests/specs/npm/typescript_file_in_package/typescript_file_in_package/main.out deleted file mode 100644 index b3faa879008340..00000000000000 --- a/tests/specs/npm/typescript_file_in_package/typescript_file_in_package/main.out +++ /dev/null @@ -1,6 +0,0 @@ -Download http://localhost:4260/@denotest%2ftypescript-file -Download http://localhost:4260/@denotest/typescript-file/1.0.0.tgz -error: Could not resolve 'npm:@denotest/typescript-file@1.0.0'. - -Caused by: - TypeScript files are not supported in npm packages: file:///[WILDCARD]/@denotest/typescript-file/1.0.0/index.ts From 41b77427db52461a4aaa4b1767ef0389321d274b Mon Sep 17 00:00:00 2001 From: David Sherret Date: Tue, 29 Oct 2024 20:16:36 -0400 Subject: [PATCH 30/42] lint --- cli/cache/mod.rs | 1 - cli/cache/module_info.rs | 6 +-- cli/factory.rs | 1 - cli/lsp/analysis.rs | 1 - cli/lsp/language_server.rs | 1 - cli/lsp/resolver.rs | 1 - cli/main.rs | 2 +- cli/module_loader.rs | 3 -- cli/node.rs | 2 +- cli/npm/managed/resolvers/global.rs | 1 - cli/npm/mod.rs | 9 +---- cli/resolver.rs | 63 ++++++++++------------------- cli/standalone/mod.rs | 1 - cli/tools/coverage/mod.rs | 3 +- cli/tsc/diagnostics.rs | 2 +- cli/tsc/mod.rs | 1 - cli/worker.rs | 6 --- resolvers/deno/npm/byonm.rs | 2 +- 18 files changed, 31 insertions(+), 75 deletions(-) diff --git a/cli/cache/mod.rs b/cli/cache/mod.rs index 87892fb9080d41..50fc135ddffcd3 100644 --- a/cli/cache/mod.rs +++ b/cli/cache/mod.rs @@ -8,7 +8,6 @@ use crate::file_fetcher::FetchOptions; use crate::file_fetcher::FetchPermissionsOptionRef; use crate::file_fetcher::FileFetcher; use crate::file_fetcher::FileOrRedirect; -use crate::npm::CliNpmResolver; use crate::util::fs::atomic_write_file_with_retries; use crate::util::fs::atomic_write_file_with_retries_and_fs; use crate::util::fs::AtomicWriteFileFsAdapter; diff --git a/cli/cache/module_info.rs b/cli/cache/module_info.rs index fb2e36181ffd20..060a6f4f0c94c4 100644 --- a/cli/cache/module_info.rs +++ b/cli/cache/module_info.rs @@ -128,7 +128,7 @@ impl ModuleInfoCache { Ok(()) } - pub fn as_module_analyzer<'a>(&'a self) -> ModuleInfoCacheModuleAnalyzer<'a> { + pub fn as_module_analyzer(&self) -> ModuleInfoCacheModuleAnalyzer { ModuleInfoCacheModuleAnalyzer { module_info_cache: self, parsed_source_cache: &self.parsed_source_cache, @@ -177,7 +177,7 @@ impl<'a> ModuleInfoCacheModuleAnalyzer<'a> { specifier, media_type, source_hash, - &module_info, + module_info, ) { log::debug!( "Error saving module cache info for {}. {:#}", @@ -205,7 +205,7 @@ impl<'a> ModuleInfoCacheModuleAnalyzer<'a> { let parser = self.parsed_source_cache.as_capturing_parser(); let analyzer = ParserModuleAnalyzer::new(&parser); let module_info = - analyzer.analyze_sync(&specifier, source.clone(), media_type)?; + analyzer.analyze_sync(specifier, source.clone(), media_type)?; // then attempt to cache it self.save_module_info_to_cache( diff --git a/cli/factory.rs b/cli/factory.rs index 509044db88032a..00a76ec8ca49fd 100644 --- a/cli/factory.rs +++ b/cli/factory.rs @@ -895,7 +895,6 @@ impl CliFactory { Ok(CliMainWorkerFactory::new( self.blob_store().clone(), - cjs_tracker.clone(), if cli_options.code_cache_enabled() { Some(self.code_cache()?.clone()) } else { diff --git a/cli/lsp/analysis.rs b/cli/lsp/analysis.rs index ab5b3ca1ac64d3..0769c90477530d 100644 --- a/cli/lsp/analysis.rs +++ b/cli/lsp/analysis.rs @@ -36,7 +36,6 @@ use deno_semver::package::PackageReq; use deno_semver::package::PackageReqReference; use deno_semver::Version; use import_map::ImportMap; -use node_resolver::NpmResolver; use once_cell::sync::Lazy; use regex::Regex; use std::borrow::Cow; diff --git a/cli/lsp/language_server.rs b/cli/lsp/language_server.rs index e2e11bd128d95d..6c1e6d4ed0f1f6 100644 --- a/cli/lsp/language_server.rs +++ b/cli/lsp/language_server.rs @@ -110,7 +110,6 @@ use crate::tools::fmt::format_file; use crate::tools::fmt::format_parsed_source; use crate::tools::upgrade::check_for_upgrades_for_lsp; use crate::tools::upgrade::upgrade_check_enabled; -use crate::tsc::TypeCheckingCjsTracker; use crate::util::fs::remove_dir_all_if_exists; use crate::util::path::is_importable_ext; use crate::util::path::to_percent_decoded_str; diff --git a/cli/lsp/resolver.rs b/cli/lsp/resolver.rs index 9e914f2392e55e..f5802142aa9d0b 100644 --- a/cli/lsp/resolver.rs +++ b/cli/lsp/resolver.rs @@ -24,7 +24,6 @@ use deno_semver::package::PackageReq; use indexmap::IndexMap; use node_resolver::errors::ClosestPkgJsonError; use node_resolver::InNpmPackageChecker; -use node_resolver::NodeResolution; use node_resolver::NodeResolutionMode; use std::borrow::Cow; use std::collections::BTreeMap; diff --git a/cli/main.rs b/cli/main.rs index c2639c90888893..04daff6700ac7f 100644 --- a/cli/main.rs +++ b/cli/main.rs @@ -135,7 +135,7 @@ async fn run_subcommand(flags: Arc) -> Result { tools::compile::compile(flags, compile_flags).await }), DenoSubcommand::Coverage(coverage_flags) => spawn_subcommand(async { - tools::coverage::cover_files(flags, coverage_flags).await + tools::coverage::cover_files(flags, coverage_flags) }), DenoSubcommand::Fmt(fmt_flags) => { spawn_subcommand( diff --git a/cli/module_loader.rs b/cli/module_loader.rs index 905d6df955ec00..e7c89c56be1cfa 100644 --- a/cli/module_loader.rs +++ b/cli/module_loader.rs @@ -50,7 +50,6 @@ use deno_core::error::AnyError; use deno_core::futures::future::FutureExt; use deno_core::futures::Future; use deno_core::resolve_url; -use deno_core::url::Url; use deno_core::ModuleCodeString; use deno_core::ModuleLoader; use deno_core::ModuleSource; @@ -69,7 +68,6 @@ use deno_graph::Module; use deno_graph::ModuleGraph; use deno_graph::Resolution; use deno_runtime::code_cache; -use deno_runtime::deno_fs; use deno_runtime::deno_fs::FileSystem; use deno_runtime::deno_node::create_host_defined_options; use deno_runtime::deno_node::NodeRequireLoader; @@ -77,7 +75,6 @@ use deno_runtime::deno_permissions::PermissionsContainer; use deno_semver::npm::NpmPackageReqReference; use node_resolver::InNpmPackageChecker; use node_resolver::NodeResolutionMode; -use thiserror::Error; pub struct ModuleLoadPreparer { options: Arc, diff --git a/cli/node.rs b/cli/node.rs index d911d17fc2df81..44e7508a8203f4 100644 --- a/cli/node.rs +++ b/cli/node.rs @@ -100,7 +100,7 @@ impl CliCjsCodeAnalyzer { } let cjs_tracker = self.cjs_tracker.clone(); - let is_maybe_cjs = cjs_tracker.is_maybe_cjs(&specifier, media_type)?; + let is_maybe_cjs = cjs_tracker.is_maybe_cjs(specifier, media_type)?; let analysis = if is_maybe_cjs { let maybe_parsed_source = self .parsed_source_cache diff --git a/cli/npm/managed/resolvers/global.rs b/cli/npm/managed/resolvers/global.rs index d4c9d13f7cc6fe..f0193e78e95cf2 100644 --- a/cli/npm/managed/resolvers/global.rs +++ b/cli/npm/managed/resolvers/global.rs @@ -11,7 +11,6 @@ use crate::colors; use async_trait::async_trait; use deno_ast::ModuleSpecifier; use deno_core::error::AnyError; -use deno_core::url::Url; use deno_npm::NpmPackageCacheFolderId; use deno_npm::NpmPackageId; use deno_npm::NpmResolutionPackage; diff --git a/cli/npm/mod.rs b/cli/npm/mod.rs index 249be1be313fb2..4eaf2c7262149c 100644 --- a/cli/npm/mod.rs +++ b/cli/npm/mod.rs @@ -11,8 +11,6 @@ use std::sync::Arc; use common::maybe_auth_header_for_npm_registry; use dashmap::DashMap; -use deno_ast::MediaType; -use deno_ast::ModuleKind; use deno_ast::ModuleSpecifier; use deno_core::error::AnyError; use deno_core::serde_json; @@ -21,9 +19,7 @@ use deno_npm::registry::NpmPackageInfo; use deno_resolver::npm::ByonmInNpmPackageChecker; use deno_resolver::npm::ByonmNpmResolver; use deno_resolver::npm::ByonmResolvePkgFolderFromDenoReqError; -use deno_runtime::deno_fs::FileSystem; use deno_runtime::deno_node::NodePermissions; -use deno_runtime::deno_node::NodeRequireLoader; use deno_runtime::ops::process::NpmProcessStateProvider; use deno_semver::package::PackageNv; use deno_semver::package::PackageReq; @@ -33,7 +29,6 @@ use node_resolver::InNpmPackageChecker; use node_resolver::NpmResolver; use thiserror::Error; -use crate::emit::Emitter; use crate::file_fetcher::FileFetcher; pub use self::byonm::CliByonmNpmResolver; @@ -90,9 +85,7 @@ pub fn create_in_npm_pkg_checker( CreateInNpmPkgCheckerOptions::Managed(options) => { create_managed_in_npm_pkg_checker(options) } - CreateInNpmPkgCheckerOptions::Byonm => { - Arc::new(ByonmInNpmPackageChecker::default()) - } + CreateInNpmPkgCheckerOptions::Byonm => Arc::new(ByonmInNpmPackageChecker), } } diff --git a/cli/resolver.rs b/cli/resolver.rs index fe0a87d2a4bf7e..c772459d8404ed 100644 --- a/cli/resolver.rs +++ b/cli/resolver.rs @@ -30,7 +30,6 @@ use deno_runtime::colors; use deno_runtime::deno_fs; use deno_runtime::deno_fs::FileSystem; use deno_runtime::deno_node::is_builtin_node_module; -use deno_runtime::deno_node::NodeRequireLoader; use deno_runtime::deno_node::NodeResolver; use deno_runtime::deno_node::PackageJsonResolver; use deno_semver::npm::NpmPackageReqReference; @@ -55,7 +54,6 @@ use thiserror::Error; use crate::args::JsxImportSourceConfig; use crate::args::DENO_DISABLE_PEDANTIC_NODE_WARNINGS; -use crate::emit::Emitter; use crate::node::CliNodeCodeTranslator; use crate::npm::CliNpmResolver; use crate::npm::InnerCliNpmResolverRef; @@ -512,12 +510,12 @@ impl CjsTracker { MediaType::Dts => { // dts files are always determined based on the package.json because // they contain imports/exports even when considered CJS - if let Some(value) = self.known.get(specifier).map(|v| v.clone()) { + if let Some(value) = self.known.get(specifier).map(|v| *v) { Some(value) } else { let value = self.check_based_on_pkg_json(specifier).ok(); if let Some(value) = value { - self.known.insert(specifier.clone(), value.clone()); + self.known.insert(specifier.clone(), value); } Some(value.unwrap_or(ModuleKind::Esm)) } @@ -532,7 +530,7 @@ impl CjsTracker { | MediaType::Css | MediaType::SourceMap | MediaType::Unknown => { - if let Some(value) = self.known.get(specifier).map(|v| v.clone()) { + if let Some(value) = self.known.get(specifier).map(|v| *v) { if value.is_cjs() && is_script == Some(false) { // we now know this is actually esm self.known.insert(specifier.clone(), ModuleKind::Esm); @@ -577,19 +575,6 @@ impl CjsTracker { Ok(ModuleKind::Esm) } } - - pub fn mark_kind(&self, specifier: ModuleSpecifier, kind: ModuleKind) { - self.known.insert(specifier, kind); - } - - pub fn snapshot(&self) -> CjsTracker { - CjsTracker { - in_npm_pkg_checker: self.in_npm_pkg_checker.clone(), - pkg_json_resolver: self.pkg_json_resolver.clone(), - unstable_detect_cjs: self.unstable_detect_cjs, - known: self.known.clone(), - } - } } pub type CliSloppyImportsResolver = @@ -803,19 +788,17 @@ impl Resolver for CliGraphResolver { ) .map_err(|e| ResolveError::Other(e.into())) .and_then(|pkg_folder| { - Ok( - self - .node_resolver - .as_ref() - .unwrap() - .resolve_package_sub_path_from_deno_module( - pkg_folder, - sub_path.as_deref(), - Some(referrer), - to_node_mode(mode), - ) - .map_err(|e| ResolveError::Other(e.into()))?, - ) + self + .node_resolver + .as_ref() + .unwrap() + .resolve_package_sub_path_from_deno_module( + pkg_folder, + sub_path.as_deref(), + Some(referrer), + to_node_mode(mode), + ) + .map_err(|e| ResolveError::Other(e.into())) }), }) } @@ -855,16 +838,14 @@ impl Resolver for CliGraphResolver { npm_req_ref.req(), ) { - return Ok( - node_resolver - .resolve_package_sub_path_from_deno_module( - pkg_folder, - npm_req_ref.sub_path(), - Some(referrer), - to_node_mode(mode), - ) - .map_err(|e| ResolveError::Other(e.into()))?, - ); + return node_resolver + .resolve_package_sub_path_from_deno_module( + pkg_folder, + npm_req_ref.sub_path(), + Some(referrer), + to_node_mode(mode), + ) + .map_err(|e| ResolveError::Other(e.into())); } // do npm resolution for byonm diff --git a/cli/standalone/mod.rs b/cli/standalone/mod.rs index ef64aa1283f630..8f519bfe79dba4 100644 --- a/cli/standalone/mod.rs +++ b/cli/standalone/mod.rs @@ -751,7 +751,6 @@ pub async fn run(data: StandaloneData) -> Result { }); let worker_factory = CliMainWorkerFactory::new( Arc::new(BlobStore::default()), - cjs_tracker, // Code cache is not supported for standalone binary yet. None, feature_checker, diff --git a/cli/tools/coverage/mod.rs b/cli/tools/coverage/mod.rs index 1a5b11d17285c7..8684c9d6d6327f 100644 --- a/cli/tools/coverage/mod.rs +++ b/cli/tools/coverage/mod.rs @@ -6,7 +6,6 @@ use crate::args::FileFlags; use crate::args::Flags; use crate::cdp; use crate::factory::CliFactory; -use crate::npm::CliNpmResolver; use crate::tools::fmt::format_json; use crate::tools::test::is_supported_test_path; use crate::util::text_encoding::source_map_from_code; @@ -481,7 +480,7 @@ fn filter_coverages( .collect::>() } -pub async fn cover_files( +pub fn cover_files( flags: Arc, coverage_flags: CoverageFlags, ) -> Result<(), AnyError> { diff --git a/cli/tsc/diagnostics.rs b/cli/tsc/diagnostics.rs index 7d276259f1c657..d3795706eb7619 100644 --- a/cli/tsc/diagnostics.rs +++ b/cli/tsc/diagnostics.rs @@ -323,7 +323,7 @@ impl Diagnostics { // todo(dsherret): use a short lived cache to prevent parsing // source maps so often if let Ok(source_map) = - SourceMap::from_slice(&fast_check_module.source_map.as_bytes()) + SourceMap::from_slice(fast_check_module.source_map.as_bytes()) { if let Some(start) = d.start.as_mut() { let maybe_token = source_map diff --git a/cli/tsc/mod.rs b/cli/tsc/mod.rs index ddcd0eee960949..dc7fc38f7a0172 100644 --- a/cli/tsc/mod.rs +++ b/cli/tsc/mod.rs @@ -41,7 +41,6 @@ use node_resolver::errors::NodeJsErrorCode; use node_resolver::errors::NodeJsErrorCoded; use node_resolver::errors::PackageSubpathResolveError; use node_resolver::NodeModuleKind; -use node_resolver::NodeResolution; use node_resolver::NodeResolutionMode; use once_cell::sync::Lazy; use std::borrow::Cow; diff --git a/cli/worker.rs b/cli/worker.rs index 3e859da23e5f9c..b0bab2876b5218 100644 --- a/cli/worker.rs +++ b/cli/worker.rs @@ -21,7 +21,6 @@ use deno_core::SharedArrayBufferStore; use deno_runtime::code_cache; use deno_runtime::deno_broadcast_channel::InMemoryBroadcastChannel; use deno_runtime::deno_fs; -use deno_runtime::deno_node; use deno_runtime::deno_node::NodeExtInitServices; use deno_runtime::deno_node::NodeRequireLoader; use deno_runtime::deno_node::NodeRequireLoaderRc; @@ -45,7 +44,6 @@ use deno_runtime::WorkerExecutionMode; use deno_runtime::WorkerLogLevel; use deno_semver::npm::NpmPackageReqReference; use deno_terminal::colors; -use node_resolver::NodeResolution; use node_resolver::NodeResolutionMode; use tokio::select; @@ -54,7 +52,6 @@ use crate::args::DenoSubcommand; use crate::args::StorageKeyResolver; use crate::errors; use crate::npm::CliNpmResolver; -use crate::resolver::CjsTracker; use crate::util::checksum; use crate::util::file_watcher::WatcherCommunicator; use crate::util::file_watcher::WatcherRestartMode; @@ -129,7 +126,6 @@ pub struct CliMainWorkerOptions { struct SharedWorkerState { blob_store: Arc, broadcast_channel: InMemoryBroadcastChannel, - cjs_tracker: Arc, code_cache: Option>, compiled_wasm_module_store: CompiledWasmModuleStore, feature_checker: Arc, @@ -406,7 +402,6 @@ impl CliMainWorkerFactory { #[allow(clippy::too_many_arguments)] pub fn new( blob_store: Arc, - cjs_tracker: Arc, code_cache: Option>, feature_checker: Arc, fs: Arc, @@ -427,7 +422,6 @@ impl CliMainWorkerFactory { shared: Arc::new(SharedWorkerState { blob_store, broadcast_channel: Default::default(), - cjs_tracker, code_cache, compiled_wasm_module_store: Default::default(), feature_checker, diff --git a/resolvers/deno/npm/byonm.rs b/resolvers/deno/npm/byonm.rs index e15a721dc60e90..b85117052c75f8 100644 --- a/resolvers/deno/npm/byonm.rs +++ b/resolvers/deno/npm/byonm.rs @@ -356,7 +356,7 @@ impl< } } -#[derive(Default, Debug)] +#[derive(Debug)] pub struct ByonmInNpmPackageChecker; impl InNpmPackageChecker for ByonmInNpmPackageChecker { From 5582eda476127c01984ffa69a2e4001340a3fc80 Mon Sep 17 00:00:00 2001 From: David Sherret Date: Tue, 29 Oct 2024 20:19:57 -0400 Subject: [PATCH 31/42] bump deno_ast --- Cargo.lock | 4 +++- Cargo.toml | 5 +---- 2 files changed, 4 insertions(+), 5 deletions(-) diff --git a/Cargo.lock b/Cargo.lock index a101afc404d401..b8e6842a5d65d5 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -1279,7 +1279,9 @@ dependencies = [ [[package]] name = "deno_ast" -version = "0.43.0" +version = "0.43.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "01cdd89f7a3327ee0ff50d472c5a841efb5d092f93a2406aae73f8f881cd5f13" dependencies = [ "base64 0.21.7", "deno_media_type", diff --git a/Cargo.toml b/Cargo.toml index f84663bb8cf373..0bc511c8072984 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -45,7 +45,7 @@ license = "MIT" repository = "https://github.com/denoland/deno" [workspace.dependencies] -deno_ast = { version = "=0.43.0", features = ["transpiling"] } +deno_ast = { version = "=0.43.1", features = ["transpiling"] } deno_core = { version = "0.314.2" } deno_bench_util = { version = "0.168.0", path = "./bench_util" } @@ -326,6 +326,3 @@ opt-level = 3 opt-level = 3 [profile.release.package.zstd-sys] opt-level = 3 - -[patch.crates-io] -deno_ast = { path = "../deno_ast" } From fc4ff68dce52dba7671f4fe620d78c036359a344 Mon Sep 17 00:00:00 2001 From: David Sherret Date: Wed, 30 Oct 2024 11:54:20 -0400 Subject: [PATCH 32/42] Fix node compat tests --- cli/args/mod.rs | 8 ++++++++ cli/factory.rs | 1 + cli/module_loader.rs | 24 ++++++++++++++++-------- cli/node.rs | 12 ++++++++++-- cli/standalone/mod.rs | 1 + cli/tools/bench/mod.rs | 2 +- cli/tools/test/mod.rs | 2 +- cli/worker.rs | 22 +++++----------------- 8 files changed, 43 insertions(+), 29 deletions(-) diff --git a/cli/args/mod.rs b/cli/args/mod.rs index 9406a9ad8fd20d..2db881f0e4edbf 100644 --- a/cli/args/mod.rs +++ b/cli/args/mod.rs @@ -1206,6 +1206,14 @@ impl CliOptions { } } + // If the main module should be treated as being in an npm package. + // This is triggered via a secret environment variable which is used + // for functionality like child_process.fork. Users should NOT depend + // on this functionality. + pub fn is_npm_main(&self) -> bool { + NPM_PROCESS_STATE.is_some() + } + pub fn has_node_modules_dir(&self) -> bool { self.maybe_node_modules_folder.is_some() } diff --git a/cli/factory.rs b/cli/factory.rs index 00a76ec8ca49fd..379116f4956f29 100644 --- a/cli/factory.rs +++ b/cli/factory.rs @@ -651,6 +651,7 @@ impl CliFactory { self.cjs_tracker()?.clone(), self.fs().clone(), Some(self.parsed_source_cache().clone()), + self.cli_options()?.is_npm_main(), ); Ok(Arc::new(NodeCodeTranslator::new( diff --git a/cli/module_loader.rs b/cli/module_loader.rs index e7c89c56be1cfa..c021f66127565e 100644 --- a/cli/module_loader.rs +++ b/cli/module_loader.rs @@ -206,6 +206,7 @@ struct SharedCliModuleLoaderState { lib_worker: TsTypeLib, initial_cwd: PathBuf, is_inspecting: bool, + is_npm_main: bool, is_repl: bool, cjs_tracker: Arc, code_cache: Option>, @@ -251,6 +252,7 @@ impl CliModuleLoaderFactory { lib_worker: options.ts_type_lib_worker(), initial_cwd: options.initial_cwd().to_path_buf(), is_inspecting: options.is_inspecting(), + is_npm_main: options.is_npm_main(), is_repl: matches!( options.sub_command(), DenoSubcommand::Repl(_) | DenoSubcommand::Jupyter(_) @@ -284,6 +286,7 @@ impl CliModuleLoaderFactory { Rc::new(CliModuleLoader(Rc::new(CliModuleLoaderInner { lib, is_worker, + is_npm_main: self.shared.is_npm_main, parent_permissions, permissions, graph_container, @@ -339,6 +342,7 @@ impl ModuleLoaderFactory for CliModuleLoaderFactory { struct CliModuleLoaderInner { lib: TsTypeLib, + is_npm_main: bool, is_worker: bool, /// The initial set of permissions used to resolve the static imports in the /// worker. These are "allow all" for main worker, and parent thread @@ -663,11 +667,14 @@ impl is_script, .. })) => { - if self.shared.cjs_tracker.is_cjs_with_known_is_script( - specifier, - *media_type, - *is_script, - )? { + // todo(dsherret): revert in https://github.com/denoland/deno/pull/26439 + if self.is_npm_main && *is_script + || self.shared.cjs_tracker.is_cjs_with_known_is_script( + specifier, + *media_type, + *is_script, + )? + { return Ok(Some(CodeOrDeferredEmit::Cjs { specifier, media_type: *media_type, @@ -745,6 +752,7 @@ impl // at this point, we no longer need the parsed source in memory, so free it self.parsed_source_cache.free(specifier); Ok(ModuleCodeStringSource { + // todo(dsherret): perf: if text is borrowed then we can clone the original source code: ModuleSourceCode::String(text.into_owned().into()), found_url: specifier.clone(), media_type, @@ -1042,11 +1050,11 @@ impl NodeRequireLoader for CliNodeRequireLoader { } fn load_text_file_lossy(&self, path: &Path) -> Result { - // todo(dsherret): use the preloaded module from the graph if available - let specifier = deno_path_util::url_from_file_path(path)?; - let media_type = MediaType::from_specifier(&specifier); + // todo(dsherret): use the preloaded module from the graph if available? + let media_type = MediaType::from_path(path); let text = self.fs.read_text_file_lossy_sync(path, None)?; if media_type.is_emittable() { + let specifier = deno_path_util::url_from_file_path(path)?; if self.in_npm_pkg_checker.in_npm_package(&specifier) { return Err( NotSupportedKindInNpmError { diff --git a/cli/node.rs b/cli/node.rs index 44e7508a8203f4..00ba4a5779e24a 100644 --- a/cli/node.rs +++ b/cli/node.rs @@ -62,6 +62,10 @@ pub struct CliCjsCodeAnalyzer { cjs_tracker: Arc, fs: deno_fs::FileSystemRc, parsed_source_cache: Option>, + // todo(dsherret): hack, remove in https://github.com/denoland/deno/pull/26439 + // For example, this does not properly handle if cjs analysis was already done + // and has been cached. + is_npm_main: bool, } impl CliCjsCodeAnalyzer { @@ -70,12 +74,14 @@ impl CliCjsCodeAnalyzer { cjs_tracker: Arc, fs: deno_fs::FileSystemRc, parsed_source_cache: Option>, + is_npm_main: bool, ) -> Self { Self { cache, cjs_tracker, fs, parsed_source_cache, + is_npm_main, } } @@ -100,7 +106,9 @@ impl CliCjsCodeAnalyzer { } let cjs_tracker = self.cjs_tracker.clone(); - let is_maybe_cjs = cjs_tracker.is_maybe_cjs(specifier, media_type)?; + let is_npm_main = self.is_npm_main; + let is_maybe_cjs = + cjs_tracker.is_maybe_cjs(specifier, media_type)? || is_npm_main; let analysis = if is_maybe_cjs { let maybe_parsed_source = self .parsed_source_cache @@ -126,7 +134,7 @@ impl CliCjsCodeAnalyzer { parsed_source.specifier(), media_type, parsed_source.is_script(), - )?; + )? || parsed_source.is_script() && is_npm_main; if is_cjs { let analysis = parsed_source.analyze_cjs(); Ok(CliCjsAnalysis::Cjs { diff --git a/cli/standalone/mod.rs b/cli/standalone/mod.rs index 8f519bfe79dba4..3b63bcd9850c34 100644 --- a/cli/standalone/mod.rs +++ b/cli/standalone/mod.rs @@ -641,6 +641,7 @@ pub async fn run(data: StandaloneData) -> Result { cjs_tracker.clone(), fs.clone(), None, + false, ); let node_code_translator = Arc::new(NodeCodeTranslator::new( cjs_esm_code_analyzer, diff --git a/cli/tools/bench/mod.rs b/cli/tools/bench/mod.rs index be5d0ad0e1e026..272d06335512a6 100644 --- a/cli/tools/bench/mod.rs +++ b/cli/tools/bench/mod.rs @@ -193,7 +193,7 @@ async fn bench_specifier_inner( .await?; // We execute the main module as a side module so that import.meta.main is not set. - worker.execute_side_module_possibly_with_npm().await?; + worker.execute_side_module().await?; let mut worker = worker.into_main_worker(); diff --git a/cli/tools/test/mod.rs b/cli/tools/test/mod.rs index e81abad0b26987..fa849614fafab6 100644 --- a/cli/tools/test/mod.rs +++ b/cli/tools/test/mod.rs @@ -631,7 +631,7 @@ async fn configure_main_worker( "Deno[Deno.internal].core.setLeakTracingEnabled(true);", )?; } - let res = worker.execute_side_module_possibly_with_npm().await; + let res = worker.execute_side_module().await; let mut worker = worker.into_main_worker(); match res { Ok(()) => Ok(()), diff --git a/cli/worker.rs b/cli/worker.rs index b0bab2876b5218..baacd681a16764 100644 --- a/cli/worker.rs +++ b/cli/worker.rs @@ -14,7 +14,6 @@ use deno_core::v8; use deno_core::CompiledWasmModuleStore; use deno_core::Extension; use deno_core::FeatureChecker; -use deno_core::ModuleId; use deno_core::ModuleLoader; use deno_core::PollEventLoopOptions; use deno_core::SharedArrayBufferStore; @@ -186,7 +185,7 @@ impl CliMainWorker { log::debug!("main_module {}", self.main_module); - self.execute_main_module_possibly_with_npm().await?; + self.execute_main_module().await?; self.worker.dispatch_load_event()?; loop { @@ -274,7 +273,7 @@ impl CliMainWorker { /// Execute the given main module emitting load and unload events before and after execution /// respectively. pub async fn execute(&mut self) -> Result<(), AnyError> { - self.inner.execute_main_module_possibly_with_npm().await?; + self.inner.execute_main_module().await?; self.inner.worker.dispatch_load_event()?; self.pending_unload = true; @@ -315,24 +314,13 @@ impl CliMainWorker { executor.execute().await } - pub async fn execute_main_module_possibly_with_npm( - &mut self, - ) -> Result<(), AnyError> { + pub async fn execute_main_module(&mut self) -> Result<(), AnyError> { let id = self.worker.preload_main_module(&self.main_module).await?; - self.evaluate_module_possibly_with_npm(id).await + self.worker.evaluate_module(id).await } - pub async fn execute_side_module_possibly_with_npm( - &mut self, - ) -> Result<(), AnyError> { + pub async fn execute_side_module(&mut self) -> Result<(), AnyError> { let id = self.worker.preload_side_module(&self.main_module).await?; - self.evaluate_module_possibly_with_npm(id).await - } - - async fn evaluate_module_possibly_with_npm( - &mut self, - id: ModuleId, - ) -> Result<(), AnyError> { self.worker.evaluate_module(id).await } From 220e782aa84ceb31d8b573c3054c52dddd66a96f Mon Sep 17 00:00:00 2001 From: David Sherret Date: Wed, 30 Oct 2024 12:45:36 -0400 Subject: [PATCH 33/42] do not require --allow-read if requiring modules found in the module graph --- cli/module_loader.rs | 22 ++++++++++++--- cli/standalone/binary.rs | 4 +++ cli/standalone/mod.rs | 5 ++++ ext/node/lib.rs | 6 ++--- ext/node/ops/require.rs | 5 +--- ext/node/polyfills/01_require.js | 27 +++++++++++++++++-- tests/integration/run_tests.rs | 2 +- tests/specs/compile/cjs/__test__.jsonc | 4 +-- tests/specs/compile/cjs/divide.cts | 1 + tests/specs/compile/cjs/multiply.cts | 2 +- tests/specs/compile/detect_cjs/__test__.jsonc | 7 +++-- tests/specs/compile/detect_cjs/add.js | 2 +- tests/specs/compile/detect_cjs/package.json | 5 +++- tests/specs/compile/detect_cjs/subtract.ts | 2 ++ 14 files changed, 72 insertions(+), 22 deletions(-) create mode 100644 tests/specs/compile/cjs/divide.cts create mode 100644 tests/specs/compile/detect_cjs/subtract.ts diff --git a/cli/module_loader.rs b/cli/module_loader.rs index c021f66127565e..9f286feae3fe52 100644 --- a/cli/module_loader.rs +++ b/cli/module_loader.rs @@ -289,7 +289,7 @@ impl CliModuleLoaderFactory { is_npm_main: self.shared.is_npm_main, parent_permissions, permissions, - graph_container, + graph_container: graph_container.clone(), node_code_translator: self.shared.node_code_translator.clone(), emitter: self.shared.emitter.clone(), parsed_source_cache: self.shared.parsed_source_cache.clone(), @@ -298,6 +298,7 @@ impl CliModuleLoaderFactory { let node_require_loader = Rc::new(CliNodeRequireLoader::new( self.shared.emitter.clone(), self.shared.fs.clone(), + graph_container, self.shared.in_npm_pkg_checker.clone(), self.shared.npm_resolver.clone(), )); @@ -1017,35 +1018,48 @@ impl ModuleGraphUpdatePermit for WorkerModuleGraphUpdatePermit { } #[derive(Debug)] -struct CliNodeRequireLoader { +struct CliNodeRequireLoader { emitter: Arc, fs: Arc, + graph_container: TGraphContainer, in_npm_pkg_checker: Arc, npm_resolver: Arc, } -impl CliNodeRequireLoader { +impl + CliNodeRequireLoader +{ pub fn new( emitter: Arc, fs: Arc, + graph_container: TGraphContainer, in_npm_pkg_checker: Arc, npm_resolver: Arc, ) -> Self { Self { emitter, fs, + graph_container, in_npm_pkg_checker, npm_resolver, } } } -impl NodeRequireLoader for CliNodeRequireLoader { +impl NodeRequireLoader + for CliNodeRequireLoader +{ fn ensure_read_permission<'a>( &self, permissions: &mut dyn deno_runtime::deno_node::NodePermissions, path: &'a Path, ) -> Result, AnyError> { + if let Ok(url) = deno_path_util::url_from_file_path(path) { + if self.graph_container.graph().get(&url).is_some() { + // allow reading if it's in the module graph + return Ok(std::borrow::Cow::Borrowed(path)); + } + } self.npm_resolver.ensure_read_permission(permissions, path) } diff --git a/cli/standalone/binary.rs b/cli/standalone/binary.rs index fdc57b7af1ecab..77d44549d3a3e5 100644 --- a/cli/standalone/binary.rs +++ b/cli/standalone/binary.rs @@ -259,6 +259,10 @@ impl StandaloneModules { } } + pub fn has_file(&self, path: &Path) -> bool { + self.vfs.file_entry(path).is_ok() + } + pub fn read<'a>( &'a self, specifier: &'a ModuleSpecifier, diff --git a/cli/standalone/mod.rs b/cli/standalone/mod.rs index 3b63bcd9850c34..733280090ba00c 100644 --- a/cli/standalone/mod.rs +++ b/cli/standalone/mod.rs @@ -412,6 +412,11 @@ impl NodeRequireLoader for EmbeddedModuleLoader { permissions: &mut dyn deno_runtime::deno_node::NodePermissions, path: &'a std::path::Path, ) -> Result, AnyError> { + if self.shared.modules.has_file(path) { + // allow reading if the file is in the snapshot + return Ok(Cow::Borrowed(path)); + } + self .shared .npm_resolver diff --git a/ext/node/lib.rs b/ext/node/lib.rs index 085208e721c40a..ddff414b666f32 100644 --- a/ext/node/lib.rs +++ b/ext/node/lib.rs @@ -14,8 +14,6 @@ use deno_core::url::Url; #[allow(unused_imports)] use deno_core::v8; use deno_core::v8::ExternalReference; -use deno_fs::sync::MaybeSend; -use deno_fs::sync::MaybeSync; use node_resolver::NpmResolverRc; use once_cell::sync::Lazy; @@ -125,7 +123,7 @@ impl NodePermissions for deno_permissions::PermissionsContainer { #[allow(clippy::disallowed_types)] pub type NodeRequireLoaderRc = std::rc::Rc; -pub trait NodeRequireLoader: std::fmt::Debug + MaybeSend + MaybeSync { +pub trait NodeRequireLoader { #[must_use = "the resolved return value to mitigate time-of-check to time-of-use issues"] fn ensure_read_permission<'a>( &self, @@ -151,7 +149,7 @@ fn op_node_build_os() -> String { env!("TARGET").split('-').nth(2).unwrap().to_string() } -#[derive(Debug, Clone)] +#[derive(Clone)] pub struct NodeExtInitServices { pub node_require_loader: NodeRequireLoaderRc, pub node_resolver: NodeResolverRc, diff --git a/ext/node/ops/require.rs b/ext/node/ops/require.rs index c0204947daa42c..3edb3acf8c96f4 100644 --- a/ext/node/ops/require.rs +++ b/ext/node/ops/require.rs @@ -137,9 +137,6 @@ where normalize_path(current_dir.join(from)) }; - let from = ensure_read_permission::

(state, &from) - .map_err(RequireError::Permission)?; - if cfg!(windows) { // return root node_modules when path is 'D:\\'. let from_str = from.to_str().unwrap(); @@ -160,7 +157,7 @@ where } let mut paths = Vec::with_capacity(from.components().count()); - let mut current_path = from.as_ref(); + let mut current_path = from.as_path(); let mut maybe_parent = Some(current_path); while let Some(parent) = maybe_parent { if !parent.ends_with("node_modules") { diff --git a/ext/node/polyfills/01_require.js b/ext/node/polyfills/01_require.js index 5b0980c310f6cf..026f470d4d607c 100644 --- a/ext/node/polyfills/01_require.js +++ b/ext/node/polyfills/01_require.js @@ -1075,13 +1075,33 @@ Module._extensions[".js"] = function (module, filename) { } else if (pkg?.type === "commonjs") { format = "commonjs"; } - } else if (StringPrototypeEndsWith(filename, ".cjs")) { + } + + module._compile(content, filename, format); +}; + +Module._extensions[".ts"] = function (module, filename) { + const content = op_require_read_file(filename); + + let format; + const pkg = op_require_read_closest_package_json(filename); + if (pkg?.type === "module") { + format = "module"; + } else if (pkg?.type === "commonjs") { format = "commonjs"; } module._compile(content, filename, format); }; +Module._extensions[".cjs"] = Module._extensions[".cts"] = function ( + module, + filename, +) { + const content = op_require_read_file(filename); + module._compile(content, filename, "commonjs"); +}; + function loadESMFromCJS(module, filename, code) { const namespace = op_import_sync( url.pathToFileURL(filename).toString(), @@ -1091,7 +1111,10 @@ function loadESMFromCJS(module, filename, code) { module.exports = namespace; } -Module._extensions[".mjs"] = function (module, filename) { +Module._extensions[".mjs"] = Module._extensions[".mts"] = function ( + module, + filename, +) { loadESMFromCJS(module, filename); }; diff --git a/tests/integration/run_tests.rs b/tests/integration/run_tests.rs index d6a2078d617b80..db9f79556ef9fb 100644 --- a/tests/integration/run_tests.rs +++ b/tests/integration/run_tests.rs @@ -3605,7 +3605,7 @@ fn running_declaration_files() { temp_dir.write(file, ""); context .new_command() - .args_vec(["run", "--allow-read", file]) + .args_vec(["run", file]) .run() .skip_output_check() .assert_exit_code(0); diff --git a/tests/specs/compile/cjs/__test__.jsonc b/tests/specs/compile/cjs/__test__.jsonc index 00bb55e8db7a54..9bdcf4724f5567 100644 --- a/tests/specs/compile/cjs/__test__.jsonc +++ b/tests/specs/compile/cjs/__test__.jsonc @@ -2,7 +2,7 @@ "tempDir": true, "steps": [{ "if": "unix", - "args": "compile -A --output main main.js", + "args": "compile --output main main.js", "output": "[WILDCARD]" }, { "if": "unix", @@ -12,7 +12,7 @@ "exitCode": 0 }, { "if": "windows", - "args": "compile -A --output main.exe main.js", + "args": "compile --output main.exe main.js", "output": "[WILDCARD]" }, { "if": "windows", diff --git a/tests/specs/compile/cjs/divide.cts b/tests/specs/compile/cjs/divide.cts new file mode 100644 index 00000000000000..d89a600a4eb91f --- /dev/null +++ b/tests/specs/compile/cjs/divide.cts @@ -0,0 +1 @@ +module.exports.divide = (a: number, b: number) => a / b; diff --git a/tests/specs/compile/cjs/multiply.cts b/tests/specs/compile/cjs/multiply.cts index 1fed324c790130..3c0618cfc54826 100644 --- a/tests/specs/compile/cjs/multiply.cts +++ b/tests/specs/compile/cjs/multiply.cts @@ -1,4 +1,4 @@ /// exports.multiply = function (a: number, b: number): number { - return a * b; + return require("./divide.cts").divide(a, 1 / b); }; diff --git a/tests/specs/compile/detect_cjs/__test__.jsonc b/tests/specs/compile/detect_cjs/__test__.jsonc index 6d4368899b9201..0abf121f05b1c3 100644 --- a/tests/specs/compile/detect_cjs/__test__.jsonc +++ b/tests/specs/compile/detect_cjs/__test__.jsonc @@ -1,8 +1,11 @@ { "tempDir": true, "steps": [{ + "args": "install", + "output": "[WILDCARD]" + }, { "if": "unix", - "args": "compile --allow-read --output main main.js", + "args": "compile --output main main.js", "output": "compile.out" }, { "if": "unix", @@ -12,7 +15,7 @@ "exitCode": 0 }, { "if": "windows", - "args": "compile --allow-read --output main.exe main.js", + "args": "compile --output main.exe main.js", "output": "compile.out" }, { "if": "windows", diff --git a/tests/specs/compile/detect_cjs/add.js b/tests/specs/compile/detect_cjs/add.js index 2a886fbc18bcbc..94b0263f0edd0d 100644 --- a/tests/specs/compile/detect_cjs/add.js +++ b/tests/specs/compile/detect_cjs/add.js @@ -1,3 +1,3 @@ module.exports.add = function (a, b) { - return a + b; + return require("./subtract.ts").subtract(a, -b); }; diff --git a/tests/specs/compile/detect_cjs/package.json b/tests/specs/compile/detect_cjs/package.json index 5bbefffbabee39..6e65b32ed55279 100644 --- a/tests/specs/compile/detect_cjs/package.json +++ b/tests/specs/compile/detect_cjs/package.json @@ -1,3 +1,6 @@ { - "type": "commonjs" + "type": "commonjs", + "dependencies": { + "@types/node": "*" + } } diff --git a/tests/specs/compile/detect_cjs/subtract.ts b/tests/specs/compile/detect_cjs/subtract.ts new file mode 100644 index 00000000000000..e4f6760b770698 --- /dev/null +++ b/tests/specs/compile/detect_cjs/subtract.ts @@ -0,0 +1,2 @@ +/// +module.exports.subtract = (a: number, b: number) => a - b; From 9aaa4d7367286c1964ced601d8e449909fd43d1e Mon Sep 17 00:00:00 2001 From: David Sherret Date: Wed, 30 Oct 2024 14:37:54 -0400 Subject: [PATCH 34/42] more tests --- .../reexport_non_analyzable}/__test__.jsonc | 0 .../reexport_non_analyzable}/deno.json | 0 .../reexport_non_analyzable}/main.ts | 0 .../node_modules/foo.cjs | 0 .../unprepared}/__test__.jsonc | 0 .../{cjs_unprepared => cjs/unprepared}/file.cjs | 0 .../{cjs_unprepared => cjs/unprepared}/main.out | 0 .../{cjs_unprepared => cjs/unprepared}/main.ts | 0 .../unprepared}/output.cjs | 0 .../specs/run/cts/cjs_import_cts/__test__.jsonc | 13 +++++++++++++ tests/specs/run/cts/cjs_import_cts/add.cts | 3 +++ tests/specs/run/cts/cjs_import_cts/check.out | 17 +++++++++++++++++ tests/specs/run/cts/cjs_import_cts/main.js | 3 +++ tests/specs/run/cts/cjs_import_cts/main.out | 1 + tests/specs/run/cts/cjs_import_cts/subtract.cjs | 3 +++ tests/specs/run/cts/main/__test__.jsonc | 4 ++++ tests/specs/run/cts/main/import_main.cjs | 1 + tests/specs/run/cts/main/main.cts | 5 +++++ tests/specs/run/cts/main/main.out | 1 + 19 files changed, 51 insertions(+) rename tests/specs/run/{cjs_reexport_non_analyzable => cjs/reexport_non_analyzable}/__test__.jsonc (100%) rename tests/specs/run/{cjs_reexport_non_analyzable => cjs/reexport_non_analyzable}/deno.json (100%) rename tests/specs/run/{cjs_reexport_non_analyzable => cjs/reexport_non_analyzable}/main.ts (100%) rename tests/specs/run/{cjs_reexport_non_analyzable => cjs/reexport_non_analyzable}/node_modules/foo.cjs (100%) rename tests/specs/run/{cjs_unprepared => cjs/unprepared}/__test__.jsonc (100%) rename tests/specs/run/{cjs_unprepared => cjs/unprepared}/file.cjs (100%) rename tests/specs/run/{cjs_unprepared => cjs/unprepared}/main.out (100%) rename tests/specs/run/{cjs_unprepared => cjs/unprepared}/main.ts (100%) rename tests/specs/run/{cjs_unprepared => cjs/unprepared}/output.cjs (100%) create mode 100644 tests/specs/run/cts/cjs_import_cts/__test__.jsonc create mode 100644 tests/specs/run/cts/cjs_import_cts/add.cts create mode 100644 tests/specs/run/cts/cjs_import_cts/check.out create mode 100644 tests/specs/run/cts/cjs_import_cts/main.js create mode 100644 tests/specs/run/cts/cjs_import_cts/main.out create mode 100644 tests/specs/run/cts/cjs_import_cts/subtract.cjs create mode 100644 tests/specs/run/cts/main/__test__.jsonc create mode 100644 tests/specs/run/cts/main/import_main.cjs create mode 100644 tests/specs/run/cts/main/main.cts create mode 100644 tests/specs/run/cts/main/main.out diff --git a/tests/specs/run/cjs_reexport_non_analyzable/__test__.jsonc b/tests/specs/run/cjs/reexport_non_analyzable/__test__.jsonc similarity index 100% rename from tests/specs/run/cjs_reexport_non_analyzable/__test__.jsonc rename to tests/specs/run/cjs/reexport_non_analyzable/__test__.jsonc diff --git a/tests/specs/run/cjs_reexport_non_analyzable/deno.json b/tests/specs/run/cjs/reexport_non_analyzable/deno.json similarity index 100% rename from tests/specs/run/cjs_reexport_non_analyzable/deno.json rename to tests/specs/run/cjs/reexport_non_analyzable/deno.json diff --git a/tests/specs/run/cjs_reexport_non_analyzable/main.ts b/tests/specs/run/cjs/reexport_non_analyzable/main.ts similarity index 100% rename from tests/specs/run/cjs_reexport_non_analyzable/main.ts rename to tests/specs/run/cjs/reexport_non_analyzable/main.ts diff --git a/tests/specs/run/cjs_reexport_non_analyzable/node_modules/foo.cjs b/tests/specs/run/cjs/reexport_non_analyzable/node_modules/foo.cjs similarity index 100% rename from tests/specs/run/cjs_reexport_non_analyzable/node_modules/foo.cjs rename to tests/specs/run/cjs/reexport_non_analyzable/node_modules/foo.cjs diff --git a/tests/specs/run/cjs_unprepared/__test__.jsonc b/tests/specs/run/cjs/unprepared/__test__.jsonc similarity index 100% rename from tests/specs/run/cjs_unprepared/__test__.jsonc rename to tests/specs/run/cjs/unprepared/__test__.jsonc diff --git a/tests/specs/run/cjs_unprepared/file.cjs b/tests/specs/run/cjs/unprepared/file.cjs similarity index 100% rename from tests/specs/run/cjs_unprepared/file.cjs rename to tests/specs/run/cjs/unprepared/file.cjs diff --git a/tests/specs/run/cjs_unprepared/main.out b/tests/specs/run/cjs/unprepared/main.out similarity index 100% rename from tests/specs/run/cjs_unprepared/main.out rename to tests/specs/run/cjs/unprepared/main.out diff --git a/tests/specs/run/cjs_unprepared/main.ts b/tests/specs/run/cjs/unprepared/main.ts similarity index 100% rename from tests/specs/run/cjs_unprepared/main.ts rename to tests/specs/run/cjs/unprepared/main.ts diff --git a/tests/specs/run/cjs_unprepared/output.cjs b/tests/specs/run/cjs/unprepared/output.cjs similarity index 100% rename from tests/specs/run/cjs_unprepared/output.cjs rename to tests/specs/run/cjs/unprepared/output.cjs diff --git a/tests/specs/run/cts/cjs_import_cts/__test__.jsonc b/tests/specs/run/cts/cjs_import_cts/__test__.jsonc new file mode 100644 index 00000000000000..2205183d07ef6b --- /dev/null +++ b/tests/specs/run/cts/cjs_import_cts/__test__.jsonc @@ -0,0 +1,13 @@ +{ + "tests": { + "no_check": { + "args": "run --allow-read main.js", + "output": "main.out" + }, + "check": { + "args": "check main.js", + "output": "check.out", + "exitCode": 1 + } + } +} diff --git a/tests/specs/run/cts/cjs_import_cts/add.cts b/tests/specs/run/cts/cjs_import_cts/add.cts new file mode 100644 index 00000000000000..2a886fbc18bcbc --- /dev/null +++ b/tests/specs/run/cts/cjs_import_cts/add.cts @@ -0,0 +1,3 @@ +module.exports.add = function (a, b) { + return a + b; +}; diff --git a/tests/specs/run/cts/cjs_import_cts/check.out b/tests/specs/run/cts/cjs_import_cts/check.out new file mode 100644 index 00000000000000..a27e8d7af75083 --- /dev/null +++ b/tests/specs/run/cts/cjs_import_cts/check.out @@ -0,0 +1,17 @@ +Check file:///[WILDLINE]main.js +error: TS2580 [ERROR]: Cannot find name 'module'. +module.exports.add = function (a, b) { +~~~~~~ + at file:///[WILDLINE] + +TS7006 [ERROR]: Parameter 'a' implicitly has an 'any' type. +module.exports.add = function (a, b) { + ^ + at file:///[WILDLINE] + +TS7006 [ERROR]: Parameter 'b' implicitly has an 'any' type. +module.exports.add = function (a, b) { + ^ + at file:///[WILDLINE] + +Found 3 errors. diff --git a/tests/specs/run/cts/cjs_import_cts/main.js b/tests/specs/run/cts/cjs_import_cts/main.js new file mode 100644 index 00000000000000..9546a0fea5a528 --- /dev/null +++ b/tests/specs/run/cts/cjs_import_cts/main.js @@ -0,0 +1,3 @@ +import { subtract } from "./subtract.cjs"; + +console.log(subtract(1, 2)); diff --git a/tests/specs/run/cts/cjs_import_cts/main.out b/tests/specs/run/cts/cjs_import_cts/main.out new file mode 100644 index 00000000000000..3a2e3f4984a0ee --- /dev/null +++ b/tests/specs/run/cts/cjs_import_cts/main.out @@ -0,0 +1 @@ +-1 diff --git a/tests/specs/run/cts/cjs_import_cts/subtract.cjs b/tests/specs/run/cts/cjs_import_cts/subtract.cjs new file mode 100644 index 00000000000000..7dee54346f7a9b --- /dev/null +++ b/tests/specs/run/cts/cjs_import_cts/subtract.cjs @@ -0,0 +1,3 @@ +module.exports.subtract = function (a, b) { + return require("./add.cts").add(a, -b); +}; diff --git a/tests/specs/run/cts/main/__test__.jsonc b/tests/specs/run/cts/main/__test__.jsonc new file mode 100644 index 00000000000000..0157b44e965749 --- /dev/null +++ b/tests/specs/run/cts/main/__test__.jsonc @@ -0,0 +1,4 @@ +{ + "args": "run --allow-read main.cts", + "output": "main.out" +} diff --git a/tests/specs/run/cts/main/import_main.cjs b/tests/specs/run/cts/main/import_main.cjs new file mode 100644 index 00000000000000..f61e0da2a6a9ec --- /dev/null +++ b/tests/specs/run/cts/main/import_main.cjs @@ -0,0 +1 @@ +require("./main.cts").sayHello(); diff --git a/tests/specs/run/cts/main/main.cts b/tests/specs/run/cts/main/main.cts new file mode 100644 index 00000000000000..0b0330686b916c --- /dev/null +++ b/tests/specs/run/cts/main/main.cts @@ -0,0 +1,5 @@ +module.exports.sayHello = function () { + console.log("Hello"); +}; + +require("./import_main.cjs"); diff --git a/tests/specs/run/cts/main/main.out b/tests/specs/run/cts/main/main.out new file mode 100644 index 00000000000000..e965047ad7c578 --- /dev/null +++ b/tests/specs/run/cts/main/main.out @@ -0,0 +1 @@ +Hello From 5049c3c4ea639e37cfddac05d294fdd8553fe16f Mon Sep 17 00:00:00 2001 From: David Sherret Date: Wed, 30 Oct 2024 17:32:54 -0400 Subject: [PATCH 35/42] fix small issues and add jsx tests --- cli/emit.rs | 23 +++++++---- cli/factory.rs | 1 + cli/graph_util.rs | 4 +- cli/module_loader.rs | 33 ++++++++++----- cli/resolver.rs | 4 ++ cli/standalone/binary.rs | 2 +- cli/standalone/mod.rs | 6 +-- ext/node/polyfills/01_require.js | 40 ++++++++++--------- tests/integration/run_tests.rs | 3 +- .../commonjs/{ => basic}/__test__.jsonc | 0 .../commonjs/{ => basic}/add.js | 0 .../commonjs/{ => basic}/deno.jsonc | 0 .../{ => basic}/import_import_meta.js | 0 .../commonjs/{ => basic}/import_meta.js | 0 .../commonjs/{ => basic}/main_cjs.js | 0 .../commonjs/{ => basic}/main_esm.js | 0 .../{ => basic}/main_esm_import_meta.js | 0 .../{ => basic}/main_esm_import_meta.out | 0 .../commonjs/{ => basic}/main_mix.js | 0 .../commonjs/{ => basic}/main_mix.out | 0 .../commonjs/{ => basic}/not_import_meta.js | 0 .../commonjs/{ => basic}/package.json | 0 .../commonjs/{ => basic}/tla.js | 0 .../commonjs/jsx/__test__.jsonc | 5 +++ .../run/package_json_type/commonjs/jsx/add.js | 3 ++ .../package_json_type/commonjs/jsx/deno.jsonc | 10 +++++ .../package_json_type/commonjs/jsx/main.jsx | 7 ++++ .../package_json_type/commonjs/jsx/main.out | 4 ++ .../commonjs/jsx/package.json | 7 ++++ .../package_json_type/commonjs/jsx/tsx.tsx | 5 +++ 30 files changed, 115 insertions(+), 42 deletions(-) rename tests/specs/run/package_json_type/commonjs/{ => basic}/__test__.jsonc (100%) rename tests/specs/run/package_json_type/commonjs/{ => basic}/add.js (100%) rename tests/specs/run/package_json_type/commonjs/{ => basic}/deno.jsonc (100%) rename tests/specs/run/package_json_type/commonjs/{ => basic}/import_import_meta.js (100%) rename tests/specs/run/package_json_type/commonjs/{ => basic}/import_meta.js (100%) rename tests/specs/run/package_json_type/commonjs/{ => basic}/main_cjs.js (100%) rename tests/specs/run/package_json_type/commonjs/{ => basic}/main_esm.js (100%) rename tests/specs/run/package_json_type/commonjs/{ => basic}/main_esm_import_meta.js (100%) rename tests/specs/run/package_json_type/commonjs/{ => basic}/main_esm_import_meta.out (100%) rename tests/specs/run/package_json_type/commonjs/{ => basic}/main_mix.js (100%) rename tests/specs/run/package_json_type/commonjs/{ => basic}/main_mix.out (100%) rename tests/specs/run/package_json_type/commonjs/{ => basic}/not_import_meta.js (100%) rename tests/specs/run/package_json_type/commonjs/{ => basic}/package.json (100%) rename tests/specs/run/package_json_type/commonjs/{ => basic}/tla.js (100%) create mode 100644 tests/specs/run/package_json_type/commonjs/jsx/__test__.jsonc create mode 100644 tests/specs/run/package_json_type/commonjs/jsx/add.js create mode 100644 tests/specs/run/package_json_type/commonjs/jsx/deno.jsonc create mode 100644 tests/specs/run/package_json_type/commonjs/jsx/main.jsx create mode 100644 tests/specs/run/package_json_type/commonjs/jsx/main.out create mode 100644 tests/specs/run/package_json_type/commonjs/jsx/package.json create mode 100644 tests/specs/run/package_json_type/commonjs/jsx/tsx.tsx diff --git a/cli/emit.rs b/cli/emit.rs index c018588f244e4e..8c4f2091cf021f 100644 --- a/cli/emit.rs +++ b/cli/emit.rs @@ -3,6 +3,7 @@ use crate::cache::EmitCache; use crate::cache::FastInsecureHasher; use crate::cache::ParsedSourceCache; +use crate::resolver::CjsTracker; use deno_ast::ModuleKind; use deno_ast::SourceMapOption; @@ -23,6 +24,7 @@ use std::sync::Arc; #[derive(Debug)] pub struct Emitter { + cjs_tracker: Arc, emit_cache: Arc, parsed_source_cache: Arc, transpile_and_emit_options: @@ -33,6 +35,7 @@ pub struct Emitter { impl Emitter { pub fn new( + cjs_tracker: Arc, emit_cache: Arc, parsed_source_cache: Arc, transpile_options: deno_ast::TranspileOptions, @@ -45,6 +48,7 @@ impl Emitter { hasher.finish() }; Self { + cjs_tracker, emit_cache, parsed_source_cache, transpile_and_emit_options: Arc::new((transpile_options, emit_options)), @@ -67,9 +71,14 @@ impl Emitter { self .emit_parsed_source( &module.specifier, - // todo(THIS PR): don't assume this is esm - ModuleKind::Esm, module.media_type, + ModuleKind::from_is_cjs( + self.cjs_tracker.is_cjs_with_known_is_script( + &module.specifier, + module.media_type, + module.is_script, + )?, + ), &module.source, ) .boxed_local(), @@ -98,8 +107,8 @@ impl Emitter { pub async fn emit_parsed_source( &self, specifier: &ModuleSpecifier, - module_kind: deno_ast::ModuleKind, media_type: MediaType, + module_kind: deno_ast::ModuleKind, source: &Arc, ) -> Result { // Note: keep this in sync with the sync version below @@ -117,8 +126,8 @@ impl Emitter { EmitParsedSourceHelper::transpile( &parsed_source_cache, &specifier, - module_kind, media_type, + module_kind, source.clone(), &transpile_and_emit_options.0, &transpile_and_emit_options.1, @@ -140,8 +149,8 @@ impl Emitter { pub fn emit_parsed_source_sync( &self, specifier: &ModuleSpecifier, - module_kind: deno_ast::ModuleKind, media_type: MediaType, + module_kind: deno_ast::ModuleKind, source: &Arc, ) -> Result { // Note: keep this in sync with the async version above @@ -152,8 +161,8 @@ impl Emitter { let transpiled_source = EmitParsedSourceHelper::transpile( &self.parsed_source_cache, specifier, - module_kind, media_type, + module_kind, source.clone(), &self.transpile_and_emit_options.0, &self.transpile_and_emit_options.1, @@ -264,8 +273,8 @@ impl<'a> EmitParsedSourceHelper<'a> { pub fn transpile( parsed_source_cache: &ParsedSourceCache, specifier: &ModuleSpecifier, - module_kind: deno_ast::ModuleKind, media_type: MediaType, + module_kind: deno_ast::ModuleKind, source: Arc, transpile_options: &deno_ast::TranspileOptions, emit_options: &deno_ast::EmitOptions, diff --git a/cli/factory.rs b/cli/factory.rs index 379116f4956f29..4a36c75ba2af12 100644 --- a/cli/factory.rs +++ b/cli/factory.rs @@ -602,6 +602,7 @@ impl CliFactory { ts_config_result.ts_config, )?; Ok(Arc::new(Emitter::new( + self.cjs_tracker()?.clone(), self.emit_cache()?.clone(), self.parsed_source_cache().clone(), transpile_options, diff --git a/cli/graph_util.rs b/cli/graph_util.rs index 7501048ba9dbfa..46257cf785bab4 100644 --- a/cli/graph_util.rs +++ b/cli/graph_util.rs @@ -671,10 +671,10 @@ impl ModuleGraphBuilder { graph.build_fast_check_type_graph( deno_graph::BuildFastCheckTypeGraphOptions { - jsr_url_provider: &CliJsrUrlProvider, + es_parser: Some(&parser), fast_check_cache: fast_check_cache.as_ref().map(|c| c as _), fast_check_dts: false, - es_parser: Some(&parser), + jsr_url_provider: &CliJsrUrlProvider, resolver: Some(graph_resolver), npm_resolver: Some(&graph_npm_resolver), workspace_fast_check: options.workspace_fast_check, diff --git a/cli/module_loader.rs b/cli/module_loader.rs index 9f286feae3fe52..43c9e1aa0750e0 100644 --- a/cli/module_loader.rs +++ b/cli/module_loader.rs @@ -573,7 +573,7 @@ impl }) => { let transpile_result = self .emitter - .emit_parsed_source(specifier, ModuleKind::Esm, media_type, source) + .emit_parsed_source(specifier, media_type, ModuleKind::Esm, source) .await?; // at this point, we no longer need the parsed source in memory, so free it @@ -613,8 +613,8 @@ impl }) => { let transpile_result = self.emitter.emit_parsed_source_sync( specifier, - ModuleKind::Esm, media_type, + ModuleKind::Esm, source, )?; @@ -734,17 +734,22 @@ impl &self, specifier: &ModuleSpecifier, media_type: MediaType, - source: &Arc, + original_source: &Arc, ) -> Result { let js_source = if media_type.is_emittable() { Cow::Owned( self .emitter - .emit_parsed_source(specifier, ModuleKind::Cjs, media_type, source) + .emit_parsed_source( + specifier, + media_type, + ModuleKind::Cjs, + original_source, + ) .await?, ) } else { - Cow::Borrowed(source.as_ref()) + Cow::Borrowed(original_source.as_ref()) }; let text = self .node_code_translator @@ -753,8 +758,15 @@ impl // at this point, we no longer need the parsed source in memory, so free it self.parsed_source_cache.free(specifier); Ok(ModuleCodeStringSource { - // todo(dsherret): perf: if text is borrowed then we can clone the original source - code: ModuleSourceCode::String(text.into_owned().into()), + code: match text { + // perf: if the text is borrowed, that means it didn't make any changes + // to the original source, so we can just provide that instead of cloning + // the borrowed text + Cow::Borrowed(_) => { + ModuleSourceCode::String(original_source.clone().into()) + } + Cow::Owned(text) => ModuleSourceCode::String(text.into()), + }, found_url: specifier.clone(), media_type, }) @@ -1055,8 +1067,8 @@ impl NodeRequireLoader path: &'a Path, ) -> Result, AnyError> { if let Ok(url) = deno_path_util::url_from_file_path(path) { + // allow reading if it's in the module graph if self.graph_container.graph().get(&url).is_some() { - // allow reading if it's in the module graph return Ok(std::borrow::Cow::Borrowed(path)); } } @@ -1080,8 +1092,11 @@ impl NodeRequireLoader } self.emitter.emit_parsed_source_sync( &specifier, - ModuleKind::Cjs, media_type, + // this is probably not super accurate due to require esm, but probably ok. + // If we find this causes a lot of churn in the emit cache then we should + // investigate how we can make this better + ModuleKind::Cjs, &text.into(), ) } else { diff --git a/cli/resolver.rs b/cli/resolver.rs index c772459d8404ed..710b97509345a6 100644 --- a/cli/resolver.rs +++ b/cli/resolver.rs @@ -424,6 +424,10 @@ pub struct CjsTrackerOptions { } /// Keeps track of what module specifiers were resolved as CJS. +/// +/// Modules that are `.js` or `.ts` are only known to be CJS or +/// ESM after they're loaded based on their contents. So these files +/// will be "maybe CJS" until they're loaded. #[derive(Debug)] pub struct CjsTracker { in_npm_pkg_checker: Arc, diff --git a/cli/standalone/binary.rs b/cli/standalone/binary.rs index 77d44549d3a3e5..9e265122687f03 100644 --- a/cli/standalone/binary.rs +++ b/cli/standalone/binary.rs @@ -619,8 +619,8 @@ impl<'a> DenoCompileBinaryWriter<'a> { .emitter .emit_parsed_source( &m.specifier, - module_kind, m.media_type, + module_kind, &m.source, ) .await?; diff --git a/cli/standalone/mod.rs b/cli/standalone/mod.rs index 733280090ba00c..85610f4c20f1f7 100644 --- a/cli/standalone/mod.rs +++ b/cli/standalone/mod.rs @@ -332,19 +332,19 @@ impl ModuleLoader for EmbeddedModuleLoader { let media_type = module.media_type; let (module_specifier, module_type, module_source) = module.into_parts(); - let is_cjs = match self + let is_maybe_cjs = match self .shared .cjs_tracker .is_maybe_cjs(original_specifier, media_type) { - Ok(is_cjs) => is_cjs, + Ok(is_maybe_cjs) => is_maybe_cjs, Err(err) => { return deno_core::ModuleLoadResponse::Sync(Err(type_error( format!("{:?}", err), ))); } }; - if is_cjs { + if is_maybe_cjs { let original_specifier = original_specifier.clone(); let module_specifier = module_specifier.clone(); let shared = self.shared.clone(); diff --git a/ext/node/polyfills/01_require.js b/ext/node/polyfills/01_require.js index 026f470d4d607c..a715117198487a 100644 --- a/ext/node/polyfills/01_require.js +++ b/ext/node/polyfills/01_require.js @@ -1080,27 +1080,29 @@ Module._extensions[".js"] = function (module, filename) { module._compile(content, filename, format); }; -Module._extensions[".ts"] = function (module, filename) { - const content = op_require_read_file(filename); - - let format; - const pkg = op_require_read_closest_package_json(filename); - if (pkg?.type === "module") { - format = "module"; - } else if (pkg?.type === "commonjs") { - format = "commonjs"; - } +Module._extensions[".ts"] = + Module._extensions[".jsx"] = + Module._extensions[".tsx"] = + function (module, filename) { + const content = op_require_read_file(filename); + + let format; + const pkg = op_require_read_closest_package_json(filename); + if (pkg?.type === "module") { + format = "module"; + } else if (pkg?.type === "commonjs") { + format = "commonjs"; + } - module._compile(content, filename, format); -}; + module._compile(content, filename, format); + }; -Module._extensions[".cjs"] = Module._extensions[".cts"] = function ( - module, - filename, -) { - const content = op_require_read_file(filename); - module._compile(content, filename, "commonjs"); -}; +Module._extensions[".cjs"] = + Module._extensions[".cts"] = + function (module, filename) { + const content = op_require_read_file(filename); + module._compile(content, filename, "commonjs"); + }; function loadESMFromCJS(module, filename, code) { const namespace = op_import_sync( diff --git a/tests/integration/run_tests.rs b/tests/integration/run_tests.rs index db9f79556ef9fb..a07dd56c833aca 100644 --- a/tests/integration/run_tests.rs +++ b/tests/integration/run_tests.rs @@ -3605,7 +3605,8 @@ fn running_declaration_files() { temp_dir.write(file, ""); context .new_command() - .args_vec(["run", file]) + // todo(dsherret): investigate why --allow-read is required here + .args_vec(["run", "--allow-read", file]) .run() .skip_output_check() .assert_exit_code(0); diff --git a/tests/specs/run/package_json_type/commonjs/__test__.jsonc b/tests/specs/run/package_json_type/commonjs/basic/__test__.jsonc similarity index 100% rename from tests/specs/run/package_json_type/commonjs/__test__.jsonc rename to tests/specs/run/package_json_type/commonjs/basic/__test__.jsonc diff --git a/tests/specs/run/package_json_type/commonjs/add.js b/tests/specs/run/package_json_type/commonjs/basic/add.js similarity index 100% rename from tests/specs/run/package_json_type/commonjs/add.js rename to tests/specs/run/package_json_type/commonjs/basic/add.js diff --git a/tests/specs/run/package_json_type/commonjs/deno.jsonc b/tests/specs/run/package_json_type/commonjs/basic/deno.jsonc similarity index 100% rename from tests/specs/run/package_json_type/commonjs/deno.jsonc rename to tests/specs/run/package_json_type/commonjs/basic/deno.jsonc diff --git a/tests/specs/run/package_json_type/commonjs/import_import_meta.js b/tests/specs/run/package_json_type/commonjs/basic/import_import_meta.js similarity index 100% rename from tests/specs/run/package_json_type/commonjs/import_import_meta.js rename to tests/specs/run/package_json_type/commonjs/basic/import_import_meta.js diff --git a/tests/specs/run/package_json_type/commonjs/import_meta.js b/tests/specs/run/package_json_type/commonjs/basic/import_meta.js similarity index 100% rename from tests/specs/run/package_json_type/commonjs/import_meta.js rename to tests/specs/run/package_json_type/commonjs/basic/import_meta.js diff --git a/tests/specs/run/package_json_type/commonjs/main_cjs.js b/tests/specs/run/package_json_type/commonjs/basic/main_cjs.js similarity index 100% rename from tests/specs/run/package_json_type/commonjs/main_cjs.js rename to tests/specs/run/package_json_type/commonjs/basic/main_cjs.js diff --git a/tests/specs/run/package_json_type/commonjs/main_esm.js b/tests/specs/run/package_json_type/commonjs/basic/main_esm.js similarity index 100% rename from tests/specs/run/package_json_type/commonjs/main_esm.js rename to tests/specs/run/package_json_type/commonjs/basic/main_esm.js diff --git a/tests/specs/run/package_json_type/commonjs/main_esm_import_meta.js b/tests/specs/run/package_json_type/commonjs/basic/main_esm_import_meta.js similarity index 100% rename from tests/specs/run/package_json_type/commonjs/main_esm_import_meta.js rename to tests/specs/run/package_json_type/commonjs/basic/main_esm_import_meta.js diff --git a/tests/specs/run/package_json_type/commonjs/main_esm_import_meta.out b/tests/specs/run/package_json_type/commonjs/basic/main_esm_import_meta.out similarity index 100% rename from tests/specs/run/package_json_type/commonjs/main_esm_import_meta.out rename to tests/specs/run/package_json_type/commonjs/basic/main_esm_import_meta.out diff --git a/tests/specs/run/package_json_type/commonjs/main_mix.js b/tests/specs/run/package_json_type/commonjs/basic/main_mix.js similarity index 100% rename from tests/specs/run/package_json_type/commonjs/main_mix.js rename to tests/specs/run/package_json_type/commonjs/basic/main_mix.js diff --git a/tests/specs/run/package_json_type/commonjs/main_mix.out b/tests/specs/run/package_json_type/commonjs/basic/main_mix.out similarity index 100% rename from tests/specs/run/package_json_type/commonjs/main_mix.out rename to tests/specs/run/package_json_type/commonjs/basic/main_mix.out diff --git a/tests/specs/run/package_json_type/commonjs/not_import_meta.js b/tests/specs/run/package_json_type/commonjs/basic/not_import_meta.js similarity index 100% rename from tests/specs/run/package_json_type/commonjs/not_import_meta.js rename to tests/specs/run/package_json_type/commonjs/basic/not_import_meta.js diff --git a/tests/specs/run/package_json_type/commonjs/package.json b/tests/specs/run/package_json_type/commonjs/basic/package.json similarity index 100% rename from tests/specs/run/package_json_type/commonjs/package.json rename to tests/specs/run/package_json_type/commonjs/basic/package.json diff --git a/tests/specs/run/package_json_type/commonjs/tla.js b/tests/specs/run/package_json_type/commonjs/basic/tla.js similarity index 100% rename from tests/specs/run/package_json_type/commonjs/tla.js rename to tests/specs/run/package_json_type/commonjs/basic/tla.js diff --git a/tests/specs/run/package_json_type/commonjs/jsx/__test__.jsonc b/tests/specs/run/package_json_type/commonjs/jsx/__test__.jsonc new file mode 100644 index 00000000000000..f815fd72cb55cd --- /dev/null +++ b/tests/specs/run/package_json_type/commonjs/jsx/__test__.jsonc @@ -0,0 +1,5 @@ +{ + "tempDir": true, + "args": "run -A --quiet main.jsx", + "output": "main.out" +} diff --git a/tests/specs/run/package_json_type/commonjs/jsx/add.js b/tests/specs/run/package_json_type/commonjs/jsx/add.js new file mode 100644 index 00000000000000..2a886fbc18bcbc --- /dev/null +++ b/tests/specs/run/package_json_type/commonjs/jsx/add.js @@ -0,0 +1,3 @@ +module.exports.add = function (a, b) { + return a + b; +}; diff --git a/tests/specs/run/package_json_type/commonjs/jsx/deno.jsonc b/tests/specs/run/package_json_type/commonjs/jsx/deno.jsonc new file mode 100644 index 00000000000000..192ddb98c4130f --- /dev/null +++ b/tests/specs/run/package_json_type/commonjs/jsx/deno.jsonc @@ -0,0 +1,10 @@ +{ + "nodeModulesDir": "auto", + "compilerOptions": { + "jsx": "react-jsx", + "jsxImportSource": "react" + }, + "unstable": [ + "detect-cjs" + ] +} diff --git a/tests/specs/run/package_json_type/commonjs/jsx/main.jsx b/tests/specs/run/package_json_type/commonjs/jsx/main.jsx new file mode 100644 index 00000000000000..1922fce1bbe3b7 --- /dev/null +++ b/tests/specs/run/package_json_type/commonjs/jsx/main.jsx @@ -0,0 +1,7 @@ +const { add } = require("./add.js"); + +console.log(add(1, 2)); + +console.log(

!= null); + +require("./tsx.tsx"); diff --git a/tests/specs/run/package_json_type/commonjs/jsx/main.out b/tests/specs/run/package_json_type/commonjs/jsx/main.out new file mode 100644 index 00000000000000..cdcf0454188aa5 --- /dev/null +++ b/tests/specs/run/package_json_type/commonjs/jsx/main.out @@ -0,0 +1,4 @@ +3 +true +4 +true diff --git a/tests/specs/run/package_json_type/commonjs/jsx/package.json b/tests/specs/run/package_json_type/commonjs/jsx/package.json new file mode 100644 index 00000000000000..88afcdcd95401b --- /dev/null +++ b/tests/specs/run/package_json_type/commonjs/jsx/package.json @@ -0,0 +1,7 @@ +{ + "type": "commonjs", + "dependencies": { + "@types/react": "*", + "react": "*" + } +} diff --git a/tests/specs/run/package_json_type/commonjs/jsx/tsx.tsx b/tests/specs/run/package_json_type/commonjs/jsx/tsx.tsx new file mode 100644 index 00000000000000..88e56bc663bda2 --- /dev/null +++ b/tests/specs/run/package_json_type/commonjs/jsx/tsx.tsx @@ -0,0 +1,5 @@ +const mod = require("./add.js"); + +console.log(mod.add(2, 2)); + +console.log(
!= null); From adcf6d3bb34162baaaf8059afe4a16dbc2f5c03e Mon Sep 17 00:00:00 2001 From: David Sherret Date: Wed, 30 Oct 2024 18:39:54 -0400 Subject: [PATCH 36/42] Bump ci From 2636535bed8d5eb4a96e1909f22e7206b1f0c233 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Bartek=20Iwa=C5=84czuk?= Date: Thu, 31 Oct 2024 15:43:46 +0100 Subject: [PATCH 37/42] mark main CJS module --- resolvers/node/analyze.rs | 9 +++++++-- 1 file changed, 7 insertions(+), 2 deletions(-) diff --git a/resolvers/node/analyze.rs b/resolvers/node/analyze.rs index d369f39a68d5ec..c7415933d74094 100644 --- a/resolvers/node/analyze.rs +++ b/resolvers/node/analyze.rs @@ -115,7 +115,7 @@ impl }; let mut source = vec![ - r#"import {createRequire as __internalCreateRequire} from "node:module"; + r#"import {createRequire as __internalCreateRequire, Module as __internalModule } from "node:module"; const require = __internalCreateRequire(import.meta.url);"# .to_string(), ]; @@ -142,7 +142,12 @@ impl } source.push(format!( - "const mod = require(\"{}\");", + r#"let mod; + if (import.meta.main) {{ + mod = __internalModule._load("{0}", null, true) + }} else {{ + mod = require("{0}"); + }}"#, url_to_file_path(entry_specifier) .unwrap() .to_str() From d47411b9adc9c5cb0ec3529386ae5f5c14652d5d Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Bartek=20Iwa=C5=84czuk?= Date: Thu, 31 Oct 2024 16:00:54 +0100 Subject: [PATCH 38/42] add a test for main cjs module --- tests/specs/run/cjs/main_module/__test__.jsonc | 4 ++++ tests/specs/run/cjs/main_module/main.cjs | 1 + tests/specs/run/cjs/main_module/main.out | 5 +++++ 3 files changed, 10 insertions(+) create mode 100644 tests/specs/run/cjs/main_module/__test__.jsonc create mode 100644 tests/specs/run/cjs/main_module/main.cjs create mode 100644 tests/specs/run/cjs/main_module/main.out diff --git a/tests/specs/run/cjs/main_module/__test__.jsonc b/tests/specs/run/cjs/main_module/__test__.jsonc new file mode 100644 index 00000000000000..e756a63627c594 --- /dev/null +++ b/tests/specs/run/cjs/main_module/__test__.jsonc @@ -0,0 +1,4 @@ +{ + "args": "run --allow-read main.cjs", + "output": "main.out" +} diff --git a/tests/specs/run/cjs/main_module/main.cjs b/tests/specs/run/cjs/main_module/main.cjs new file mode 100644 index 00000000000000..2a4c57ab1ca31f --- /dev/null +++ b/tests/specs/run/cjs/main_module/main.cjs @@ -0,0 +1 @@ +console.log(require.main); diff --git a/tests/specs/run/cjs/main_module/main.out b/tests/specs/run/cjs/main_module/main.out new file mode 100644 index 00000000000000..93b86d27e3d7dd --- /dev/null +++ b/tests/specs/run/cjs/main_module/main.out @@ -0,0 +1,5 @@ +Module { + id: ".", + [WILDCARD] + filename: "[WILDCARD]main.cjs", +[WILDCARD] \ No newline at end of file From 66c139d03cd846dc0a0fcfda3939523d3981b48c Mon Sep 17 00:00:00 2001 From: David Sherret Date: Thu, 31 Oct 2024 11:06:14 -0400 Subject: [PATCH 39/42] Update --- Cargo.lock | 8 ++++---- Cargo.toml | 2 +- cli/Cargo.toml | 2 +- cli/lsp/resolver.rs | 2 +- cli/node.rs | 5 +++-- tests/specs/run/package_json_type/commonjs/jsx/tsx.tsx | 2 +- 6 files changed, 11 insertions(+), 10 deletions(-) diff --git a/Cargo.lock b/Cargo.lock index acda66ba0cf9af..0e977c926adbbd 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -1279,9 +1279,9 @@ dependencies = [ [[package]] name = "deno_ast" -version = "0.43.1" +version = "0.43.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "01cdd89f7a3327ee0ff50d472c5a841efb5d092f93a2406aae73f8f881cd5f13" +checksum = "cbbf350300650502744d50bdd31eca7ea094c25216758a5a02a2b5ac66e62cf9" dependencies = [ "base64 0.21.7", "deno_media_type", @@ -1606,9 +1606,9 @@ dependencies = [ [[package]] name = "deno_graph" -version = "0.84.0" +version = "0.84.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "c9d6c7edec696810563d8002e4330e87b63ecbdd48996a18a94946b9a0f4db50" +checksum = "cd4f4a14aa069087be41c2998077b0453f0191747898f96e6343f700abfc2c18" dependencies = [ "anyhow", "async-trait", diff --git a/Cargo.toml b/Cargo.toml index 338c27a5759606..78059f8ad70ef2 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -45,7 +45,7 @@ license = "MIT" repository = "https://github.com/denoland/deno" [workspace.dependencies] -deno_ast = { version = "=0.43.1", features = ["transpiling"] } +deno_ast = { version = "=0.43.2", features = ["transpiling"] } deno_core = { version = "0.316.0" } deno_bench_util = { version = "0.169.0", path = "./bench_util" } diff --git a/cli/Cargo.toml b/cli/Cargo.toml index 57a5a48701565f..a3e2b71f0021e3 100644 --- a/cli/Cargo.toml +++ b/cli/Cargo.toml @@ -73,7 +73,7 @@ deno_cache_dir = { workspace = true } deno_config = { version = "=0.37.2", features = ["workspace", "sync"] } deno_core = { workspace = true, features = ["include_js_files_for_snapshotting"] } deno_doc = { version = "0.156.0", default-features = false, features = ["rust", "html", "syntect"] } -deno_graph = { version = "=0.84.0" } +deno_graph = { version = "=0.84.1" } deno_lint = { version = "=0.68.0", features = ["docs"] } deno_lockfile.workspace = true deno_npm.workspace = true diff --git a/cli/lsp/resolver.rs b/cli/lsp/resolver.rs index 1114c8db5b2b28..9ce76005e50078 100644 --- a/cli/lsp/resolver.rs +++ b/cli/lsp/resolver.rs @@ -818,7 +818,7 @@ impl LspCjsTracker { { module_kind.is_cjs() } else { - let maybe_is_script = maybe_parsed_source.map(|p| p.is_script()); + let maybe_is_script = maybe_parsed_source.map(|p| p.compute_is_script()); maybe_is_script .and_then(|is_script| { self diff --git a/cli/node.rs b/cli/node.rs index 00ba4a5779e24a..1d410a726af601 100644 --- a/cli/node.rs +++ b/cli/node.rs @@ -130,11 +130,12 @@ impl CliCjsCodeAnalyzer { maybe_syntax: None, }) })?; + let is_script = parsed_source.compute_is_script(); let is_cjs = cjs_tracker.is_cjs_with_known_is_script( parsed_source.specifier(), media_type, - parsed_source.is_script(), - )? || parsed_source.is_script() && is_npm_main; + is_script, + )? || is_script && is_npm_main; if is_cjs { let analysis = parsed_source.analyze_cjs(); Ok(CliCjsAnalysis::Cjs { diff --git a/tests/specs/run/package_json_type/commonjs/jsx/tsx.tsx b/tests/specs/run/package_json_type/commonjs/jsx/tsx.tsx index 88e56bc663bda2..ad8f0c0a9cddf5 100644 --- a/tests/specs/run/package_json_type/commonjs/jsx/tsx.tsx +++ b/tests/specs/run/package_json_type/commonjs/jsx/tsx.tsx @@ -1,4 +1,4 @@ -const mod = require("./add.js"); +import mod = require("./add.js"); console.log(mod.add(2, 2)); From b2500ddc9fe653e58ac2c93195b2daf2cb5cd91d Mon Sep 17 00:00:00 2001 From: David Sherret Date: Thu, 31 Oct 2024 11:08:14 -0400 Subject: [PATCH 40/42] Support TsImportEquals --- Cargo.lock | 8 ++++---- Cargo.toml | 2 +- cli/Cargo.toml | 2 +- cli/lsp/resolver.rs | 2 +- cli/node.rs | 5 +++-- tests/specs/run/package_json_type/commonjs/jsx/tsx.tsx | 2 +- 6 files changed, 11 insertions(+), 10 deletions(-) diff --git a/Cargo.lock b/Cargo.lock index acda66ba0cf9af..0e977c926adbbd 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -1279,9 +1279,9 @@ dependencies = [ [[package]] name = "deno_ast" -version = "0.43.1" +version = "0.43.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "01cdd89f7a3327ee0ff50d472c5a841efb5d092f93a2406aae73f8f881cd5f13" +checksum = "cbbf350300650502744d50bdd31eca7ea094c25216758a5a02a2b5ac66e62cf9" dependencies = [ "base64 0.21.7", "deno_media_type", @@ -1606,9 +1606,9 @@ dependencies = [ [[package]] name = "deno_graph" -version = "0.84.0" +version = "0.84.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "c9d6c7edec696810563d8002e4330e87b63ecbdd48996a18a94946b9a0f4db50" +checksum = "cd4f4a14aa069087be41c2998077b0453f0191747898f96e6343f700abfc2c18" dependencies = [ "anyhow", "async-trait", diff --git a/Cargo.toml b/Cargo.toml index 338c27a5759606..78059f8ad70ef2 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -45,7 +45,7 @@ license = "MIT" repository = "https://github.com/denoland/deno" [workspace.dependencies] -deno_ast = { version = "=0.43.1", features = ["transpiling"] } +deno_ast = { version = "=0.43.2", features = ["transpiling"] } deno_core = { version = "0.316.0" } deno_bench_util = { version = "0.169.0", path = "./bench_util" } diff --git a/cli/Cargo.toml b/cli/Cargo.toml index 57a5a48701565f..a3e2b71f0021e3 100644 --- a/cli/Cargo.toml +++ b/cli/Cargo.toml @@ -73,7 +73,7 @@ deno_cache_dir = { workspace = true } deno_config = { version = "=0.37.2", features = ["workspace", "sync"] } deno_core = { workspace = true, features = ["include_js_files_for_snapshotting"] } deno_doc = { version = "0.156.0", default-features = false, features = ["rust", "html", "syntect"] } -deno_graph = { version = "=0.84.0" } +deno_graph = { version = "=0.84.1" } deno_lint = { version = "=0.68.0", features = ["docs"] } deno_lockfile.workspace = true deno_npm.workspace = true diff --git a/cli/lsp/resolver.rs b/cli/lsp/resolver.rs index 1114c8db5b2b28..9ce76005e50078 100644 --- a/cli/lsp/resolver.rs +++ b/cli/lsp/resolver.rs @@ -818,7 +818,7 @@ impl LspCjsTracker { { module_kind.is_cjs() } else { - let maybe_is_script = maybe_parsed_source.map(|p| p.is_script()); + let maybe_is_script = maybe_parsed_source.map(|p| p.compute_is_script()); maybe_is_script .and_then(|is_script| { self diff --git a/cli/node.rs b/cli/node.rs index 00ba4a5779e24a..1d410a726af601 100644 --- a/cli/node.rs +++ b/cli/node.rs @@ -130,11 +130,12 @@ impl CliCjsCodeAnalyzer { maybe_syntax: None, }) })?; + let is_script = parsed_source.compute_is_script(); let is_cjs = cjs_tracker.is_cjs_with_known_is_script( parsed_source.specifier(), media_type, - parsed_source.is_script(), - )? || parsed_source.is_script() && is_npm_main; + is_script, + )? || is_script && is_npm_main; if is_cjs { let analysis = parsed_source.analyze_cjs(); Ok(CliCjsAnalysis::Cjs { diff --git a/tests/specs/run/package_json_type/commonjs/jsx/tsx.tsx b/tests/specs/run/package_json_type/commonjs/jsx/tsx.tsx index 88e56bc663bda2..ad8f0c0a9cddf5 100644 --- a/tests/specs/run/package_json_type/commonjs/jsx/tsx.tsx +++ b/tests/specs/run/package_json_type/commonjs/jsx/tsx.tsx @@ -1,4 +1,4 @@ -const mod = require("./add.js"); +import mod = require("./add.js"); console.log(mod.add(2, 2)); From c21c283dc53ad1a2eae97eaf17b069ba7e427dd3 Mon Sep 17 00:00:00 2001 From: David Sherret Date: Thu, 31 Oct 2024 13:21:52 -0400 Subject: [PATCH 41/42] proper support for ts import equals and export equals --- Cargo.lock | 4 ++-- Cargo.toml | 2 +- tests/specs/run/cts/import_export_equals/__test__.jsonc | 4 ++++ tests/specs/run/cts/import_export_equals/add.cts | 3 +++ tests/specs/run/cts/import_export_equals/main.cts | 3 +++ tests/specs/run/cts/import_export_equals/main.out | 2 ++ 6 files changed, 15 insertions(+), 3 deletions(-) create mode 100644 tests/specs/run/cts/import_export_equals/__test__.jsonc create mode 100644 tests/specs/run/cts/import_export_equals/add.cts create mode 100644 tests/specs/run/cts/import_export_equals/main.cts create mode 100644 tests/specs/run/cts/import_export_equals/main.out diff --git a/Cargo.lock b/Cargo.lock index 0e977c926adbbd..8c7b35ca9f27d6 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -1279,9 +1279,9 @@ dependencies = [ [[package]] name = "deno_ast" -version = "0.43.2" +version = "0.43.3" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "cbbf350300650502744d50bdd31eca7ea094c25216758a5a02a2b5ac66e62cf9" +checksum = "48d00b724e06d2081a141ec1155756a0b465d413d8e2a7515221f61d482eb2ee" dependencies = [ "base64 0.21.7", "deno_media_type", diff --git a/Cargo.toml b/Cargo.toml index 78059f8ad70ef2..f57563e0b86b53 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -45,7 +45,7 @@ license = "MIT" repository = "https://github.com/denoland/deno" [workspace.dependencies] -deno_ast = { version = "=0.43.2", features = ["transpiling"] } +deno_ast = { version = "=0.43.3", features = ["transpiling"] } deno_core = { version = "0.316.0" } deno_bench_util = { version = "0.169.0", path = "./bench_util" } diff --git a/tests/specs/run/cts/import_export_equals/__test__.jsonc b/tests/specs/run/cts/import_export_equals/__test__.jsonc new file mode 100644 index 00000000000000..a3aa19b4fd82d7 --- /dev/null +++ b/tests/specs/run/cts/import_export_equals/__test__.jsonc @@ -0,0 +1,4 @@ +{ + "args": "run --check --allow-read=. main.cts", + "output": "main.out" +} diff --git a/tests/specs/run/cts/import_export_equals/add.cts b/tests/specs/run/cts/import_export_equals/add.cts new file mode 100644 index 00000000000000..adf3503ac5fdda --- /dev/null +++ b/tests/specs/run/cts/import_export_equals/add.cts @@ -0,0 +1,3 @@ +export = function (a: number, b: number) { + return a + b; +}; diff --git a/tests/specs/run/cts/import_export_equals/main.cts b/tests/specs/run/cts/import_export_equals/main.cts new file mode 100644 index 00000000000000..e5c45f92e08648 --- /dev/null +++ b/tests/specs/run/cts/import_export_equals/main.cts @@ -0,0 +1,3 @@ +import add = require("./add.cts"); + +console.log(add(1, 2)); diff --git a/tests/specs/run/cts/import_export_equals/main.out b/tests/specs/run/cts/import_export_equals/main.out new file mode 100644 index 00000000000000..e7a973a7d04d12 --- /dev/null +++ b/tests/specs/run/cts/import_export_equals/main.out @@ -0,0 +1,2 @@ +Check file:///[WILDLINE]/main.cts +3 From 0411a7317525ff6588b60b9a85dc7cc086f16a48 Mon Sep 17 00:00:00 2001 From: David Sherret Date: Thu, 31 Oct 2024 16:48:37 -0400 Subject: [PATCH 42/42] Add some more import equals tests --- .../run/cts/import_export_equals/__test__.jsonc | 17 +++++++++++++++-- .../specs/run/cts/import_export_equals/mod.mts | 3 +++ .../cts/import_export_equals/mod.mts.check.out | 5 +++++ .../run/cts/import_export_equals/mod.mts.out | 1 + 4 files changed, 24 insertions(+), 2 deletions(-) create mode 100644 tests/specs/run/cts/import_export_equals/mod.mts create mode 100644 tests/specs/run/cts/import_export_equals/mod.mts.check.out create mode 100644 tests/specs/run/cts/import_export_equals/mod.mts.out diff --git a/tests/specs/run/cts/import_export_equals/__test__.jsonc b/tests/specs/run/cts/import_export_equals/__test__.jsonc index a3aa19b4fd82d7..6a5c6db42013ba 100644 --- a/tests/specs/run/cts/import_export_equals/__test__.jsonc +++ b/tests/specs/run/cts/import_export_equals/__test__.jsonc @@ -1,4 +1,17 @@ { - "args": "run --check --allow-read=. main.cts", - "output": "main.out" + "tests": { + "main": { + "args": "run --check --allow-read=. main.cts", + "output": "main.out" + }, + "mts": { + "args": "run --allow-read=. mod.mts", + "output": "mod.mts.out" + }, + "mts_check": { + "args": "check mod.mts", + "output": "mod.mts.check.out", + "exitCode": 1 + } + } } diff --git a/tests/specs/run/cts/import_export_equals/mod.mts b/tests/specs/run/cts/import_export_equals/mod.mts new file mode 100644 index 00000000000000..5fbbd6c6a0554b --- /dev/null +++ b/tests/specs/run/cts/import_export_equals/mod.mts @@ -0,0 +1,3 @@ +import add from "./add.cts"; + +console.log(add(1, "test")); diff --git a/tests/specs/run/cts/import_export_equals/mod.mts.check.out b/tests/specs/run/cts/import_export_equals/mod.mts.check.out new file mode 100644 index 00000000000000..8703539019d616 --- /dev/null +++ b/tests/specs/run/cts/import_export_equals/mod.mts.check.out @@ -0,0 +1,5 @@ +Check file:///[WILDLINE]/mod.mts +error: TS2345 [ERROR]: Argument of type 'string' is not assignable to parameter of type 'number'. +console.log(add(1, "test")); + ~~~~~~ + at file:///[WILDLINE]/mod.mts:3:20 diff --git a/tests/specs/run/cts/import_export_equals/mod.mts.out b/tests/specs/run/cts/import_export_equals/mod.mts.out new file mode 100644 index 00000000000000..208465a08080ba --- /dev/null +++ b/tests/specs/run/cts/import_export_equals/mod.mts.out @@ -0,0 +1 @@ +1test