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';
|
2020-05-25 07:56:26 -05:00
|
|
|
import * as ra from './lsp_ext';
|
2020-05-30 21:13:08 -05:00
|
|
|
import * as toolchain from "./toolchain";
|
2019-03-18 16:30:23 -05:00
|
|
|
|
2020-05-25 05:02:30 -05:00
|
|
|
import { Ctx, Cmd } from './ctx';
|
|
|
|
import { startDebugSession, getDebugConfiguration } from './debug';
|
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) => {
|
|
|
|
(async () => await makeDebugConfig(ctx, quickPick.activeItems[0]))();
|
|
|
|
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
|
|
|
|
|
|
|
export function runSingle(ctx: Ctx): Cmd {
|
2020-02-24 16:56:38 -06:00
|
|
|
return async (runnable: ra.Runnable) => {
|
2019-12-30 12:58:44 -06:00
|
|
|
const editor = ctx.activeRustEditor;
|
2019-12-30 13:07:04 -06:00
|
|
|
if (!editor) return;
|
2019-12-30 12:58:44 -06:00
|
|
|
|
|
|
|
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);
|
2019-12-30 13:07:04 -06:00
|
|
|
};
|
2019-12-30 12:58:44 -06:00
|
|
|
}
|
2018-10-07 15:44:25 -05:00
|
|
|
|
2020-05-11 08:06:57 -05:00
|
|
|
export function debug(ctx: Ctx): Cmd {
|
|
|
|
let prevDebuggee: RunnableQuickPick | undefined;
|
2020-05-06 08:01:35 -05:00
|
|
|
|
2020-05-11 08:06:57 -05:00
|
|
|
return async () => {
|
2020-05-17 12:29:59 -05:00
|
|
|
const item = await selectRunnable(ctx, prevDebuggee, true);
|
2020-05-11 08:06:57 -05:00
|
|
|
if (!item) return;
|
2020-05-06 08:01:35 -05:00
|
|
|
|
2020-05-11 08:06:57 -05:00
|
|
|
item.detail = 'restart';
|
|
|
|
prevDebuggee = item;
|
|
|
|
return await startDebugSession(ctx, item.runnable);
|
|
|
|
};
|
2020-05-06 08:01:35 -05:00
|
|
|
}
|
|
|
|
|
2020-03-09 16:06:45 -05:00
|
|
|
export function debugSingle(ctx: Ctx): Cmd {
|
|
|
|
return async (config: ra.Runnable) => {
|
2020-05-11 08:06:57 -05:00
|
|
|
await startDebugSession(ctx, config);
|
2020-03-09 16:06:45 -05:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2020-05-14 05:22:52 -05:00
|
|
|
async function makeDebugConfig(ctx: Ctx, item: RunnableQuickPick): Promise<void> {
|
|
|
|
const scope = ctx.activeRustEditor?.document.uri;
|
|
|
|
if (!scope) return;
|
2020-05-11 10:00:15 -05:00
|
|
|
|
2020-05-14 05:22:52 -05:00
|
|
|
const debugConfig = await getDebugConfiguration(ctx, item.runnable);
|
|
|
|
if (!debugConfig) return;
|
2020-05-11 10:00:15 -05:00
|
|
|
|
2020-05-14 05:22:52 -05:00
|
|
|
const wsLaunchSection = vscode.workspace.getConfiguration("launch", scope);
|
|
|
|
const configurations = wsLaunchSection.get<any[]>("configurations") || [];
|
2020-05-11 10:00:15 -05:00
|
|
|
|
2020-05-14 05:22:52 -05:00
|
|
|
const index = configurations.findIndex(c => c.name === debugConfig.name);
|
|
|
|
if (index !== -1) {
|
|
|
|
const answer = await vscode.window.showErrorMessage(`Launch configuration '${debugConfig.name}' already exists!`, 'Cancel', 'Update');
|
|
|
|
if (answer === "Cancel") return;
|
2020-05-11 10:00:15 -05:00
|
|
|
|
2020-05-14 05:22:52 -05:00
|
|
|
configurations[index] = debugConfig;
|
|
|
|
} else {
|
|
|
|
configurations.push(debugConfig);
|
|
|
|
}
|
2020-05-11 10:00:15 -05:00
|
|
|
|
2020-05-14 05:22:52 -05:00
|
|
|
await wsLaunchSection.update("configurations", configurations);
|
|
|
|
}
|
|
|
|
|
|
|
|
export function newDebugConfig(ctx: Ctx): Cmd {
|
|
|
|
return async () => {
|
2020-05-17 12:29:59 -05:00
|
|
|
const item = await selectRunnable(ctx, undefined, true, false);
|
2020-05-14 05:22:52 -05:00
|
|
|
if (!item) return;
|
2020-05-11 10:00:15 -05:00
|
|
|
|
2020-05-14 05:22:52 -05:00
|
|
|
await makeDebugConfig(ctx, item);
|
2020-05-11 10:00:15 -05: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
|
|
|
|
2020-02-24 16:56:38 -06: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
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
interface CargoTaskDefinition extends vscode.TaskDefinition {
|
|
|
|
type: 'cargo';
|
|
|
|
label: string;
|
|
|
|
command: string;
|
2018-10-07 15:59:02 -05:00
|
|
|
args: string[];
|
2018-10-07 15:44:25 -05:00
|
|
|
env?: { [key: string]: string };
|
|
|
|
}
|
|
|
|
|
2020-05-25 05:02:30 -05:00
|
|
|
export function createTask(spec: ra.Runnable): vscode.Task {
|
2018-10-07 15:44:25 -05:00
|
|
|
const TASK_SOURCE = 'Rust';
|
2018-10-07 15:59:02 -05:00
|
|
|
const definition: CargoTaskDefinition = {
|
2018-10-07 15:44:25 -05:00
|
|
|
type: 'cargo',
|
2019-03-18 15:04:33 -05:00
|
|
|
label: spec.label,
|
2020-05-30 21:13:08 -05:00
|
|
|
command: toolchain.getPathForExecutable(spec.kind),
|
2020-03-09 16:06:45 -05:00
|
|
|
args: spec.extraArgs ? [...spec.args, '--', ...spec.extraArgs] : spec.args,
|
2020-05-31 06:41:52 -05:00
|
|
|
env: Object.assign({}, process.env, spec.env),
|
2018-10-07 15:59:02 -05:00
|
|
|
};
|
2018-10-07 15:44:25 -05:00
|
|
|
|
2018-10-07 15:59:02 -05:00
|
|
|
const execOption: vscode.ShellExecutionOptions = {
|
2019-04-13 12:45:21 -05:00
|
|
|
cwd: spec.cwd || '.',
|
2019-12-09 12:57:55 -06:00
|
|
|
env: definition.env,
|
2018-10-07 15:44:25 -05:00
|
|
|
};
|
2019-01-12 17:49:07 -06:00
|
|
|
const exec = new vscode.ShellExecution(
|
|
|
|
definition.command,
|
|
|
|
definition.args,
|
2019-12-09 12:57:55 -06:00
|
|
|
execOption,
|
2019-01-12 17:49:07 -06:00
|
|
|
);
|
2018-10-07 15:44:25 -05:00
|
|
|
|
2018-10-07 15:59:02 -05:00
|
|
|
const f = vscode.workspace.workspaceFolders![0];
|
2018-10-08 16:38:33 -05:00
|
|
|
const t = new vscode.Task(
|
|
|
|
definition,
|
|
|
|
f,
|
|
|
|
definition.label,
|
|
|
|
TASK_SOURCE,
|
|
|
|
exec,
|
2019-12-09 12:57:55 -06:00
|
|
|
['$rustc'],
|
2018-10-08 16:38:33 -05:00
|
|
|
);
|
2019-01-12 17:49:07 -06:00
|
|
|
t.presentationOptions.clear = true;
|
2018-10-07 15:44:25 -05:00
|
|
|
return t;
|
|
|
|
}
|