-
Notifications
You must be signed in to change notification settings - Fork 51
/
Copy pathtests.ts
167 lines (144 loc) · 5.18 KB
/
tests.ts
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
import * as vscode from "vscode";
import { ExecResult, exec, extensionConfiguration } from "./utils";
import { Tests, DebugEnvironmentConfiguration } from "./types";
import { getMesonTests, getMesonTargets } from "./introspection";
import { workspaceState } from "./extension";
export async function rebuildTests(controller: vscode.TestController) {
let tests = await getMesonTests(workspaceState.get<string>("mesonbuild.buildDir")!);
controller.items.forEach((item) => {
if (!tests.some((test) => item.id == test.name)) {
controller.items.delete(item.id);
}
});
for (let testDescr of tests) {
let testItem = controller.createTestItem(testDescr.name, testDescr.name);
controller.items.add(testItem);
}
}
export async function testRunHandler(
controller: vscode.TestController,
request: vscode.TestRunRequest,
token: vscode.CancellationToken,
) {
const run = controller.createTestRun(request, undefined, false);
const queue: vscode.TestItem[] = [];
if (request.include) {
request.include.forEach((test) => queue.push(test));
} else {
controller.items.forEach((test) => queue.push(test));
}
const buildDir = workspaceState.get<string>("mesonbuild.buildDir")!;
for (let test of queue) {
run.started(test);
let starttime = Date.now();
try {
await exec(
extensionConfiguration("mesonPath"),
["test", "-C", buildDir, "--print-errorlog", `"${test.id}"`],
extensionConfiguration("testEnvironment"),
);
let duration = Date.now() - starttime;
run.passed(test, duration);
} catch (e) {
const execResult = e as ExecResult;
run.appendOutput(execResult.stdout);
let duration = Date.now() - starttime;
if (execResult.error?.code == 125) {
vscode.window.showErrorMessage("Failed to build tests. Results will not be updated");
run.errored(test, new vscode.TestMessage(execResult.stderr));
} else {
run.failed(test, new vscode.TestMessage(execResult.stderr), duration);
}
}
}
run.end();
}
export async function testDebugHandler(
controller: vscode.TestController,
request: vscode.TestRunRequest,
token: vscode.CancellationToken,
) {
const run = controller.createTestRun(request, undefined, false);
const queue: vscode.TestItem[] = [];
if (request.include) {
request.include.forEach((test) => queue.push(test));
} else {
controller.items.forEach((test) => queue.push(test));
}
const buildDir = workspaceState.get<string>("mesonbuild.buildDir")!;
const tests: Tests = await getMesonTests(buildDir);
const targets = await getMesonTargets(buildDir);
/* while meson has the --gdb arg to test, but IMO we should go the actual debugger route.
* We still want stuff to be built though... Without going through weird dances */
const relevantTests = tests.filter((test) => queue.some((candidate) => candidate.id == test.name));
const requiredTargets = targets.filter((target) =>
relevantTests.some((test) => test.depends.some((dep) => dep == target.id)),
);
let args = ["compile", "-C", buildDir];
requiredTargets.forEach((target) => {
args.push(target.name);
});
try {
await exec(extensionConfiguration("mesonPath"), args);
} catch (e) {
vscode.window.showErrorMessage("Failed to build tests. Results will not be updated");
run.end();
return;
}
let debugType = null;
const cppTools = vscode.extensions.getExtension("ms-vscode.cpptools");
if (cppTools && cppTools.isActive) {
debugType = "cppdbg";
} else {
const codelldb = vscode.extensions.getExtension("vadimcn.vscode-lldb");
if (codelldb && codelldb.isActive) {
debugType = "lldb";
}
}
if (!debugType) {
vscode.window.showErrorMessage("No debugger extension found. Please install one and try again");
run.end();
return;
}
const configDebugOptions = extensionConfiguration("debugOptions");
const sourceDir = workspaceState.get<string>("mesonbuild.sourceDir")!;
/* We already figured out which tests we want to run.
* We don't use the actual test either way, as we don't get the result here... */
for (let test of relevantTests) {
let args = [...test.cmd];
args.shift();
let debugEnvironmentConfiguration: DebugEnvironmentConfiguration;
/* cppdbg uses 'environment' key, all others use 'env' key */
if (debugType == "cppdbg") {
const debugEnv = [];
if (test.env instanceof Object) {
/* convert from dict of key = value to array of {name: key, value: value} */
for (const [key, val] of Object.entries(test.env)) {
debugEnv.push({ name: key, value: val });
}
}
debugEnvironmentConfiguration = {
environment: debugEnv,
};
} else {
debugEnvironmentConfiguration = {
env: test.env,
};
}
let debugConfiguration = {
name: `meson-debug-${test.name}`,
type: debugType,
request: "launch",
cwd: test.workdir || sourceDir,
program: test.cmd[0],
args: args,
};
const configuration = {
...debugConfiguration,
...debugEnvironmentConfiguration,
...configDebugOptions,
};
await vscode.debug.startDebugging(undefined, configuration);
}
run.end();
}