// Copyright 2018-2024 the Deno authors. All rights reserved. MIT license. use crate::args::check_warn_tsconfig; use crate::args::discover_npmrc_from_workspace; use crate::args::get_root_cert_store; use crate::args::CaData; use crate::args::CliLockfile; use crate::args::CliOptions; use crate::args::DenoSubcommand; use crate::args::Flags; use crate::args::NpmInstallDepsProvider; use crate::args::ScopeOptions; 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; use crate::cache::GlobalHttpCache; use crate::cache::HttpCache; use crate::cache::LocalHttpCache; use crate::cache::ModuleInfoCache; use crate::cache::NodeAnalysisCache; use crate::cache::ParsedSourceCache; use crate::emit::Emitter; use crate::file_fetcher::File; use crate::file_fetcher::FileFetcher; use crate::graph_container::MainModuleGraphContainer; use crate::graph_util::FileWatcherReporter; use crate::graph_util::ModuleGraphBuilder; use crate::graph_util::ModuleGraphCreator; use crate::http_util::HttpClientProvider; use crate::module_loader::CliModuleLoaderFactory; 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::CliNpmResolverManagedSnapshotOption; use crate::npm::CreateInNpmPkgCheckerOptions; use crate::resolver::CjsTracker; use crate::resolver::CliDenoResolver; use crate::resolver::CliDenoResolverFs; use crate::resolver::CliNpmReqResolver; use crate::resolver::CliResolver; use crate::resolver::CliResolverOptions; use crate::resolver::CliSloppyImportsResolver; use crate::resolver::NpmModuleLoader; use crate::resolver::SloppyImportsCachedFs; use crate::standalone::DenoCompileBinaryWriter; use crate::tools::check::MaybeDiagnostics; use crate::tools::check::TypeChecker; use crate::tools::coverage::CoverageCollector; use crate::tools::lint::LintRuleProvider; use crate::tools::run::hmr::HmrRunner; use crate::tsc::Diagnostics; 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; use crate::util::progress_bar::ProgressBarStyle; use crate::worker::CliMainWorkerFactory; use crate::worker::CliMainWorkerOptions; use std::collections::BTreeSet; use std::path::PathBuf; use deno_ast::ModuleSpecifier; use deno_cache_dir::npm::NpmCacheDir; use deno_config::glob::FilePatterns; use deno_config::workspace::PackageJsonDepResolution; use deno_config::workspace::WorkspaceDirectory; use deno_config::workspace::WorkspaceResolver; use deno_core::anyhow::anyhow; use deno_core::error::AnyError; use deno_core::futures::FutureExt; use deno_core::FeatureChecker; use deno_resolver::cjs::IsCjsResolutionMode; use deno_resolver::npm::NpmReqResolverOptions; use deno_resolver::DenoResolverOptions; use deno_resolver::NodeAndNpmReqResolver; 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; use deno_runtime::deno_tls::RootCertStoreProvider; use deno_runtime::deno_web::BlobStore; 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; struct CliRootCertStoreProvider { cell: OnceCell, maybe_root_path: Option, maybe_ca_stores: Option>, maybe_ca_data: Option, } impl CliRootCertStoreProvider { pub fn new( maybe_root_path: Option, maybe_ca_stores: Option>, maybe_ca_data: Option, ) -> Self { Self { cell: Default::default(), maybe_root_path, maybe_ca_stores, maybe_ca_data, } } } impl RootCertStoreProvider for CliRootCertStoreProvider { fn get_or_try_init(&self) -> Result<&RootCertStore, AnyError> { self .cell .get_or_try_init(|| { get_root_cert_store( self.maybe_root_path.clone(), self.maybe_ca_stores.clone(), self.maybe_ca_data.clone(), ) }) .map_err(|e| e.into()) } } pub struct Deferred(once_cell::unsync::OnceCell); impl Default for Deferred { fn default() -> Self { Self(once_cell::unsync::OnceCell::default()) } } impl Deferred { pub fn from_value(value: T) -> Self { Self(once_cell::unsync::OnceCell::from(value)) } #[inline(always)] pub fn get_or_try_init( &self, create: impl FnOnce() -> Result, ) -> Result<&T, AnyError> { self.0.get_or_try_init(create) } #[inline(always)] pub fn get_or_init(&self, create: impl FnOnce() -> T) -> &T { self.0.get_or_init(create) } pub async fn get_or_try_init_async( &self, // some futures passed here are boxed because it was discovered // that they were called a lot, causing other futures to get // really big causing stack overflows on Windows create: impl Future>, ) -> Result<&T, AnyError> { if self.0.get().is_none() { // todo(dsherret): it would be more ideal if this enforced a // single executor and then we could make some initialization // concurrent let val = create.await?; _ = self.0.set(val); } Ok(self.0.get().unwrap()) } } #[derive(Default)] struct CliFactoryServices { blob_store: Deferred>, caches: Deferred>, cjs_tracker: Deferred>, cli_options: Deferred>, code_cache: Deferred>, deno_resolver: 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>, in_npm_pkg_checker: Deferred>, main_graph_container: 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_req_resolver: 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>, workspace_resolver: Deferred>, } pub struct CliFactory { watcher_communicator: Option>, flags: Arc, services: CliFactoryServices, } impl CliFactory { pub fn from_flags(flags: Arc) -> Self { Self { flags, watcher_communicator: None, services: Default::default(), } } pub fn from_cli_options(cli_options: Arc) -> Self { let (cli_options, flags) = cli_options.into_self_and_flags(); CliFactory { watcher_communicator: None, flags, services: CliFactoryServices { cli_options: Deferred::from_value(cli_options), ..Default::default() }, } } pub fn from_flags_for_watcher( flags: Arc, watcher_communicator: Arc, ) -> Self { CliFactory { watcher_communicator: Some(watcher_communicator), flags, services: Default::default(), } } pub fn cli_options(&self) -> Result<&Arc, AnyError> { self.services.cli_options.get_or_try_init(|| { CliOptions::from_flags(self.flags.clone()).map(Arc::new) }) } pub fn deno_dir_provider(&self) -> Result<&Arc, AnyError> { Ok(&self.cli_options()?.deno_dir_provider) } pub fn deno_dir(&self) -> Result<&DenoDir, AnyError> { Ok(self.deno_dir_provider()?.get_or_create()?) } pub fn caches(&self) -> Result<&Arc, AnyError> { self.services.caches.get_or_try_init(|| { let cli_options = self.cli_options()?; let caches = Arc::new(Caches::new(self.deno_dir_provider()?.clone())); // Warm up the caches we know we'll likely need based on the CLI mode match cli_options.sub_command() { DenoSubcommand::Run(_) | DenoSubcommand::Serve(_) | DenoSubcommand::Bench(_) | DenoSubcommand::Test(_) | DenoSubcommand::Check(_) => { _ = caches.dep_analysis_db(); _ = caches.node_analysis_db(); if cli_options.type_check_mode().is_true() { _ = caches.fast_check_db(); _ = caches.type_checking_cache_db(); } if cli_options.code_cache_enabled() { _ = caches.code_cache_db(); } } _ => {} } Ok(caches) }) } pub fn blob_store(&self) -> &Arc { self.services.blob_store.get_or_init(Default::default) } pub fn root_cert_store_provider(&self) -> &Arc { self.services.root_cert_store_provider.get_or_init(|| { Arc::new(CliRootCertStoreProvider::new( None, self.flags.ca_stores.clone(), self.flags.ca_data.clone(), )) }) } pub fn text_only_progress_bar(&self) -> &ProgressBar { self .services .text_only_progress_bar .get_or_init(|| ProgressBar::new(ProgressBarStyle::TextOnly)) } pub fn global_http_cache(&self) -> Result<&Arc, AnyError> { self.services.global_http_cache.get_or_try_init(|| { Ok(Arc::new(GlobalHttpCache::new( self.deno_dir()?.remote_folder_path(), crate::cache::RealDenoCacheEnv, ))) }) } pub fn http_cache(&self) -> Result<&Arc, AnyError> { self.services.http_cache.get_or_try_init(|| { let global_cache = self.global_http_cache()?.clone(); match self.cli_options()?.vendor_dir_path() { Some(local_path) => { let local_cache = LocalHttpCache::new( local_path.clone(), global_cache, deno_cache_dir::GlobalToLocalCopy::Allow, ); Ok(Arc::new(local_cache)) } None => Ok(global_cache), } }) } pub fn http_client_provider(&self) -> &Arc { self.services.http_client_provider.get_or_init(|| { Arc::new(HttpClientProvider::new( Some(self.root_cert_store_provider().clone()), self.flags.unsafely_ignore_certificate_errors.clone(), )) }) } pub fn file_fetcher(&self) -> Result<&Arc, AnyError> { self.services.file_fetcher.get_or_try_init(|| { let cli_options = self.cli_options()?; Ok(Arc::new(FileFetcher::new( self.http_cache()?.clone(), cli_options.cache_setting(), !cli_options.no_remote(), self.http_client_provider().clone(), self.blob_store().clone(), Some(self.text_only_progress_bar().clone()), ))) }) } pub fn fs(&self) -> &Arc { 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()?; 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"), }, ), }, ) } 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 } pub fn sloppy_imports_resolver( &self, ) -> Result>, AnyError> { self .services .sloppy_imports_resolver .get_or_try_init(|| { Ok(self.cli_options()?.unstable_sloppy_imports().then(|| { Arc::new(CliSloppyImportsResolver::new(SloppyImportsCachedFs::new( self.fs().clone(), ))) })) }) .map(|maybe| maybe.as_ref()) } pub async fn workspace_resolver( &self, ) -> Result<&Arc, AnyError> { self .services .workspace_resolver .get_or_try_init_async(async { let cli_options = self.cli_options()?; let resolver = cli_options .create_workspace_resolver( self.file_fetcher()?, if cli_options.use_byonm() && !matches!( cli_options.sub_command(), DenoSubcommand::Publish(_) ) { PackageJsonDepResolution::Disabled } else { // todo(dsherret): this should be false for nodeModulesDir: true PackageJsonDepResolution::Enabled }, ) .await?; if !resolver.diagnostics().is_empty() { warn!( "Import map diagnostics:\n{}", resolver .diagnostics() .iter() .map(|d| format!(" - {d}")) .collect::>() .join("\n") ); } Ok(Arc::new(resolver)) }) .await } pub async fn deno_resolver(&self) -> Result<&Arc, AnyError> { self .services .deno_resolver .get_or_try_init_async(async { let cli_options = self.cli_options()?; Ok(Arc::new(CliDenoResolver::new(DenoResolverOptions { in_npm_pkg_checker: self.in_npm_pkg_checker()?.clone(), node_and_req_resolver: if cli_options.no_npm() { None } else { Some(NodeAndNpmReqResolver { node_resolver: self.node_resolver().await?.clone(), npm_req_resolver: self.npm_req_resolver().await?.clone(), }) }, sloppy_imports_resolver: self.sloppy_imports_resolver()?.cloned(), workspace_resolver: self.workspace_resolver().await?.clone(), is_byonm: cli_options.use_byonm(), maybe_vendor_dir: cli_options.vendor_dir_path(), }))) }) .await } pub async fn resolver(&self) -> Result<&Arc, AnyError> { self .services .resolver .get_or_try_init_async( async { let cli_options = self.cli_options()?; Ok(Arc::new(CliResolver::new(CliResolverOptions { npm_resolver: if cli_options.no_npm() { None } else { Some(self.npm_resolver().await?.clone()) }, bare_node_builtins_enabled: cli_options .unstable_bare_node_builtins(), deno_resolver: self.deno_resolver().await?.clone(), }))) } .boxed_local(), ) .await } pub fn maybe_file_watcher_reporter(&self) -> &Option { let maybe_file_watcher_reporter = self .watcher_communicator .as_ref() .map(|i| FileWatcherReporter::new(i.clone())); self .services .maybe_file_watcher_reporter .get_or_init(|| maybe_file_watcher_reporter) } pub fn emit_cache(&self) -> Result<&Arc, AnyError> { self.services.emit_cache.get_or_try_init(|| { Ok(Arc::new(EmitCache::new(self.deno_dir()?.gen_cache.clone()))) }) } pub fn module_info_cache(&self) -> Result<&Arc, AnyError> { self.services.module_info_cache.get_or_try_init(|| { Ok(Arc::new(ModuleInfoCache::new( self.caches()?.dep_analysis_db(), self.parsed_source_cache().clone(), ))) }) } pub fn code_cache(&self) -> Result<&Arc, AnyError> { self.services.code_cache.get_or_try_init(|| { Ok(Arc::new(CodeCache::new(self.caches()?.code_cache_db()))) }) } pub fn parsed_source_cache(&self) -> &Arc { self .services .parsed_source_cache .get_or_init(Default::default) } pub fn emitter(&self) -> Result<&Arc, AnyError> { self.services.emitter.get_or_try_init(|| { let cli_options = self.cli_options()?; let ts_config_result = cli_options.resolve_ts_config_for_emit(TsConfigType::Emit)?; check_warn_tsconfig(&ts_config_result); let (transpile_options, emit_options) = crate::args::ts_config_to_transpile_and_emit_options( 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, emit_options, ))) }) } pub async fn lint_rule_provider(&self) -> Result { Ok(LintRuleProvider::new( self.sloppy_imports_resolver()?.cloned(), Some(self.workspace_resolver().await?.clone()), )) } pub async fn node_resolver(&self) -> Result<&Arc, AnyError> { self .services .node_resolver .get_or_try_init_async( async { Ok(Arc::new(NodeResolver::new( DenoFsNodeResolverEnv::new(self.fs().clone()), self.in_npm_pkg_checker()?.clone(), self .npm_resolver() .await? .clone() .into_npm_pkg_folder_resolver(), self.pkg_json_resolver().clone(), ))) } .boxed_local(), ) .await } pub async fn node_code_translator( &self, ) -> Result<&Arc, AnyError> { self .services .node_code_translator .get_or_try_init_async(async { let caches = self.caches()?; let node_analysis_cache = NodeAnalysisCache::new(caches.node_analysis_db()); let node_resolver = self.node_resolver().await?.clone(); let cjs_esm_analyzer = CliCjsCodeAnalyzer::new( node_analysis_cache, self.cjs_tracker()?.clone(), self.fs().clone(), Some(self.parsed_source_cache().clone()), ); 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_pkg_folder_resolver(), self.pkg_json_resolver().clone(), ))) }) .await } pub async fn npm_req_resolver( &self, ) -> Result<&Arc, AnyError> { self .services .npm_req_resolver .get_or_try_init_async(async { let npm_resolver = self.npm_resolver().await?; Ok(Arc::new(CliNpmReqResolver::new(NpmReqResolverOptions { byonm_resolver: (npm_resolver.clone()).into_maybe_byonm(), fs: CliDenoResolverFs(self.fs().clone()), in_npm_pkg_checker: self.in_npm_pkg_checker()?.clone(), node_resolver: self.node_resolver().await?.clone(), npm_req_resolver: npm_resolver.clone().into_npm_req_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 .type_checker .get_or_try_init_async(async { let cli_options = self.cli_options()?; Ok(Arc::new(TypeChecker::new( self.caches()?.clone(), Arc::new(TypeCheckingCjsTracker::new( self.cjs_tracker()?.clone(), self.module_info_cache()?.clone(), )), cli_options.clone(), self.module_graph_builder().await?.clone(), self.node_resolver().await?.clone(), self.npm_resolver().await?.clone(), ))) }) .await } pub async fn module_graph_builder( &self, ) -> Result<&Arc, AnyError> { self .services .module_graph_builder .get_or_try_init_async(async { let cli_options = self.cli_options()?; Ok(Arc::new(ModuleGraphBuilder::new( self.caches()?.clone(), self.cjs_tracker()?.clone(), cli_options.clone(), self.file_fetcher()?.clone(), self.fs().clone(), self.global_http_cache()?.clone(), self.in_npm_pkg_checker()?.clone(), cli_options.maybe_lockfile().cloned(), self.maybe_file_watcher_reporter().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(), ))) }) .await } pub async fn module_graph_creator( &self, ) -> Result<&Arc, AnyError> { self .services .module_graph_creator .get_or_try_init_async(async { let cli_options = self.cli_options()?; Ok(Arc::new(ModuleGraphCreator::new( cli_options.clone(), self.npm_resolver().await?.clone(), self.module_graph_builder().await?.clone(), self.type_checker().await?.clone(), ))) }) .await } pub async fn main_module_graph_container( &self, ) -> Result<&Arc, AnyError> { self .services .main_graph_container .get_or_try_init_async(async { Ok(Arc::new(MainModuleGraphContainer::new( self.cli_options()?.clone(), self.module_load_preparer().await?.clone(), self.root_permissions_container()?.clone(), ))) }) .await } pub fn maybe_inspector_server( &self, ) -> Result<&Option>, AnyError> { self.services.maybe_inspector_server.get_or_try_init(|| { let cli_options = self.cli_options()?; match cli_options.resolve_inspector_server() { Ok(server) => Ok(server.map(Arc::new)), Err(err) => Err(err), } }) } pub async fn module_load_preparer( &self, ) -> Result<&Arc, AnyError> { self .services .module_load_preparer .get_or_try_init_async(async { let cli_options = self.cli_options()?; Ok(Arc::new(ModuleLoadPreparer::new( cli_options.clone(), cli_options.maybe_lockfile().cloned(), self.module_graph_builder().await?.clone(), self.text_only_progress_bar().clone(), self.type_checker().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( self.in_npm_pkg_checker()?.clone(), self.pkg_json_resolver().clone(), if options.is_node_main() || options.unstable_detect_cjs() { IsCjsResolutionMode::ImplicitTypeCommonJs } else if options.detect_cjs() { IsCjsResolutionMode::ExplicitTypeCommonJs } else { IsCjsResolutionMode::Disabled }, ))) }) } pub fn permission_desc_parser( &self, ) -> Result<&Arc, AnyError> { self.services.permission_desc_parser.get_or_try_init(|| { let fs = self.fs().clone(); Ok(Arc::new(RuntimePermissionDescriptorParser::new(fs))) }) } pub fn feature_checker(&self) -> Result<&Arc, AnyError> { self.services.feature_checker.get_or_try_init(|| { let cli_options = self.cli_options()?; let mut checker = FeatureChecker::default(); checker.set_exit_cb(Box::new(crate::unstable_exit_cb)); let unstable_features = cli_options.unstable_features(); for granular_flag in crate::UNSTABLE_GRANULAR_FLAGS { if unstable_features.contains(&granular_flag.name.to_string()) { checker.enable_feature(granular_flag.name); } } Ok(Arc::new(checker)) }) } pub async fn create_compile_binary_writer( &self, ) -> Result { let cli_options = self.cli_options()?; Ok(DenoCompileBinaryWriter::new( self.cjs_tracker()?, self.cli_options()?, self.deno_dir()?, self.emitter()?, self.file_fetcher()?, self.http_client_provider(), self.npm_resolver().await?.as_ref(), self.workspace_resolver().await?.as_ref(), cli_options.npm_system_info(), )) } pub fn root_permissions_container( &self, ) -> Result<&PermissionsContainer, AnyError> { self .services .root_permissions_container .get_or_try_init(|| { let desc_parser = self.permission_desc_parser()?.clone(); let permissions = Permissions::from_options( desc_parser.as_ref(), &self.cli_options()?.permissions_options(), )?; Ok(PermissionsContainer::new(desc_parser, permissions)) }) } pub async fn create_cli_main_worker_factory( &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 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 { None }; let node_code_translator = self.node_code_translator().await?; let cjs_tracker = self.cjs_tracker()?.clone(); let pkg_json_resolver = self.pkg_json_resolver().clone(); let npm_req_resolver = self.npm_req_resolver().await?; Ok(CliMainWorkerFactory::new( self.blob_store().clone(), if cli_options.code_cache_enabled() { Some(self.code_cache()?.clone()) } else { None }, self.feature_checker()?.clone(), fs.clone(), maybe_file_watcher_communicator, self.maybe_inspector_server()?.clone(), 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 { None }, 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(), node_resolver.clone(), npm_req_resolver.clone(), cli_npm_resolver.clone(), NpmModuleLoader::new( self.cjs_tracker()?.clone(), fs.clone(), node_code_translator.clone(), ), self.parsed_source_cache().clone(), self.resolver().await?.clone(), )), 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()?, self.cli_options()?.otel_config(), )) } 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 fn_: crate::worker::CreateHmrRunnerCb = Box::new(move |session| { Box::new(HmrRunner::new( emitter.clone(), session, watcher_communicator.clone(), )) }); Some(fn_) } else { None }; let create_coverage_collector = if let Some(coverage_dir) = cli_options.coverage_dir() { let coverage_dir = PathBuf::from(coverage_dir); let fn_: crate::worker::CreateCoverageCollectorCb = Box::new(move |session| { Box::new(CoverageCollector::new(coverage_dir.clone(), session)) }); Some(fn_) } else { None }; Ok(CliMainWorkerOptions { argv: cli_options.argv().clone(), // This optimization is only available for "run" subcommand // because we need to register new ops for testing and jupyter // integration. skip_op_registration: cli_options.sub_command().is_run(), log_level: cli_options.log_level().unwrap_or(log::Level::Info).into(), enable_op_summary_metrics: cli_options.enable_op_summary_metrics(), enable_testing_features: cli_options.enable_testing_features(), has_node_modules_dir: cli_options.has_node_modules_dir(), hmr: cli_options.has_hmr(), inspect_brk: cli_options.inspect_brk().is_some(), inspect_wait: cli_options.inspect_wait().is_some(), strace_ops: cli_options.strace_ops().clone(), is_inspecting: cli_options.is_inspecting(), 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 argv0: cli_options .take_binary_npm_command_name() .or(std::env::args().next()), node_debug: std::env::var("NODE_DEBUG").ok(), origin_data_folder_path: Some(self.deno_dir()?.origin_data_folder_path()), seed: cli_options.seed(), unsafely_ignore_certificate_errors: cli_options .unsafely_ignore_certificate_errors() .clone(), create_hmr_runner, create_coverage_collector, node_ipc: cli_options.node_ipc_fd(), serve_port: cli_options.serve_port(), serve_host: cli_options.serve_host(), }) } } #[derive(Debug, Copy, Clone)] pub struct SpecifierInfo { /// Type check as an ES module. pub check: bool, /// Type check virtual modules from doc snippets. If this is set but `check` /// is not, this may be a markdown file for example. pub check_doc: bool, } struct WorkspaceFileContainerEntry { specifiers: Vec<(ModuleSpecifier, SpecifierInfo)>, doc_snippet_specifiers: Vec, main_graph_container: Arc, worker_factory: Arc, ext_flag: Option, } impl WorkspaceFileContainerEntry { fn checked_specifiers(&self) -> impl Iterator { self .specifiers .iter() .filter_map(|(s, i)| i.check.then_some(s)) .chain(self.doc_snippet_specifiers.iter()) } } pub struct WorkspaceFileContainer { entries: Vec, } impl WorkspaceFileContainer { #[allow(clippy::type_complexity)] pub async fn from_workspace_dirs_with_files( mut workspace_dirs_with_files: Vec<(Arc, FilePatterns)>, cli_options: &CliOptions, extract_doc_files: fn(File) -> Result, AnyError>, collect_specifiers: fn( FilePatterns, Arc, Arc, T, ) -> std::pin::Pin< Box< dyn Future< Output = Result, AnyError>, >, >, >, args: T, ) -> Result { workspace_dirs_with_files.sort_by_cached_key(|(d, _)| d.dir_url().clone()); let all_scopes = Arc::new( workspace_dirs_with_files .iter() .filter(|(d, _)| d.has_deno_or_pkg_json()) .map(|(d, _)| d.dir_url().clone()) .collect::>(), ); let dir_count = workspace_dirs_with_files.len(); let mut entries = Vec::with_capacity(dir_count); for (workspace_dir, files) in workspace_dirs_with_files { let (npmrc, _) = discover_npmrc_from_workspace(&workspace_dir.workspace)?; let lockfile = CliLockfile::discover(&cli_options.flags, &workspace_dir.workspace)?; let scope_options = (dir_count > 1).then(|| ScopeOptions { scope: workspace_dir .has_deno_or_pkg_json() .then(|| workspace_dir.dir_url().clone()), all_scopes: all_scopes.clone(), }); let cli_options = Arc::new(CliOptions::new( cli_options.flags.clone(), cli_options.initial_cwd().to_path_buf(), lockfile.map(Arc::new), npmrc, workspace_dir, false, scope_options.map(Arc::new), )?); let ext_flag = cli_options.ext_flag().clone(); let factory = CliFactory::from_cli_options(cli_options.clone()); let file_fetcher = factory.file_fetcher()?; let specifiers = collect_specifiers( files, cli_options, file_fetcher.clone(), args.clone(), ) .await?; let mut doc_snippet_specifiers = vec![]; let root_permissions = factory.root_permissions_container()?; for (s, _) in specifiers.iter().filter(|(_, i)| i.check_doc) { let file = file_fetcher.fetch(s, root_permissions).await?; let snippet_files = extract_doc_files(file)?; for snippet_file in snippet_files { doc_snippet_specifiers.push(snippet_file.specifier.clone()); file_fetcher.insert_memory_files(snippet_file); } } let main_graph_container = factory.main_module_graph_container().await?.clone(); let worker_factory = Arc::new(factory.create_cli_main_worker_factory().await?); entries.push(WorkspaceFileContainerEntry { specifiers, doc_snippet_specifiers, main_graph_container, worker_factory, ext_flag, }); } Ok(Self { entries }) } pub async fn check(&self) -> Result<(), AnyError> { let mut diagnostics = vec![]; let mut all_errors = vec![]; for entry in &self.entries { let specifiers_for_typecheck = entry.checked_specifiers().cloned().collect::>(); if specifiers_for_typecheck.is_empty() { continue; } if let Err(err) = entry .main_graph_container .check_specifiers(&specifiers_for_typecheck, entry.ext_flag.as_ref()) .await { match err { MaybeDiagnostics::Diagnostics(Diagnostics(d)) => { diagnostics.extend(d) } MaybeDiagnostics::Other(err) => all_errors.push(err), } } } if !diagnostics.is_empty() { all_errors.push(AnyError::from(Diagnostics(diagnostics))); } if !all_errors.is_empty() { return Err(anyhow!( "{}", all_errors .into_iter() .map(|e| e.to_string()) .collect::>() .join("\n\n"), )); } Ok(()) } pub fn found_specifiers(&self) -> bool { self.entries.iter().any(|e| !e.specifiers.is_empty()) } pub fn checked_specifiers_with_worker_factories( &self, ) -> Vec<(Vec, Arc)> { self .entries .iter() .map(|e| { ( e.checked_specifiers().cloned().collect(), e.worker_factory.clone(), ) }) .collect() } }