rust-analyzer/editors/code/src/config.ts

266 lines
9 KiB
TypeScript
Raw Normal View History

import * as os from "os";
2018-10-07 20:59:02 +00:00
import * as vscode from 'vscode';
import { BinarySource } from "./installation/interfaces";
2018-10-07 20:59:02 +00:00
const RA_LSP_DEBUG = process.env.__RA_LSP_SERVER_DEBUG;
export interface CargoWatchOptions {
enable: boolean;
arguments: string[];
command: string;
allTargets: boolean;
2019-04-02 06:43:02 +00:00
}
2019-12-13 10:16:34 +00:00
export interface CargoFeatures {
noDefaultFeatures: boolean;
allFeatures: boolean;
features: string[];
}
2018-10-07 20:59:02 +00:00
export class Config {
langServerSource!: null | BinarySource;
2019-12-31 16:28:27 +00:00
highlightingOn = true;
rainbowHighlightingOn = false;
enableEnhancedTyping = true;
lruCapacity: null | number = null;
displayInlayHints = true;
maxInlayHintLength: null | number = null;
2020-02-04 22:13:46 +00:00
excludeGlobs: string[] = [];
2019-12-31 16:28:27 +00:00
useClientWatching = true;
2020-02-04 22:13:46 +00:00
featureFlags: Record<string, boolean> = {};
// for internal use
2019-12-31 16:28:27 +00:00
withSysroot: null | boolean = null;
cargoWatchOptions: CargoWatchOptions = {
enable: true,
arguments: [],
command: '',
allTargets: true,
2019-04-02 06:43:02 +00:00
};
2019-12-31 16:28:27 +00:00
cargoFeatures: CargoFeatures = {
2019-12-13 10:16:34 +00:00
noDefaultFeatures: false,
allFeatures: true,
2019-12-13 10:16:34 +00:00
features: [],
};
2018-10-07 20:59:02 +00:00
private prevEnhancedTyping: null | boolean = null;
2019-12-13 10:16:34 +00:00
private prevCargoFeatures: null | CargoFeatures = null;
private prevCargoWatchOptions: null | CargoWatchOptions = null;
2019-12-31 16:34:52 +00:00
constructor(ctx: vscode.ExtensionContext) {
vscode.workspace.onDidChangeConfiguration(_ => this.refresh(ctx), null, ctx.subscriptions);
this.refresh(ctx);
2018-10-08 21:36:47 +00:00
}
2018-10-07 20:59:02 +00:00
private static expandPathResolving(path: string) {
if (path.startsWith('~/')) {
return path.replace('~', os.homedir());
}
return path;
}
/**
* Name of the binary artifact for `ra_lsp_server` that is published for
* `platform` on GitHub releases. (It is also stored under the same name when
* downloaded by the extension).
*/
private static prebuiltLangServerFileName(platform: NodeJS.Platform): null | string {
switch (platform) {
case "linux": return "ra_lsp_server-linux";
case "darwin": return "ra_lsp_server-mac";
case "win32": return "ra_lsp_server-windows.exe";
// Users on these platforms yet need to manually build from sources
case "aix":
case "android":
case "freebsd":
case "openbsd":
case "sunos":
case "cygwin":
case "netbsd": return null;
2020-02-08 22:28:32 +00:00
// The list of platforms is exhaustive (see `NodeJS.Platform` type definition)
}
}
private static langServerBinarySource(
ctx: vscode.ExtensionContext,
config: vscode.WorkspaceConfiguration
): null | BinarySource {
const langServerPath = RA_LSP_DEBUG ?? config.get<null | string>("raLspServerPath");
if (langServerPath) {
return {
type: BinarySource.Type.ExplicitPath,
path: Config.expandPathResolving(langServerPath)
};
}
const prebuiltBinaryName = Config.prebuiltLangServerFileName(process.platform);
return !prebuiltBinaryName ? null : {
type: BinarySource.Type.GithubRelease,
dir: ctx.globalStoragePath,
file: prebuiltBinaryName,
repo: {
name: "rust-analyzer",
owner: "rust-analyzer",
}
};
}
// FIXME: revisit the logic for `if (.has(...)) config.get(...)` set default
// values only in one place (i.e. remove default values from non-readonly members declarations)
private refresh(ctx: vscode.ExtensionContext) {
2019-01-28 11:43:07 +00:00
const config = vscode.workspace.getConfiguration('rust-analyzer');
2019-12-13 10:16:34 +00:00
let requireReloadMessage = null;
2018-10-08 21:38:33 +00:00
if (config.has('highlightingOn')) {
this.highlightingOn = config.get('highlightingOn') as boolean;
}
2019-05-27 09:26:15 +00:00
if (config.has('rainbowHighlightingOn')) {
this.rainbowHighlightingOn = config.get(
2019-12-09 18:57:55 +00:00
'rainbowHighlightingOn',
2019-05-27 09:26:15 +00:00
) as boolean;
}
if (config.has('enableEnhancedTyping')) {
2019-02-07 10:54:41 +00:00
this.enableEnhancedTyping = config.get(
2019-12-09 18:57:55 +00:00
'enableEnhancedTyping',
2019-02-07 10:54:41 +00:00
) as boolean;
if (this.prevEnhancedTyping === null) {
this.prevEnhancedTyping = this.enableEnhancedTyping;
}
} else if (this.prevEnhancedTyping === null) {
this.prevEnhancedTyping = this.enableEnhancedTyping;
}
if (this.prevEnhancedTyping !== this.enableEnhancedTyping) {
requireReloadMessage =
'Changing enhanced typing setting requires a reload';
this.prevEnhancedTyping = this.enableEnhancedTyping;
}
this.langServerSource = Config.langServerBinarySource(ctx, config);
if (config.has('cargo-watch.enable')) {
this.cargoWatchOptions.enable = config.get<boolean>(
'cargo-watch.enable',
true,
2019-04-02 06:43:02 +00:00
);
}
2019-04-02 05:07:40 +00:00
2019-06-24 10:50:34 +00:00
if (config.has('cargo-watch.arguments')) {
this.cargoWatchOptions.arguments = config.get<string[]>(
2019-06-24 10:50:34 +00:00
'cargo-watch.arguments',
[],
2019-04-02 06:43:02 +00:00
);
}
2019-06-24 10:02:20 +00:00
2019-06-24 10:50:34 +00:00
if (config.has('cargo-watch.command')) {
this.cargoWatchOptions.command = config.get<string>(
2019-06-24 10:50:34 +00:00
'cargo-watch.command',
2019-12-09 18:57:55 +00:00
'',
2019-06-24 10:02:20 +00:00
);
}
if (config.has('cargo-watch.allTargets')) {
this.cargoWatchOptions.allTargets = config.get<boolean>(
'cargo-watch.allTargets',
true,
);
}
2019-06-07 17:49:29 +00:00
if (config.has('lruCapacity')) {
this.lruCapacity = config.get('lruCapacity') as number;
}
2019-07-23 13:38:21 +00:00
if (config.has('displayInlayHints')) {
this.displayInlayHints = config.get('displayInlayHints') as boolean;
}
2019-10-18 11:40:03 +00:00
if (config.has('maxInlayHintLength')) {
this.maxInlayHintLength = config.get(
2019-12-09 18:57:55 +00:00
'maxInlayHintLength',
2019-10-18 11:40:03 +00:00
) as number;
}
if (config.has('excludeGlobs')) {
this.excludeGlobs = config.get('excludeGlobs') || [];
}
2019-09-06 13:25:24 +00:00
if (config.has('useClientWatching')) {
2019-12-17 11:41:44 +00:00
this.useClientWatching = config.get('useClientWatching') || true;
2019-09-06 13:25:24 +00:00
}
2019-08-22 11:44:16 +00:00
if (config.has('featureFlags')) {
this.featureFlags = config.get('featureFlags') || {};
}
if (config.has('withSysroot')) {
this.withSysroot = config.get('withSysroot') || false;
}
2019-12-13 10:16:34 +00:00
if (config.has('cargoFeatures.noDefaultFeatures')) {
this.cargoFeatures.noDefaultFeatures = config.get(
'cargoFeatures.noDefaultFeatures',
false,
);
}
if (config.has('cargoFeatures.allFeatures')) {
this.cargoFeatures.allFeatures = config.get(
'cargoFeatures.allFeatures',
true,
2019-12-13 10:16:34 +00:00
);
}
if (config.has('cargoFeatures.features')) {
this.cargoFeatures.features = config.get(
'cargoFeatures.features',
[],
);
}
if (
this.prevCargoFeatures !== null &&
(this.cargoFeatures.allFeatures !==
this.prevCargoFeatures.allFeatures ||
this.cargoFeatures.noDefaultFeatures !==
2019-12-30 22:30:35 +00:00
this.prevCargoFeatures.noDefaultFeatures ||
this.cargoFeatures.features.length !==
2019-12-30 22:30:35 +00:00
this.prevCargoFeatures.features.length ||
this.cargoFeatures.features.some(
(v, i) => v !== this.prevCargoFeatures!.features[i],
))
) {
2019-12-13 10:16:34 +00:00
requireReloadMessage = 'Changing cargo features requires a reload';
}
this.prevCargoFeatures = { ...this.cargoFeatures };
2020-01-15 03:52:49 +00:00
if (this.prevCargoWatchOptions !== null) {
const changed =
this.cargoWatchOptions.enable !== this.prevCargoWatchOptions.enable ||
this.cargoWatchOptions.command !== this.prevCargoWatchOptions.command ||
this.cargoWatchOptions.allTargets !== this.prevCargoWatchOptions.allTargets ||
this.cargoWatchOptions.arguments.length !== this.prevCargoWatchOptions.arguments.length ||
this.cargoWatchOptions.arguments.some(
(v, i) => v !== this.prevCargoWatchOptions!.arguments[i],
2020-01-15 03:52:49 +00:00
);
if (changed) {
requireReloadMessage = 'Changing cargo-watch options requires a reload';
}
}
this.prevCargoWatchOptions = { ...this.cargoWatchOptions };
2019-12-13 10:16:34 +00:00
if (requireReloadMessage !== null) {
const reloadAction = 'Reload now';
vscode.window
.showInformationMessage(requireReloadMessage, reloadAction)
.then(selectedAction => {
if (selectedAction === reloadAction) {
vscode.commands.executeCommand(
'workbench.action.reloadWindow',
);
}
});
}
2018-10-07 20:59:02 +00:00
}
}