mirror of
https://github.com/rust-lang/rust-analyzer
synced 2025-01-09 03:38:47 +00:00
211 lines
5.9 KiB
TypeScript
211 lines
5.9 KiB
TypeScript
import * as child_process from 'child_process';
|
|
|
|
import * as util from 'util';
|
|
import * as vscode from 'vscode';
|
|
import * as lc from 'vscode-languageclient';
|
|
|
|
import { Server } from '../server';
|
|
import { CargoWatchProvider, registerCargoWatchProvider } from './cargo_watch';
|
|
|
|
interface RunnablesParams {
|
|
textDocument: lc.TextDocumentIdentifier;
|
|
position?: lc.Position;
|
|
}
|
|
|
|
interface Runnable {
|
|
label: string;
|
|
bin: string;
|
|
args: string[];
|
|
env: { [index: string]: string };
|
|
cwd?: string;
|
|
}
|
|
|
|
class RunnableQuickPick implements vscode.QuickPickItem {
|
|
public label: string;
|
|
public description?: string | undefined;
|
|
public detail?: string | undefined;
|
|
public picked?: boolean | undefined;
|
|
|
|
constructor(public runnable: Runnable) {
|
|
this.label = runnable.label;
|
|
}
|
|
}
|
|
|
|
interface CargoTaskDefinition extends vscode.TaskDefinition {
|
|
type: 'cargo';
|
|
label: string;
|
|
command: string;
|
|
args: string[];
|
|
env?: { [key: string]: string };
|
|
}
|
|
|
|
function createTask(spec: Runnable): vscode.Task {
|
|
const TASK_SOURCE = 'Rust';
|
|
const definition: CargoTaskDefinition = {
|
|
type: 'cargo',
|
|
label: spec.label,
|
|
command: spec.bin,
|
|
args: spec.args,
|
|
env: spec.env
|
|
};
|
|
|
|
const execOption: vscode.ShellExecutionOptions = {
|
|
cwd: spec.cwd || '.',
|
|
env: definition.env
|
|
};
|
|
const exec = new vscode.ShellExecution(
|
|
definition.command,
|
|
definition.args,
|
|
execOption
|
|
);
|
|
|
|
const f = vscode.workspace.workspaceFolders![0];
|
|
const t = new vscode.Task(
|
|
definition,
|
|
f,
|
|
definition.label,
|
|
TASK_SOURCE,
|
|
exec,
|
|
['$rustc']
|
|
);
|
|
t.presentationOptions.clear = true;
|
|
return t;
|
|
}
|
|
|
|
let prevRunnable: RunnableQuickPick | undefined;
|
|
export async function handle() {
|
|
const editor = vscode.window.activeTextEditor;
|
|
if (editor == null || editor.document.languageId !== 'rust') {
|
|
return;
|
|
}
|
|
const textDocument: lc.TextDocumentIdentifier = {
|
|
uri: editor.document.uri.toString()
|
|
};
|
|
const params: RunnablesParams = {
|
|
textDocument,
|
|
position: Server.client.code2ProtocolConverter.asPosition(
|
|
editor.selection.active
|
|
)
|
|
};
|
|
const runnables = await Server.client.sendRequest<Runnable[]>(
|
|
'rust-analyzer/runnables',
|
|
params
|
|
);
|
|
const items: RunnableQuickPick[] = [];
|
|
if (prevRunnable) {
|
|
items.push(prevRunnable);
|
|
}
|
|
for (const r of runnables) {
|
|
if (
|
|
prevRunnable &&
|
|
JSON.stringify(prevRunnable.runnable) === JSON.stringify(r)
|
|
) {
|
|
continue;
|
|
}
|
|
items.push(new RunnableQuickPick(r));
|
|
}
|
|
const item = await vscode.window.showQuickPick(items);
|
|
if (item) {
|
|
item.detail = 'rerun';
|
|
prevRunnable = item;
|
|
const task = createTask(item.runnable);
|
|
return await vscode.tasks.executeTask(task);
|
|
}
|
|
}
|
|
|
|
export async function handleSingle(runnable: Runnable) {
|
|
const editor = vscode.window.activeTextEditor;
|
|
if (editor == null || editor.document.languageId !== 'rust') {
|
|
return;
|
|
}
|
|
|
|
const task = createTask(runnable);
|
|
task.group = vscode.TaskGroup.Build;
|
|
task.presentationOptions = {
|
|
reveal: vscode.TaskRevealKind.Always,
|
|
panel: vscode.TaskPanelKind.Dedicated,
|
|
clear: true
|
|
};
|
|
|
|
return vscode.tasks.executeTask(task);
|
|
}
|
|
|
|
/**
|
|
* Interactively asks the user whether we should run `cargo check` in order to
|
|
* provide inline diagnostics; the user is met with a series of dialog boxes
|
|
* that, when accepted, allow us to `cargo install cargo-watch` and then run it.
|
|
*/
|
|
export async function interactivelyStartCargoWatch(
|
|
context: vscode.ExtensionContext
|
|
): Promise<CargoWatchProvider | undefined> {
|
|
if (Server.config.cargoWatchOptions.enableOnStartup === 'disabled') {
|
|
return;
|
|
}
|
|
|
|
if (Server.config.cargoWatchOptions.enableOnStartup === 'ask') {
|
|
const watch = await vscode.window.showInformationMessage(
|
|
'Start watching changes with cargo? (Executes `cargo watch`, provides inline diagnostics)',
|
|
'yes',
|
|
'no'
|
|
);
|
|
if (watch !== 'yes') {
|
|
return;
|
|
}
|
|
}
|
|
|
|
return startCargoWatch(context);
|
|
}
|
|
|
|
export async function startCargoWatch(
|
|
context: vscode.ExtensionContext
|
|
): Promise<CargoWatchProvider | undefined> {
|
|
const execPromise = util.promisify(child_process.exec);
|
|
|
|
const { stderr } = await execPromise('cargo watch --version').catch(e => e);
|
|
|
|
if (stderr.includes('no such subcommand: `watch`')) {
|
|
const msg =
|
|
'The `cargo-watch` subcommand is not installed. Install? (takes ~1-2 minutes)';
|
|
const install = await vscode.window.showInformationMessage(
|
|
msg,
|
|
'yes',
|
|
'no'
|
|
);
|
|
if (install !== 'yes') {
|
|
return;
|
|
}
|
|
|
|
const label = 'install-cargo-watch';
|
|
const taskFinished = new Promise((resolve, reject) => {
|
|
const disposable = vscode.tasks.onDidEndTask(({ execution }) => {
|
|
if (execution.task.name === label) {
|
|
disposable.dispose();
|
|
resolve();
|
|
}
|
|
});
|
|
});
|
|
|
|
vscode.tasks.executeTask(
|
|
createTask({
|
|
label,
|
|
bin: 'cargo',
|
|
args: ['install', 'cargo-watch'],
|
|
env: {}
|
|
})
|
|
);
|
|
await taskFinished;
|
|
const output = await execPromise('cargo watch --version').catch(e => e);
|
|
if (output.stderr !== '') {
|
|
vscode.window.showErrorMessage(
|
|
`Couldn't install \`cargo-\`watch: ${output.stderr}`
|
|
);
|
|
return;
|
|
}
|
|
}
|
|
|
|
const provider = await registerCargoWatchProvider(context.subscriptions);
|
|
if (provider) {
|
|
provider.start();
|
|
}
|
|
return provider;
|
|
}
|