mirror of
https://github.com/denoland/deno.git
synced 2024-11-21 15:04:11 -05:00
feat(lsp): jsr support first pass (#22382)
This implementation heavily depends on there being a lockfile, meaning JSR specifiers will always diagnose as uncached unless it's there. In practice this affects cases where a `deno.json` isn't being used. Our NPM specifier support isn't subject to this. The reason for this is that the version constraint solving code is currently buried in `deno_graph` and not usable from the LSP, so the only way to reuse that logic is the solved-version map in the lockfile's `packages.specifiers`.
This commit is contained in:
parent
f60720090c
commit
49d82e609f
9 changed files with 265 additions and 15 deletions
4
Cargo.lock
generated
4
Cargo.lock
generated
|
@ -1404,9 +1404,9 @@ dependencies = [
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "deno_graph"
|
name = "deno_graph"
|
||||||
version = "0.65.1"
|
version = "0.65.2"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "06fadc0cd21f54794ab22e852d6c92487a65f0dd70b40eadbf3a8e249c760223"
|
checksum = "1f5d44da4195e0908b78bbf5db30dbc7defc8173f8d3034462863beccb368df4"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"anyhow",
|
"anyhow",
|
||||||
"async-trait",
|
"async-trait",
|
||||||
|
|
|
@ -65,7 +65,7 @@ deno_config = "=0.9.2"
|
||||||
deno_core = { workspace = true, features = ["include_js_files_for_snapshotting"] }
|
deno_core = { workspace = true, features = ["include_js_files_for_snapshotting"] }
|
||||||
deno_doc = { version = "=0.103.0", features = ["html"] }
|
deno_doc = { version = "=0.103.0", features = ["html"] }
|
||||||
deno_emit = "=0.36.0"
|
deno_emit = "=0.36.0"
|
||||||
deno_graph = "=0.65.1"
|
deno_graph = "=0.65.2"
|
||||||
deno_lint = { version = "=0.56.0", features = ["docs"] }
|
deno_lint = { version = "=0.56.0", features = ["docs"] }
|
||||||
deno_lockfile.workspace = true
|
deno_lockfile.workspace = true
|
||||||
deno_npm = "=0.16.0"
|
deno_npm = "=0.16.0"
|
||||||
|
|
|
@ -42,6 +42,7 @@ use deno_lint::rules::LintRule;
|
||||||
use deno_runtime::deno_fs;
|
use deno_runtime::deno_fs;
|
||||||
use deno_runtime::deno_node;
|
use deno_runtime::deno_node;
|
||||||
use deno_runtime::tokio_util::create_basic_runtime;
|
use deno_runtime::tokio_util::create_basic_runtime;
|
||||||
|
use deno_semver::jsr::JsrPackageReqReference;
|
||||||
use deno_semver::npm::NpmPackageReqReference;
|
use deno_semver::npm::NpmPackageReqReference;
|
||||||
use deno_semver::package::PackageReq;
|
use deno_semver::package::PackageReq;
|
||||||
use log::error;
|
use log::error;
|
||||||
|
@ -328,6 +329,8 @@ impl DiagnosticsState {
|
||||||
for diagnostic in diagnostics {
|
for diagnostic in diagnostics {
|
||||||
if diagnostic.code
|
if diagnostic.code
|
||||||
== Some(lsp::NumberOrString::String("no-cache".to_string()))
|
== Some(lsp::NumberOrString::String("no-cache".to_string()))
|
||||||
|
|| diagnostic.code
|
||||||
|
== Some(lsp::NumberOrString::String("no-cache-jsr".to_string()))
|
||||||
|| diagnostic.code
|
|| diagnostic.code
|
||||||
== Some(lsp::NumberOrString::String("no-cache-npm".to_string()))
|
== Some(lsp::NumberOrString::String("no-cache-npm".to_string()))
|
||||||
{
|
{
|
||||||
|
@ -968,6 +971,8 @@ pub enum DenoDiagnostic {
|
||||||
NoAttributeType,
|
NoAttributeType,
|
||||||
/// A remote module was not found in the cache.
|
/// A remote module was not found in the cache.
|
||||||
NoCache(ModuleSpecifier),
|
NoCache(ModuleSpecifier),
|
||||||
|
/// A remote jsr package reference was not found in the cache.
|
||||||
|
NoCacheJsr(PackageReq, ModuleSpecifier),
|
||||||
/// A remote npm package reference was not found in the cache.
|
/// A remote npm package reference was not found in the cache.
|
||||||
NoCacheNpm(PackageReq, ModuleSpecifier),
|
NoCacheNpm(PackageReq, ModuleSpecifier),
|
||||||
/// A local module was not found on the local file system.
|
/// A local module was not found on the local file system.
|
||||||
|
@ -994,6 +999,7 @@ impl DenoDiagnostic {
|
||||||
Self::InvalidAttributeType(_) => "invalid-attribute-type",
|
Self::InvalidAttributeType(_) => "invalid-attribute-type",
|
||||||
Self::NoAttributeType => "no-attribute-type",
|
Self::NoAttributeType => "no-attribute-type",
|
||||||
Self::NoCache(_) => "no-cache",
|
Self::NoCache(_) => "no-cache",
|
||||||
|
Self::NoCacheJsr(_, _) => "no-cache-jsr",
|
||||||
Self::NoCacheNpm(_, _) => "no-cache-npm",
|
Self::NoCacheNpm(_, _) => "no-cache-npm",
|
||||||
Self::NoLocal(_) => "no-local",
|
Self::NoLocal(_) => "no-local",
|
||||||
Self::Redirect { .. } => "redirect",
|
Self::Redirect { .. } => "redirect",
|
||||||
|
@ -1072,7 +1078,7 @@ impl DenoDiagnostic {
|
||||||
}),
|
}),
|
||||||
..Default::default()
|
..Default::default()
|
||||||
},
|
},
|
||||||
"no-cache" | "no-cache-npm" => {
|
"no-cache" | "no-cache-jsr" | "no-cache-npm" => {
|
||||||
let data = diagnostic
|
let data = diagnostic
|
||||||
.data
|
.data
|
||||||
.clone()
|
.clone()
|
||||||
|
@ -1188,6 +1194,7 @@ impl DenoDiagnostic {
|
||||||
match code.as_str() {
|
match code.as_str() {
|
||||||
"import-map-remap"
|
"import-map-remap"
|
||||||
| "no-cache"
|
| "no-cache"
|
||||||
|
| "no-cache-jsr"
|
||||||
| "no-cache-npm"
|
| "no-cache-npm"
|
||||||
| "no-attribute-type"
|
| "no-attribute-type"
|
||||||
| "redirect"
|
| "redirect"
|
||||||
|
@ -1226,6 +1233,7 @@ impl DenoDiagnostic {
|
||||||
Self::InvalidAttributeType(assert_type) => (lsp::DiagnosticSeverity::ERROR, format!("The module is a JSON module and expected an attribute type of \"json\". Instead got \"{assert_type}\"."), None),
|
Self::InvalidAttributeType(assert_type) => (lsp::DiagnosticSeverity::ERROR, format!("The module is a JSON module and expected an attribute type of \"json\". Instead got \"{assert_type}\"."), None),
|
||||||
Self::NoAttributeType => (lsp::DiagnosticSeverity::ERROR, "The module is a JSON module and not being imported with an import attribute. Consider adding `with { type: \"json\" }` to the import statement.".to_string(), None),
|
Self::NoAttributeType => (lsp::DiagnosticSeverity::ERROR, "The module is a JSON module and not being imported with an import attribute. Consider adding `with { type: \"json\" }` to the import statement.".to_string(), None),
|
||||||
Self::NoCache(specifier) => (lsp::DiagnosticSeverity::ERROR, format!("Uncached or missing remote URL: {specifier}"), Some(json!({ "specifier": specifier }))),
|
Self::NoCache(specifier) => (lsp::DiagnosticSeverity::ERROR, format!("Uncached or missing remote URL: {specifier}"), Some(json!({ "specifier": specifier }))),
|
||||||
|
Self::NoCacheJsr(pkg_req, specifier) => (lsp::DiagnosticSeverity::ERROR, format!("Uncached or missing jsr package: {}", pkg_req), Some(json!({ "specifier": specifier }))),
|
||||||
Self::NoCacheNpm(pkg_req, specifier) => (lsp::DiagnosticSeverity::ERROR, format!("Uncached or missing npm package: {}", pkg_req), Some(json!({ "specifier": specifier }))),
|
Self::NoCacheNpm(pkg_req, specifier) => (lsp::DiagnosticSeverity::ERROR, format!("Uncached or missing npm package: {}", pkg_req), Some(json!({ "specifier": specifier }))),
|
||||||
Self::NoLocal(specifier) => {
|
Self::NoLocal(specifier) => {
|
||||||
let sloppy_resolution = SloppyImportsResolver::resolve_with_fs(&deno_fs::RealFs, specifier);
|
let sloppy_resolution = SloppyImportsResolver::resolve_with_fs(&deno_fs::RealFs, specifier);
|
||||||
|
@ -1310,7 +1318,7 @@ fn diagnose_resolution(
|
||||||
// If the module was redirected, we want to issue an informational
|
// If the module was redirected, we want to issue an informational
|
||||||
// diagnostic that indicates this. This then allows us to issue a code
|
// diagnostic that indicates this. This then allows us to issue a code
|
||||||
// action to replace the specifier with the final redirected one.
|
// action to replace the specifier with the final redirected one.
|
||||||
if doc_specifier != specifier {
|
if specifier.scheme() != "jsr" && doc_specifier != specifier {
|
||||||
diagnostics.push(DenoDiagnostic::Redirect {
|
diagnostics.push(DenoDiagnostic::Redirect {
|
||||||
from: specifier.clone(),
|
from: specifier.clone(),
|
||||||
to: doc_specifier.clone(),
|
to: doc_specifier.clone(),
|
||||||
|
@ -1332,8 +1340,11 @@ fn diagnose_resolution(
|
||||||
None => diagnostics.push(DenoDiagnostic::NoAttributeType),
|
None => diagnostics.push(DenoDiagnostic::NoAttributeType),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} else if specifier.scheme() == "jsr" {
|
} else if let Ok(pkg_ref) =
|
||||||
// TODO(nayeemrmn): Check if jsr specifiers are cached.
|
JsrPackageReqReference::from_specifier(specifier)
|
||||||
|
{
|
||||||
|
let req = pkg_ref.into_inner().req;
|
||||||
|
diagnostics.push(DenoDiagnostic::NoCacheJsr(req, specifier.clone()));
|
||||||
} else if let Ok(pkg_ref) =
|
} else if let Ok(pkg_ref) =
|
||||||
NpmPackageReqReference::from_specifier(specifier)
|
NpmPackageReqReference::from_specifier(specifier)
|
||||||
{
|
{
|
||||||
|
|
|
@ -2,6 +2,7 @@
|
||||||
|
|
||||||
use super::cache::calculate_fs_version;
|
use super::cache::calculate_fs_version;
|
||||||
use super::cache::calculate_fs_version_at_path;
|
use super::cache::calculate_fs_version_at_path;
|
||||||
|
use super::jsr_resolver::JsrResolver;
|
||||||
use super::language_server::StateNpmSnapshot;
|
use super::language_server::StateNpmSnapshot;
|
||||||
use super::text::LineIndex;
|
use super::text::LineIndex;
|
||||||
use super::tsc;
|
use super::tsc;
|
||||||
|
@ -38,6 +39,7 @@ use deno_core::ModuleSpecifier;
|
||||||
use deno_graph::source::ResolutionMode;
|
use deno_graph::source::ResolutionMode;
|
||||||
use deno_graph::GraphImport;
|
use deno_graph::GraphImport;
|
||||||
use deno_graph::Resolution;
|
use deno_graph::Resolution;
|
||||||
|
use deno_lockfile::Lockfile;
|
||||||
use deno_runtime::deno_fs::RealFs;
|
use deno_runtime::deno_fs::RealFs;
|
||||||
use deno_runtime::deno_node;
|
use deno_runtime::deno_node;
|
||||||
use deno_runtime::deno_node::NodeResolution;
|
use deno_runtime::deno_node::NodeResolution;
|
||||||
|
@ -855,6 +857,7 @@ pub struct UpdateDocumentConfigOptions<'a> {
|
||||||
pub maybe_import_map: Option<Arc<import_map::ImportMap>>,
|
pub maybe_import_map: Option<Arc<import_map::ImportMap>>,
|
||||||
pub maybe_config_file: Option<&'a ConfigFile>,
|
pub maybe_config_file: Option<&'a ConfigFile>,
|
||||||
pub maybe_package_json: Option<&'a PackageJson>,
|
pub maybe_package_json: Option<&'a PackageJson>,
|
||||||
|
pub maybe_lockfile: Option<Arc<Mutex<Lockfile>>>,
|
||||||
pub node_resolver: Option<Arc<NodeResolver>>,
|
pub node_resolver: Option<Arc<NodeResolver>>,
|
||||||
pub npm_resolver: Option<Arc<dyn CliNpmResolver>>,
|
pub npm_resolver: Option<Arc<dyn CliNpmResolver>>,
|
||||||
}
|
}
|
||||||
|
@ -893,6 +896,7 @@ pub struct Documents {
|
||||||
/// A resolver that takes into account currently loaded import map and JSX
|
/// A resolver that takes into account currently loaded import map and JSX
|
||||||
/// settings.
|
/// settings.
|
||||||
resolver: Arc<CliGraphResolver>,
|
resolver: Arc<CliGraphResolver>,
|
||||||
|
jsr_resolver: Arc<JsrResolver>,
|
||||||
/// The npm package requirements found in npm specifiers.
|
/// The npm package requirements found in npm specifiers.
|
||||||
npm_specifier_reqs: Arc<Vec<PackageReq>>,
|
npm_specifier_reqs: Arc<Vec<PackageReq>>,
|
||||||
/// Gets if any document had a node: specifier such that a @types/node package
|
/// Gets if any document had a node: specifier such that a @types/node package
|
||||||
|
@ -927,6 +931,7 @@ impl Documents {
|
||||||
bare_node_builtins_enabled: false,
|
bare_node_builtins_enabled: false,
|
||||||
sloppy_imports_resolver: None,
|
sloppy_imports_resolver: None,
|
||||||
})),
|
})),
|
||||||
|
jsr_resolver: Default::default(),
|
||||||
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)),
|
||||||
|
@ -1084,7 +1089,11 @@ impl Documents {
|
||||||
.into_owned(),
|
.into_owned(),
|
||||||
)
|
)
|
||||||
} else {
|
} else {
|
||||||
self.redirect_resolver.resolve(specifier)
|
let specifier = match self.jsr_resolver.jsr_to_registry_url(specifier) {
|
||||||
|
Some(url) => Cow::Owned(url),
|
||||||
|
None => Cow::Borrowed(specifier),
|
||||||
|
};
|
||||||
|
self.redirect_resolver.resolve(&specifier)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1425,6 +1434,10 @@ impl Documents {
|
||||||
// specifier for free.
|
// specifier for free.
|
||||||
sloppy_imports_resolver: None,
|
sloppy_imports_resolver: None,
|
||||||
}));
|
}));
|
||||||
|
self.jsr_resolver = Arc::new(JsrResolver::from_cache_and_lockfile(
|
||||||
|
self.cache.clone(),
|
||||||
|
options.maybe_lockfile,
|
||||||
|
));
|
||||||
self.redirect_resolver =
|
self.redirect_resolver =
|
||||||
Arc::new(RedirectResolver::new(self.cache.clone()));
|
Arc::new(RedirectResolver::new(self.cache.clone()));
|
||||||
self.imports = Arc::new(
|
self.imports = Arc::new(
|
||||||
|
@ -2252,6 +2265,7 @@ console.log(b, "hello deno");
|
||||||
maybe_import_map: Some(Arc::new(import_map)),
|
maybe_import_map: Some(Arc::new(import_map)),
|
||||||
maybe_config_file: None,
|
maybe_config_file: None,
|
||||||
maybe_package_json: None,
|
maybe_package_json: None,
|
||||||
|
maybe_lockfile: None,
|
||||||
node_resolver: None,
|
node_resolver: None,
|
||||||
npm_resolver: None,
|
npm_resolver: None,
|
||||||
});
|
});
|
||||||
|
@ -2295,6 +2309,7 @@ console.log(b, "hello deno");
|
||||||
maybe_import_map: Some(Arc::new(import_map)),
|
maybe_import_map: Some(Arc::new(import_map)),
|
||||||
maybe_config_file: None,
|
maybe_config_file: None,
|
||||||
maybe_package_json: None,
|
maybe_package_json: None,
|
||||||
|
maybe_lockfile: None,
|
||||||
node_resolver: None,
|
node_resolver: None,
|
||||||
npm_resolver: None,
|
npm_resolver: None,
|
||||||
});
|
});
|
||||||
|
|
120
cli/lsp/jsr_resolver.rs
Normal file
120
cli/lsp/jsr_resolver.rs
Normal file
|
@ -0,0 +1,120 @@
|
||||||
|
// Copyright 2018-2024 the Deno authors. All rights reserved. MIT license.
|
||||||
|
|
||||||
|
use crate::args::deno_registry_url;
|
||||||
|
use deno_cache_dir::HttpCache;
|
||||||
|
use deno_core::parking_lot::Mutex;
|
||||||
|
use deno_core::serde_json;
|
||||||
|
use deno_core::serde_json::json;
|
||||||
|
use deno_core::ModuleSpecifier;
|
||||||
|
use deno_graph::packages::JsrPackageVersionInfo;
|
||||||
|
use deno_lockfile::Lockfile;
|
||||||
|
use deno_semver::jsr::JsrPackageReqReference;
|
||||||
|
use deno_semver::package::PackageNv;
|
||||||
|
use deno_semver::package::PackageReq;
|
||||||
|
use std::borrow::Cow;
|
||||||
|
use std::collections::HashMap;
|
||||||
|
use std::sync::Arc;
|
||||||
|
|
||||||
|
#[derive(Debug, Default)]
|
||||||
|
pub struct JsrResolver {
|
||||||
|
nv_by_req: HashMap<PackageReq, PackageNv>,
|
||||||
|
/// The `module_graph` field of the version infos should be forcibly absent.
|
||||||
|
/// It can be large and we don't want to store it.
|
||||||
|
info_by_nv: HashMap<PackageNv, JsrPackageVersionInfo>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl JsrResolver {
|
||||||
|
pub fn from_cache_and_lockfile(
|
||||||
|
cache: Arc<dyn HttpCache>,
|
||||||
|
lockfile: Option<Arc<Mutex<Lockfile>>>,
|
||||||
|
) -> Self {
|
||||||
|
let mut nv_by_req = HashMap::new();
|
||||||
|
let mut info_by_nv = HashMap::new();
|
||||||
|
if let Some(lockfile) = lockfile {
|
||||||
|
for (req_url, nv_url) in &lockfile.lock().content.packages.specifiers {
|
||||||
|
let Some(req) = req_url.strip_prefix("jsr:") else {
|
||||||
|
continue;
|
||||||
|
};
|
||||||
|
let Some(nv) = nv_url.strip_prefix("jsr:") else {
|
||||||
|
continue;
|
||||||
|
};
|
||||||
|
let Ok(req) = PackageReq::from_str(req) else {
|
||||||
|
continue;
|
||||||
|
};
|
||||||
|
let Ok(nv) = PackageNv::from_str(nv) else {
|
||||||
|
continue;
|
||||||
|
};
|
||||||
|
nv_by_req.insert(req, nv);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
for nv in nv_by_req.values() {
|
||||||
|
if info_by_nv.contains_key(nv) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
let Ok(meta_url) = deno_registry_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 {
|
||||||
|
nv_by_req,
|
||||||
|
info_by_nv,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn jsr_to_registry_url(
|
||||||
|
&self,
|
||||||
|
specifier: &ModuleSpecifier,
|
||||||
|
) -> Option<ModuleSpecifier> {
|
||||||
|
let req_ref = JsrPackageReqReference::from_str(specifier.as_str()).ok()?;
|
||||||
|
let nv = self.nv_by_req.get(req_ref.req())?;
|
||||||
|
let info = self.info_by_nv.get(nv)?;
|
||||||
|
let path = info.export(&normalize_export_name(req_ref.sub_path()))?;
|
||||||
|
deno_registry_url()
|
||||||
|
.join(&format!("{}/{}/{}", &nv.name, &nv.version, &path))
|
||||||
|
.ok()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// TODO(nayeemrmn): This is duplicated from a private function in deno_graph
|
||||||
|
// 0.65.1. Make it public or cleanup otherwise.
|
||||||
|
fn normalize_export_name(sub_path: Option<&str>) -> Cow<str> {
|
||||||
|
let Some(sub_path) = sub_path else {
|
||||||
|
return Cow::Borrowed(".");
|
||||||
|
};
|
||||||
|
if sub_path.is_empty() || matches!(sub_path, "/" | ".") {
|
||||||
|
Cow::Borrowed(".")
|
||||||
|
} else {
|
||||||
|
let sub_path = if sub_path.starts_with('/') {
|
||||||
|
Cow::Owned(format!(".{}", sub_path))
|
||||||
|
} else if !sub_path.starts_with("./") {
|
||||||
|
Cow::Owned(format!("./{}", sub_path))
|
||||||
|
} else {
|
||||||
|
Cow::Borrowed(sub_path)
|
||||||
|
};
|
||||||
|
if let Some(prefix) = sub_path.strip_suffix('/') {
|
||||||
|
Cow::Owned(prefix.to_string())
|
||||||
|
} else {
|
||||||
|
sub_path
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -362,9 +362,24 @@ impl LanguageServer {
|
||||||
.client
|
.client
|
||||||
.show_message(MessageType::WARNING, err);
|
.show_message(MessageType::WARNING, err);
|
||||||
}
|
}
|
||||||
// do npm resolution in a write—we should have everything
|
let mut lockfile_content_changed = false;
|
||||||
// cached by this point anyway
|
if let Some(lockfile) = self.0.read().await.config.maybe_lockfile() {
|
||||||
|
let lockfile = lockfile.lock();
|
||||||
|
let path = lockfile.filename.clone();
|
||||||
|
if let Ok(new_lockfile) = Lockfile::new(path, false) {
|
||||||
|
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;
|
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;
|
||||||
}
|
}
|
||||||
|
@ -1330,6 +1345,7 @@ impl Inner {
|
||||||
maybe_import_map: self.maybe_import_map.clone(),
|
maybe_import_map: self.maybe_import_map.clone(),
|
||||||
maybe_config_file: self.config.maybe_config_file(),
|
maybe_config_file: self.config.maybe_config_file(),
|
||||||
maybe_package_json: self.maybe_package_json.as_ref(),
|
maybe_package_json: self.maybe_package_json.as_ref(),
|
||||||
|
maybe_lockfile: self.config.maybe_lockfile().cloned(),
|
||||||
node_resolver: self.npm.node_resolver.clone(),
|
node_resolver: self.npm.node_resolver.clone(),
|
||||||
npm_resolver: self.npm.resolver.clone(),
|
npm_resolver: self.npm.resolver.clone(),
|
||||||
});
|
});
|
||||||
|
@ -2009,6 +2025,8 @@ impl Inner {
|
||||||
Some("deno") => {
|
Some("deno") => {
|
||||||
if diagnostic.code
|
if diagnostic.code
|
||||||
== Some(NumberOrString::String("no-cache".to_string()))
|
== Some(NumberOrString::String("no-cache".to_string()))
|
||||||
|
|| diagnostic.code
|
||||||
|
== Some(NumberOrString::String("no-cache-jsr".to_string()))
|
||||||
|| diagnostic.code
|
|| diagnostic.code
|
||||||
== Some(NumberOrString::String("no-cache-npm".to_string()))
|
== Some(NumberOrString::String("no-cache-npm".to_string()))
|
||||||
{
|
{
|
||||||
|
|
|
@ -21,6 +21,7 @@ mod completions;
|
||||||
mod config;
|
mod config;
|
||||||
mod diagnostics;
|
mod diagnostics;
|
||||||
mod documents;
|
mod documents;
|
||||||
|
mod jsr_resolver;
|
||||||
pub mod language_server;
|
pub mod language_server;
|
||||||
mod logging;
|
mod logging;
|
||||||
mod lsp_custom;
|
mod lsp_custom;
|
||||||
|
|
|
@ -1,6 +1,7 @@
|
||||||
// Copyright 2018-2024 the Deno authors. All rights reserved. MIT license.
|
// Copyright 2018-2024 the Deno authors. All rights reserved. MIT license.
|
||||||
|
|
||||||
use crate::deno_exe_path;
|
use crate::deno_exe_path;
|
||||||
|
use crate::jsr_registry_url;
|
||||||
use crate::npm_registry_url;
|
use crate::npm_registry_url;
|
||||||
use crate::PathRef;
|
use crate::PathRef;
|
||||||
|
|
||||||
|
@ -523,6 +524,7 @@ impl LspClientBuilder {
|
||||||
command
|
command
|
||||||
.env("DENO_DIR", deno_dir.path())
|
.env("DENO_DIR", deno_dir.path())
|
||||||
.env("NPM_CONFIG_REGISTRY", npm_registry_url())
|
.env("NPM_CONFIG_REGISTRY", npm_registry_url())
|
||||||
|
.env("DENO_REGISTRY_URL", jsr_registry_url())
|
||||||
// turn on diagnostic synchronization communication
|
// turn on diagnostic synchronization communication
|
||||||
.env(
|
.env(
|
||||||
"DENO_DONT_USE_INTERNAL_LSP_DIAGNOSTIC_SYNC_FLAG",
|
"DENO_DONT_USE_INTERNAL_LSP_DIAGNOSTIC_SYNC_FLAG",
|
||||||
|
|
|
@ -4658,12 +4658,15 @@ fn lsp_code_actions_deno_cache() {
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn lsp_jsr_uncached() {
|
fn lsp_code_actions_deno_cache_jsr() {
|
||||||
let context = TestContextBuilder::new()
|
let context = TestContextBuilder::new()
|
||||||
.use_http_server()
|
.use_http_server()
|
||||||
.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!({
|
||||||
|
@ -4671,11 +4674,91 @@ fn lsp_jsr_uncached() {
|
||||||
"uri": temp_dir.uri().join("file.ts").unwrap(),
|
"uri": temp_dir.uri().join("file.ts").unwrap(),
|
||||||
"languageId": "typescript",
|
"languageId": "typescript",
|
||||||
"version": 1,
|
"version": 1,
|
||||||
"text": r#"import "jsr:@foo/bar";"#,
|
"text": r#"
|
||||||
|
import { add } from "jsr:@denotest/add@1";
|
||||||
|
console.log(add(1, 2));
|
||||||
|
"#,
|
||||||
},
|
},
|
||||||
}));
|
}));
|
||||||
// TODO(nayeemrmn): This should check if the jsr dep is cached and give a
|
assert_eq!(
|
||||||
// diagnostic.
|
json!(diagnostics.messages_with_source("deno")),
|
||||||
|
json!({
|
||||||
|
"uri": temp_dir.uri().join("file.ts").unwrap(),
|
||||||
|
"diagnostics": [{
|
||||||
|
"range": {
|
||||||
|
"start": { "line": 1, "character": 28 },
|
||||||
|
"end": { "line": 1, "character": 49 },
|
||||||
|
},
|
||||||
|
"severity": 1,
|
||||||
|
"code": "no-cache-jsr",
|
||||||
|
"source": "deno",
|
||||||
|
"message": "Uncached or missing jsr package: @denotest/add@1",
|
||||||
|
"data": { "specifier": "jsr:@denotest/add@1" },
|
||||||
|
}],
|
||||||
|
"version": 1,
|
||||||
|
})
|
||||||
|
);
|
||||||
|
let res = client.write_request(
|
||||||
|
"textDocument/codeAction",
|
||||||
|
json!({
|
||||||
|
"textDocument": { "uri": temp_dir.uri().join("file.ts").unwrap() },
|
||||||
|
"range": {
|
||||||
|
"start": { "line": 1, "character": 28 },
|
||||||
|
"end": { "line": 1, "character": 49 },
|
||||||
|
},
|
||||||
|
"context": {
|
||||||
|
"diagnostics": [{
|
||||||
|
"range": {
|
||||||
|
"start": { "line": 1, "character": 28 },
|
||||||
|
"end": { "line": 1, "character": 49 },
|
||||||
|
},
|
||||||
|
"severity": 1,
|
||||||
|
"code": "no-cache-jsr",
|
||||||
|
"source": "deno",
|
||||||
|
"message": "Uncached or missing jsr package: @denotest/add@1",
|
||||||
|
"data": { "specifier": "jsr:@denotest/add@1" },
|
||||||
|
}],
|
||||||
|
"only": ["quickfix"],
|
||||||
|
}
|
||||||
|
}),
|
||||||
|
);
|
||||||
|
assert_eq!(
|
||||||
|
res,
|
||||||
|
json!([{
|
||||||
|
"title": "Cache \"jsr:@denotest/add@1\" and its dependencies.",
|
||||||
|
"kind": "quickfix",
|
||||||
|
"diagnostics": [{
|
||||||
|
"range": {
|
||||||
|
"start": { "line": 1, "character": 28 },
|
||||||
|
"end": { "line": 1, "character": 49 },
|
||||||
|
},
|
||||||
|
"severity": 1,
|
||||||
|
"code": "no-cache-jsr",
|
||||||
|
"source": "deno",
|
||||||
|
"message": "Uncached or missing jsr package: @denotest/add@1",
|
||||||
|
"data": { "specifier": "jsr:@denotest/add@1" },
|
||||||
|
}],
|
||||||
|
"command": {
|
||||||
|
"title": "",
|
||||||
|
"command": "deno.cache",
|
||||||
|
"arguments": [
|
||||||
|
["jsr:@denotest/add@1"],
|
||||||
|
temp_dir.uri().join("file.ts").unwrap(),
|
||||||
|
],
|
||||||
|
},
|
||||||
|
}])
|
||||||
|
);
|
||||||
|
client.write_request(
|
||||||
|
"workspace/executeCommand",
|
||||||
|
json!({
|
||||||
|
"command": "deno.cache",
|
||||||
|
"arguments": [
|
||||||
|
["jsr:@denotest/add@1"],
|
||||||
|
temp_dir.uri().join("file.ts").unwrap(),
|
||||||
|
],
|
||||||
|
}),
|
||||||
|
);
|
||||||
|
let diagnostics = client.read_diagnostics();
|
||||||
assert_eq!(json!(diagnostics.all()), json!([]));
|
assert_eq!(json!(diagnostics.all()), json!([]));
|
||||||
client.shutdown();
|
client.shutdown();
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue