mirror of
https://github.com/denoland/deno.git
synced 2025-01-10 08:09:06 -05:00
393 lines
12 KiB
Rust
393 lines
12 KiB
Rust
// Copyright 2018-2025 the Deno authors. MIT license.
|
|
|
|
use std::collections::HashMap;
|
|
use std::sync::Arc;
|
|
|
|
use dashmap::DashMap;
|
|
use deno_cache_dir::HttpCache;
|
|
use deno_core::anyhow::anyhow;
|
|
use deno_core::error::AnyError;
|
|
use deno_core::serde_json;
|
|
use deno_graph::packages::JsrPackageInfo;
|
|
use deno_graph::packages::JsrPackageInfoVersion;
|
|
use deno_graph::packages::JsrPackageVersionInfo;
|
|
use deno_graph::ModuleSpecifier;
|
|
use deno_semver::jsr::JsrPackageReqReference;
|
|
use deno_semver::package::PackageNv;
|
|
use deno_semver::package::PackageReq;
|
|
use deno_semver::StackString;
|
|
use deno_semver::Version;
|
|
use serde::Deserialize;
|
|
|
|
use super::config::ConfigData;
|
|
use super::search::PackageSearchApi;
|
|
use crate::args::jsr_api_url;
|
|
use crate::args::jsr_url;
|
|
use crate::file_fetcher::CliFileFetcher;
|
|
use crate::file_fetcher::TextDecodedFile;
|
|
use crate::jsr::partial_jsr_package_version_info_from_slice;
|
|
use crate::jsr::JsrFetchResolver;
|
|
|
|
/// Keep in sync with `JsrFetchResolver`!
|
|
#[derive(Debug)]
|
|
pub struct JsrCacheResolver {
|
|
nv_by_req: DashMap<PackageReq, Option<PackageNv>>,
|
|
/// The `module_graph` fields of the version infos should be forcibly absent.
|
|
/// It can be large and we don't want to store it.
|
|
info_by_nv: DashMap<PackageNv, Option<Arc<JsrPackageVersionInfo>>>,
|
|
info_by_name: DashMap<StackString, Option<Arc<JsrPackageInfo>>>,
|
|
workspace_scope_by_name: HashMap<StackString, ModuleSpecifier>,
|
|
cache: Arc<dyn HttpCache>,
|
|
}
|
|
|
|
impl JsrCacheResolver {
|
|
pub fn new(
|
|
cache: Arc<dyn HttpCache>,
|
|
config_data: Option<&ConfigData>,
|
|
) -> Self {
|
|
let nv_by_req = DashMap::new();
|
|
let info_by_nv = DashMap::new();
|
|
let info_by_name = DashMap::new();
|
|
let mut workspace_scope_by_name = HashMap::new();
|
|
if let Some(config_data) = config_data {
|
|
for jsr_pkg_config in config_data.member_dir.workspace.jsr_packages() {
|
|
let Some(exports) = &jsr_pkg_config.config_file.json.exports else {
|
|
continue;
|
|
};
|
|
let Some(version) = &jsr_pkg_config.config_file.json.version else {
|
|
continue;
|
|
};
|
|
let Ok(version) = Version::parse_standard(version) else {
|
|
continue;
|
|
};
|
|
let nv = PackageNv {
|
|
name: jsr_pkg_config.name.as_str().into(),
|
|
version: version.clone(),
|
|
};
|
|
info_by_name.insert(
|
|
nv.name.clone(),
|
|
Some(Arc::new(JsrPackageInfo {
|
|
versions: [(
|
|
nv.version.clone(),
|
|
JsrPackageInfoVersion { yanked: false },
|
|
)]
|
|
.into_iter()
|
|
.collect(),
|
|
})),
|
|
);
|
|
info_by_nv.insert(
|
|
nv.clone(),
|
|
Some(Arc::new(JsrPackageVersionInfo {
|
|
exports: exports.clone(),
|
|
module_graph_1: None,
|
|
module_graph_2: None,
|
|
manifest: Default::default(),
|
|
})),
|
|
);
|
|
workspace_scope_by_name.insert(
|
|
nv.name.clone(),
|
|
ModuleSpecifier::from_directory_path(
|
|
jsr_pkg_config.config_file.dir_path(),
|
|
)
|
|
.unwrap(),
|
|
);
|
|
}
|
|
}
|
|
if let Some(lockfile) = config_data.and_then(|d| d.lockfile.as_ref()) {
|
|
for (dep_req, version) in &lockfile.lock().content.packages.specifiers {
|
|
let req = match dep_req.kind {
|
|
deno_semver::package::PackageKind::Jsr => &dep_req.req,
|
|
deno_semver::package::PackageKind::Npm => {
|
|
continue;
|
|
}
|
|
};
|
|
let Ok(version) = Version::parse_standard(version) else {
|
|
continue;
|
|
};
|
|
nv_by_req.insert(
|
|
req.clone(),
|
|
Some(PackageNv {
|
|
name: req.name.clone(),
|
|
version,
|
|
}),
|
|
);
|
|
}
|
|
}
|
|
Self {
|
|
nv_by_req,
|
|
info_by_nv,
|
|
info_by_name,
|
|
workspace_scope_by_name,
|
|
cache: cache.clone(),
|
|
}
|
|
}
|
|
|
|
pub 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 = || {
|
|
let name = &req.name;
|
|
let package_info = self.package_info(name)?;
|
|
// 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| {
|
|
if req.version_req.tag().is_some() || !req.version_req.matches(v) {
|
|
return false;
|
|
}
|
|
let nv = PackageNv {
|
|
name: name.clone(),
|
|
version: (*v).clone(),
|
|
};
|
|
self.package_version_info(&nv).is_some()
|
|
})
|
|
.cloned()?;
|
|
Some(PackageNv {
|
|
name: name.clone(),
|
|
version,
|
|
})
|
|
};
|
|
let nv = maybe_get_nv();
|
|
self.nv_by_req.insert(req.clone(), nv.clone());
|
|
nv
|
|
}
|
|
|
|
pub fn jsr_to_resource_url(
|
|
&self,
|
|
req_ref: &JsrPackageReqReference,
|
|
) -> Option<ModuleSpecifier> {
|
|
let req = req_ref.req().clone();
|
|
let maybe_nv = self.req_to_nv(&req);
|
|
let nv = maybe_nv.as_ref()?;
|
|
let info = self.package_version_info(nv)?;
|
|
let path = info.export(&req_ref.export_name())?;
|
|
if let Some(workspace_scope) = self.workspace_scope_by_name.get(&nv.name) {
|
|
workspace_scope.join(path).ok()
|
|
} else {
|
|
jsr_url()
|
|
.join(&format!("{}/{}/{}", &nv.name, &nv.version, &path))
|
|
.ok()
|
|
}
|
|
}
|
|
|
|
pub fn lookup_export_for_path(
|
|
&self,
|
|
nv: &PackageNv,
|
|
path: &str,
|
|
) -> Option<String> {
|
|
let info = self.package_version_info(nv)?;
|
|
let path = path.strip_prefix("./").unwrap_or(path);
|
|
let mut sloppy_fallback = None;
|
|
for (export, path_) in info.exports() {
|
|
let path_ = path_.strip_prefix("./").unwrap_or(path_);
|
|
if path_ == path {
|
|
return Some(export.strip_prefix("./").unwrap_or(export).to_string());
|
|
}
|
|
// TSC in some cases will suggest a `.js` import path for a `.d.ts` source
|
|
// file.
|
|
if sloppy_fallback.is_none() {
|
|
let path = path
|
|
.strip_suffix(".js")
|
|
.or_else(|| path.strip_suffix(".mjs"))
|
|
.or_else(|| path.strip_suffix(".cjs"))
|
|
.unwrap_or(path);
|
|
let path_ = path_
|
|
.strip_suffix(".d.ts")
|
|
.or_else(|| path_.strip_suffix(".d.mts"))
|
|
.or_else(|| path_.strip_suffix(".d.cts"))
|
|
.unwrap_or(path_);
|
|
if path_ == path {
|
|
sloppy_fallback =
|
|
Some(export.strip_prefix("./").unwrap_or(export).to_string());
|
|
}
|
|
}
|
|
}
|
|
sloppy_fallback
|
|
}
|
|
|
|
pub fn lookup_req_for_nv(&self, nv: &PackageNv) -> Option<PackageReq> {
|
|
for entry in self.nv_by_req.iter() {
|
|
let Some(nv_) = entry.value() else {
|
|
continue;
|
|
};
|
|
if nv_ == nv {
|
|
return Some(entry.key().clone());
|
|
}
|
|
}
|
|
None
|
|
}
|
|
|
|
pub fn package_info(
|
|
&self,
|
|
name: &StackString,
|
|
) -> Option<Arc<JsrPackageInfo>> {
|
|
if let Some(info) = self.info_by_name.get(name) {
|
|
return info.value().clone();
|
|
}
|
|
let read_cached_package_info = || {
|
|
let meta_url = jsr_url().join(&format!("{}/meta.json", name)).ok()?;
|
|
let meta_bytes = read_cached_url(&meta_url, &self.cache)?;
|
|
serde_json::from_slice::<JsrPackageInfo>(&meta_bytes).ok()
|
|
};
|
|
let info = read_cached_package_info().map(Arc::new);
|
|
self.info_by_name.insert(name.clone(), info.clone());
|
|
info
|
|
}
|
|
|
|
pub fn package_version_info(
|
|
&self,
|
|
nv: &PackageNv,
|
|
) -> Option<Arc<JsrPackageVersionInfo>> {
|
|
if let Some(info) = self.info_by_nv.get(nv) {
|
|
return info.value().clone();
|
|
}
|
|
let read_cached_package_version_info = || {
|
|
let meta_url = jsr_url()
|
|
.join(&format!("{}/{}_meta.json", &nv.name, &nv.version))
|
|
.ok()?;
|
|
let meta_bytes = read_cached_url(&meta_url, &self.cache)?;
|
|
partial_jsr_package_version_info_from_slice(&meta_bytes).ok()
|
|
};
|
|
let info = read_cached_package_version_info().map(Arc::new);
|
|
self.info_by_nv.insert(nv.clone(), info.clone());
|
|
info
|
|
}
|
|
|
|
pub fn did_cache(&self) {
|
|
self.nv_by_req.retain(|_, nv| nv.is_some());
|
|
self.info_by_nv.retain(|_, info| info.is_some());
|
|
self.info_by_name.retain(|_, info| info.is_some());
|
|
}
|
|
}
|
|
|
|
fn read_cached_url(
|
|
url: &ModuleSpecifier,
|
|
cache: &Arc<dyn HttpCache>,
|
|
) -> Option<Vec<u8>> {
|
|
cache
|
|
.get(&cache.cache_item_key(url).ok()?, None)
|
|
.ok()?
|
|
.map(|f| f.content.into_owned())
|
|
}
|
|
|
|
#[derive(Debug)]
|
|
pub struct CliJsrSearchApi {
|
|
file_fetcher: Arc<CliFileFetcher>,
|
|
resolver: JsrFetchResolver,
|
|
search_cache: DashMap<String, Arc<Vec<String>>>,
|
|
versions_cache: DashMap<String, Arc<Vec<Version>>>,
|
|
exports_cache: DashMap<PackageNv, Arc<Vec<String>>>,
|
|
}
|
|
|
|
impl CliJsrSearchApi {
|
|
pub fn new(file_fetcher: Arc<CliFileFetcher>) -> Self {
|
|
let resolver = JsrFetchResolver::new(file_fetcher.clone());
|
|
Self {
|
|
file_fetcher,
|
|
resolver,
|
|
search_cache: Default::default(),
|
|
versions_cache: Default::default(),
|
|
exports_cache: Default::default(),
|
|
}
|
|
}
|
|
|
|
pub fn get_resolver(&self) -> &JsrFetchResolver {
|
|
&self.resolver
|
|
}
|
|
|
|
pub fn clear_cache(&self) {
|
|
self.file_fetcher.clear_memory_files();
|
|
self.search_cache.clear();
|
|
self.versions_cache.clear();
|
|
self.exports_cache.clear();
|
|
}
|
|
}
|
|
|
|
#[async_trait::async_trait]
|
|
impl PackageSearchApi for CliJsrSearchApi {
|
|
async fn search(&self, query: &str) -> Result<Arc<Vec<String>>, AnyError> {
|
|
if let Some(names) = self.search_cache.get(query) {
|
|
return Ok(names.clone());
|
|
}
|
|
let mut search_url = jsr_api_url().join("packages")?;
|
|
search_url.query_pairs_mut().append_pair("query", query);
|
|
let file_fetcher = self.file_fetcher.clone();
|
|
// spawn due to the lsp's `Send` requirement
|
|
let file = deno_core::unsync::spawn(async move {
|
|
let file = file_fetcher.fetch_bypass_permissions(&search_url).await?;
|
|
TextDecodedFile::decode(file)
|
|
})
|
|
.await??;
|
|
let names = Arc::new(parse_jsr_search_response(&file.source)?);
|
|
self.search_cache.insert(query.to_string(), names.clone());
|
|
Ok(names)
|
|
}
|
|
|
|
async fn versions(&self, name: &str) -> Result<Arc<Vec<Version>>, AnyError> {
|
|
if let Some(versions) = self.versions_cache.get(name) {
|
|
return Ok(versions.clone());
|
|
}
|
|
let info = self
|
|
.resolver
|
|
.package_info(name)
|
|
.await
|
|
.ok_or_else(|| anyhow!("JSR package info not found: {}", name))?;
|
|
let mut versions = info.versions.keys().cloned().collect::<Vec<_>>();
|
|
versions.sort();
|
|
versions.reverse();
|
|
let versions = Arc::new(versions);
|
|
self
|
|
.versions_cache
|
|
.insert(name.to_string(), versions.clone());
|
|
Ok(versions)
|
|
}
|
|
|
|
async fn exports(
|
|
&self,
|
|
nv: &PackageNv,
|
|
) -> Result<Arc<Vec<String>>, AnyError> {
|
|
if let Some(exports) = self.exports_cache.get(nv) {
|
|
return Ok(exports.clone());
|
|
}
|
|
let info = self
|
|
.resolver
|
|
.package_version_info(nv)
|
|
.await
|
|
.ok_or_else(|| anyhow!("JSR package version info not found: {}", nv))?;
|
|
let mut exports = info
|
|
.exports()
|
|
.map(|(n, _)| n.to_string())
|
|
.collect::<Vec<_>>();
|
|
exports.sort();
|
|
let exports = Arc::new(exports);
|
|
self.exports_cache.insert(nv.clone(), exports.clone());
|
|
Ok(exports)
|
|
}
|
|
}
|
|
|
|
fn parse_jsr_search_response(source: &str) -> Result<Vec<String>, AnyError> {
|
|
#[derive(Debug, Deserialize)]
|
|
#[serde(rename_all = "camelCase")]
|
|
struct Item {
|
|
scope: String,
|
|
name: String,
|
|
version_count: usize,
|
|
}
|
|
#[derive(Debug, Deserialize)]
|
|
#[serde(rename_all = "camelCase")]
|
|
struct Response {
|
|
items: Vec<Item>,
|
|
}
|
|
let items = serde_json::from_str::<Response>(source)?.items;
|
|
Ok(
|
|
items
|
|
.into_iter()
|
|
.filter(|i| i.version_count > 0)
|
|
.map(|i| format!("@{}/{}", i.scope, i.name))
|
|
.collect(),
|
|
)
|
|
}
|