测试 API
测试 API 允许 Visual Studio Code 扩展发现工作区中的测试并发布结果。用户可以在“测试资源管理器”视图、装饰器和命令内部执行测试。借助这些新 API,Visual Studio Code 可以比以往任何时候都更丰富地显示输出和差异。
注意:测试 API 在 VS Code 版本 1.59 及更高版本中可用。
示例
VS Code 团队维护着两个测试提供程序
发现测试
测试由 TestController 提供,需要一个全局唯一 ID 和易于阅读的标签来创建。
const controller = vscode.tests.createTestController(
'helloWorldTests',
'Hello World Tests'
);
要发布测试,您需要将 TestItem 添加为控制器 items 集合的子项。TestItem 是 TestItem 接口中测试 API 的基础,它是一个通用类型,可以描述代码中存在的测试用例、套件或树项。它们反过来可以拥有自己的 children,从而形成一个层次结构。例如,以下是示例测试扩展创建测试的简化版本:
parseMarkdown(content, {
onTest: (range, numberA, mathOperator, numberB, expectedValue) => {
// If this is a top-level test, add it to its parent's children. If not,
// add it to the controller's top level items.
const collection = parent ? parent.children : controller.items;
// Create a new ID that's unique among the parent's children:
const id = [numberA, mathOperator, numberB, expectedValue].join(' ');
// Finally, create the test item:
const test = controller.createTestItem(id, data.getLabel(), item.uri);
test.range = range;
collection.add(test);
}
// ...
});
与 Diagnostics 类似,测试的发现主要由扩展控制。一个简单的扩展可能会在激活时监视整个工作区并解析所有文件中的所有测试。但是,立即解析所有内容对于大型工作区来说可能会很慢。您可以采取以下两种措施:
- 通过监视
vscode.workspace.onDidOpenTextDocument,在文件编辑器中打开时主动发现该文件的测试。 - 设置
item.canResolveChildren = true并设置controller.resolveHandler。如果用户采取了要求发现测试的操作(例如,通过在“测试资源管理器”中展开某个项),则会调用resolveHandler。
以下是在惰性解析文件的扩展中,此策略可能是什么样的:
// First, create the `resolveHandler`. This may initially be called with
// "undefined" to ask for all tests in the workspace to be discovered, usually
// when the user opens the Test Explorer for the first time.
controller.resolveHandler = async test => {
if (!test) {
await discoverAllFilesInWorkspace();
} else {
await parseTestsInFileContents(test);
}
};
// When text documents are open, parse tests in them.
vscode.workspace.onDidOpenTextDocument(parseTestsInDocument);
// We could also listen to document changes to re-parse unsaved changes:
vscode.workspace.onDidChangeTextDocument(e => parseTestsInDocument(e.document));
// In this function, we'll get the file TestItem if we've already found it,
// otherwise we'll create it with `canResolveChildren = true` to indicate it
// can be passed to the `controller.resolveHandler` to gets its children.
function getOrCreateFile(uri: vscode.Uri) {
const existing = controller.items.get(uri.toString());
if (existing) {
return existing;
}
const file = controller.createTestItem(uri.toString(), uri.path.split('/').pop()!, uri);
file.canResolveChildren = true;
return file;
}
function parseTestsInDocument(e: vscode.TextDocument) {
if (e.uri.scheme === 'file' && e.uri.path.endsWith('.md')) {
parseTestsInFileContents(getOrCreateFile(e.uri), e.getText());
}
}
async function parseTestsInFileContents(file: vscode.TestItem, contents?: string) {
// If a document is open, VS Code already knows its contents. If this is being
// called from the resolveHandler when a document isn't open, we'll need to
// read them from disk ourselves.
if (contents === undefined) {
const rawContent = await vscode.workspace.fs.readFile(file.uri);
contents = new TextDecoder().decode(rawContent);
}
// some custom logic to fill in test.children from the contents...
}
discoverAllFilesInWorkspace 的实现可以使用 VS Code 现有的文件监视功能来构建。当调用 resolveHandler 时,您应该继续监视更改,以便“测试资源管理器”中的数据保持最新。
async function discoverAllFilesInWorkspace() {
if (!vscode.workspace.workspaceFolders) {
return []; // handle the case of no open folders
}
return Promise.all(
vscode.workspace.workspaceFolders.map(async workspaceFolder => {
const pattern = new vscode.RelativePattern(workspaceFolder, '**/*.md');
const watcher = vscode.workspace.createFileSystemWatcher(pattern);
// When files are created, make sure there's a corresponding "file" node in the tree
watcher.onDidCreate(uri => getOrCreateFile(uri));
// When files change, re-parse them. Note that you could optimize this so
// that you only re-parse children that have been resolved in the past.
watcher.onDidChange(uri => parseTestsInFileContents(getOrCreateFile(uri)));
// And, finally, delete TestItems for removed files. This is simple, since
// we use the URI as the TestItem's ID.
watcher.onDidDelete(uri => controller.items.delete(uri.toString()));
for (const file of await vscode.workspace.findFiles(pattern)) {
getOrCreateFile(file);
}
return watcher;
})
);
}
TestItem 接口很简单,没有自定义数据的空间。如果您需要将额外信息与 TestItem 相关联,可以使用 WeakMap。
const testData = new WeakMap<vscode.TestItem, MyCustomData>();
// to associate data:
const item = controller.createTestItem(id, label);
testData.set(item, new MyCustomData());
// to get it back later:
const myData = testData.get(item);
可以保证传递给所有 TestController 相关方法的 TestItem 实例与从 createTestItem 最初创建的实例是相同的,因此您可以确定从 testData map 中获取的项是有效的。
在本示例中,我们只存储每个项的类型。
enum ItemType {
File,
TestCase
}
const testData = new WeakMap<vscode.TestItem, ItemType>();
const getType = (testItem: vscode.TestItem) => testData.get(testItem)!;
运行测试
测试通过 TestRunProfile 执行。每个配置文件都属于一个特定的执行 kind:运行 (run)、调试 (debug) 或覆盖率 (coverage)。大多数测试扩展在这几组中最多有一个配置文件,但允许更多。例如,如果您的扩展在多个平台上运行测试,您可以为每个平台和 kind 的组合创建一个配置文件。每个配置文件都有一个 runHandler,在请求该类型的运行时会调用它。
function runHandler(
shouldDebug: boolean,
request: vscode.TestRunRequest,
token: vscode.CancellationToken
) {
// todo
}
const runProfile = controller.createRunProfile(
'Run',
vscode.TestRunProfileKind.Run,
(request, token) => {
runHandler(false, request, token);
}
);
const debugProfile = controller.createRunProfile(
'Debug',
vscode.TestRunProfileKind.Debug,
(request, token) => {
runHandler(true, request, token);
}
);
runHandler 至少应调用一次 controller.createTestRun,并传递原始请求。请求包含要 include 在测试运行中的测试(如果用户请求运行所有测试,则省略),以及可能要 exclude 从运行中排除的测试。扩展应使用生成的 TestRun 对象来更新测试运行所涉及的测试的状态。例如:
async function runHandler(
shouldDebug: boolean,
request: vscode.TestRunRequest,
token: vscode.CancellationToken
) {
const run = controller.createTestRun(request);
const queue: vscode.TestItem[] = [];
// Loop through all included tests, or all known tests, and add them to our queue
if (request.include) {
request.include.forEach(test => queue.push(test));
} else {
controller.items.forEach(test => queue.push(test));
}
// For every test that was queued, try to run it. Call run.passed() or run.failed().
// The `TestMessage` can contain extra information, like a failing location or
// a diff output. But here we'll just give it a textual message.
while (queue.length > 0 && !token.isCancellationRequested) {
const test = queue.pop()!;
// Skip tests the user asked to exclude
if (request.exclude?.includes(test)) {
continue;
}
switch (getType(test)) {
case ItemType.File:
// If we're running a file and don't know what it contains yet, parse it now
if (test.children.size === 0) {
await parseTestsInFileContents(test);
}
break;
case ItemType.TestCase:
// Otherwise, just run the test case. Note that we don't need to manually
// set the state of parent tests; they'll be set automatically.
const start = Date.now();
try {
await assertTestPasses(test);
run.passed(test, Date.now() - start);
} catch (e) {
run.failed(test, new vscode.TestMessage(e.message), Date.now() - start);
}
break;
}
test.children.forEach(test => queue.push(test));
}
// Make sure to end the run after all tests have been executed:
run.end();
}
除了 runHandler 之外,您还可以为 TestRunProfile 设置 configureHandler。如果存在,VS Code 将提供 UI 以允许用户配置测试运行,并在用户进行配置时调用该处理程序。在此,您可以打开文件、显示快速选择菜单,或执行适合您的测试框架的任何操作。
VS Code 故意以不同于调试或任务配置的方式处理测试配置。这些传统上是编辑器或 IDE 的核心功能,并且在
.vscode文件夹中的特殊文件中进行配置。然而,测试传统上是从命令行执行的,并且大多数测试框架都有现有的配置策略。因此,在 VS Code 中,我们避免重复配置,而是将其留给扩展来处理。
测试输出
除了传递给 TestRun.failed 或 TestRun.errored 的消息之外,您还可以使用 run.appendOutput(str) 追加通用输出。此输出可以在终端中使用“测试: 显示输出”以及 UI 中的各种按钮(如“测试资源管理器”视图中的终端图标)进行显示。
由于字符串将在终端中渲染,因此您可以使用完整的ANSI 代码集,包括ansi-styles npm 包中可用的样式。请记住,由于它在终端中,行必须使用 CRLF (\r\n) 换行,而不是仅使用 LF (\n),后者可能是某些工具的默认输出。
测试覆盖率
测试覆盖率通过 run.addCoverage() 方法与 TestRun 相关联。通常,这应该由 TestRunProfileKind.Coverage 配置文件的 runHandler 完成,但也可以在任何测试运行期间调用它。addCoverage 方法接受一个 FileCoverage 对象,该对象是该文件中覆盖率数据的摘要。
async function runHandler(
shouldDebug: boolean,
request: vscode.TestRunRequest,
token: vscode.CancellationToken
) {
// ...
for await (const file of readCoverageOutput()) {
run.addCoverage(new vscode.FileCoverage(file.uri, file.statementCoverage));
}
}
FileCoverage 包含每个文件中语句、分支和声明的总体覆盖和未覆盖计数。根据您的运行时和覆盖率格式,您可能会看到语句覆盖率称为行覆盖率,或声明覆盖率称为函数或方法覆盖率。您可以为单个 URI 添加多次文件覆盖率,在这种情况下,新信息将替换旧信息。
一旦用户打开一个带有覆盖率的文件或在“测试覆盖率”视图中展开一个文件,VS Code 就会请求该文件的更多信息。它通过在 TestRunProfile 上调用扩展定义的 loadDetailedCoverage 方法来完成此操作,并传递 TestRun、FileCoverage 和 CancellationToken。请注意,测试运行和文件覆盖率实例与 run.addCoverage 中使用的实例相同,这对于关联数据很有用。例如,您可以创建 FileCoverage 对象到您自己的数据的映射。
const coverageData = new WeakMap<vscode.FileCoverage, MyCoverageDetails>();
profile.loadDetailedCoverage = (testRun, fileCoverage, token) => {
return coverageData.get(fileCoverage).load(token);
};
async function runHandler(
shouldDebug: boolean,
request: vscode.TestRunRequest,
token: vscode.CancellationToken
) {
// ...
for await (const file of readCoverageOutput()) {
const coverage = new vscode.FileCoverage(file.uri, file.statementCoverage);
coverageData.set(coverage, file);
run.addCoverage(coverage);
}
}
或者,您可能需要继承 FileCoverage 并实现包含该数据。
class MyFileCoverage extends vscode.FileCoverage {
// ...
}
profile.loadDetailedCoverage = async (testRun, fileCoverage, token) => {
return fileCoverage instanceof MyFileCoverage ? await fileCoverage.load() : [];
};
async function runHandler(
shouldDebug: boolean,
request: vscode.TestRunRequest,
token: vscode.CancellationToken
) {
// ...
for await (const file of readCoverageOutput()) {
// 'file' is MyFileCoverage:
run.addCoverage(file);
}
}
loadDetailedCoverage 预计会返回一个 Promise,该 Promise 解析为 DeclarationCoverage 和/或 StatementCoverage 对象数组。这两个对象都包含它们在源文件中可以找到的 Position 或 Range。DeclarationCoverage 对象包含被声明内容的名称(如函数或方法名称)以及该声明被输入或调用的次数。语句包括它们被执行的次数,以及零个或多个关联的分支。有关更多信息,请参阅 vscode.d.ts 中的类型定义。
在许多情况下,您可能会在测试运行中留下持久化的文件。最佳实践是将此类覆盖率输出放在系统的临时目录中(可以通过 require('os').tmpdir() 获取),但您也可以通过侦听 VS Code 不再需要保留测试运行的提示来积极地清理它们。
import { promises as fs } from 'fs';
async function runHandler(
shouldDebug: boolean,
request: vscode.TestRunRequest,
token: vscode.CancellationToken
) {
// ...
run.onDidDispose(async () => {
await fs.rm(coverageOutputDirectory, { recursive: true, force: true });
});
}
测试标签
有时测试只能在特定配置下运行,或者根本不能运行。对于这些用例,您可以使用测试标签。TestRunProfile 可以选择性地带有一个关联的标签,如果它们有,则只有具有该标签的测试才能在该配置文件下运行。同样,如果不存在可以运行、调试或收集特定测试覆盖率的合格配置文件,则 UI 中将不会显示这些选项。
// Create a new tag with an ID of "runnable"
const runnableTag = new TestTag('runnable');
// Assign it to a profile. Now this profile can only execute tests with that tag.
runProfile.tag = runnableTag;
// Add the "runnable" tag to all applicable tests.
for (const test of getAllRunnableTests()) {
test.tags = [...test.tags, runnableTag];
}
用户也可以在“测试资源管理器”UI 中按标签进行筛选。
仅发布控制器
运行配置文件的存在是可选的。控制器可以创建测试,在 runHandler 之外调用 createTestRun,并更新运行中测试的状态,而无需配置文件。这种情况的常见用例是加载其结果来自外部源(如 CI 或摘要文件)的控制器。
在这种情况下,这些控制器通常应将可选的 name 参数传递给 createTestRun,并将 persist 参数传递为 false。将 false 传递给这里会指示 VS Code 不要保留测试结果,就像对待编辑器中的运行一样,因为这些结果可以从外部重新加载。
const controller = vscode.tests.createTestController(
'myCoverageFileTests',
'Coverage File Tests'
);
vscode.commands.registerCommand('myExtension.loadTestResultFile', async file => {
const info = await readFile(file);
// set the controller items to those read from the file:
controller.items.replace(readTestsFromInfo(info));
// create your own custom test run, then you can immediately set the state of
// items in the run and end it to publish results:
const run = controller.createTestRun(
new vscode.TestRunRequest(),
path.basename(file),
false
);
for (const result of info) {
if (result.passed) {
run.passed(result.item);
} else {
run.failed(result.item, new vscode.TestMessage(result.message));
}
}
run.end();
});
从 Test Explorer UI 迁移
如果您有一个使用 Test Explorer UI 的现有扩展,我们建议您迁移到原生体验以获得更多功能和效率。我们整理了一个包含 Test Adapter 示例迁移的存储库,其Git 历史记录。您可以通过选择提交名称(从 [1] Create a native TestController 开始)来查看每个步骤。
总之,一般步骤是:
-
而不是检索
TestAdapter并将其注册到 Test Explorer UI 的TestHub,而是调用const controller = vscode.tests.createTestController(...)。 -
不要在发现或重新发现测试时触发
testAdapter.tests,而是将测试创建并推送到controller.items中,例如通过调用controller.items.replace并传入一个通过调用vscode.test.createTestItem创建的已发现测试数组。请注意,随着测试的变化,您可以修改测试项的属性并更新其子项,更改将自动反映在 VS Code 的 UI 中。 -
要初步加载测试,而不是等待
testAdapter.load()方法调用,请设置controller.resolveHandler = () => { /* discover tests */ }。有关测试发现如何工作的更多信息,请参阅发现测试。 -
要运行测试,您应该创建一个具有处理程序函数的运行配置文件,该函数调用
const run = controller.createTestRun(request)。而不是触发testStates事件,而是将TestItem传递给run上的方法来更新它们的状态。
其他贡献点
testing/item/context 菜单贡献点可用于向“测试资源管理器”视图中的测试添加菜单项。将菜单项放置在 inline 组中即可使其内联显示。所有其他菜单项组将显示在可通过鼠标右键单击访问的上下文菜单中。
在菜单项的 when 子句中,可以使用其他上下文键:testId、controllerId 和 testItemHasUri。对于更复杂的 when 场景,如果您希望操作对不同的 Test Item 可选可用,请考虑使用in 条件运算符。
如果您想在资源管理器中显示一个测试,可以将该测试传递给命令 vscode.commands.executeCommand('vscode.revealTestInExplorer', testItem)。