rust/editors/code/src/commands/runnables.ts

186 lines
5.5 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 '../rust-analyzer-api';
2020-04-29 08:52:53 -05:00
import * as os from "os";
2019-12-30 12:58:44 -06:00
import { Ctx, Cmd } from '../ctx';
import { Cargo } from '../cargo';
2019-12-30 12:58:44 -06:00
export function run(ctx: Ctx): Cmd {
let prevRunnable: RunnableQuickPick | undefined;
return async () => {
const editor = ctx.activeRustEditor;
2019-12-31 11:50:32 -06:00
const client = ctx.client;
if (!editor || !client) return;
2019-12-30 12:58:44 -06:00
const textDocument: lc.TextDocumentIdentifier = {
uri: editor.document.uri.toString(),
};
const runnables = await client.sendRequest(ra.runnables, {
2019-12-30 12:58:44 -06:00
textDocument,
2019-12-31 11:50:32 -06:00
position: client.code2ProtocolConverter.asPosition(
2019-12-30 12:58:44 -06:00
editor.selection.active,
),
});
2019-12-30 12:58:44 -06:00
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) return;
item.detail = 'rerun';
prevRunnable = item;
const task = createTask(item.runnable);
return await vscode.tasks.executeTask(task);
2019-12-30 13:07:04 -06:00
};
2019-12-30 12:58:44 -06:00
}
export function runSingle(ctx: Ctx): Cmd {
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
function getLldbDebugConfig(config: ra.Runnable, sourceFileMap: Record<string, string>): vscode.DebugConfiguration {
return {
type: "lldb",
request: "launch",
name: config.label,
cargo: {
args: config.args,
},
args: config.extraArgs,
cwd: config.cwd,
sourceMap: sourceFileMap
};
}
const debugOutput = vscode.window.createOutputChannel("Debug");
async function getCppvsDebugConfig(config: ra.Runnable, sourceFileMap: Record<string, string>): Promise<vscode.DebugConfiguration> {
debugOutput.clear();
2020-04-30 10:41:48 -05:00
const cargo = new Cargo(config.cwd || '.', debugOutput);
const executable = await cargo.executableFromArgs(config.args);
// if we are here, there were no compilation errors.
return {
2020-04-29 08:52:53 -05:00
type: (os.platform() === "win32") ? "cppvsdbg" : 'cppdbg',
request: "launch",
name: config.label,
program: executable,
args: config.extraArgs,
cwd: config.cwd,
sourceFileMap: sourceFileMap,
};
}
2020-03-09 16:06:45 -05:00
export function debugSingle(ctx: Ctx): Cmd {
return async (config: ra.Runnable) => {
const editor = ctx.activeRustEditor;
if (!editor) return;
const lldbId = "vadimcn.vscode-lldb";
const cpptoolsId = "ms-vscode.cpptools";
2020-04-30 10:41:48 -05:00
const debugEngineId = ctx.config.debug.engine;
let debugEngine = null;
2020-04-30 10:41:48 -05:00
if (debugEngineId === "auto") {
debugEngine = vscode.extensions.getExtension(lldbId);
if (!debugEngine) {
debugEngine = vscode.extensions.getExtension(cpptoolsId);
}
}
else {
debugEngine = vscode.extensions.getExtension(debugEngineId);
}
if (!debugEngine) {
vscode.window.showErrorMessage(`Install [CodeLLDB](https://marketplace.visualstudio.com/items?itemName=${lldbId})`
2020-04-30 10:41:48 -05:00
+ ` or [MS C++ tools](https://marketplace.visualstudio.com/items?itemName=${cpptoolsId}) extension for debugging.`);
2020-04-02 05:56:14 -05:00
return;
}
2020-03-09 16:06:45 -05:00
2020-04-30 10:41:48 -05:00
const debugConfig = lldbId === debugEngine.id
? getLldbDebugConfig(config, ctx.config.debug.sourceFileMap)
: await getCppvsDebugConfig(config, ctx.config.debug.sourceFileMap);
2020-03-12 15:00:40 -05:00
return vscode.debug.startDebugging(undefined, debugConfig);
2020-03-09 16:06:45 -05:00
};
}
2018-10-07 15:44:25 -05:00
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
}
}
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 };
}
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',
label: spec.label,
2018-10-07 15:44:25 -05:00
command: spec.bin,
2020-03-09 16:06:45 -05:00
args: spec.extraArgs ? [...spec.args, '--', ...spec.extraArgs] : spec.args,
2019-12-09 12:57:55 -06:00
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 = {
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;
}