rust/editors/code/src/run.ts

150 lines
5.1 KiB
TypeScript
Raw Normal View History

2018-10-07 15:44:25 -05:00
import * as vscode from 'vscode';
2018-10-07 15:59:02 -05:00
import * as lc from 'vscode-languageclient';
import * as ra from './lsp_ext';
2020-06-18 14:20:13 -05:00
import * as tasks from './tasks';
2020-06-02 07:33:47 -05:00
import { Ctx } from './ctx';
import { makeDebugConfig } from './debug';
2020-07-02 13:33:26 -05:00
import { Config, RunnableEnvCfg } from './config';
2020-05-11 08:06:57 -05:00
2020-05-14 05:22:52 -05:00
const quickPickButtons = [{ iconPath: new vscode.ThemeIcon("save"), tooltip: "Save as a launch.json configurtation." }];
2020-05-25 05:02:30 -05:00
export async function selectRunnable(ctx: Ctx, prevRunnable?: RunnableQuickPick, debuggeeOnly = false, showButtons: boolean = true): Promise<RunnableQuickPick | undefined> {
2020-05-11 08:06:57 -05:00
const editor = ctx.activeRustEditor;
const client = ctx.client;
if (!editor || !client) return;
const textDocument: lc.TextDocumentIdentifier = {
uri: editor.document.uri.toString(),
};
const runnables = await client.sendRequest(ra.runnables, {
textDocument,
position: client.code2ProtocolConverter.asPosition(
editor.selection.active,
),
});
const items: RunnableQuickPick[] = [];
if (prevRunnable) {
items.push(prevRunnable);
}
for (const r of runnables) {
if (
prevRunnable &&
JSON.stringify(prevRunnable.runnable) === JSON.stringify(r)
) {
continue;
}
2020-05-17 12:29:59 -05:00
if (debuggeeOnly && (r.label.startsWith('doctest') || r.label.startsWith('cargo'))) {
continue;
}
2020-05-11 08:06:57 -05:00
items.push(new RunnableQuickPick(r));
}
2020-05-14 05:22:52 -05:00
2020-05-17 12:38:50 -05:00
if (items.length === 0) {
2020-05-17 12:29:59 -05:00
// it is the debug case, run always has at least 'cargo check ...'
// see crates\rust-analyzer\src\main_loop\handlers.rs, handle_runnables
vscode.window.showErrorMessage("There's no debug target!");
return;
}
2020-05-14 05:22:52 -05:00
return await new Promise((resolve) => {
const disposables: vscode.Disposable[] = [];
const close = (result?: RunnableQuickPick) => {
resolve(result);
disposables.forEach(d => d.dispose());
};
const quickPick = vscode.window.createQuickPick<RunnableQuickPick>();
quickPick.items = items;
quickPick.title = "Select Runnable";
if (showButtons) {
quickPick.buttons = quickPickButtons;
}
disposables.push(
quickPick.onDidHide(() => close()),
quickPick.onDidAccept(() => close(quickPick.selectedItems[0])),
quickPick.onDidTriggerButton((_button) => {
2020-06-02 07:33:47 -05:00
(async () => await makeDebugConfig(ctx, quickPick.activeItems[0].runnable))();
2020-05-14 05:22:52 -05:00
close();
}),
quickPick.onDidChangeActive((active) => {
if (showButtons && active.length > 0) {
if (active[0].label.startsWith('cargo')) {
// save button makes no sense for `cargo test` or `cargo check`
quickPick.buttons = [];
} else if (quickPick.buttons.length === 0) {
quickPick.buttons = quickPickButtons;
}
}
}),
quickPick
);
quickPick.show();
});
2020-05-11 08:06:57 -05:00
}
2019-12-30 12:58:44 -06:00
2020-05-25 05:02:30 -05:00
export class RunnableQuickPick implements vscode.QuickPickItem {
2018-10-07 15:59:02 -05:00
public label: string;
public description?: string | undefined;
public detail?: string | undefined;
public picked?: boolean | undefined;
2018-10-07 15:44:25 -05:00
constructor(public runnable: ra.Runnable) {
2018-10-07 15:59:02 -05:00
this.label = runnable.label;
2018-10-07 15:44:25 -05:00
}
}
2020-07-02 13:33:26 -05:00
export function prepareEnv(runnable: ra.Runnable, runnableEnvCfg: RunnableEnvCfg): Record<string, string> {
2020-07-02 11:47:40 -05:00
const env: Record<string, string> = { "RUST_BACKTRACE": "short" };
if (runnable.args.expectTest) {
env["UPDATE_EXPECT"] = "1";
}
2020-07-03 06:56:30 -05:00
Object.assign(env, process.env as { [key: string]: string });
2020-07-02 13:33:26 -05:00
if (runnableEnvCfg) {
if (Array.isArray(runnableEnvCfg)) {
for (const it of runnableEnvCfg) {
2020-07-02 11:47:40 -05:00
if (!it.mask || new RegExp(it.mask).test(runnable.label)) {
Object.assign(env, it.env);
}
}
} else {
2020-07-02 14:08:33 -05:00
Object.assign(env, runnableEnvCfg);
2020-07-02 11:47:40 -05:00
}
}
return env;
}
2020-06-18 14:20:13 -05:00
export async function createTask(runnable: ra.Runnable, config: Config): Promise<vscode.Task> {
if (runnable.kind !== "cargo") {
// rust-analyzer supports only one kind, "cargo"
// do not use tasks.TASK_TYPE here, these are completely different meanings.
2020-06-02 10:22:23 -05:00
2020-06-18 14:20:13 -05:00
throw `Unexpected runnable kind: ${runnable.kind}`;
2020-06-02 10:22:23 -05:00
}
2020-06-18 14:20:13 -05:00
2020-06-06 07:12:17 -05:00
const args = [...runnable.args.cargoArgs]; // should be a copy!
2020-06-02 10:22:23 -05:00
if (runnable.args.executableArgs.length > 0) {
args.push('--', ...runnable.args.executableArgs);
}
2020-07-02 11:47:40 -05:00
2020-06-18 14:20:13 -05:00
const definition: tasks.CargoTaskDefinition = {
type: tasks.TASK_TYPE,
command: args[0], // run, test, etc...
args: args.slice(1),
2020-07-03 06:56:30 -05:00
cwd: runnable.args.workspaceRoot || ".",
2020-07-02 13:33:26 -05:00
env: prepareEnv(runnable, config.runnableEnv),
2018-10-07 15:59:02 -05:00
};
2018-10-07 15:44:25 -05:00
const target = vscode.workspace.workspaceFolders![0]; // safe, see main activate()
const cargoTask = await tasks.buildCargoTask(target, definition, runnable.label, args, config.cargoRunner, true);
2020-06-18 14:20:13 -05:00
cargoTask.presentationOptions.clear = true;
return cargoTask;
2018-10-07 15:44:25 -05:00
}