1
0
Fork 0
mirror of https://github.com/denoland/deno.git synced 2024-12-22 15:24:46 -05:00

refactor(lsp): optimise static assets (#8771)

Fixes #8158
This commit is contained in:
Kitson Kelly 2020-12-16 06:34:39 +11:00 committed by GitHub
parent 6356345365
commit 892d6cc997
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
7 changed files with 142 additions and 98 deletions

View file

@ -31,8 +31,11 @@ fn get_line_index(
specifier: &ModuleSpecifier, specifier: &ModuleSpecifier,
) -> Result<Vec<u32>, AnyError> { ) -> Result<Vec<u32>, AnyError> {
let line_index = if specifier.as_url().scheme() == "asset" { let line_index = if specifier.as_url().scheme() == "asset" {
if let Some(source) = tsc::get_asset(specifier.as_url().path()) { let server_state = state.snapshot();
text::index_lines(source) if let Some(source) =
tsc::get_asset(specifier, &mut state.ts_runtime, &server_state)?
{
text::index_lines(&source)
} else { } else {
return Err(custom_error( return Err(custom_error(
"NotFound", "NotFound",
@ -256,7 +259,7 @@ pub fn handle_references(
} }
pub fn handle_virtual_text_document( pub fn handle_virtual_text_document(
state: ServerStateSnapshot, state: &mut ServerState,
params: lsp_extensions::VirtualTextDocumentParams, params: lsp_extensions::VirtualTextDocumentParams,
) -> Result<String, AnyError> { ) -> Result<String, AnyError> {
let specifier = utils::normalize_url(params.text_document.uri); let specifier = utils::normalize_url(params.text_document.uri);
@ -274,8 +277,11 @@ pub fn handle_virtual_text_document(
} else { } else {
match url.scheme() { match url.scheme() {
"asset" => { "asset" => {
if let Some(text) = tsc::get_asset(url.path()) { let server_state = state.snapshot();
text.to_string() if let Some(text) =
tsc::get_asset(&specifier, &mut state.ts_runtime, &server_state)?
{
text
} else { } else {
error!("Missing asset: {}", specifier); error!("Missing asset: {}", specifier);
"".to_string() "".to_string()

View file

@ -408,10 +408,10 @@ impl ServerState {
.on_sync::<lsp_types::request::HoverRequest>(handlers::handle_hover)? .on_sync::<lsp_types::request::HoverRequest>(handlers::handle_hover)?
.on_sync::<lsp_types::request::Completion>(handlers::handle_completion)? .on_sync::<lsp_types::request::Completion>(handlers::handle_completion)?
.on_sync::<lsp_types::request::References>(handlers::handle_references)? .on_sync::<lsp_types::request::References>(handlers::handle_references)?
.on::<lsp_types::request::Formatting>(handlers::handle_formatting) .on_sync::<lsp_extensions::VirtualTextDocument>(
.on::<lsp_extensions::VirtualTextDocument>(
handlers::handle_virtual_text_document, handlers::handle_virtual_text_document,
) )?
.on::<lsp_types::request::Formatting>(handlers::handle_formatting)
.finish(); .finish();
Ok(()) Ok(())

View file

@ -192,6 +192,7 @@ impl DocumentData {
/// An immutable snapshot of the server state at a point in time. /// An immutable snapshot of the server state at a point in time.
#[derive(Debug, Clone, Default)] #[derive(Debug, Clone, Default)]
pub struct ServerStateSnapshot { pub struct ServerStateSnapshot {
pub assets: Arc<RwLock<HashMap<ModuleSpecifier, Option<String>>>>,
pub config: Config, pub config: Config,
pub diagnostics: DiagnosticCollection, pub diagnostics: DiagnosticCollection,
pub doc_data: HashMap<ModuleSpecifier, DocumentData>, pub doc_data: HashMap<ModuleSpecifier, DocumentData>,
@ -200,6 +201,7 @@ pub struct ServerStateSnapshot {
} }
pub struct ServerState { pub struct ServerState {
pub assets: Arc<RwLock<HashMap<ModuleSpecifier, Option<String>>>>,
pub config: Config, pub config: Config,
pub diagnostics: DiagnosticCollection, pub diagnostics: DiagnosticCollection,
pub doc_data: HashMap<ModuleSpecifier, DocumentData>, pub doc_data: HashMap<ModuleSpecifier, DocumentData>,
@ -230,6 +232,7 @@ impl ServerState {
let ts_runtime = tsc::start(false).expect("could not start tsc"); let ts_runtime = tsc::start(false).expect("could not start tsc");
Self { Self {
assets: Default::default(),
config, config,
diagnostics: Default::default(), diagnostics: Default::default(),
doc_data: Default::default(), doc_data: Default::default(),
@ -315,6 +318,7 @@ impl ServerState {
pub fn snapshot(&self) -> ServerStateSnapshot { pub fn snapshot(&self) -> ServerStateSnapshot {
ServerStateSnapshot { ServerStateSnapshot {
assets: Arc::clone(&self.assets),
config: self.config.clone(), config: self.config.clone(),
diagnostics: self.diagnostics.clone(), diagnostics: self.diagnostics.clone(),
doc_data: self.doc_data.clone(), doc_data: self.doc_data.clone(),

View file

@ -7,6 +7,7 @@ use super::utils;
use crate::js; use crate::js;
use crate::media_type::MediaType; use crate::media_type::MediaType;
use crate::tsc;
use crate::tsc::ResolveArgs; use crate::tsc::ResolveArgs;
use crate::tsc_config::TsConfig; use crate::tsc_config::TsConfig;
@ -27,92 +28,26 @@ use regex::Regex;
use std::borrow::Cow; use std::borrow::Cow;
use std::collections::HashMap; use std::collections::HashMap;
/// Provide static assets for the language server. /// Optionally returns an internal asset, first checking for any static assets
/// /// in Rust, then checking any previously retrieved static assets from the
/// TODO(@kitsonk) this should be DRY'ed up with `cli/tsc.rs` and the /// isolate, and then finally, the tsc isolate itself.
/// `cli/build.rs` pub fn get_asset(
pub fn get_asset(asset: &str) -> Option<&'static str> { specifier: &ModuleSpecifier,
macro_rules! inc { runtime: &mut JsRuntime,
($e:expr) => { server_state: &ServerStateSnapshot,
Some(include_str!(concat!("../dts/", $e))) ) -> Result<Option<String>, AnyError> {
}; let specifier_str = specifier.to_string().replace("asset:///", "");
} if let Some(asset_text) = tsc::get_asset(&specifier_str) {
match asset { Ok(Some(asset_text.to_string()))
// These are not included in the snapshot } else {
"/lib.dom.d.ts" => inc!("lib.dom.d.ts"), let mut assets = server_state.assets.write().unwrap();
"/lib.dom.iterable.d.ts" => inc!("lib.dom.iterable.d.ts"), if let Some(asset) = assets.get(specifier) {
"/lib.es6.d.ts" => inc!("lib.es6.d.ts"), Ok(asset.clone())
"/lib.es2016.full.d.ts" => inc!("lib.es2016.full.d.ts"), } else {
"/lib.es2017.full.d.ts" => inc!("lib.es2017.full.d.ts"), let asset = request_asset(specifier, runtime, server_state)?;
"/lib.es2018.full.d.ts" => inc!("lib.es2018.full.d.ts"), assets.insert(specifier.clone(), asset.clone());
"/lib.es2019.full.d.ts" => inc!("lib.es2019.full.d.ts"), Ok(asset)
"/lib.es2020.full.d.ts" => inc!("lib.es2020.full.d.ts"),
"/lib.esnext.full.d.ts" => inc!("lib.esnext.full.d.ts"),
"/lib.scripthost.d.ts" => inc!("lib.scripthost.d.ts"),
"/lib.webworker.d.ts" => inc!("lib.webworker.d.ts"),
"/lib.webworker.importscripts.d.ts" => {
inc!("lib.webworker.importscripts.d.ts")
} }
"/lib.webworker.iterable.d.ts" => inc!("lib.webworker.iterable.d.ts"),
// These come from op crates
// TODO(@kitsonk) these is even hackier than the rest of this...
"/lib.deno.web.d.ts" => Some(js::DENO_WEB_LIB),
"/lib.deno.fetch.d.ts" => Some(js::DENO_FETCH_LIB),
// These are included in the snapshot for TypeScript, and could be retrieved
// from there?
"/lib.d.ts" => inc!("lib.d.ts"),
"/lib.deno.ns.d.ts" => inc!("lib.deno.ns.d.ts"),
"/lib.deno.shared_globals.d.ts" => inc!("lib.deno.shared_globals.d.ts"),
"/lib.deno.unstable.d.ts" => inc!("lib.deno.unstable.d.ts"),
"/lib.deno.window.d.ts" => inc!("lib.deno.window.d.ts"),
"/lib.deno.worker.d.ts" => inc!("lib.deno.worker.d.ts"),
"/lib.es5.d.ts" => inc!("lib.es5.d.ts"),
"/lib.es2015.collection.d.ts" => inc!("lib.es2015.collection.d.ts"),
"/lib.es2015.core.d.ts" => inc!("lib.es2015.core.d.ts"),
"/lib.es2015.d.ts" => inc!("lib.es2015.d.ts"),
"/lib.es2015.generator.d.ts" => inc!("lib.es2015.generator.d.ts"),
"/lib.es2015.iterable.d.ts" => inc!("lib.es2015.iterable.d.ts"),
"/lib.es2015.promise.d.ts" => inc!("lib.es2015.promise.d.ts"),
"/lib.es2015.proxy.d.ts" => inc!("lib.es2015.proxy.d.ts"),
"/lib.es2015.reflect.d.ts" => inc!("lib.es2015.reflect.d.ts"),
"/lib.es2015.symbol.d.ts" => inc!("lib.es2015.symbol.d.ts"),
"/lib.es2015.symbol.wellknown.d.ts" => {
inc!("lib.es2015.symbol.wellknown.d.ts")
}
"/lib.es2016.array.include.d.ts" => inc!("lib.es2016.array.include.d.ts"),
"/lib.es2016.d.ts" => inc!("lib.es2016.d.ts"),
"/lib.es2017.d.ts" => inc!("lib.es2017.d.ts"),
"/lib.es2017.intl.d.ts" => inc!("lib.es2017.intl.d.ts"),
"/lib.es2017.object.d.ts" => inc!("lib.es2017.object.d.ts"),
"/lib.es2017.sharedmemory.d.ts" => inc!("lib.es2017.sharedmemory.d.ts"),
"/lib.es2017.string.d.ts" => inc!("lib.es2017.string.d.ts"),
"/lib.es2017.typedarrays.d.ts" => inc!("lib.es2017.typedarrays.d.ts"),
"/lib.es2018.asyncgenerator.d.ts" => inc!("lib.es2018.asyncgenerator.d.ts"),
"/lib.es2018.asynciterable.d.ts" => inc!("lib.es2018.asynciterable.d.ts"),
"/lib.es2018.d.ts" => inc!("lib.es2018.d.ts"),
"/lib.es2018.intl.d.ts" => inc!("lib.es2018.intl.d.ts"),
"/lib.es2018.promise.d.ts" => inc!("lib.es2018.promise.d.ts"),
"/lib.es2018.regexp.d.ts" => inc!("lib.es2018.regexp.d.ts"),
"/lib.es2019.array.d.ts" => inc!("lib.es2019.array.d.ts"),
"/lib.es2019.d.ts" => inc!("lib.es2019.d.ts"),
"/lib.es2019.object.d.ts" => inc!("lib.es2019.object.d.ts"),
"/lib.es2019.string.d.ts" => inc!("lib.es2019.string.d.ts"),
"/lib.es2019.symbol.d.ts" => inc!("lib.es2019.symbol.d.ts"),
"/lib.es2020.bigint.d.ts" => inc!("lib.es2020.bigint.d.ts"),
"/lib.es2020.d.ts" => inc!("lib.es2020.d.ts"),
"/lib.es2020.intl.d.ts" => inc!("lib.es2020.intl.d.ts"),
"/lib.es2020.promise.d.ts" => inc!("lib.es2020.promise.d.ts"),
"/lib.es2020.sharedmemory.d.ts" => inc!("lib.es2020.sharedmemory.d.ts"),
"/lib.es2020.string.d.ts" => inc!("lib.es2020.string.d.ts"),
"/lib.es2020.symbol.wellknown.d.ts" => {
inc!("lib.es2020.symbol.wellknown.d.ts")
}
"/lib.esnext.d.ts" => inc!("lib.esnext.d.ts"),
"/lib.esnext.intl.d.ts" => inc!("lib.esnext.intl.d.ts"),
"/lib.esnext.promise.d.ts" => inc!("lib.esnext.promise.d.ts"),
"/lib.esnext.string.d.ts" => inc!("lib.esnext.string.d.ts"),
"/lib.esnext.weakref.d.ts" => inc!("lib.esnext.weakref.d.ts"),
_ => None,
} }
} }
@ -661,6 +596,7 @@ struct Response {
} }
struct State<'a> { struct State<'a> {
asset: Option<String>,
last_id: usize, last_id: usize,
response: Option<Response>, response: Option<Response>,
server_state: ServerStateSnapshot, server_state: ServerStateSnapshot,
@ -670,6 +606,7 @@ struct State<'a> {
impl<'a> State<'a> { impl<'a> State<'a> {
fn new(server_state: ServerStateSnapshot) -> Self { fn new(server_state: ServerStateSnapshot) -> Self {
Self { Self {
asset: None,
last_id: 1, last_id: 1,
response: None, response: None,
server_state, server_state,
@ -928,6 +865,18 @@ fn script_version(state: &mut State, args: Value) -> Result<Value, AnyError> {
Ok(json!(None::<String>)) Ok(json!(None::<String>))
} }
#[derive(Debug, Deserialize)]
#[serde(rename_all = "camelCase")]
struct SetAssetArgs {
text: Option<String>,
}
fn set_asset(state: &mut State, args: Value) -> Result<Value, AnyError> {
let v: SetAssetArgs = serde_json::from_value(args)?;
state.asset = v.text;
Ok(json!(true))
}
/// Create and setup a JsRuntime based on a snapshot. It is expected that the /// Create and setup a JsRuntime based on a snapshot. It is expected that the
/// supplied snapshot is an isolate that contains the TypeScript language /// supplied snapshot is an isolate that contains the TypeScript language
/// server. /// server.
@ -951,6 +900,7 @@ pub fn start(debug: bool) -> Result<JsRuntime, AnyError> {
runtime.register_op("op_respond", op(respond)); runtime.register_op("op_respond", op(respond));
runtime.register_op("op_script_names", op(script_names)); runtime.register_op("op_script_names", op(script_names));
runtime.register_op("op_script_version", op(script_version)); runtime.register_op("op_script_version", op(script_version));
runtime.register_op("op_set_asset", op(set_asset));
let init_config = json!({ "debug": debug }); let init_config = json!({ "debug": debug });
let init_src = format!("globalThis.serverInit({});", init_config); let init_src = format!("globalThis.serverInit({});", init_config);
@ -1028,6 +978,8 @@ pub struct UserPreferences {
pub enum RequestMethod { pub enum RequestMethod {
/// Configure the compilation settings for the server. /// Configure the compilation settings for the server.
Configure(TsConfig), Configure(TsConfig),
/// Retrieve the text of an assets that exists in memory in the isolate.
GetAsset(ModuleSpecifier),
/// Return semantic diagnostics for given file. /// Return semantic diagnostics for given file.
GetSemanticDiagnostics(ModuleSpecifier), GetSemanticDiagnostics(ModuleSpecifier),
/// Returns suggestion diagnostics for given file. /// Returns suggestion diagnostics for given file.
@ -1054,6 +1006,11 @@ impl RequestMethod {
"method": "configure", "method": "configure",
"compilerOptions": config, "compilerOptions": config,
}), }),
RequestMethod::GetAsset(specifier) => json!({
"id": id,
"method": "getAsset",
"specifier": specifier,
}),
RequestMethod::GetSemanticDiagnostics(specifier) => json!({ RequestMethod::GetSemanticDiagnostics(specifier) => json!({
"id": id, "id": id,
"method": "getSemanticDiagnostics", "method": "getSemanticDiagnostics",
@ -1144,6 +1101,30 @@ pub fn request(
} }
} }
fn request_asset(
specifier: &ModuleSpecifier,
runtime: &mut JsRuntime,
server_state: &ServerStateSnapshot,
) -> Result<Option<String>, AnyError> {
let id = {
let op_state = runtime.op_state();
let mut op_state = op_state.borrow_mut();
let state = op_state.borrow_mut::<State>();
state.server_state = server_state.clone();
state.last_id += 1;
state.last_id
};
let request_params = RequestMethod::GetAsset(specifier.clone()).to_value(id);
let request_src = format!("globalThis.serverRequest({});", request_params);
runtime.execute("[native_code]", &request_src)?;
let op_state = runtime.op_state();
let mut op_state = op_state.borrow_mut();
let state = op_state.borrow_mut::<State>();
Ok(state.asset.clone())
}
#[cfg(test)] #[cfg(test)]
mod tests { mod tests {
use super::super::memory_cache::MemoryCache; use super::super::memory_cache::MemoryCache;
@ -1167,6 +1148,7 @@ mod tests {
} }
let file_cache = Arc::new(RwLock::new(file_cache)); let file_cache = Arc::new(RwLock::new(file_cache));
ServerStateSnapshot { ServerStateSnapshot {
assets: Default::default(),
config: Default::default(), config: Default::default(),
diagnostics: Default::default(), diagnostics: Default::default(),
doc_data, doc_data,
@ -1422,9 +1404,45 @@ mod tests {
&server_state, &server_state,
RequestMethod::GetSyntacticDiagnostics(specifier), RequestMethod::GetSyntacticDiagnostics(specifier),
); );
println!("{:?}", result); assert!(result.is_ok());
// assert!(result.is_ok()); let response = result.unwrap();
// let response = result.unwrap(); assert_eq!(
// assert_eq!(response, json!([])); response,
json!([{
"start": {
"line": 8,
"character": 29
},
"end": {
"line": 8,
"character": 29
},
"fileName": "file:///a.ts",
"messageText": "Expression expected.",
"sourceLine": " import * as test from",
"category": 1,
"code": 1109
}])
);
}
#[test]
fn test_request_asset() {
let (mut runtime, server_state) = setup(
false,
json!({
"target": "esnext",
"module": "esnext",
"lib": ["deno.ns", "deno.window"],
"noEmit": true,
}),
vec![],
);
let specifier = ModuleSpecifier::resolve_url("asset:///lib.esnext.d.ts")
.expect("could not resolve url");
let result = request_asset(&specifier, &mut runtime, &server_state);
assert!(result.is_ok());
let response = result.unwrap();
assert!(response.is_some());
} }
} }

View file

@ -26,7 +26,7 @@ use std::path::PathBuf;
use std::rc::Rc; use std::rc::Rc;
/// Provide static assets that are not preloaded in the compiler snapshot. /// Provide static assets that are not preloaded in the compiler snapshot.
fn get_asset(asset: &str) -> Option<&'static str> { pub fn get_asset(asset: &str) -> Option<&'static str> {
macro_rules! inc { macro_rules! inc {
($e:expr) => { ($e:expr) => {
Some(include_str!(concat!("dts/", $e))) Some(include_str!(concat!("dts/", $e)))

View file

@ -487,6 +487,16 @@ delete Object.prototype.__proto__;
compilationSettings = options; compilationSettings = options;
return respond(id, true); return respond(id, true);
} }
case "getAsset": {
const sourceFile = host.getSourceFile(
request.specifier,
ts.ScriptTarget.ESNext,
);
return core.jsonOpSync(
"op_set_asset",
{ text: sourceFile && sourceFile.text },
);
}
case "getSemanticDiagnostics": { case "getSemanticDiagnostics": {
const diagnostics = languageService.getSemanticDiagnostics( const diagnostics = languageService.getSemanticDiagnostics(
request.specifier, request.specifier,

View file

@ -42,6 +42,7 @@ declare global {
type LanguageServerRequest = type LanguageServerRequest =
| ConfigureRequest | ConfigureRequest
| GetAsset
| GetSyntacticDiagnosticsRequest | GetSyntacticDiagnosticsRequest
| GetSemanticDiagnosticsRequest | GetSemanticDiagnosticsRequest
| GetSuggestionDiagnosticsRequest | GetSuggestionDiagnosticsRequest
@ -62,6 +63,11 @@ declare global {
compilerOptions: Record<string, any>; compilerOptions: Record<string, any>;
} }
interface GetAsset extends BaseLanguageServerRequest {
method: "getAsset";
specifier: string;
}
interface GetSyntacticDiagnosticsRequest extends BaseLanguageServerRequest { interface GetSyntacticDiagnosticsRequest extends BaseLanguageServerRequest {
method: "getSyntacticDiagnostics"; method: "getSyntacticDiagnostics";
specifier: string; specifier: string;