mirror of
https://github.com/microsoft/vscode.git
synced 2026-05-08 17:19:48 +01:00
526 lines
20 KiB
TypeScript
526 lines
20 KiB
TypeScript
/*---------------------------------------------------------------------------------------------
|
|
* Copyright (c) Microsoft Corporation. All rights reserved.
|
|
* Licensed under the MIT License. See License.txt in the project root for license information.
|
|
*--------------------------------------------------------------------------------------------*/
|
|
|
|
'use strict';
|
|
|
|
import * as assert from 'assert';
|
|
import { setUnexpectedErrorHandler, errorHandler } from 'vs/base/common/errors';
|
|
import { TestInstantiationService } from 'vs/platform/instantiation/test/common/instantiationServiceMock';
|
|
import URI from 'vs/base/common/uri';
|
|
import { TPromise } from 'vs/base/common/winjs.base';
|
|
import * as types from 'vs/workbench/api/node/extHostTypes';
|
|
import { TextModel as EditorModel } from 'vs/editor/common/model/textModel';
|
|
import { TestRPCProtocol } from './testRPCProtocol';
|
|
import { MarkerService } from 'vs/platform/markers/common/markerService';
|
|
import { IMarkerService } from 'vs/platform/markers/common/markers';
|
|
import { ICommandService, CommandsRegistry } from 'vs/platform/commands/common/commands';
|
|
import { IModelService } from 'vs/editor/common/services/modelService';
|
|
import { ExtHostLanguageFeatures } from 'vs/workbench/api/node/extHostLanguageFeatures';
|
|
import { MainThreadLanguageFeatures } from 'vs/workbench/api/electron-browser/mainThreadLanguageFeatures';
|
|
import { IHeapService } from 'vs/workbench/api/electron-browser/mainThreadHeapService';
|
|
import { ExtHostApiCommands } from 'vs/workbench/api/node/extHostApiCommands';
|
|
import { ExtHostCommands } from 'vs/workbench/api/node/extHostCommands';
|
|
import { ExtHostHeapService } from 'vs/workbench/api/node/extHostHeapService';
|
|
import { MainThreadCommands } from 'vs/workbench/api/electron-browser/mainThreadCommands';
|
|
import { ExtHostDocuments } from 'vs/workbench/api/node/extHostDocuments';
|
|
import { ExtHostDocumentsAndEditors } from 'vs/workbench/api/node/extHostDocumentsAndEditors';
|
|
import { MainContext, ExtHostContext } from 'vs/workbench/api/node/extHost.protocol';
|
|
import { ExtHostDiagnostics } from 'vs/workbench/api/node/extHostDiagnostics';
|
|
import * as vscode from 'vscode';
|
|
import { IInstantiationService } from 'vs/platform/instantiation/common/instantiation';
|
|
import 'vs/workbench/parts/search/electron-browser/search.contribution';
|
|
import { NullLogService } from 'vs/platform/log/common/log';
|
|
import { ITextModel } from 'vs/editor/common/model';
|
|
import { ExtHostWorkspace } from 'vs/workbench/api/node/extHostWorkspace';
|
|
import { generateUuid } from 'vs/base/common/uuid';
|
|
import { ExtHostTask } from 'vs/workbench/api/node/extHostTask';
|
|
|
|
const defaultSelector = { scheme: 'far' };
|
|
const model: ITextModel = EditorModel.createFromString(
|
|
[
|
|
'This is the first line',
|
|
'This is the second line',
|
|
'This is the third line',
|
|
].join('\n'),
|
|
undefined,
|
|
undefined,
|
|
URI.parse('far://testing/file.b'));
|
|
|
|
let rpcProtocol: TestRPCProtocol;
|
|
let extHost: ExtHostLanguageFeatures;
|
|
let mainThread: MainThreadLanguageFeatures;
|
|
let commands: ExtHostCommands;
|
|
let task: ExtHostTask;
|
|
let workspace: ExtHostWorkspace;
|
|
let disposables: vscode.Disposable[] = [];
|
|
let originalErrorHandler: (e: any) => any;
|
|
|
|
suite('ExtHostLanguageFeatureCommands', function () {
|
|
|
|
suiteSetup(() => {
|
|
|
|
originalErrorHandler = errorHandler.getUnexpectedErrorHandler();
|
|
setUnexpectedErrorHandler(() => { });
|
|
|
|
// Use IInstantiationService to get typechecking when instantiating
|
|
let inst: IInstantiationService;
|
|
{
|
|
let instantiationService = new TestInstantiationService();
|
|
rpcProtocol = new TestRPCProtocol();
|
|
instantiationService.stub(IHeapService, {
|
|
_serviceBrand: undefined,
|
|
trackRecursive(args) {
|
|
// nothing
|
|
return args;
|
|
}
|
|
});
|
|
instantiationService.stub(ICommandService, {
|
|
_serviceBrand: undefined,
|
|
executeCommand(id, args): any {
|
|
if (!CommandsRegistry.getCommands()[id]) {
|
|
return TPromise.wrapError(new Error(id + ' NOT known'));
|
|
}
|
|
let { handler } = CommandsRegistry.getCommands()[id];
|
|
return TPromise.as(instantiationService.invokeFunction(handler, args));
|
|
}
|
|
});
|
|
instantiationService.stub(IMarkerService, new MarkerService());
|
|
instantiationService.stub(IModelService, <IModelService>{
|
|
_serviceBrand: IModelService,
|
|
getModel(): any { return model; },
|
|
createModel(): any { throw new Error(); },
|
|
updateModel(): any { throw new Error(); },
|
|
setMode(): any { throw new Error(); },
|
|
destroyModel(): any { throw new Error(); },
|
|
getModels(): any { throw new Error(); },
|
|
onModelAdded: undefined,
|
|
onModelModeChanged: undefined,
|
|
onModelRemoved: undefined,
|
|
getCreationOptions(): any { throw new Error(); }
|
|
});
|
|
inst = instantiationService;
|
|
}
|
|
|
|
const extHostDocumentsAndEditors = new ExtHostDocumentsAndEditors(rpcProtocol);
|
|
extHostDocumentsAndEditors.$acceptDocumentsAndEditorsDelta({
|
|
addedDocuments: [{
|
|
isDirty: false,
|
|
versionId: model.getVersionId(),
|
|
modeId: model.getLanguageIdentifier().language,
|
|
uri: model.uri,
|
|
lines: model.getValue().split(model.getEOL()),
|
|
EOL: model.getEOL(),
|
|
}]
|
|
});
|
|
const extHostDocuments = new ExtHostDocuments(rpcProtocol, extHostDocumentsAndEditors);
|
|
rpcProtocol.set(ExtHostContext.ExtHostDocuments, extHostDocuments);
|
|
|
|
const heapService = new ExtHostHeapService();
|
|
|
|
commands = new ExtHostCommands(rpcProtocol, heapService, new NullLogService());
|
|
workspace = new ExtHostWorkspace(rpcProtocol, { id: generateUuid(), name: 'Test', folders: [] }, new NullLogService());
|
|
task = new ExtHostTask(rpcProtocol, workspace);
|
|
rpcProtocol.set(ExtHostContext.ExtHostCommands, commands);
|
|
rpcProtocol.set(MainContext.MainThreadCommands, inst.createInstance(MainThreadCommands, rpcProtocol));
|
|
ExtHostApiCommands.register(commands, task);
|
|
|
|
const diagnostics = new ExtHostDiagnostics(rpcProtocol);
|
|
rpcProtocol.set(ExtHostContext.ExtHostDiagnostics, diagnostics);
|
|
|
|
extHost = new ExtHostLanguageFeatures(rpcProtocol, extHostDocuments, commands, heapService, diagnostics);
|
|
rpcProtocol.set(ExtHostContext.ExtHostLanguageFeatures, extHost);
|
|
|
|
mainThread = rpcProtocol.set(MainContext.MainThreadLanguageFeatures, inst.createInstance(MainThreadLanguageFeatures, rpcProtocol));
|
|
|
|
return rpcProtocol.sync();
|
|
});
|
|
|
|
suiteTeardown(() => {
|
|
setUnexpectedErrorHandler(originalErrorHandler);
|
|
model.dispose();
|
|
mainThread.dispose();
|
|
});
|
|
|
|
teardown(function () {
|
|
while (disposables.length) {
|
|
disposables.pop().dispose();
|
|
}
|
|
return rpcProtocol.sync();
|
|
});
|
|
|
|
// --- workspace symbols
|
|
|
|
test('WorkspaceSymbols, invalid arguments', function () {
|
|
let promises = [
|
|
commands.executeCommand('vscode.executeWorkspaceSymbolProvider'),
|
|
commands.executeCommand('vscode.executeWorkspaceSymbolProvider', null),
|
|
commands.executeCommand('vscode.executeWorkspaceSymbolProvider', undefined),
|
|
commands.executeCommand('vscode.executeWorkspaceSymbolProvider', true)
|
|
];
|
|
|
|
return TPromise.join(<any[]>promises).then(undefined, (err: any[]) => {
|
|
assert.equal(err.length, 4);
|
|
});
|
|
});
|
|
|
|
test('WorkspaceSymbols, back and forth', function () {
|
|
|
|
disposables.push(extHost.registerWorkspaceSymbolProvider(<vscode.WorkspaceSymbolProvider>{
|
|
provideWorkspaceSymbols(query): any {
|
|
return [
|
|
new types.SymbolInformation(query, types.SymbolKind.Array, new types.Range(0, 0, 1, 1), URI.parse('far://testing/first')),
|
|
new types.SymbolInformation(query, types.SymbolKind.Array, new types.Range(0, 0, 1, 1), URI.parse('far://testing/second'))
|
|
];
|
|
}
|
|
}));
|
|
|
|
disposables.push(extHost.registerWorkspaceSymbolProvider(<vscode.WorkspaceSymbolProvider>{
|
|
provideWorkspaceSymbols(query): any {
|
|
return [
|
|
new types.SymbolInformation(query, types.SymbolKind.Array, new types.Range(0, 0, 1, 1), URI.parse('far://testing/first'))
|
|
];
|
|
}
|
|
}));
|
|
|
|
return rpcProtocol.sync().then(() => {
|
|
return commands.executeCommand<vscode.SymbolInformation[]>('vscode.executeWorkspaceSymbolProvider', 'testing').then(value => {
|
|
|
|
for (let info of value) {
|
|
assert.ok(info instanceof types.SymbolInformation);
|
|
assert.equal(info.name, 'testing');
|
|
assert.equal(info.kind, types.SymbolKind.Array);
|
|
}
|
|
assert.equal(value.length, 3);
|
|
});
|
|
});
|
|
});
|
|
|
|
test('executeWorkspaceSymbolProvider should accept empty string, #39522', async function () {
|
|
|
|
disposables.push(extHost.registerWorkspaceSymbolProvider({
|
|
provideWorkspaceSymbols(query) {
|
|
return [new types.SymbolInformation('hello', types.SymbolKind.Array, new types.Range(0, 0, 0, 0), URI.parse('foo:bar'))];
|
|
}
|
|
}));
|
|
|
|
await rpcProtocol.sync();
|
|
let symbols = await commands.executeCommand<vscode.SymbolInformation[]>('vscode.executeWorkspaceSymbolProvider', '');
|
|
assert.equal(symbols.length, 1);
|
|
|
|
await rpcProtocol.sync();
|
|
symbols = await commands.executeCommand<vscode.SymbolInformation[]>('vscode.executeWorkspaceSymbolProvider', '*');
|
|
assert.equal(symbols.length, 1);
|
|
});
|
|
|
|
// --- definition
|
|
|
|
test('Definition, invalid arguments', function () {
|
|
let promises = [
|
|
commands.executeCommand('vscode.executeDefinitionProvider'),
|
|
commands.executeCommand('vscode.executeDefinitionProvider', null),
|
|
commands.executeCommand('vscode.executeDefinitionProvider', undefined),
|
|
commands.executeCommand('vscode.executeDefinitionProvider', true, false)
|
|
];
|
|
|
|
return TPromise.join(<any[]>promises).then(undefined, (err: any[]) => {
|
|
assert.equal(err.length, 4);
|
|
});
|
|
});
|
|
|
|
test('Definition, back and forth', function () {
|
|
|
|
disposables.push(extHost.registerDefinitionProvider(defaultSelector, <vscode.DefinitionProvider>{
|
|
provideDefinition(doc: any): any {
|
|
return new types.Location(doc.uri, new types.Range(0, 0, 0, 0));
|
|
}
|
|
}));
|
|
disposables.push(extHost.registerDefinitionProvider(defaultSelector, <vscode.DefinitionProvider>{
|
|
provideDefinition(doc: any): any {
|
|
return [
|
|
new types.Location(doc.uri, new types.Range(0, 0, 0, 0)),
|
|
new types.Location(doc.uri, new types.Range(0, 0, 0, 0)),
|
|
new types.Location(doc.uri, new types.Range(0, 0, 0, 0)),
|
|
];
|
|
}
|
|
}));
|
|
|
|
return rpcProtocol.sync().then(() => {
|
|
return commands.executeCommand<vscode.Location[]>('vscode.executeDefinitionProvider', model.uri, new types.Position(0, 0)).then(values => {
|
|
assert.equal(values.length, 4);
|
|
for (let v of values) {
|
|
assert.ok(v.uri instanceof URI);
|
|
assert.ok(v.range instanceof types.Range);
|
|
}
|
|
});
|
|
});
|
|
});
|
|
|
|
// --- type definition
|
|
|
|
test('Type Definition, invalid arguments', function () {
|
|
const promises = [
|
|
commands.executeCommand('vscode.executeTypeDefinitionProvider'),
|
|
commands.executeCommand('vscode.executeTypeDefinitionProvider', null),
|
|
commands.executeCommand('vscode.executeTypeDefinitionProvider', undefined),
|
|
commands.executeCommand('vscode.executeTypeDefinitionProvider', true, false)
|
|
];
|
|
|
|
return TPromise.join(<any[]>promises).then(undefined, (err: any[]) => {
|
|
assert.equal(err.length, 4);
|
|
});
|
|
});
|
|
|
|
test('Type Definition, back and forth', function () {
|
|
|
|
disposables.push(extHost.registerTypeDefinitionProvider(defaultSelector, <vscode.TypeDefinitionProvider>{
|
|
provideTypeDefinition(doc: any): any {
|
|
return new types.Location(doc.uri, new types.Range(0, 0, 0, 0));
|
|
}
|
|
}));
|
|
disposables.push(extHost.registerTypeDefinitionProvider(defaultSelector, <vscode.TypeDefinitionProvider>{
|
|
provideTypeDefinition(doc: any): any {
|
|
return [
|
|
new types.Location(doc.uri, new types.Range(0, 0, 0, 0)),
|
|
new types.Location(doc.uri, new types.Range(0, 0, 0, 0)),
|
|
new types.Location(doc.uri, new types.Range(0, 0, 0, 0)),
|
|
];
|
|
}
|
|
}));
|
|
|
|
return rpcProtocol.sync().then(() => {
|
|
return commands.executeCommand<vscode.Location[]>('vscode.executeTypeDefinitionProvider', model.uri, new types.Position(0, 0)).then(values => {
|
|
assert.equal(values.length, 4);
|
|
for (const v of values) {
|
|
assert.ok(v.uri instanceof URI);
|
|
assert.ok(v.range instanceof types.Range);
|
|
}
|
|
});
|
|
});
|
|
});
|
|
|
|
// --- references
|
|
|
|
test('reference search, back and forth', function () {
|
|
|
|
disposables.push(extHost.registerReferenceProvider(defaultSelector, <vscode.ReferenceProvider>{
|
|
provideReferences(doc: any) {
|
|
return [
|
|
new types.Location(URI.parse('some:uri/path'), new types.Range(0, 1, 0, 5))
|
|
];
|
|
}
|
|
}));
|
|
|
|
return commands.executeCommand<vscode.Location[]>('vscode.executeReferenceProvider', model.uri, new types.Position(0, 0)).then(values => {
|
|
assert.equal(values.length, 1);
|
|
let [first] = values;
|
|
assert.equal(first.uri.toString(), 'some:uri/path');
|
|
assert.equal(first.range.start.line, 0);
|
|
assert.equal(first.range.start.character, 1);
|
|
assert.equal(first.range.end.line, 0);
|
|
assert.equal(first.range.end.character, 5);
|
|
});
|
|
});
|
|
|
|
// --- outline
|
|
|
|
test('Outline, back and forth', function () {
|
|
disposables.push(extHost.registerDocumentSymbolProvider(defaultSelector, <vscode.DocumentSymbolProvider>{
|
|
provideDocumentSymbols(): any {
|
|
return [
|
|
new types.SymbolInformation('testing1', types.SymbolKind.Enum, new types.Range(1, 0, 1, 0)),
|
|
new types.SymbolInformation('testing2', types.SymbolKind.Enum, new types.Range(0, 1, 0, 3)),
|
|
];
|
|
}
|
|
}));
|
|
|
|
return rpcProtocol.sync().then(() => {
|
|
return commands.executeCommand<vscode.SymbolInformation[]>('vscode.executeDocumentSymbolProvider', model.uri).then(values => {
|
|
assert.equal(values.length, 2);
|
|
let [first, second] = values;
|
|
assert.equal(first.name, 'testing2');
|
|
assert.equal(second.name, 'testing1');
|
|
});
|
|
});
|
|
});
|
|
|
|
// --- suggest
|
|
|
|
test('Suggest, back and forth', function () {
|
|
disposables.push(extHost.registerCompletionItemProvider(defaultSelector, <vscode.CompletionItemProvider>{
|
|
provideCompletionItems(doc, pos): any {
|
|
let a = new types.CompletionItem('item1');
|
|
let b = new types.CompletionItem('item2');
|
|
b.textEdit = types.TextEdit.replace(new types.Range(0, 4, 0, 8), 'foo'); // overwite after
|
|
let c = new types.CompletionItem('item3');
|
|
c.textEdit = types.TextEdit.replace(new types.Range(0, 1, 0, 6), 'foobar'); // overwite before & after
|
|
|
|
// snippet string!
|
|
let d = new types.CompletionItem('item4');
|
|
d.range = new types.Range(0, 1, 0, 4);// overwite before
|
|
d.insertText = new types.SnippetString('foo$0bar');
|
|
return [a, b, c, d];
|
|
}
|
|
}, []));
|
|
|
|
return rpcProtocol.sync().then(() => {
|
|
return commands.executeCommand<vscode.CompletionList>('vscode.executeCompletionItemProvider', model.uri, new types.Position(0, 4)).then(list => {
|
|
|
|
assert.ok(list instanceof types.CompletionList);
|
|
let values = list.items;
|
|
assert.ok(Array.isArray(values));
|
|
assert.equal(values.length, 4);
|
|
let [first, second, third, fourth] = values;
|
|
assert.equal(first.label, 'item1');
|
|
assert.equal(first.textEdit.newText, 'item1');
|
|
assert.equal(first.textEdit.range.start.line, 0);
|
|
assert.equal(first.textEdit.range.start.character, 0);
|
|
assert.equal(first.textEdit.range.end.line, 0);
|
|
assert.equal(first.textEdit.range.end.character, 4);
|
|
|
|
assert.equal(second.label, 'item2');
|
|
assert.equal(second.textEdit.newText, 'foo');
|
|
assert.equal(second.textEdit.range.start.line, 0);
|
|
assert.equal(second.textEdit.range.start.character, 4);
|
|
assert.equal(second.textEdit.range.end.line, 0);
|
|
assert.equal(second.textEdit.range.end.character, 8);
|
|
|
|
assert.equal(third.label, 'item3');
|
|
assert.equal(third.textEdit.newText, 'foobar');
|
|
assert.equal(third.textEdit.range.start.line, 0);
|
|
assert.equal(third.textEdit.range.start.character, 1);
|
|
assert.equal(third.textEdit.range.end.line, 0);
|
|
assert.equal(third.textEdit.range.end.character, 6);
|
|
|
|
assert.equal(fourth.label, 'item4');
|
|
assert.equal(fourth.textEdit, undefined);
|
|
assert.equal(fourth.range.start.line, 0);
|
|
assert.equal(fourth.range.start.character, 1);
|
|
assert.equal(fourth.range.end.line, 0);
|
|
assert.equal(fourth.range.end.character, 4);
|
|
assert.ok(fourth.insertText instanceof types.SnippetString);
|
|
assert.equal((<types.SnippetString>fourth.insertText).value, 'foo$0bar');
|
|
});
|
|
});
|
|
});
|
|
|
|
test('Suggest, return CompletionList !array', function () {
|
|
disposables.push(extHost.registerCompletionItemProvider(defaultSelector, <vscode.CompletionItemProvider>{
|
|
provideCompletionItems(): any {
|
|
let a = new types.CompletionItem('item1');
|
|
let b = new types.CompletionItem('item2');
|
|
return new types.CompletionList(<any>[a, b], true);
|
|
}
|
|
}, []));
|
|
|
|
return rpcProtocol.sync().then(() => {
|
|
return commands.executeCommand<vscode.CompletionList>('vscode.executeCompletionItemProvider', model.uri, new types.Position(0, 4)).then(list => {
|
|
assert.ok(list instanceof types.CompletionList);
|
|
assert.equal(list.isIncomplete, true);
|
|
});
|
|
});
|
|
});
|
|
|
|
// --- quickfix
|
|
|
|
test('QuickFix, back and forth', function () {
|
|
disposables.push(extHost.registerCodeActionProvider(defaultSelector, {
|
|
provideCodeActions(): vscode.Command[] {
|
|
return [{ command: 'testing', title: 'Title', arguments: [1, 2, true] }];
|
|
}
|
|
}));
|
|
|
|
return rpcProtocol.sync().then(() => {
|
|
return commands.executeCommand<vscode.Command[]>('vscode.executeCodeActionProvider', model.uri, new types.Range(0, 0, 1, 1)).then(value => {
|
|
assert.equal(value.length, 1);
|
|
let [first] = value;
|
|
assert.equal(first.title, 'Title');
|
|
assert.equal(first.command, 'testing');
|
|
assert.deepEqual(first.arguments, [1, 2, true]);
|
|
});
|
|
});
|
|
});
|
|
|
|
test('vscode.executeCodeActionProvider results seem to be missing their `command` property #45124', function () {
|
|
disposables.push(extHost.registerCodeActionProvider(defaultSelector, {
|
|
provideCodeActions(document, range): vscode.CodeAction[] {
|
|
return [{
|
|
command: {
|
|
arguments: [document, range],
|
|
command: 'command',
|
|
title: 'command_title',
|
|
},
|
|
kind: types.CodeActionKind.Empty.append('foo'),
|
|
title: 'title',
|
|
}];
|
|
}
|
|
}));
|
|
|
|
return rpcProtocol.sync().then(() => {
|
|
return commands.executeCommand<vscode.CodeAction[]>('vscode.executeCodeActionProvider', model.uri, new types.Range(0, 0, 1, 1)).then(value => {
|
|
assert.equal(value.length, 1);
|
|
let [first] = value;
|
|
assert.ok(first.command);
|
|
assert.equal(first.command.command, 'command');
|
|
assert.equal(first.command.title, 'command_title');
|
|
assert.equal(first.kind.value, 'foo');
|
|
assert.equal(first.title, 'title');
|
|
|
|
});
|
|
});
|
|
});
|
|
|
|
// --- code lens
|
|
|
|
test('CodeLens, back and forth', function () {
|
|
|
|
const complexArg = {
|
|
foo() { },
|
|
bar() { },
|
|
big: extHost
|
|
};
|
|
|
|
disposables.push(extHost.registerCodeLensProvider(defaultSelector, <vscode.CodeLensProvider>{
|
|
provideCodeLenses(): any {
|
|
return [new types.CodeLens(new types.Range(0, 0, 1, 1), { title: 'Title', command: 'cmd', arguments: [1, true, complexArg] })];
|
|
}
|
|
}));
|
|
|
|
return rpcProtocol.sync().then(() => {
|
|
return commands.executeCommand<vscode.CodeLens[]>('vscode.executeCodeLensProvider', model.uri).then(value => {
|
|
assert.equal(value.length, 1);
|
|
let [first] = value;
|
|
|
|
assert.equal(first.command.title, 'Title');
|
|
assert.equal(first.command.command, 'cmd');
|
|
assert.equal(first.command.arguments[0], 1);
|
|
assert.equal(first.command.arguments[1], true);
|
|
assert.equal(first.command.arguments[2], complexArg);
|
|
});
|
|
});
|
|
});
|
|
|
|
test('Links, back and forth', function () {
|
|
|
|
disposables.push(extHost.registerDocumentLinkProvider(defaultSelector, <vscode.DocumentLinkProvider>{
|
|
provideDocumentLinks(): any {
|
|
return [new types.DocumentLink(new types.Range(0, 0, 0, 20), URI.parse('foo:bar'))];
|
|
}
|
|
}));
|
|
|
|
return rpcProtocol.sync().then(() => {
|
|
return commands.executeCommand<vscode.DocumentLink[]>('vscode.executeLinkProvider', model.uri).then(value => {
|
|
assert.equal(value.length, 1);
|
|
let [first] = value;
|
|
|
|
assert.equal(first.target.toString(), 'foo:bar');
|
|
assert.equal(first.range.start.line, 0);
|
|
assert.equal(first.range.start.character, 0);
|
|
assert.equal(first.range.end.line, 0);
|
|
assert.equal(first.range.end.character, 20);
|
|
});
|
|
});
|
|
});
|
|
});
|