1
0
Fork 0
mirror of https://github.com/denoland/deno.git synced 2025-01-12 00:54:02 -05:00

feat(unstable/pm): support npm packages in 'deno add' (#22715)

This commit is contained in:
Nayeem Rahman 2024-03-06 13:24:15 +00:00 committed by GitHub
parent 8b1f160bb5
commit 01bc2f530e
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
9 changed files with 135 additions and 67 deletions

View file

@ -75,7 +75,7 @@ use deno_config::FmtConfig;
use deno_config::LintConfig; use deno_config::LintConfig;
use deno_config::TestConfig; use deno_config::TestConfig;
pub fn npm_registry_default_url() -> &'static Url { pub fn npm_registry_url() -> &'static Url {
static NPM_REGISTRY_DEFAULT_URL: Lazy<Url> = Lazy::new(|| { static NPM_REGISTRY_DEFAULT_URL: Lazy<Url> = Lazy::new(|| {
let env_var_name = "NPM_CONFIG_REGISTRY"; let env_var_name = "NPM_CONFIG_REGISTRY";
if let Ok(registry_url) = std::env::var(env_var_name) { if let Ok(registry_url) = std::env::var(env_var_name) {

View file

@ -443,7 +443,7 @@ impl CliFactory {
self.package_json_deps_provider().clone(), self.package_json_deps_provider().clone(),
), ),
npm_system_info: self.options.npm_system_info(), npm_system_info: self.options.npm_system_info(),
npm_registry_url: crate::args::npm_registry_default_url().to_owned(), npm_registry_url: crate::args::npm_registry_url().to_owned(),
}) })
}).await }).await
}.boxed_local()) }.boxed_local())

View file

@ -226,7 +226,7 @@ impl JsrFetchResolver {
if let Some(info) = self.info_by_name.get(name) { if let Some(info) = self.info_by_name.get(name) {
return info.value().clone(); return info.value().clone();
} }
let read_cached_package_info = || async { let fetch_package_info = || async {
let meta_url = jsr_url().join(&format!("{}/meta.json", name)).ok()?; let meta_url = jsr_url().join(&format!("{}/meta.json", name)).ok()?;
let file = self let file = self
.file_fetcher .file_fetcher
@ -235,7 +235,7 @@ impl JsrFetchResolver {
.ok()?; .ok()?;
serde_json::from_slice::<JsrPackageInfo>(&file.source).ok() serde_json::from_slice::<JsrPackageInfo>(&file.source).ok()
}; };
let info = read_cached_package_info().await.map(Arc::new); let info = fetch_package_info().await.map(Arc::new);
self.info_by_name.insert(name.to_string(), info.clone()); self.info_by_name.insert(name.to_string(), info.clone());
info info
} }
@ -247,7 +247,7 @@ impl JsrFetchResolver {
if let Some(info) = self.info_by_nv.get(nv) { if let Some(info) = self.info_by_nv.get(nv) {
return info.value().clone(); return info.value().clone();
} }
let read_cached_package_version_info = || async { let fetch_package_version_info = || async {
let meta_url = jsr_url() let meta_url = jsr_url()
.join(&format!("{}/{}_meta.json", &nv.name, &nv.version)) .join(&format!("{}/{}_meta.json", &nv.name, &nv.version))
.ok()?; .ok()?;
@ -258,7 +258,7 @@ impl JsrFetchResolver {
.ok()?; .ok()?;
partial_jsr_package_version_info_from_slice(&file.source).ok() partial_jsr_package_version_info_from_slice(&file.source).ok()
}; };
let info = read_cached_package_version_info().await.map(Arc::new); let info = fetch_package_version_info().await.map(Arc::new);
self.info_by_nv.insert(nv.clone(), info.clone()); self.info_by_nv.insert(nv.clone(), info.clone());
info info
} }

View file

@ -15,20 +15,18 @@ use std::sync::Arc;
use super::search::PackageSearchApi; use super::search::PackageSearchApi;
#[derive(Debug, Clone)] #[derive(Debug)]
pub struct CliJsrSearchApi { pub struct CliJsrSearchApi {
file_fetcher: FileFetcher, file_fetcher: FileFetcher,
/// We only store this here so the completion system has access to a resolver resolver: JsrFetchResolver,
/// that always uses the global cache. search_cache: DashMap<String, Arc<Vec<String>>>,
resolver: Arc<JsrFetchResolver>, versions_cache: DashMap<String, Arc<Vec<Version>>>,
search_cache: Arc<DashMap<String, Arc<Vec<String>>>>, exports_cache: DashMap<PackageNv, Arc<Vec<String>>>,
versions_cache: Arc<DashMap<String, Arc<Vec<Version>>>>,
exports_cache: Arc<DashMap<PackageNv, Arc<Vec<String>>>>,
} }
impl CliJsrSearchApi { impl CliJsrSearchApi {
pub fn new(file_fetcher: FileFetcher) -> Self { pub fn new(file_fetcher: FileFetcher) -> Self {
let resolver = Arc::new(JsrFetchResolver::new(file_fetcher.clone())); let resolver = JsrFetchResolver::new(file_fetcher.clone());
Self { Self {
file_fetcher, file_fetcher,
resolver, resolver,
@ -38,7 +36,7 @@ impl CliJsrSearchApi {
} }
} }
pub fn get_resolver(&self) -> &Arc<JsrFetchResolver> { pub fn get_resolver(&self) -> &JsrFetchResolver {
&self.resolver &self.resolver
} }
} }
@ -49,12 +47,7 @@ impl PackageSearchApi for CliJsrSearchApi {
if let Some(names) = self.search_cache.get(query) { if let Some(names) = self.search_cache.get(query) {
return Ok(names.clone()); return Ok(names.clone());
} }
let mut search_url = jsr_api_url().clone(); let mut search_url = jsr_api_url().join("packages")?;
search_url
.path_segments_mut()
.map_err(|_| anyhow!("Custom jsr URL cannot be a base."))?
.pop_if_empty()
.push("packages");
search_url.query_pairs_mut().append_pair("query", query); search_url.query_pairs_mut().append_pair("query", query);
let file = self let file = self
.file_fetcher .file_fetcher

View file

@ -875,9 +875,8 @@ impl Inner {
None, None,
); );
deps_file_fetcher.set_download_log_level(super::logging::lsp_log_level()); deps_file_fetcher.set_download_log_level(super::logging::lsp_log_level());
self.jsr_search_api = CliJsrSearchApi::new(deps_file_fetcher); self.jsr_search_api = CliJsrSearchApi::new(deps_file_fetcher.clone());
self.npm.search_api = self.npm.search_api = CliNpmSearchApi::new(deps_file_fetcher);
CliNpmSearchApi::new(self.module_registries.file_fetcher.clone());
let maybe_local_cache = let maybe_local_cache =
self.config.maybe_vendor_dir_path().map(|local_path| { self.config.maybe_vendor_dir_path().map(|local_path| {
Arc::new(LocalLspHttpCache::new(local_path, global_cache.clone())) Arc::new(LocalLspHttpCache::new(local_path, global_cache.clone()))
@ -1182,7 +1181,7 @@ async fn create_npm_resolver(
// do not install while resolving in the lsp—leave that to the cache command // do not install while resolving in the lsp—leave that to the cache command
package_json_installer: package_json_installer:
CliNpmResolverManagedPackageJsonInstallerOption::NoInstall, CliNpmResolverManagedPackageJsonInstallerOption::NoInstall,
npm_registry_url: crate::args::npm_registry_default_url().to_owned(), npm_registry_url: crate::args::npm_registry_url().to_owned(),
npm_system_info: NpmSystemInfo::default(), npm_system_info: NpmSystemInfo::default(),
}) })
}) })

View file

@ -4,29 +4,32 @@ use dashmap::DashMap;
use deno_core::anyhow::anyhow; use deno_core::anyhow::anyhow;
use deno_core::error::AnyError; use deno_core::error::AnyError;
use deno_core::serde_json; use deno_core::serde_json;
use deno_npm::registry::NpmPackageInfo;
use deno_runtime::permissions::PermissionsContainer; use deno_runtime::permissions::PermissionsContainer;
use deno_semver::package::PackageNv; use deno_semver::package::PackageNv;
use deno_semver::Version; use deno_semver::Version;
use serde::Deserialize; use serde::Deserialize;
use std::sync::Arc; use std::sync::Arc;
use crate::args::npm_registry_default_url; use crate::args::npm_registry_url;
use crate::file_fetcher::FileFetcher; use crate::file_fetcher::FileFetcher;
use crate::npm::NpmFetchResolver;
use super::search::PackageSearchApi; use super::search::PackageSearchApi;
#[derive(Debug, Clone)] #[derive(Debug)]
pub struct CliNpmSearchApi { pub struct CliNpmSearchApi {
file_fetcher: FileFetcher, file_fetcher: FileFetcher,
search_cache: Arc<DashMap<String, Arc<Vec<String>>>>, resolver: NpmFetchResolver,
versions_cache: Arc<DashMap<String, Arc<Vec<Version>>>>, search_cache: DashMap<String, Arc<Vec<String>>>,
versions_cache: DashMap<String, Arc<Vec<Version>>>,
} }
impl CliNpmSearchApi { impl CliNpmSearchApi {
pub fn new(file_fetcher: FileFetcher) -> Self { pub fn new(file_fetcher: FileFetcher) -> Self {
let resolver = NpmFetchResolver::new(file_fetcher.clone());
Self { Self {
file_fetcher, file_fetcher,
resolver,
search_cache: Default::default(), search_cache: Default::default(),
versions_cache: Default::default(), versions_cache: Default::default(),
} }
@ -39,12 +42,7 @@ impl PackageSearchApi for CliNpmSearchApi {
if let Some(names) = self.search_cache.get(query) { if let Some(names) = self.search_cache.get(query) {
return Ok(names.clone()); return Ok(names.clone());
} }
let mut search_url = npm_registry_default_url().clone(); let mut search_url = npm_registry_url().join("-/v1/search")?;
search_url
.path_segments_mut()
.map_err(|_| anyhow!("Custom npm registry URL cannot be a base."))?
.pop_if_empty()
.extend("-/v1/search".split('/'));
search_url search_url
.query_pairs_mut() .query_pairs_mut()
.append_pair("text", &format!("{} boost-exact:false", query)); .append_pair("text", &format!("{} boost-exact:false", query));
@ -62,18 +60,12 @@ impl PackageSearchApi for CliNpmSearchApi {
if let Some(versions) = self.versions_cache.get(name) { if let Some(versions) = self.versions_cache.get(name) {
return Ok(versions.clone()); return Ok(versions.clone());
} }
let mut info_url = npm_registry_default_url().clone(); let info = self
info_url .resolver
.path_segments_mut() .package_info(name)
.map_err(|_| anyhow!("Custom npm registry URL cannot be a base."))? .await
.pop_if_empty() .ok_or_else(|| anyhow!("npm package info not found: {}", name))?;
.push(name); let mut versions = info.versions.keys().cloned().collect::<Vec<_>>();
let file = self
.file_fetcher
.fetch(&info_url, PermissionsContainer::allow_all())
.await?;
let info = serde_json::from_slice::<NpmPackageInfo>(&file.source)?;
let mut versions = info.versions.into_keys().collect::<Vec<_>>();
versions.sort(); versions.sort();
versions.reverse(); versions.reverse();
let versions = Arc::new(versions); let versions = Arc::new(versions);

View file

@ -8,11 +8,19 @@ mod managed;
use std::path::PathBuf; use std::path::PathBuf;
use std::sync::Arc; use std::sync::Arc;
use dashmap::DashMap;
use deno_ast::ModuleSpecifier; use deno_ast::ModuleSpecifier;
use deno_core::error::AnyError; use deno_core::error::AnyError;
use deno_core::serde_json;
use deno_npm::registry::NpmPackageInfo;
use deno_runtime::deno_node::NpmResolver; use deno_runtime::deno_node::NpmResolver;
use deno_runtime::permissions::PermissionsContainer;
use deno_semver::package::PackageNv;
use deno_semver::package::PackageReq; use deno_semver::package::PackageReq;
use crate::args::npm_registry_url;
use crate::file_fetcher::FileFetcher;
pub use self::byonm::ByonmCliNpmResolver; pub use self::byonm::ByonmCliNpmResolver;
pub use self::byonm::CliNpmResolverByonmCreateOptions; pub use self::byonm::CliNpmResolverByonmCreateOptions;
pub use self::cache_dir::NpmCacheDir; pub use self::cache_dir::NpmCacheDir;
@ -87,3 +95,60 @@ pub trait CliNpmResolver: NpmResolver {
/// or `None` if the state currently can't be determined. /// or `None` if the state currently can't be determined.
fn check_state_hash(&self) -> Option<u64>; fn check_state_hash(&self) -> Option<u64>;
} }
#[derive(Debug)]
pub struct NpmFetchResolver {
nv_by_req: DashMap<PackageReq, Option<PackageNv>>,
info_by_name: DashMap<String, Option<Arc<NpmPackageInfo>>>,
file_fetcher: FileFetcher,
}
impl NpmFetchResolver {
pub fn new(file_fetcher: FileFetcher) -> Self {
Self {
nv_by_req: Default::default(),
info_by_name: Default::default(),
file_fetcher,
}
}
pub async fn req_to_nv(&self, req: &PackageReq) -> Option<PackageNv> {
if let Some(nv) = self.nv_by_req.get(req) {
return nv.value().clone();
}
let maybe_get_nv = || async {
let name = req.name.clone();
let package_info = self.package_info(&name).await?;
// Find the first matching version of the package which is cached.
let mut versions = package_info.versions.keys().collect::<Vec<_>>();
versions.sort();
let version = versions
.into_iter()
.rev()
.find(|v| req.version_req.tag().is_none() && req.version_req.matches(v))
.cloned()?;
Some(PackageNv { name, version })
};
let nv = maybe_get_nv().await;
self.nv_by_req.insert(req.clone(), nv.clone());
nv
}
pub async fn package_info(&self, name: &str) -> Option<Arc<NpmPackageInfo>> {
if let Some(info) = self.info_by_name.get(name) {
return info.value().clone();
}
let fetch_package_info = || async {
let info_url = npm_registry_url().join(name).ok()?;
let file = self
.file_fetcher
.fetch(&info_url, PermissionsContainer::allow_all())
.await
.ok()?;
serde_json::from_slice::<NpmPackageInfo>(&file.source).ok()
};
let info = fetch_package_info().await.map(Arc::new);
self.info_by_name.insert(name.to_string(), info.clone());
info
}
}

View file

@ -23,6 +23,7 @@ use crate::args::Flags;
use crate::factory::CliFactory; use crate::factory::CliFactory;
use crate::file_fetcher::FileFetcher; use crate::file_fetcher::FileFetcher;
use crate::jsr::JsrFetchResolver; use crate::jsr::JsrFetchResolver;
use crate::npm::NpmFetchResolver;
pub async fn add(flags: Flags, add_flags: AddFlags) -> Result<(), AnyError> { pub async fn add(flags: Flags, add_flags: AddFlags) -> Result<(), AnyError> {
let cli_factory = CliFactory::from_flags(flags.clone()).await?; let cli_factory = CliFactory::from_flags(flags.clone()).await?;
@ -77,13 +78,18 @@ pub async fn add(flags: Flags, add_flags: AddFlags) -> Result<(), AnyError> {
None, None,
); );
deps_file_fetcher.set_download_log_level(log::Level::Trace); deps_file_fetcher.set_download_log_level(log::Level::Trace);
let jsr_resolver = Arc::new(JsrFetchResolver::new(deps_file_fetcher)); let jsr_resolver = Arc::new(JsrFetchResolver::new(deps_file_fetcher.clone()));
let npm_resolver = Arc::new(NpmFetchResolver::new(deps_file_fetcher));
let package_futures = package_reqs let package_futures = package_reqs
.into_iter() .into_iter()
.map(move |package_req| { .map(move |package_req| {
find_package_and_select_version_for_req(jsr_resolver.clone(), package_req) find_package_and_select_version_for_req(
.boxed_local() jsr_resolver.clone(),
npm_resolver.clone(),
package_req,
)
.boxed_local()
}) })
.collect::<Vec<_>>(); .collect::<Vec<_>>();
@ -183,6 +189,7 @@ enum PackageAndVersion {
async fn find_package_and_select_version_for_req( async fn find_package_and_select_version_for_req(
jsr_resolver: Arc<JsrFetchResolver>, jsr_resolver: Arc<JsrFetchResolver>,
npm_resolver: Arc<NpmFetchResolver>,
add_package_req: AddPackageReq, add_package_req: AddPackageReq,
) -> Result<PackageAndVersion, AnyError> { ) -> Result<PackageAndVersion, AnyError> {
match add_package_req { match add_package_req {
@ -203,11 +210,22 @@ async fn find_package_and_select_version_for_req(
version_req: format!("{}{}", range_symbol, &nv.version), version_req: format!("{}{}", range_symbol, &nv.version),
})) }))
} }
AddPackageReq::Npm(pkg_req) => { AddPackageReq::Npm(pkg_ref) => {
bail!( let req = pkg_ref.req();
"Adding npm: packages is currently not supported. Package: npm:{}", let npm_prefixed_name = format!("npm:{}", &req.name);
pkg_req.req().name let Some(nv) = npm_resolver.req_to_nv(req).await else {
); return Ok(PackageAndVersion::NotFound(npm_prefixed_name));
};
let range_symbol = if req.version_req.version_text().starts_with('~') {
'~'
} else {
'^'
};
Ok(PackageAndVersion::Selected(SelectedPackage {
import_name: req.name.to_string(),
package_name: npm_prefixed_name,
version_req: format!("{}{}", range_symbol, &nv.version),
}))
} }
} }
} }

View file

@ -2,9 +2,7 @@
use deno_core::serde_json::json; use deno_core::serde_json::json;
use test_util::assert_contains; use test_util::assert_contains;
use test_util::env_vars_for_jsr_tests; use test_util::env_vars_for_jsr_npm_tests;
// use test_util::env_vars_for_npm_tests;
// use test_util::itest;
use test_util::TestContextBuilder; use test_util::TestContextBuilder;
#[test] #[test]
@ -110,21 +108,24 @@ fn add_multiple() {
} }
#[test] #[test]
fn add_not_supported_npm() { fn add_npm() {
let context = pm_context_builder().build(); let context = pm_context_builder().build();
let temp_dir = context.temp_dir().path();
let output = context let output = context.new_command().args("add npm:chalk@4.1").run();
.new_command() output.assert_exit_code(0);
.args("add @denotest/add npm:express")
.run();
output.assert_exit_code(1);
let output = output.combined_output(); let output = output.combined_output();
assert_contains!(output, "error: Adding npm: packages is currently not supported. Package: npm:express"); assert_contains!(output, "Add chalk");
temp_dir.join("deno.json").assert_matches_json(json!({
"imports": {
"chalk": "npm:chalk@^4.1.2"
}
}));
} }
fn pm_context_builder() -> TestContextBuilder { fn pm_context_builder() -> TestContextBuilder {
TestContextBuilder::new() TestContextBuilder::new()
.use_http_server() .use_http_server()
.envs(env_vars_for_jsr_tests()) .envs(env_vars_for_jsr_npm_tests())
.use_temp_cwd() .use_temp_cwd()
} }