mirror of
https://github.com/denoland/deno.git
synced 2025-01-08 15:19:40 -05:00
refactor(lsp): store assets behind a mutex (#13414)
This commit is contained in:
parent
0f3a53e5d4
commit
b3545dd447
2 changed files with 126 additions and 70 deletions
|
@ -52,6 +52,7 @@ use super::text;
|
|||
use super::tsc;
|
||||
use super::tsc::AssetDocument;
|
||||
use super::tsc::Assets;
|
||||
use super::tsc::AssetsSnapshot;
|
||||
use super::tsc::TsServer;
|
||||
use super::urls;
|
||||
use crate::config_file::ConfigFile;
|
||||
|
@ -75,7 +76,7 @@ pub struct LanguageServer(Arc<tokio::sync::Mutex<Inner>>);
|
|||
|
||||
#[derive(Debug, Default)]
|
||||
pub(crate) struct StateSnapshot {
|
||||
pub assets: Assets,
|
||||
pub assets: AssetsSnapshot,
|
||||
pub config: ConfigSnapshot,
|
||||
pub documents: Documents,
|
||||
pub maybe_lint_config: Option<LintConfig>,
|
||||
|
@ -151,9 +152,10 @@ impl Inner {
|
|||
performance.clone(),
|
||||
ts_server.clone(),
|
||||
);
|
||||
let assets = Assets::new(ts_server.clone());
|
||||
|
||||
Self {
|
||||
assets: Default::default(),
|
||||
assets,
|
||||
client,
|
||||
config,
|
||||
diagnostics_server,
|
||||
|
@ -177,7 +179,7 @@ impl Inner {
|
|||
/// Searches assets and open documents which might be performed asynchronously,
|
||||
/// hydrating in memory caches for subsequent requests.
|
||||
pub(crate) async fn get_asset_or_document(
|
||||
&mut self,
|
||||
&self,
|
||||
specifier: &ModuleSpecifier,
|
||||
) -> LspResult<AssetOrDocument> {
|
||||
self
|
||||
|
@ -197,7 +199,7 @@ impl Inner {
|
|||
/// Searches assets and open documents which might be performed asynchronously,
|
||||
/// hydrating in memory caches for subsequent requests.
|
||||
pub(crate) async fn get_maybe_asset_or_document(
|
||||
&mut self,
|
||||
&self,
|
||||
specifier: &ModuleSpecifier,
|
||||
) -> LspResult<Option<AssetOrDocument>> {
|
||||
let mark = self.performance.mark(
|
||||
|
@ -205,7 +207,11 @@ impl Inner {
|
|||
Some(json!({ "specifier": specifier })),
|
||||
);
|
||||
let result = if specifier.scheme() == "asset" {
|
||||
self.get_asset(specifier).await?.map(AssetOrDocument::Asset)
|
||||
self
|
||||
.assets
|
||||
.get(specifier, || self.snapshot())
|
||||
.await?
|
||||
.map(AssetOrDocument::Asset)
|
||||
} else {
|
||||
self.documents.get(specifier).map(AssetOrDocument::Document)
|
||||
};
|
||||
|
@ -241,7 +247,7 @@ impl Inner {
|
|||
if specifier.scheme() == "asset" {
|
||||
self
|
||||
.assets
|
||||
.get(specifier)
|
||||
.get_cached(specifier)
|
||||
.map(|maybe_asset| {
|
||||
maybe_asset
|
||||
.as_ref()
|
||||
|
@ -365,7 +371,7 @@ impl Inner {
|
|||
}
|
||||
|
||||
fn merge_user_tsconfig(
|
||||
&mut self,
|
||||
&self,
|
||||
tsconfig: &mut TsConfig,
|
||||
) -> Result<(), AnyError> {
|
||||
if let Some(config_file) = self.maybe_config_file.as_ref() {
|
||||
|
@ -383,7 +389,7 @@ impl Inner {
|
|||
|
||||
pub(crate) fn snapshot(&self) -> LspResult<Arc<StateSnapshot>> {
|
||||
Ok(Arc::new(StateSnapshot {
|
||||
assets: self.assets.clone(),
|
||||
assets: self.assets.snapshot(),
|
||||
config: self.config.snapshot().map_err(|err| {
|
||||
error!("{}", err);
|
||||
LspError::internal_error()
|
||||
|
@ -595,25 +601,6 @@ impl Inner {
|
|||
self.performance.measure(mark);
|
||||
Ok(())
|
||||
}
|
||||
|
||||
async fn get_asset(
|
||||
&mut self,
|
||||
specifier: &ModuleSpecifier,
|
||||
) -> LspResult<Option<AssetDocument>> {
|
||||
if let Some(maybe_asset) = self.assets.get(specifier) {
|
||||
Ok(maybe_asset.clone())
|
||||
} else {
|
||||
let maybe_asset =
|
||||
tsc::get_asset(specifier, &self.ts_server, self.snapshot()?)
|
||||
.await
|
||||
.map_err(|err| {
|
||||
error!("Error getting asset {}: {}", specifier, err);
|
||||
LspError::internal_error()
|
||||
})?;
|
||||
self.assets.insert(specifier.clone(), maybe_asset.clone());
|
||||
Ok(maybe_asset)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// lspower::LanguageServer methods. This file's LanguageServer delegates to us.
|
||||
|
@ -1078,7 +1065,7 @@ impl Inner {
|
|||
}
|
||||
}
|
||||
|
||||
async fn hover(&mut self, params: HoverParams) -> LspResult<Option<Hover>> {
|
||||
async fn hover(&self, params: HoverParams) -> LspResult<Option<Hover>> {
|
||||
let specifier = self
|
||||
.url_map
|
||||
.normalize_url(¶ms.text_document_position_params.text_document.uri);
|
||||
|
|
121
cli/lsp/tsc.rs
121
cli/lsp/tsc.rs
|
@ -26,6 +26,7 @@ use deno_core::error::custom_error;
|
|||
use deno_core::error::AnyError;
|
||||
use deno_core::located_script_name;
|
||||
use deno_core::op_sync;
|
||||
use deno_core::parking_lot::Mutex;
|
||||
use deno_core::resolve_url;
|
||||
use deno_core::serde::de;
|
||||
use deno_core::serde::Deserialize;
|
||||
|
@ -40,6 +41,7 @@ use deno_core::ModuleSpecifier;
|
|||
use deno_core::OpFn;
|
||||
use deno_core::RuntimeOptions;
|
||||
use deno_runtime::tokio_util::create_basic_runtime;
|
||||
use log::error;
|
||||
use log::warn;
|
||||
use lspower::jsonrpc::Error as LspError;
|
||||
use lspower::jsonrpc::Result as LspResult;
|
||||
|
@ -187,11 +189,9 @@ impl AssetDocument {
|
|||
}
|
||||
}
|
||||
|
||||
#[derive(Debug, Clone)]
|
||||
pub struct Assets(HashMap<ModuleSpecifier, Option<AssetDocument>>);
|
||||
type AssetsMap = HashMap<ModuleSpecifier, Option<AssetDocument>>;
|
||||
|
||||
impl Default for Assets {
|
||||
fn default() -> Self {
|
||||
fn new_assets_map() -> Arc<Mutex<AssetsMap>> {
|
||||
let assets = tsc::STATIC_ASSETS
|
||||
.iter()
|
||||
.map(|(k, v)| {
|
||||
|
@ -201,34 +201,102 @@ impl Default for Assets {
|
|||
(specifier, Some(asset))
|
||||
})
|
||||
.collect();
|
||||
Self(assets)
|
||||
Arc::new(Mutex::new(assets))
|
||||
}
|
||||
|
||||
/// Snapshot of Assets.
|
||||
#[derive(Debug, Clone)]
|
||||
pub struct AssetsSnapshot(Arc<Mutex<AssetsMap>>);
|
||||
|
||||
impl Default for AssetsSnapshot {
|
||||
fn default() -> Self {
|
||||
Self(new_assets_map())
|
||||
}
|
||||
}
|
||||
|
||||
impl Assets {
|
||||
impl AssetsSnapshot {
|
||||
pub fn contains_key(&self, k: &ModuleSpecifier) -> bool {
|
||||
self.0.contains_key(k)
|
||||
self.0.lock().contains_key(k)
|
||||
}
|
||||
|
||||
pub fn get(&self, k: &ModuleSpecifier) -> Option<&Option<AssetDocument>> {
|
||||
self.0.get(k)
|
||||
}
|
||||
|
||||
pub fn insert(
|
||||
&mut self,
|
||||
k: ModuleSpecifier,
|
||||
v: Option<AssetDocument>,
|
||||
pub fn get_cached(
|
||||
&self,
|
||||
k: &ModuleSpecifier,
|
||||
) -> Option<Option<AssetDocument>> {
|
||||
self.0.insert(k, v)
|
||||
self.0.lock().get(k).cloned()
|
||||
}
|
||||
}
|
||||
|
||||
/// Assets are never updated and so we can safely use this struct across
|
||||
/// multiple threads without needing to worry about race conditions.
|
||||
#[derive(Debug, Clone)]
|
||||
pub struct Assets {
|
||||
ts_server: Arc<TsServer>,
|
||||
assets: Arc<Mutex<AssetsMap>>,
|
||||
}
|
||||
|
||||
impl Assets {
|
||||
pub fn new(ts_server: Arc<TsServer>) -> Self {
|
||||
Self {
|
||||
ts_server,
|
||||
assets: new_assets_map(),
|
||||
}
|
||||
}
|
||||
|
||||
pub fn snapshot(&self) -> AssetsSnapshot {
|
||||
// it's ok to not make a complete copy for snapshotting purposes
|
||||
// because assets are static
|
||||
AssetsSnapshot(self.assets.clone())
|
||||
}
|
||||
|
||||
pub fn contains_key(&self, k: &ModuleSpecifier) -> bool {
|
||||
self.assets.lock().contains_key(k)
|
||||
}
|
||||
|
||||
pub fn get_cached(
|
||||
&self,
|
||||
k: &ModuleSpecifier,
|
||||
) -> Option<Option<AssetDocument>> {
|
||||
self.assets.lock().get(k).cloned()
|
||||
}
|
||||
|
||||
pub(crate) async fn get(
|
||||
&self,
|
||||
specifier: &ModuleSpecifier,
|
||||
// todo(dsherret): this shouldn't be a parameter, but instead retrieved via
|
||||
// a constructor dependency
|
||||
get_snapshot: impl Fn() -> LspResult<Arc<StateSnapshot>>,
|
||||
) -> LspResult<Option<AssetDocument>> {
|
||||
// Race conditions are ok to happen here since the assets are static
|
||||
if let Some(maybe_asset) = self.get_cached(specifier) {
|
||||
Ok(maybe_asset)
|
||||
} else {
|
||||
let maybe_asset = get_asset(specifier, &self.ts_server, get_snapshot()?)
|
||||
.await
|
||||
.map_err(|err| {
|
||||
error!("Error getting asset {}: {}", specifier, err);
|
||||
LspError::internal_error()
|
||||
})?;
|
||||
// if another thread has inserted into the cache, return the asset
|
||||
// that already exists in the cache so that we don't store duplicate
|
||||
// assets in memory anywhere
|
||||
let mut assets = self.assets.lock();
|
||||
if let Some(maybe_asset) = assets.get(specifier) {
|
||||
Ok(maybe_asset.clone())
|
||||
} else {
|
||||
assets.insert(specifier.clone(), maybe_asset.clone());
|
||||
Ok(maybe_asset)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
pub fn cache_navigation_tree(
|
||||
&mut self,
|
||||
&self,
|
||||
specifier: &ModuleSpecifier,
|
||||
navigation_tree: Arc<NavigationTree>,
|
||||
) -> Result<(), AnyError> {
|
||||
let maybe_doc = self
|
||||
.0
|
||||
let mut assets = self.assets.lock();
|
||||
let maybe_doc = assets
|
||||
.get_mut(specifier)
|
||||
.ok_or_else(|| anyhow!("Missing asset."))?;
|
||||
let doc = maybe_doc
|
||||
|
@ -242,7 +310,7 @@ impl Assets {
|
|||
/// Optionally returns an internal asset, first checking for any static assets
|
||||
/// in Rust, then checking any previously retrieved static assets from the
|
||||
/// isolate, and then finally, the tsc isolate itself.
|
||||
pub(crate) async fn get_asset(
|
||||
async fn get_asset(
|
||||
specifier: &ModuleSpecifier,
|
||||
ts_server: &TsServer,
|
||||
state_snapshot: Arc<StateSnapshot>,
|
||||
|
@ -1137,7 +1205,7 @@ pub struct FileTextChanges {
|
|||
impl FileTextChanges {
|
||||
pub(crate) async fn to_text_document_edit(
|
||||
&self,
|
||||
language_server: &mut language_server::Inner,
|
||||
language_server: &language_server::Inner,
|
||||
) -> Result<lsp::TextDocumentEdit, AnyError> {
|
||||
let specifier = normalize_specifier(&self.file_name)?;
|
||||
let asset_or_doc =
|
||||
|
@ -1158,7 +1226,7 @@ impl FileTextChanges {
|
|||
|
||||
pub(crate) async fn to_text_document_change_ops(
|
||||
&self,
|
||||
language_server: &mut language_server::Inner,
|
||||
language_server: &language_server::Inner,
|
||||
) -> Result<Vec<lsp::DocumentChangeOperation>, AnyError> {
|
||||
let mut ops = Vec::<lsp::DocumentChangeOperation>::new();
|
||||
let specifier = normalize_specifier(&self.file_name)?;
|
||||
|
@ -1394,7 +1462,7 @@ pub struct RefactorEditInfo {
|
|||
impl RefactorEditInfo {
|
||||
pub(crate) async fn to_workspace_edit(
|
||||
&self,
|
||||
language_server: &mut language_server::Inner,
|
||||
language_server: &language_server::Inner,
|
||||
) -> Result<Option<lsp::WorkspaceEdit>, AnyError> {
|
||||
let mut all_ops = Vec::<lsp::DocumentChangeOperation>::new();
|
||||
for edit in self.edits.iter() {
|
||||
|
@ -2376,7 +2444,8 @@ fn op_get_length(
|
|||
) -> Result<usize, AnyError> {
|
||||
let mark = state.performance.mark("op_get_length", Some(&args));
|
||||
let specifier = state.normalize_specifier(args.specifier)?;
|
||||
let r = if let Some(Some(asset)) = state.state_snapshot.assets.get(&specifier)
|
||||
let r = if let Some(Some(asset)) =
|
||||
state.state_snapshot.assets.get_cached(&specifier)
|
||||
{
|
||||
Ok(asset.length())
|
||||
} else {
|
||||
|
@ -2406,8 +2475,8 @@ fn op_get_text(
|
|||
) -> Result<String, AnyError> {
|
||||
let mark = state.performance.mark("op_get_text", Some(&args));
|
||||
let specifier = state.normalize_specifier(args.specifier)?;
|
||||
let content =
|
||||
if let Some(Some(content)) = state.state_snapshot.assets.get(&specifier) {
|
||||
let maybe_asset = state.state_snapshot.assets.get_cached(&specifier);
|
||||
let content = if let Some(Some(content)) = &maybe_asset {
|
||||
content.text_str()
|
||||
} else {
|
||||
cache_snapshot(state, &specifier, args.version.clone())?;
|
||||
|
|
Loading…
Reference in a new issue