mirror of
https://github.com/denoland/deno.git
synced 2025-01-11 16:42:21 -05:00
feat(lsp): jsr support with cache probing (#22418)
This commit is contained in:
parent
66baff763f
commit
1ad754b412
7 changed files with 161 additions and 61 deletions
|
@ -931,7 +931,10 @@ impl Documents {
|
||||||
bare_node_builtins_enabled: false,
|
bare_node_builtins_enabled: false,
|
||||||
sloppy_imports_resolver: None,
|
sloppy_imports_resolver: None,
|
||||||
})),
|
})),
|
||||||
jsr_resolver: Default::default(),
|
jsr_resolver: Arc::new(JsrResolver::from_cache_and_lockfile(
|
||||||
|
cache.clone(),
|
||||||
|
None,
|
||||||
|
)),
|
||||||
npm_specifier_reqs: Default::default(),
|
npm_specifier_reqs: Default::default(),
|
||||||
has_injected_types_node_package: false,
|
has_injected_types_node_package: false,
|
||||||
redirect_resolver: Arc::new(RedirectResolver::new(cache)),
|
redirect_resolver: Arc::new(RedirectResolver::new(cache)),
|
||||||
|
@ -1332,6 +1335,16 @@ impl Documents {
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn refresh_jsr_resolver(
|
||||||
|
&mut self,
|
||||||
|
lockfile: Option<Arc<Mutex<Lockfile>>>,
|
||||||
|
) {
|
||||||
|
self.jsr_resolver = Arc::new(JsrResolver::from_cache_and_lockfile(
|
||||||
|
self.cache.clone(),
|
||||||
|
lockfile,
|
||||||
|
));
|
||||||
|
}
|
||||||
|
|
||||||
pub fn update_config(&mut self, options: UpdateDocumentConfigOptions) {
|
pub fn update_config(&mut self, options: UpdateDocumentConfigOptions) {
|
||||||
#[allow(clippy::too_many_arguments)]
|
#[allow(clippy::too_many_arguments)]
|
||||||
fn calculate_resolver_config_hash(
|
fn calculate_resolver_config_hash(
|
||||||
|
|
|
@ -1,26 +1,28 @@
|
||||||
// Copyright 2018-2024 the Deno authors. All rights reserved. MIT license.
|
// Copyright 2018-2024 the Deno authors. All rights reserved. MIT license.
|
||||||
|
|
||||||
use crate::args::jsr_url;
|
use crate::args::jsr_url;
|
||||||
|
use dashmap::DashMap;
|
||||||
use deno_cache_dir::HttpCache;
|
use deno_cache_dir::HttpCache;
|
||||||
use deno_core::parking_lot::Mutex;
|
use deno_core::parking_lot::Mutex;
|
||||||
use deno_core::serde_json;
|
use deno_core::serde_json;
|
||||||
use deno_core::serde_json::json;
|
|
||||||
use deno_core::ModuleSpecifier;
|
use deno_core::ModuleSpecifier;
|
||||||
|
use deno_graph::packages::JsrPackageInfo;
|
||||||
use deno_graph::packages::JsrPackageVersionInfo;
|
use deno_graph::packages::JsrPackageVersionInfo;
|
||||||
use deno_lockfile::Lockfile;
|
use deno_lockfile::Lockfile;
|
||||||
use deno_semver::jsr::JsrPackageReqReference;
|
use deno_semver::jsr::JsrPackageReqReference;
|
||||||
use deno_semver::package::PackageNv;
|
use deno_semver::package::PackageNv;
|
||||||
use deno_semver::package::PackageReq;
|
use deno_semver::package::PackageReq;
|
||||||
use std::borrow::Cow;
|
use std::borrow::Cow;
|
||||||
use std::collections::HashMap;
|
|
||||||
use std::sync::Arc;
|
use std::sync::Arc;
|
||||||
|
|
||||||
#[derive(Debug, Default)]
|
#[derive(Debug)]
|
||||||
pub struct JsrResolver {
|
pub struct JsrResolver {
|
||||||
nv_by_req: HashMap<PackageReq, PackageNv>,
|
nv_by_req: DashMap<PackageReq, Option<PackageNv>>,
|
||||||
/// The `module_graph` field of the version infos should be forcibly absent.
|
/// The `module_graph` field of the version infos should be forcibly absent.
|
||||||
/// It can be large and we don't want to store it.
|
/// It can be large and we don't want to store it.
|
||||||
info_by_nv: HashMap<PackageNv, JsrPackageVersionInfo>,
|
info_by_nv: DashMap<PackageNv, Option<JsrPackageVersionInfo>>,
|
||||||
|
info_by_name: DashMap<String, Option<JsrPackageInfo>>,
|
||||||
|
cache: Arc<dyn HttpCache>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl JsrResolver {
|
impl JsrResolver {
|
||||||
|
@ -28,8 +30,7 @@ impl JsrResolver {
|
||||||
cache: Arc<dyn HttpCache>,
|
cache: Arc<dyn HttpCache>,
|
||||||
lockfile: Option<Arc<Mutex<Lockfile>>>,
|
lockfile: Option<Arc<Mutex<Lockfile>>>,
|
||||||
) -> Self {
|
) -> Self {
|
||||||
let mut nv_by_req = HashMap::new();
|
let nv_by_req = DashMap::new();
|
||||||
let mut info_by_nv = HashMap::new();
|
|
||||||
if let Some(lockfile) = lockfile {
|
if let Some(lockfile) = lockfile {
|
||||||
for (req_url, nv_url) in &lockfile.lock().content.packages.specifiers {
|
for (req_url, nv_url) in &lockfile.lock().content.packages.specifiers {
|
||||||
let Some(req) = req_url.strip_prefix("jsr:") else {
|
let Some(req) = req_url.strip_prefix("jsr:") else {
|
||||||
|
@ -44,40 +45,14 @@ impl JsrResolver {
|
||||||
let Ok(nv) = PackageNv::from_str(nv) else {
|
let Ok(nv) = PackageNv::from_str(nv) else {
|
||||||
continue;
|
continue;
|
||||||
};
|
};
|
||||||
nv_by_req.insert(req, nv);
|
nv_by_req.insert(req, Some(nv));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
for nv in nv_by_req.values() {
|
|
||||||
if info_by_nv.contains_key(nv) {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
let Ok(meta_url) =
|
|
||||||
jsr_url().join(&format!("{}/{}_meta.json", &nv.name, &nv.version))
|
|
||||||
else {
|
|
||||||
continue;
|
|
||||||
};
|
|
||||||
let Ok(meta_cache_item_key) = cache.cache_item_key(&meta_url) else {
|
|
||||||
continue;
|
|
||||||
};
|
|
||||||
let Ok(Some(meta_bytes)) = cache.read_file_bytes(&meta_cache_item_key)
|
|
||||||
else {
|
|
||||||
continue;
|
|
||||||
};
|
|
||||||
// This is a roundabout way of deserializing `JsrPackageVersionInfo`,
|
|
||||||
// because we only want the `exports` field and `module_graph` is large.
|
|
||||||
let Ok(info) = serde_json::from_slice::<serde_json::Value>(&meta_bytes)
|
|
||||||
else {
|
|
||||||
continue;
|
|
||||||
};
|
|
||||||
let info = JsrPackageVersionInfo {
|
|
||||||
exports: json!(info.as_object().and_then(|o| o.get("exports"))),
|
|
||||||
module_graph: None,
|
|
||||||
};
|
|
||||||
info_by_nv.insert(nv.clone(), info);
|
|
||||||
}
|
|
||||||
Self {
|
Self {
|
||||||
nv_by_req,
|
nv_by_req,
|
||||||
info_by_nv,
|
info_by_nv: Default::default(),
|
||||||
|
info_by_name: Default::default(),
|
||||||
|
cache: cache.clone(),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -86,8 +61,43 @@ impl JsrResolver {
|
||||||
specifier: &ModuleSpecifier,
|
specifier: &ModuleSpecifier,
|
||||||
) -> Option<ModuleSpecifier> {
|
) -> Option<ModuleSpecifier> {
|
||||||
let req_ref = JsrPackageReqReference::from_str(specifier.as_str()).ok()?;
|
let req_ref = JsrPackageReqReference::from_str(specifier.as_str()).ok()?;
|
||||||
let nv = self.nv_by_req.get(req_ref.req())?;
|
let req = req_ref.req().clone();
|
||||||
let info = self.info_by_nv.get(nv)?;
|
let maybe_nv = self.nv_by_req.entry(req.clone()).or_insert_with(|| {
|
||||||
|
let name = req.name.clone();
|
||||||
|
let maybe_package_info = self
|
||||||
|
.info_by_name
|
||||||
|
.entry(name.clone())
|
||||||
|
.or_insert_with(|| read_cached_package_info(&name, &self.cache));
|
||||||
|
let package_info = maybe_package_info.as_ref()?;
|
||||||
|
// Find the first matching version of the package which is cached.
|
||||||
|
let version = package_info
|
||||||
|
.versions
|
||||||
|
.keys()
|
||||||
|
.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
|
||||||
|
.info_by_nv
|
||||||
|
.entry(nv.clone())
|
||||||
|
.or_insert_with(|| {
|
||||||
|
read_cached_package_version_info(&nv, &self.cache)
|
||||||
|
})
|
||||||
|
.is_some()
|
||||||
|
})
|
||||||
|
.cloned()?;
|
||||||
|
Some(PackageNv { name, version })
|
||||||
|
});
|
||||||
|
let nv = maybe_nv.as_ref()?;
|
||||||
|
let maybe_info = self
|
||||||
|
.info_by_nv
|
||||||
|
.entry(nv.clone())
|
||||||
|
.or_insert_with(|| read_cached_package_version_info(nv, &self.cache));
|
||||||
|
let info = maybe_info.as_ref()?;
|
||||||
let path = info.export(&normalize_export_name(req_ref.sub_path()))?;
|
let path = info.export(&normalize_export_name(req_ref.sub_path()))?;
|
||||||
jsr_url()
|
jsr_url()
|
||||||
.join(&format!("{}/{}/{}", &nv.name, &nv.version, &path))
|
.join(&format!("{}/{}/{}", &nv.name, &nv.version, &path))
|
||||||
|
@ -95,6 +105,35 @@ impl JsrResolver {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn read_cached_package_info(
|
||||||
|
name: &str,
|
||||||
|
cache: &Arc<dyn HttpCache>,
|
||||||
|
) -> Option<JsrPackageInfo> {
|
||||||
|
let meta_url = jsr_url().join(&format!("{}/meta.json", name)).ok()?;
|
||||||
|
let meta_cache_item_key = cache.cache_item_key(&meta_url).ok()?;
|
||||||
|
let meta_bytes = cache.read_file_bytes(&meta_cache_item_key).ok()??;
|
||||||
|
serde_json::from_slice::<JsrPackageInfo>(&meta_bytes).ok()
|
||||||
|
}
|
||||||
|
|
||||||
|
fn read_cached_package_version_info(
|
||||||
|
nv: &PackageNv,
|
||||||
|
cache: &Arc<dyn HttpCache>,
|
||||||
|
) -> Option<JsrPackageVersionInfo> {
|
||||||
|
let meta_url = jsr_url()
|
||||||
|
.join(&format!("{}/{}_meta.json", &nv.name, &nv.version))
|
||||||
|
.ok()?;
|
||||||
|
let meta_cache_item_key = cache.cache_item_key(&meta_url).ok()?;
|
||||||
|
let meta_bytes = cache.read_file_bytes(&meta_cache_item_key).ok()??;
|
||||||
|
// This is a roundabout way of deserializing `JsrPackageVersionInfo`,
|
||||||
|
// because we only want the `exports` field and `module_graph` is large.
|
||||||
|
let mut info =
|
||||||
|
serde_json::from_slice::<serde_json::Value>(&meta_bytes).ok()?;
|
||||||
|
Some(JsrPackageVersionInfo {
|
||||||
|
exports: info.as_object_mut()?.remove("exports")?,
|
||||||
|
module_graph: None,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
// TODO(nayeemrmn): This is duplicated from a private function in deno_graph
|
// TODO(nayeemrmn): This is duplicated from a private function in deno_graph
|
||||||
// 0.65.1. Make it public or cleanup otherwise.
|
// 0.65.1. Make it public or cleanup otherwise.
|
||||||
fn normalize_export_name(sub_path: Option<&str>) -> Cow<str> {
|
fn normalize_export_name(sub_path: Option<&str>) -> Cow<str> {
|
||||||
|
|
|
@ -362,23 +362,11 @@ impl LanguageServer {
|
||||||
.client
|
.client
|
||||||
.show_message(MessageType::WARNING, err);
|
.show_message(MessageType::WARNING, err);
|
||||||
}
|
}
|
||||||
let mut lockfile_content_changed = false;
|
{
|
||||||
if let Some(lockfile) = self.0.read().await.config.maybe_lockfile() {
|
let mut inner = self.0.write().await;
|
||||||
let lockfile = lockfile.lock();
|
let lockfile = inner.config.maybe_lockfile().cloned();
|
||||||
let path = lockfile.filename.clone();
|
inner.documents.refresh_jsr_resolver(lockfile);
|
||||||
if let Ok(new_lockfile) = Lockfile::new(path, false) {
|
inner.refresh_npm_specifiers().await;
|
||||||
lockfile_content_changed = FastInsecureHasher::hash(&*lockfile)
|
|
||||||
!= FastInsecureHasher::hash(new_lockfile);
|
|
||||||
} else {
|
|
||||||
lockfile_content_changed = true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if lockfile_content_changed {
|
|
||||||
// TODO(nayeemrmn): Remove this branch when the documents config no
|
|
||||||
// longer depends on the lockfile for JSR resolution.
|
|
||||||
self.0.write().await.refresh_documents_config().await;
|
|
||||||
} else {
|
|
||||||
self.0.write().await.refresh_npm_specifiers().await;
|
|
||||||
}
|
}
|
||||||
// now refresh the data in a read
|
// now refresh the data in a read
|
||||||
self.0.read().await.post_cache(result.mark).await;
|
self.0.read().await.post_cache(result.mark).await;
|
||||||
|
|
|
@ -4664,9 +4664,6 @@ fn lsp_code_actions_deno_cache_jsr() {
|
||||||
.use_temp_cwd()
|
.use_temp_cwd()
|
||||||
.build();
|
.build();
|
||||||
let temp_dir = context.temp_dir();
|
let temp_dir = context.temp_dir();
|
||||||
// TODO(nayeemrmn): JSR resolution currently depends on a lockfile being
|
|
||||||
// created on cache. Remove this when that's not the case.
|
|
||||||
temp_dir.write("deno.json", "{}");
|
|
||||||
let mut client = context.new_lsp_command().build();
|
let mut client = context.new_lsp_command().build();
|
||||||
client.initialize_default();
|
client.initialize_default();
|
||||||
let diagnostics = client.did_open(json!({
|
let diagnostics = client.did_open(json!({
|
||||||
|
@ -4763,6 +4760,56 @@ fn lsp_code_actions_deno_cache_jsr() {
|
||||||
client.shutdown();
|
client.shutdown();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn lsp_jsr_lockfile() {
|
||||||
|
let context = TestContextBuilder::new()
|
||||||
|
.use_http_server()
|
||||||
|
.use_temp_cwd()
|
||||||
|
.build();
|
||||||
|
let temp_dir = context.temp_dir();
|
||||||
|
temp_dir.write("./deno.json", json!({}).to_string());
|
||||||
|
temp_dir.write(
|
||||||
|
"./deno.lock",
|
||||||
|
json!({
|
||||||
|
"version": "3",
|
||||||
|
"packages": {
|
||||||
|
"specifiers": {
|
||||||
|
// This is an old version of the package which exports `sum()` instead
|
||||||
|
// of `add()`.
|
||||||
|
"jsr:@denotest/add": "jsr:@denotest/add@0.2.0",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
})
|
||||||
|
.to_string(),
|
||||||
|
);
|
||||||
|
let mut client = context.new_lsp_command().build();
|
||||||
|
client.initialize_default();
|
||||||
|
client.did_open(json!({
|
||||||
|
"textDocument": {
|
||||||
|
"uri": temp_dir.uri().join("file.ts").unwrap(),
|
||||||
|
"languageId": "typescript",
|
||||||
|
"version": 1,
|
||||||
|
"text": r#"
|
||||||
|
import { add } from "jsr:@denotest/add";
|
||||||
|
console.log(add(1, 2));
|
||||||
|
"#,
|
||||||
|
},
|
||||||
|
}));
|
||||||
|
client.write_request(
|
||||||
|
"workspace/executeCommand",
|
||||||
|
json!({
|
||||||
|
"command": "deno.cache",
|
||||||
|
"arguments": [
|
||||||
|
[],
|
||||||
|
temp_dir.uri().join("file.ts").unwrap(),
|
||||||
|
],
|
||||||
|
}),
|
||||||
|
);
|
||||||
|
let diagnostics = client.read_diagnostics();
|
||||||
|
assert_eq!(json!(diagnostics.all()), json!([]));
|
||||||
|
client.shutdown();
|
||||||
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn lsp_code_actions_deno_cache_npm() {
|
fn lsp_code_actions_deno_cache_npm() {
|
||||||
let context = TestContextBuilder::new().use_temp_cwd().build();
|
let context = TestContextBuilder::new().use_temp_cwd().build();
|
||||||
|
|
4
tests/testdata/jsr/registry/@denotest/add/0.2.0/mod.ts
vendored
Normal file
4
tests/testdata/jsr/registry/@denotest/add/0.2.0/mod.ts
vendored
Normal file
|
@ -0,0 +1,4 @@
|
||||||
|
// This is renamed to `add()` in 1.0.0.
|
||||||
|
export function sum(a: number, b: number): number {
|
||||||
|
return a + b;
|
||||||
|
}
|
8
tests/testdata/jsr/registry/@denotest/add/0.2.0_meta.json
vendored
Normal file
8
tests/testdata/jsr/registry/@denotest/add/0.2.0_meta.json
vendored
Normal file
|
@ -0,0 +1,8 @@
|
||||||
|
{
|
||||||
|
"exports": {
|
||||||
|
".": "./mod.ts"
|
||||||
|
},
|
||||||
|
"moduleGraph1": {
|
||||||
|
"/mod.ts": {}
|
||||||
|
}
|
||||||
|
}
|
|
@ -1,5 +1,6 @@
|
||||||
{
|
{
|
||||||
"versions": {
|
"versions": {
|
||||||
"1.0.0": {}
|
"1.0.0": {},
|
||||||
|
"0.2.0": {}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue