2023-01-02 16:00:42 -05:00
|
|
|
// Copyright 2018-2023 the Deno authors. All rights reserved. MIT license.
|
2020-12-07 21:46:39 +11:00
|
|
|
|
2022-03-21 12:33:37 +11:00
|
|
|
use super::client::Client;
|
|
|
|
use super::logging::lsp_log;
|
2022-11-28 17:28:54 -05:00
|
|
|
use crate::util::path::ensure_directory_specifier;
|
|
|
|
use crate::util::path::specifier_to_file_path;
|
2021-05-20 19:56:48 +10:00
|
|
|
use deno_core::error::AnyError;
|
2020-12-07 21:46:39 +11:00
|
|
|
use deno_core::serde::Deserialize;
|
2021-07-21 11:50:43 +10:00
|
|
|
use deno_core::serde::Serialize;
|
2020-12-07 21:46:39 +11:00
|
|
|
use deno_core::serde_json;
|
|
|
|
use deno_core::serde_json::Value;
|
2021-05-10 11:16:04 +10:00
|
|
|
use deno_core::ModuleSpecifier;
|
2021-05-20 19:56:48 +10:00
|
|
|
use std::collections::BTreeMap;
|
2021-04-09 11:27:27 +10:00
|
|
|
use std::collections::HashMap;
|
2021-05-20 19:56:48 +10:00
|
|
|
use std::sync::Arc;
|
2022-04-03 12:17:30 +08:00
|
|
|
use tower_lsp::lsp_types as lsp;
|
2021-12-15 13:23:43 -05:00
|
|
|
|
2021-05-10 11:16:04 +10:00
|
|
|
pub const SETTINGS_SECTION: &str = "deno";
|
|
|
|
|
2020-12-07 21:46:39 +11:00
|
|
|
#[derive(Debug, Clone, Default)]
|
|
|
|
pub struct ClientCapabilities {
|
2021-08-10 09:56:34 +10:00
|
|
|
pub code_action_disabled_support: bool,
|
|
|
|
pub line_folding_only: bool,
|
2022-10-14 23:04:38 +11:00
|
|
|
pub snippet_support: bool,
|
2020-12-07 21:46:39 +11:00
|
|
|
pub status_notification: bool,
|
2022-03-30 09:59:27 +11:00
|
|
|
/// The client provides the `experimental.testingApi` capability, which is
|
|
|
|
/// built around VSCode's testing API. It indicates that the server should
|
|
|
|
/// send notifications about tests discovered in modules.
|
|
|
|
pub testing_api: bool,
|
2021-01-04 22:52:20 +01:00
|
|
|
pub workspace_configuration: bool,
|
|
|
|
pub workspace_did_change_watched_files: bool,
|
2020-12-07 21:46:39 +11:00
|
|
|
}
|
|
|
|
|
2021-06-07 21:38:07 +10:00
|
|
|
fn is_true() -> bool {
|
|
|
|
true
|
|
|
|
}
|
|
|
|
|
2021-07-21 11:50:43 +10:00
|
|
|
#[derive(Debug, Clone, Deserialize, Serialize, PartialEq, Eq)]
|
2021-02-01 14:30:41 +11:00
|
|
|
#[serde(rename_all = "camelCase")]
|
|
|
|
pub struct CodeLensSettings {
|
2021-02-08 21:45:10 +11:00
|
|
|
/// Flag for providing implementation code lenses.
|
|
|
|
#[serde(default)]
|
|
|
|
pub implementations: bool,
|
|
|
|
/// Flag for providing reference code lenses.
|
2021-02-01 14:30:41 +11:00
|
|
|
#[serde(default)]
|
|
|
|
pub references: bool,
|
|
|
|
/// Flag for providing reference code lens on all functions. For this to have
|
|
|
|
/// an impact, the `references` flag needs to be `true`.
|
|
|
|
#[serde(default)]
|
|
|
|
pub references_all_functions: bool,
|
2021-06-07 21:38:07 +10:00
|
|
|
/// Flag for providing test code lens on `Deno.test` statements. There is
|
|
|
|
/// also the `test_args` setting, but this is not used by the server.
|
|
|
|
#[serde(default = "is_true")]
|
|
|
|
pub test: bool,
|
2021-02-01 14:30:41 +11:00
|
|
|
}
|
|
|
|
|
2021-03-16 09:01:41 +11:00
|
|
|
impl Default for CodeLensSettings {
|
|
|
|
fn default() -> Self {
|
|
|
|
Self {
|
|
|
|
implementations: false,
|
|
|
|
references: false,
|
|
|
|
references_all_functions: false,
|
2021-06-07 21:38:07 +10:00
|
|
|
test: true,
|
2021-03-16 09:01:41 +11:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-06-07 21:38:07 +10:00
|
|
|
#[derive(Debug, Clone, Deserialize, PartialEq, Eq)]
|
|
|
|
#[serde(rename_all = "camelCase")]
|
|
|
|
pub struct CodeLensSpecifierSettings {
|
|
|
|
/// Flag for providing test code lens on `Deno.test` statements. There is
|
|
|
|
/// also the `test_args` setting, but this is not used by the server.
|
|
|
|
#[serde(default = "is_true")]
|
|
|
|
pub test: bool,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Default for CodeLensSpecifierSettings {
|
|
|
|
fn default() -> Self {
|
|
|
|
Self { test: true }
|
|
|
|
}
|
2021-06-01 21:53:08 +10:00
|
|
|
}
|
|
|
|
|
2021-07-21 11:50:43 +10:00
|
|
|
#[derive(Debug, Clone, Deserialize, Serialize, PartialEq, Eq)]
|
2021-03-16 09:01:41 +11:00
|
|
|
#[serde(rename_all = "camelCase")]
|
|
|
|
pub struct CompletionSettings {
|
|
|
|
#[serde(default)]
|
|
|
|
pub complete_function_calls: bool,
|
2021-06-01 21:53:08 +10:00
|
|
|
#[serde(default = "is_true")]
|
2021-03-16 09:01:41 +11:00
|
|
|
pub names: bool,
|
2021-06-01 21:53:08 +10:00
|
|
|
#[serde(default = "is_true")]
|
2021-03-16 09:01:41 +11:00
|
|
|
pub paths: bool,
|
2021-06-01 21:53:08 +10:00
|
|
|
#[serde(default = "is_true")]
|
2021-03-16 09:01:41 +11:00
|
|
|
pub auto_imports: bool,
|
2021-04-09 11:27:27 +10:00
|
|
|
#[serde(default)]
|
|
|
|
pub imports: ImportCompletionSettings,
|
2021-03-16 09:01:41 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
impl Default for CompletionSettings {
|
|
|
|
fn default() -> Self {
|
|
|
|
Self {
|
|
|
|
complete_function_calls: false,
|
|
|
|
names: true,
|
|
|
|
paths: true,
|
|
|
|
auto_imports: true,
|
2021-04-09 11:27:27 +10:00
|
|
|
imports: ImportCompletionSettings::default(),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-10-16 13:39:43 +11:00
|
|
|
#[derive(Debug, Default, Clone, Deserialize, Serialize, PartialEq, Eq)]
|
|
|
|
#[serde(rename_all = "camelCase")]
|
|
|
|
pub struct InlayHintsSettings {
|
|
|
|
#[serde(default)]
|
|
|
|
pub parameter_names: InlayHintsParamNamesOptions,
|
|
|
|
#[serde(default)]
|
|
|
|
pub parameter_types: InlayHintsParamTypesOptions,
|
|
|
|
#[serde(default)]
|
|
|
|
pub variable_types: InlayHintsVarTypesOptions,
|
|
|
|
#[serde(default)]
|
|
|
|
pub property_declaration_types: InlayHintsPropDeclTypesOptions,
|
|
|
|
#[serde(default)]
|
|
|
|
pub function_like_return_types: InlayHintsFuncLikeReturnTypesOptions,
|
|
|
|
#[serde(default)]
|
|
|
|
pub enum_member_values: InlayHintsEnumMemberValuesOptions,
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Debug, Clone, Deserialize, Serialize, PartialEq, Eq)]
|
|
|
|
#[serde(rename_all = "camelCase")]
|
|
|
|
pub struct InlayHintsParamNamesOptions {
|
|
|
|
#[serde(default)]
|
|
|
|
pub enabled: InlayHintsParamNamesEnabled,
|
|
|
|
#[serde(default = "is_true")]
|
|
|
|
pub suppress_when_argument_matches_name: bool,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Default for InlayHintsParamNamesOptions {
|
|
|
|
fn default() -> Self {
|
|
|
|
Self {
|
|
|
|
enabled: InlayHintsParamNamesEnabled::None,
|
|
|
|
suppress_when_argument_matches_name: true,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Debug, Clone, Deserialize, Serialize, PartialEq, Eq)]
|
|
|
|
#[serde(rename_all = "camelCase")]
|
|
|
|
pub enum InlayHintsParamNamesEnabled {
|
|
|
|
None,
|
|
|
|
Literals,
|
|
|
|
All,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Default for InlayHintsParamNamesEnabled {
|
|
|
|
fn default() -> Self {
|
|
|
|
Self::None
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Debug, Default, Clone, Deserialize, Serialize, PartialEq, Eq)]
|
|
|
|
#[serde(rename_all = "camelCase")]
|
|
|
|
pub struct InlayHintsParamTypesOptions {
|
|
|
|
#[serde(default)]
|
|
|
|
pub enabled: bool,
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Debug, Clone, Deserialize, Serialize, PartialEq, Eq)]
|
|
|
|
#[serde(rename_all = "camelCase")]
|
|
|
|
pub struct InlayHintsVarTypesOptions {
|
|
|
|
#[serde(default)]
|
|
|
|
pub enabled: bool,
|
|
|
|
#[serde(default = "is_true")]
|
2022-10-28 14:48:14 -04:00
|
|
|
pub suppress_when_type_matches_name: bool,
|
2022-10-16 13:39:43 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
impl Default for InlayHintsVarTypesOptions {
|
|
|
|
fn default() -> Self {
|
|
|
|
Self {
|
|
|
|
enabled: false,
|
2022-10-28 14:48:14 -04:00
|
|
|
suppress_when_type_matches_name: true,
|
2022-10-16 13:39:43 +11:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Debug, Default, Clone, Deserialize, Serialize, PartialEq, Eq)]
|
|
|
|
#[serde(rename_all = "camelCase")]
|
|
|
|
pub struct InlayHintsPropDeclTypesOptions {
|
|
|
|
#[serde(default)]
|
|
|
|
pub enabled: bool,
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Debug, Default, Clone, Deserialize, Serialize, PartialEq, Eq)]
|
|
|
|
#[serde(rename_all = "camelCase")]
|
|
|
|
pub struct InlayHintsFuncLikeReturnTypesOptions {
|
|
|
|
#[serde(default)]
|
|
|
|
pub enabled: bool,
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Debug, Default, Clone, Deserialize, Serialize, PartialEq, Eq)]
|
|
|
|
#[serde(rename_all = "camelCase")]
|
|
|
|
pub struct InlayHintsEnumMemberValuesOptions {
|
|
|
|
#[serde(default)]
|
|
|
|
pub enabled: bool,
|
|
|
|
}
|
|
|
|
|
2021-07-21 11:50:43 +10:00
|
|
|
#[derive(Debug, Clone, Deserialize, Serialize, PartialEq, Eq)]
|
2021-04-09 11:27:27 +10:00
|
|
|
#[serde(rename_all = "camelCase")]
|
|
|
|
pub struct ImportCompletionSettings {
|
2021-06-01 21:53:08 +10:00
|
|
|
/// A flag that indicates if non-explicitly set origins should be checked for
|
|
|
|
/// supporting import suggestions.
|
|
|
|
#[serde(default = "is_true")]
|
|
|
|
pub auto_discover: bool,
|
|
|
|
/// A map of origins which have had explicitly set if import suggestions are
|
|
|
|
/// enabled.
|
2021-04-09 11:27:27 +10:00
|
|
|
#[serde(default)]
|
|
|
|
pub hosts: HashMap<String, bool>,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Default for ImportCompletionSettings {
|
|
|
|
fn default() -> Self {
|
|
|
|
Self {
|
2021-06-01 21:53:08 +10:00
|
|
|
auto_discover: true,
|
2021-04-09 11:27:27 +10:00
|
|
|
hosts: HashMap::default(),
|
2021-03-16 09:01:41 +11:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-05-10 11:16:04 +10:00
|
|
|
/// Deno language server specific settings that can be applied uniquely to a
|
|
|
|
/// specifier.
|
|
|
|
#[derive(Debug, Default, Clone, Deserialize)]
|
2021-06-07 21:38:07 +10:00
|
|
|
#[serde(rename_all = "camelCase")]
|
2021-05-10 11:16:04 +10:00
|
|
|
pub struct SpecifierSettings {
|
|
|
|
/// A flag that indicates if Deno is enabled for this specifier or not.
|
|
|
|
pub enable: bool,
|
2022-03-21 12:33:37 +11:00
|
|
|
/// A list of paths, using the workspace folder as a base that should be Deno
|
|
|
|
/// enabled.
|
|
|
|
#[serde(default)]
|
|
|
|
pub enable_paths: Vec<String>,
|
2021-06-07 21:38:07 +10:00
|
|
|
/// Code lens specific settings for the resource.
|
|
|
|
#[serde(default)]
|
|
|
|
pub code_lens: CodeLensSpecifierSettings,
|
2021-05-10 11:16:04 +10:00
|
|
|
}
|
|
|
|
|
2022-03-30 09:59:27 +11:00
|
|
|
#[derive(Debug, Clone, Deserialize, Serialize, PartialEq, Eq)]
|
|
|
|
#[serde(rename_all = "camelCase")]
|
|
|
|
pub struct TestingSettings {
|
|
|
|
/// A vector of arguments which should be used when running the tests for
|
|
|
|
/// a workspace.
|
|
|
|
#[serde(default)]
|
|
|
|
pub args: Vec<String>,
|
|
|
|
/// Enable or disable the testing API if the client is capable of supporting
|
|
|
|
/// the testing API.
|
|
|
|
#[serde(default = "is_true")]
|
|
|
|
pub enable: bool,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Default for TestingSettings {
|
|
|
|
fn default() -> Self {
|
|
|
|
Self {
|
|
|
|
args: vec!["--allow-all".to_string(), "--no-check".to_string()],
|
|
|
|
enable: true,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-05-16 13:10:08 +02:00
|
|
|
fn default_to_true() -> bool {
|
|
|
|
true
|
|
|
|
}
|
|
|
|
|
2023-01-03 16:59:48 +01:00
|
|
|
fn empty_string_none<'de, D: serde::Deserializer<'de>>(
|
|
|
|
d: D,
|
|
|
|
) -> Result<Option<String>, D::Error> {
|
|
|
|
let o: Option<String> = Option::deserialize(d)?;
|
|
|
|
Ok(o.filter(|s| !s.is_empty()))
|
|
|
|
}
|
|
|
|
|
2021-05-10 11:16:04 +10:00
|
|
|
/// Deno language server specific settings that are applied to a workspace.
|
2023-01-03 16:59:48 +01:00
|
|
|
#[derive(Debug, Clone, Deserialize, Serialize, PartialEq, Eq)]
|
2020-12-07 21:46:39 +11:00
|
|
|
#[serde(rename_all = "camelCase")]
|
|
|
|
pub struct WorkspaceSettings {
|
2021-05-10 11:16:04 +10:00
|
|
|
/// A flag that indicates if Deno is enabled for the workspace.
|
2021-06-01 21:53:08 +10:00
|
|
|
#[serde(default)]
|
2020-12-07 21:46:39 +11:00
|
|
|
pub enable: bool,
|
2021-05-10 11:16:04 +10:00
|
|
|
|
2022-03-21 12:33:37 +11:00
|
|
|
/// A list of paths, using the root_uri as a base that should be Deno enabled.
|
|
|
|
#[serde(default)]
|
|
|
|
pub enable_paths: Vec<String>,
|
|
|
|
|
2021-07-28 07:25:09 +10:00
|
|
|
/// An option that points to a path string of the path to utilise as the
|
|
|
|
/// cache/DENO_DIR for the language server.
|
2023-01-03 16:59:48 +01:00
|
|
|
#[serde(default, deserialize_with = "empty_string_none")]
|
2021-07-28 07:25:09 +10:00
|
|
|
pub cache: Option<String>,
|
|
|
|
|
2022-01-24 11:27:52 +11:00
|
|
|
/// Override the default stores used to validate certificates. This overrides
|
|
|
|
/// the environment variable `DENO_TLS_CA_STORE` if present.
|
|
|
|
pub certificate_stores: Option<Vec<String>>,
|
|
|
|
|
2021-05-10 18:16:39 +02:00
|
|
|
/// An option that points to a path string of the config file to apply to
|
2021-05-10 11:16:04 +10:00
|
|
|
/// code within the workspace.
|
2023-01-03 16:59:48 +01:00
|
|
|
#[serde(default, deserialize_with = "empty_string_none")]
|
2020-12-07 21:46:39 +11:00
|
|
|
pub config: Option<String>,
|
2021-05-10 11:16:04 +10:00
|
|
|
|
|
|
|
/// An option that points to a path string of the import map to apply to the
|
|
|
|
/// code within the workspace.
|
2023-01-03 16:59:48 +01:00
|
|
|
#[serde(default, deserialize_with = "empty_string_none")]
|
2020-12-07 21:46:39 +11:00
|
|
|
pub import_map: Option<String>,
|
2021-05-10 11:16:04 +10:00
|
|
|
|
|
|
|
/// Code lens specific settings for the workspace.
|
2021-03-16 09:01:41 +11:00
|
|
|
#[serde(default)]
|
|
|
|
pub code_lens: CodeLensSettings,
|
2021-05-10 11:16:04 +10:00
|
|
|
|
2022-10-16 13:39:43 +11:00
|
|
|
#[serde(default)]
|
|
|
|
pub inlay_hints: InlayHintsSettings,
|
|
|
|
|
2021-05-11 14:54:10 +10:00
|
|
|
/// A flag that indicates if internal debug logging should be made available.
|
|
|
|
#[serde(default)]
|
|
|
|
pub internal_debug: bool,
|
2021-01-04 22:52:20 +01:00
|
|
|
|
2021-05-10 11:16:04 +10:00
|
|
|
/// A flag that indicates if linting is enabled for the workspace.
|
2022-05-16 13:10:08 +02:00
|
|
|
#[serde(default = "default_to_true")]
|
2020-12-07 21:46:39 +11:00
|
|
|
pub lint: bool,
|
2021-05-10 11:16:04 +10:00
|
|
|
|
|
|
|
/// A flag that indicates if Dene should validate code against the unstable
|
|
|
|
/// APIs for the workspace.
|
2021-05-11 14:54:10 +10:00
|
|
|
#[serde(default)]
|
|
|
|
pub suggest: CompletionSettings,
|
|
|
|
|
2022-03-30 09:59:27 +11:00
|
|
|
/// Testing settings for the workspace.
|
|
|
|
#[serde(default)]
|
|
|
|
pub testing: TestingSettings,
|
|
|
|
|
2022-01-24 11:27:52 +11:00
|
|
|
/// An option which sets the cert file to use when attempting to fetch remote
|
|
|
|
/// resources. This overrides `DENO_CERT` if present.
|
2023-01-03 16:59:48 +01:00
|
|
|
#[serde(default, deserialize_with = "empty_string_none")]
|
2022-01-24 11:27:52 +11:00
|
|
|
pub tls_certificate: Option<String>,
|
|
|
|
|
|
|
|
/// An option, if set, will unsafely ignore certificate errors when fetching
|
|
|
|
/// remote resources.
|
|
|
|
#[serde(default)]
|
|
|
|
pub unsafely_ignore_certificate_errors: Option<Vec<String>>,
|
|
|
|
|
2021-01-04 22:52:20 +01:00
|
|
|
#[serde(default)]
|
2020-12-07 21:46:39 +11:00
|
|
|
pub unstable: bool,
|
|
|
|
}
|
|
|
|
|
2023-01-03 16:59:48 +01:00
|
|
|
impl Default for WorkspaceSettings {
|
|
|
|
fn default() -> Self {
|
|
|
|
WorkspaceSettings {
|
|
|
|
enable: false,
|
|
|
|
enable_paths: vec![],
|
|
|
|
cache: None,
|
|
|
|
certificate_stores: None,
|
|
|
|
config: None,
|
|
|
|
import_map: None,
|
|
|
|
code_lens: Default::default(),
|
|
|
|
inlay_hints: Default::default(),
|
|
|
|
internal_debug: false,
|
|
|
|
lint: true,
|
|
|
|
suggest: Default::default(),
|
|
|
|
testing: Default::default(),
|
|
|
|
tls_certificate: None,
|
|
|
|
unsafely_ignore_certificate_errors: None,
|
|
|
|
unstable: false,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-02-01 14:30:41 +11:00
|
|
|
impl WorkspaceSettings {
|
|
|
|
/// Determine if any code lenses are enabled at all. This allows short
|
|
|
|
/// circuiting when there are no code lenses enabled.
|
|
|
|
pub fn enabled_code_lens(&self) -> bool {
|
2021-03-16 09:01:41 +11:00
|
|
|
self.code_lens.implementations || self.code_lens.references
|
2021-02-01 14:30:41 +11:00
|
|
|
}
|
2022-10-16 13:39:43 +11:00
|
|
|
|
|
|
|
/// Determine if any inlay hints are enabled. This allows short circuiting
|
|
|
|
/// when there are no inlay hints enabled.
|
|
|
|
pub fn enabled_inlay_hints(&self) -> bool {
|
|
|
|
!matches!(
|
|
|
|
self.inlay_hints.parameter_names.enabled,
|
|
|
|
InlayHintsParamNamesEnabled::None
|
|
|
|
) || self.inlay_hints.parameter_types.enabled
|
|
|
|
|| self.inlay_hints.variable_types.enabled
|
|
|
|
|| self.inlay_hints.property_declaration_types.enabled
|
|
|
|
|| self.inlay_hints.function_like_return_types.enabled
|
|
|
|
|| self.inlay_hints.enum_member_values.enabled
|
|
|
|
}
|
2021-02-01 14:30:41 +11:00
|
|
|
}
|
|
|
|
|
2021-05-20 19:56:48 +10:00
|
|
|
#[derive(Debug, Clone, Default)]
|
|
|
|
pub struct ConfigSnapshot {
|
|
|
|
pub client_capabilities: ClientCapabilities,
|
2022-03-21 12:33:37 +11:00
|
|
|
pub enabled_paths: HashMap<String, Vec<String>>,
|
2021-05-20 19:56:48 +10:00
|
|
|
pub settings: Settings,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl ConfigSnapshot {
|
2022-03-21 12:33:37 +11:00
|
|
|
/// Determine if the provided specifier is enabled or not.
|
2021-05-20 19:56:48 +10:00
|
|
|
pub fn specifier_enabled(&self, specifier: &ModuleSpecifier) -> bool {
|
2022-03-21 12:33:37 +11:00
|
|
|
if !self.enabled_paths.is_empty() {
|
|
|
|
let specifier_str = specifier.to_string();
|
|
|
|
for (workspace, enabled_paths) in self.enabled_paths.iter() {
|
|
|
|
if specifier_str.starts_with(workspace) {
|
|
|
|
return enabled_paths
|
|
|
|
.iter()
|
|
|
|
.any(|path| specifier_str.starts_with(path));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if let Some((_, SpecifierSettings { enable, .. })) =
|
|
|
|
self.settings.specifiers.get(specifier)
|
|
|
|
{
|
|
|
|
*enable
|
2021-05-20 19:56:48 +10:00
|
|
|
} else {
|
|
|
|
self.settings.workspace.enable
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-01-25 10:30:38 -05:00
|
|
|
#[derive(Debug, Clone)]
|
|
|
|
pub struct SpecifierWithClientUri {
|
|
|
|
pub specifier: ModuleSpecifier,
|
|
|
|
pub client_uri: ModuleSpecifier,
|
|
|
|
}
|
|
|
|
|
2021-03-10 13:41:35 +11:00
|
|
|
#[derive(Debug, Default, Clone)]
|
2021-05-20 19:56:48 +10:00
|
|
|
pub struct Settings {
|
|
|
|
pub specifiers:
|
|
|
|
BTreeMap<ModuleSpecifier, (ModuleSpecifier, SpecifierSettings)>,
|
|
|
|
pub workspace: WorkspaceSettings,
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Debug)]
|
2020-12-07 21:46:39 +11:00
|
|
|
pub struct Config {
|
|
|
|
pub client_capabilities: ClientCapabilities,
|
2022-03-21 12:33:37 +11:00
|
|
|
enabled_paths: HashMap<String, Vec<String>>,
|
|
|
|
pub root_uri: Option<ModuleSpecifier>,
|
2022-01-25 10:30:38 -05:00
|
|
|
settings: Settings,
|
2022-03-21 12:33:37 +11:00
|
|
|
pub workspace_folders: Option<Vec<(ModuleSpecifier, lsp::WorkspaceFolder)>>,
|
2020-12-07 21:46:39 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
impl Config {
|
2022-01-25 10:30:38 -05:00
|
|
|
pub fn new() -> Self {
|
2021-05-20 19:56:48 +10:00
|
|
|
Self {
|
|
|
|
client_capabilities: ClientCapabilities::default(),
|
2022-03-21 12:33:37 +11:00
|
|
|
enabled_paths: Default::default(),
|
|
|
|
/// Root provided by the initialization parameters.
|
|
|
|
root_uri: None,
|
2022-01-25 10:30:38 -05:00
|
|
|
settings: Default::default(),
|
2021-06-22 07:18:32 +10:00
|
|
|
workspace_folders: None,
|
2021-05-20 19:56:48 +10:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn get_workspace_settings(&self) -> WorkspaceSettings {
|
2022-01-25 10:30:38 -05:00
|
|
|
self.settings.workspace.clone()
|
2021-05-20 19:56:48 +10:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Set the workspace settings directly, which occurs during initialization
|
|
|
|
/// and when the client does not support workspace configuration requests
|
2022-01-25 10:30:38 -05:00
|
|
|
pub fn set_workspace_settings(
|
|
|
|
&mut self,
|
|
|
|
value: Value,
|
|
|
|
) -> Result<(), AnyError> {
|
2021-05-20 19:56:48 +10:00
|
|
|
let workspace_settings = serde_json::from_value(value)?;
|
2022-01-25 10:30:38 -05:00
|
|
|
self.settings.workspace = workspace_settings;
|
2021-05-20 19:56:48 +10:00
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2022-01-19 17:10:14 -05:00
|
|
|
pub fn snapshot(&self) -> Arc<ConfigSnapshot> {
|
|
|
|
Arc::new(ConfigSnapshot {
|
2021-05-20 19:56:48 +10:00
|
|
|
client_capabilities: self.client_capabilities.clone(),
|
2022-03-21 12:33:37 +11:00
|
|
|
enabled_paths: self.enabled_paths.clone(),
|
2022-01-25 10:30:38 -05:00
|
|
|
settings: self.settings.clone(),
|
2021-05-20 19:56:48 +10:00
|
|
|
})
|
2021-05-10 11:16:04 +10:00
|
|
|
}
|
|
|
|
|
2022-01-25 10:30:38 -05:00
|
|
|
pub fn has_specifier_settings(&self, specifier: &ModuleSpecifier) -> bool {
|
|
|
|
self.settings.specifiers.contains_key(specifier)
|
|
|
|
}
|
|
|
|
|
2021-05-10 11:16:04 +10:00
|
|
|
pub fn specifier_enabled(&self, specifier: &ModuleSpecifier) -> bool {
|
2022-03-21 12:33:37 +11:00
|
|
|
if !self.enabled_paths.is_empty() {
|
|
|
|
let specifier_str = specifier.to_string();
|
|
|
|
for (workspace, enabled_paths) in self.enabled_paths.iter() {
|
|
|
|
if specifier_str.starts_with(workspace) {
|
|
|
|
return enabled_paths
|
|
|
|
.iter()
|
|
|
|
.any(|path| specifier_str.starts_with(path));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2022-01-25 10:30:38 -05:00
|
|
|
self
|
|
|
|
.settings
|
2021-06-07 21:38:07 +10:00
|
|
|
.specifiers
|
|
|
|
.get(specifier)
|
|
|
|
.map(|(_, s)| s.enable)
|
2022-01-25 10:30:38 -05:00
|
|
|
.unwrap_or_else(|| self.settings.workspace.enable)
|
2021-06-07 21:38:07 +10:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn specifier_code_lens_test(&self, specifier: &ModuleSpecifier) -> bool {
|
2022-01-25 10:30:38 -05:00
|
|
|
let value = self
|
|
|
|
.settings
|
2021-06-07 21:38:07 +10:00
|
|
|
.specifiers
|
|
|
|
.get(specifier)
|
|
|
|
.map(|(_, s)| s.code_lens.test)
|
2022-01-25 10:30:38 -05:00
|
|
|
.unwrap_or_else(|| self.settings.workspace.code_lens.test);
|
2021-06-07 21:38:07 +10:00
|
|
|
value
|
2020-12-07 21:46:39 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn update_capabilities(
|
|
|
|
&mut self,
|
2021-01-29 12:34:33 -07:00
|
|
|
capabilities: &lsp::ClientCapabilities,
|
2020-12-07 21:46:39 +11:00
|
|
|
) {
|
|
|
|
if let Some(experimental) = &capabilities.experimental {
|
2021-06-15 13:22:28 -04:00
|
|
|
self.client_capabilities.status_notification = experimental
|
|
|
|
.get("statusNotification")
|
|
|
|
.and_then(|it| it.as_bool())
|
2022-03-30 09:59:27 +11:00
|
|
|
== Some(true);
|
|
|
|
self.client_capabilities.testing_api =
|
|
|
|
experimental.get("testingApi").and_then(|it| it.as_bool())
|
|
|
|
== Some(true);
|
2020-12-07 21:46:39 +11:00
|
|
|
}
|
2021-01-04 22:52:20 +01:00
|
|
|
|
|
|
|
if let Some(workspace) = &capabilities.workspace {
|
|
|
|
self.client_capabilities.workspace_configuration =
|
|
|
|
workspace.configuration.unwrap_or(false);
|
|
|
|
self.client_capabilities.workspace_did_change_watched_files = workspace
|
|
|
|
.did_change_watched_files
|
|
|
|
.and_then(|it| it.dynamic_registration)
|
|
|
|
.unwrap_or(false);
|
|
|
|
}
|
2021-04-02 01:21:07 -05:00
|
|
|
|
|
|
|
if let Some(text_document) = &capabilities.text_document {
|
|
|
|
self.client_capabilities.line_folding_only = text_document
|
|
|
|
.folding_range
|
|
|
|
.as_ref()
|
|
|
|
.and_then(|it| it.line_folding_only)
|
|
|
|
.unwrap_or(false);
|
2021-08-10 09:56:34 +10:00
|
|
|
self.client_capabilities.code_action_disabled_support = text_document
|
|
|
|
.code_action
|
|
|
|
.as_ref()
|
|
|
|
.and_then(|it| it.disabled_support)
|
|
|
|
.unwrap_or(false);
|
2022-10-14 23:04:38 +11:00
|
|
|
self.client_capabilities.snippet_support =
|
|
|
|
if let Some(completion) = &text_document.completion {
|
|
|
|
completion
|
|
|
|
.completion_item
|
|
|
|
.as_ref()
|
|
|
|
.and_then(|it| it.snippet_support)
|
|
|
|
.unwrap_or(false)
|
|
|
|
} else {
|
|
|
|
false
|
|
|
|
};
|
2021-04-02 01:21:07 -05:00
|
|
|
}
|
2020-12-07 21:46:39 +11:00
|
|
|
}
|
2021-05-10 11:16:04 +10:00
|
|
|
|
2022-03-21 12:33:37 +11:00
|
|
|
/// Given the configured workspaces or root URI and the their settings,
|
|
|
|
/// update and resolve any paths that should be enabled
|
|
|
|
pub async fn update_enabled_paths(&mut self, client: Client) -> bool {
|
|
|
|
if let Some(workspace_folders) = self.workspace_folders.clone() {
|
|
|
|
let mut touched = false;
|
|
|
|
for (workspace, folder) in workspace_folders {
|
|
|
|
if let Ok(settings) = client.specifier_configuration(&folder.uri).await
|
|
|
|
{
|
|
|
|
if self.update_enabled_paths_entry(&workspace, settings.enable_paths)
|
|
|
|
{
|
|
|
|
touched = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
touched
|
|
|
|
} else if let Some(root_uri) = self.root_uri.clone() {
|
|
|
|
self.update_enabled_paths_entry(
|
|
|
|
&root_uri,
|
|
|
|
self.settings.workspace.enable_paths.clone(),
|
|
|
|
)
|
|
|
|
} else {
|
|
|
|
false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Update a specific entry in the enabled paths for a given workspace.
|
|
|
|
fn update_enabled_paths_entry(
|
|
|
|
&mut self,
|
|
|
|
workspace: &ModuleSpecifier,
|
|
|
|
enabled_paths: Vec<String>,
|
|
|
|
) -> bool {
|
2022-11-28 17:28:54 -05:00
|
|
|
let workspace = ensure_directory_specifier(workspace.clone());
|
2022-03-21 12:33:37 +11:00
|
|
|
let key = workspace.to_string();
|
|
|
|
let mut touched = false;
|
|
|
|
if !enabled_paths.is_empty() {
|
2022-11-28 17:28:54 -05:00
|
|
|
if let Ok(workspace_path) = specifier_to_file_path(&workspace) {
|
2022-03-21 12:33:37 +11:00
|
|
|
let mut paths = Vec::new();
|
|
|
|
for path in &enabled_paths {
|
|
|
|
let fs_path = workspace_path.join(path);
|
|
|
|
match ModuleSpecifier::from_file_path(fs_path) {
|
|
|
|
Ok(path_uri) => {
|
|
|
|
paths.push(path_uri.to_string());
|
|
|
|
}
|
|
|
|
Err(_) => {
|
|
|
|
lsp_log!("Unable to resolve a file path for `deno.enablePath` from \"{}\" for workspace \"{}\".", path, workspace);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if !paths.is_empty() {
|
|
|
|
touched = true;
|
|
|
|
self.enabled_paths.insert(key, paths);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
touched = true;
|
|
|
|
self.enabled_paths.remove(&key);
|
|
|
|
}
|
|
|
|
touched
|
|
|
|
}
|
|
|
|
|
2022-01-25 10:30:38 -05:00
|
|
|
pub fn get_specifiers_with_client_uris(&self) -> Vec<SpecifierWithClientUri> {
|
2021-05-20 19:56:48 +10:00
|
|
|
self
|
2022-01-25 10:30:38 -05:00
|
|
|
.settings
|
|
|
|
.specifiers
|
|
|
|
.iter()
|
|
|
|
.map(|(s, (u, _))| SpecifierWithClientUri {
|
|
|
|
specifier: s.clone(),
|
|
|
|
client_uri: u.clone(),
|
|
|
|
})
|
2022-03-21 12:33:37 +11:00
|
|
|
.collect()
|
2021-05-10 11:16:04 +10:00
|
|
|
}
|
|
|
|
|
2022-01-25 10:30:38 -05:00
|
|
|
pub fn set_specifier_settings(
|
|
|
|
&mut self,
|
|
|
|
specifier: ModuleSpecifier,
|
|
|
|
client_uri: ModuleSpecifier,
|
|
|
|
settings: SpecifierSettings,
|
|
|
|
) {
|
2021-05-20 19:56:48 +10:00
|
|
|
self
|
2022-01-25 10:30:38 -05:00
|
|
|
.settings
|
|
|
|
.specifiers
|
|
|
|
.insert(specifier, (client_uri, settings));
|
2021-05-10 11:16:04 +10:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
mod tests {
|
|
|
|
use super::*;
|
|
|
|
use deno_core::resolve_url;
|
|
|
|
use deno_core::serde_json::json;
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_config_specifier_enabled() {
|
2022-03-21 12:33:37 +11:00
|
|
|
let mut config = Config::new();
|
2021-05-10 11:16:04 +10:00
|
|
|
let specifier = resolve_url("file:///a.ts").unwrap();
|
|
|
|
assert!(!config.specifier_enabled(&specifier));
|
|
|
|
config
|
2021-05-20 19:56:48 +10:00
|
|
|
.set_workspace_settings(json!({
|
2021-05-10 11:16:04 +10:00
|
|
|
"enable": true
|
|
|
|
}))
|
|
|
|
.expect("could not update");
|
|
|
|
assert!(config.specifier_enabled(&specifier));
|
|
|
|
}
|
2021-06-01 21:53:08 +10:00
|
|
|
|
2022-03-21 12:33:37 +11:00
|
|
|
#[test]
|
|
|
|
fn test_config_snapshot_specifier_enabled() {
|
|
|
|
let mut config = Config::new();
|
|
|
|
let specifier = resolve_url("file:///a.ts").unwrap();
|
|
|
|
assert!(!config.specifier_enabled(&specifier));
|
|
|
|
config
|
|
|
|
.set_workspace_settings(json!({
|
|
|
|
"enable": true
|
|
|
|
}))
|
|
|
|
.expect("could not update");
|
|
|
|
let config_snapshot = config.snapshot();
|
|
|
|
assert!(config_snapshot.specifier_enabled(&specifier));
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_config_specifier_enabled_path() {
|
|
|
|
let mut config = Config::new();
|
|
|
|
let specifier_a = resolve_url("file:///project/worker/a.ts").unwrap();
|
|
|
|
let specifier_b = resolve_url("file:///project/other/b.ts").unwrap();
|
|
|
|
assert!(!config.specifier_enabled(&specifier_a));
|
|
|
|
assert!(!config.specifier_enabled(&specifier_b));
|
|
|
|
let mut enabled_paths = HashMap::new();
|
|
|
|
enabled_paths.insert(
|
|
|
|
"file:///project/".to_string(),
|
|
|
|
vec!["file:///project/worker/".to_string()],
|
|
|
|
);
|
|
|
|
config.enabled_paths = enabled_paths;
|
|
|
|
assert!(config.specifier_enabled(&specifier_a));
|
|
|
|
assert!(!config.specifier_enabled(&specifier_b));
|
|
|
|
let config_snapshot = config.snapshot();
|
|
|
|
assert!(config_snapshot.specifier_enabled(&specifier_a));
|
|
|
|
assert!(!config_snapshot.specifier_enabled(&specifier_b));
|
|
|
|
}
|
|
|
|
|
2021-06-01 21:53:08 +10:00
|
|
|
#[test]
|
|
|
|
fn test_set_workspace_settings_defaults() {
|
2022-03-21 12:33:37 +11:00
|
|
|
let mut config = Config::new();
|
2021-06-01 21:53:08 +10:00
|
|
|
config
|
|
|
|
.set_workspace_settings(json!({}))
|
|
|
|
.expect("could not update");
|
|
|
|
assert_eq!(
|
|
|
|
config.get_workspace_settings(),
|
|
|
|
WorkspaceSettings {
|
|
|
|
enable: false,
|
2022-03-21 12:33:37 +11:00
|
|
|
enable_paths: Vec::new(),
|
2021-07-28 07:25:09 +10:00
|
|
|
cache: None,
|
2022-01-24 11:27:52 +11:00
|
|
|
certificate_stores: None,
|
2021-06-01 21:53:08 +10:00
|
|
|
config: None,
|
|
|
|
import_map: None,
|
|
|
|
code_lens: CodeLensSettings {
|
|
|
|
implementations: false,
|
|
|
|
references: false,
|
|
|
|
references_all_functions: false,
|
2021-06-07 21:38:07 +10:00
|
|
|
test: true,
|
2021-06-01 21:53:08 +10:00
|
|
|
},
|
2022-10-16 13:39:43 +11:00
|
|
|
inlay_hints: InlayHintsSettings {
|
|
|
|
parameter_names: InlayHintsParamNamesOptions {
|
|
|
|
enabled: InlayHintsParamNamesEnabled::None,
|
|
|
|
suppress_when_argument_matches_name: true
|
|
|
|
},
|
|
|
|
parameter_types: InlayHintsParamTypesOptions { enabled: false },
|
|
|
|
variable_types: InlayHintsVarTypesOptions {
|
|
|
|
enabled: false,
|
2022-10-28 14:48:14 -04:00
|
|
|
suppress_when_type_matches_name: true
|
2022-10-16 13:39:43 +11:00
|
|
|
},
|
|
|
|
property_declaration_types: InlayHintsPropDeclTypesOptions {
|
|
|
|
enabled: false
|
|
|
|
},
|
|
|
|
function_like_return_types: InlayHintsFuncLikeReturnTypesOptions {
|
|
|
|
enabled: false
|
|
|
|
},
|
|
|
|
enum_member_values: InlayHintsEnumMemberValuesOptions {
|
|
|
|
enabled: false
|
|
|
|
},
|
|
|
|
},
|
2021-06-01 21:53:08 +10:00
|
|
|
internal_debug: false,
|
2022-05-16 13:10:08 +02:00
|
|
|
lint: true,
|
2021-06-01 21:53:08 +10:00
|
|
|
suggest: CompletionSettings {
|
|
|
|
complete_function_calls: false,
|
|
|
|
names: true,
|
|
|
|
paths: true,
|
|
|
|
auto_imports: true,
|
|
|
|
imports: ImportCompletionSettings {
|
|
|
|
auto_discover: true,
|
|
|
|
hosts: HashMap::new(),
|
|
|
|
}
|
|
|
|
},
|
2022-03-30 09:59:27 +11:00
|
|
|
testing: TestingSettings {
|
|
|
|
args: vec!["--allow-all".to_string(), "--no-check".to_string()],
|
|
|
|
enable: true
|
|
|
|
},
|
2022-01-24 11:27:52 +11:00
|
|
|
tls_certificate: None,
|
|
|
|
unsafely_ignore_certificate_errors: None,
|
2021-06-01 21:53:08 +10:00
|
|
|
unstable: false,
|
|
|
|
}
|
|
|
|
);
|
|
|
|
}
|
2023-01-03 16:59:48 +01:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_empty_cache() {
|
|
|
|
let mut config = Config::new();
|
|
|
|
config
|
|
|
|
.set_workspace_settings(json!({ "cache": "" }))
|
|
|
|
.expect("could not update");
|
|
|
|
assert_eq!(
|
|
|
|
config.get_workspace_settings(),
|
|
|
|
WorkspaceSettings::default()
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_empty_import_map() {
|
|
|
|
let mut config = Config::new();
|
|
|
|
config
|
|
|
|
.set_workspace_settings(json!({ "import_map": "" }))
|
|
|
|
.expect("could not update");
|
|
|
|
assert_eq!(
|
|
|
|
config.get_workspace_settings(),
|
|
|
|
WorkspaceSettings::default()
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_empty_tls_certificate() {
|
|
|
|
let mut config = Config::new();
|
|
|
|
config
|
|
|
|
.set_workspace_settings(json!({ "tls_certificate": "" }))
|
|
|
|
.expect("could not update");
|
|
|
|
assert_eq!(
|
|
|
|
config.get_workspace_settings(),
|
|
|
|
WorkspaceSettings::default()
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_empty_config() {
|
|
|
|
let mut config = Config::new();
|
|
|
|
config
|
|
|
|
.set_workspace_settings(json!({ "config": "" }))
|
|
|
|
.expect("could not update");
|
|
|
|
assert_eq!(
|
|
|
|
config.get_workspace_settings(),
|
|
|
|
WorkspaceSettings::default()
|
|
|
|
);
|
|
|
|
}
|
2020-12-07 21:46:39 +11:00
|
|
|
}
|