Book a Demo!
CoCalc Logo Icon
StoreFeaturesDocsShareSupportNewsAboutPoliciesSign UpSign In
microsoft
GitHub Repository: microsoft/vscode
Path: blob/main/src/vs/editor/contrib/snippet/test/browser/snippetVariables.test.ts
3296 views
1
/*---------------------------------------------------------------------------------------------
2
* Copyright (c) Microsoft Corporation. All rights reserved.
3
* Licensed under the MIT License. See License.txt in the project root for license information.
4
*--------------------------------------------------------------------------------------------*/
5
import assert from 'assert';
6
import * as sinon from 'sinon';
7
import { DisposableStore } from '../../../../../base/common/lifecycle.js';
8
import { sep } from '../../../../../base/common/path.js';
9
import { isWindows } from '../../../../../base/common/platform.js';
10
import { extUriBiasedIgnorePathCase } from '../../../../../base/common/resources.js';
11
import { URI } from '../../../../../base/common/uri.js';
12
import { mock } from '../../../../../base/test/common/mock.js';
13
import { ensureNoDisposablesAreLeakedInTestSuite } from '../../../../../base/test/common/utils.js';
14
import { Selection } from '../../../../common/core/selection.js';
15
import { TextModel } from '../../../../common/model/textModel.js';
16
import { SnippetParser, Variable, VariableResolver } from '../../browser/snippetParser.js';
17
import { ClipboardBasedVariableResolver, CompositeSnippetVariableResolver, ModelBasedVariableResolver, SelectionBasedVariableResolver, TimeBasedVariableResolver, WorkspaceBasedVariableResolver } from '../../browser/snippetVariables.js';
18
import { createTextModel } from '../../../../test/common/testTextModel.js';
19
import { ILabelService } from '../../../../../platform/label/common/label.js';
20
import { IWorkspace, IWorkspaceContextService, toWorkspaceFolder } from '../../../../../platform/workspace/common/workspace.js';
21
import { Workspace } from '../../../../../platform/workspace/test/common/testWorkspace.js';
22
import { toWorkspaceFolders } from '../../../../../platform/workspaces/common/workspaces.js';
23
24
suite('Snippet Variables Resolver', function () {
25
26
27
const labelService = new class extends mock<ILabelService>() {
28
override getUriLabel(uri: URI) {
29
return uri.fsPath;
30
}
31
};
32
33
let model: TextModel;
34
let resolver: VariableResolver;
35
36
setup(function () {
37
model = createTextModel([
38
'this is line one',
39
'this is line two',
40
' this is line three'
41
].join('\n'), undefined, undefined, URI.parse('file:///foo/files/text.txt'));
42
43
resolver = new CompositeSnippetVariableResolver([
44
new ModelBasedVariableResolver(labelService, model),
45
new SelectionBasedVariableResolver(model, new Selection(1, 1, 1, 1), 0, undefined),
46
]);
47
});
48
49
teardown(function () {
50
model.dispose();
51
});
52
53
ensureNoDisposablesAreLeakedInTestSuite();
54
55
56
function assertVariableResolve(resolver: VariableResolver, varName: string, expected?: string) {
57
const snippet = new SnippetParser().parse(`$${varName}`);
58
const variable = <Variable>snippet.children[0];
59
variable.resolve(resolver);
60
if (variable.children.length === 0) {
61
assert.strictEqual(undefined, expected);
62
} else {
63
assert.strictEqual(variable.toString(), expected);
64
}
65
}
66
67
test('editor variables, basics', function () {
68
assertVariableResolve(resolver, 'TM_FILENAME', 'text.txt');
69
assertVariableResolve(resolver, 'something', undefined);
70
});
71
72
test('editor variables, file/dir', function () {
73
74
const disposables = new DisposableStore();
75
76
assertVariableResolve(resolver, 'TM_FILENAME', 'text.txt');
77
if (!isWindows) {
78
assertVariableResolve(resolver, 'TM_DIRECTORY', '/foo/files');
79
assertVariableResolve(resolver, 'TM_FILEPATH', '/foo/files/text.txt');
80
}
81
82
resolver = new ModelBasedVariableResolver(
83
labelService,
84
disposables.add(createTextModel('', undefined, undefined, URI.parse('http://www.pb.o/abc/def/ghi')))
85
);
86
assertVariableResolve(resolver, 'TM_FILENAME', 'ghi');
87
if (!isWindows) {
88
assertVariableResolve(resolver, 'TM_DIRECTORY', '/abc/def');
89
assertVariableResolve(resolver, 'TM_FILEPATH', '/abc/def/ghi');
90
}
91
92
resolver = new ModelBasedVariableResolver(
93
labelService,
94
disposables.add(createTextModel('', undefined, undefined, URI.parse('mem:fff.ts')))
95
);
96
assertVariableResolve(resolver, 'TM_DIRECTORY', '');
97
assertVariableResolve(resolver, 'TM_FILEPATH', 'fff.ts');
98
99
disposables.dispose();
100
});
101
102
test('Path delimiters in code snippet variables aren\'t specific to remote OS #76840', function () {
103
104
const labelService = new class extends mock<ILabelService>() {
105
override getUriLabel(uri: URI) {
106
return uri.fsPath.replace(/\/|\\/g, '|');
107
}
108
};
109
110
const model = createTextModel([].join('\n'), undefined, undefined, URI.parse('foo:///foo/files/text.txt'));
111
112
const resolver = new CompositeSnippetVariableResolver([new ModelBasedVariableResolver(labelService, model)]);
113
114
assertVariableResolve(resolver, 'TM_FILEPATH', '|foo|files|text.txt');
115
116
model.dispose();
117
});
118
119
test('editor variables, selection', function () {
120
121
resolver = new SelectionBasedVariableResolver(model, new Selection(1, 2, 2, 3), 0, undefined);
122
assertVariableResolve(resolver, 'TM_SELECTED_TEXT', 'his is line one\nth');
123
assertVariableResolve(resolver, 'TM_CURRENT_LINE', 'this is line two');
124
assertVariableResolve(resolver, 'TM_LINE_INDEX', '1');
125
assertVariableResolve(resolver, 'TM_LINE_NUMBER', '2');
126
assertVariableResolve(resolver, 'CURSOR_INDEX', '0');
127
assertVariableResolve(resolver, 'CURSOR_NUMBER', '1');
128
129
resolver = new SelectionBasedVariableResolver(model, new Selection(1, 2, 2, 3), 4, undefined);
130
assertVariableResolve(resolver, 'CURSOR_INDEX', '4');
131
assertVariableResolve(resolver, 'CURSOR_NUMBER', '5');
132
133
resolver = new SelectionBasedVariableResolver(model, new Selection(2, 3, 1, 2), 0, undefined);
134
assertVariableResolve(resolver, 'TM_SELECTED_TEXT', 'his is line one\nth');
135
assertVariableResolve(resolver, 'TM_CURRENT_LINE', 'this is line one');
136
assertVariableResolve(resolver, 'TM_LINE_INDEX', '0');
137
assertVariableResolve(resolver, 'TM_LINE_NUMBER', '1');
138
139
resolver = new SelectionBasedVariableResolver(model, new Selection(1, 2, 1, 2), 0, undefined);
140
assertVariableResolve(resolver, 'TM_SELECTED_TEXT', undefined);
141
142
assertVariableResolve(resolver, 'TM_CURRENT_WORD', 'this');
143
144
resolver = new SelectionBasedVariableResolver(model, new Selection(3, 1, 3, 1), 0, undefined);
145
assertVariableResolve(resolver, 'TM_CURRENT_WORD', undefined);
146
147
});
148
149
test('TextmateSnippet, resolve variable', function () {
150
const snippet = new SnippetParser().parse('"$TM_CURRENT_WORD"', true);
151
assert.strictEqual(snippet.toString(), '""');
152
snippet.resolveVariables(resolver);
153
assert.strictEqual(snippet.toString(), '"this"');
154
155
});
156
157
test('TextmateSnippet, resolve variable with default', function () {
158
const snippet = new SnippetParser().parse('"${TM_CURRENT_WORD:foo}"', true);
159
assert.strictEqual(snippet.toString(), '"foo"');
160
snippet.resolveVariables(resolver);
161
assert.strictEqual(snippet.toString(), '"this"');
162
});
163
164
test('More useful environment variables for snippets, #32737', function () {
165
166
const disposables = new DisposableStore();
167
168
assertVariableResolve(resolver, 'TM_FILENAME_BASE', 'text');
169
170
resolver = new ModelBasedVariableResolver(
171
labelService,
172
disposables.add(createTextModel('', undefined, undefined, URI.parse('http://www.pb.o/abc/def/ghi')))
173
);
174
assertVariableResolve(resolver, 'TM_FILENAME_BASE', 'ghi');
175
176
resolver = new ModelBasedVariableResolver(
177
labelService,
178
disposables.add(createTextModel('', undefined, undefined, URI.parse('mem:.git')))
179
);
180
assertVariableResolve(resolver, 'TM_FILENAME_BASE', '.git');
181
182
resolver = new ModelBasedVariableResolver(
183
labelService,
184
disposables.add(createTextModel('', undefined, undefined, URI.parse('mem:foo.')))
185
);
186
assertVariableResolve(resolver, 'TM_FILENAME_BASE', 'foo');
187
188
disposables.dispose();
189
});
190
191
192
function assertVariableResolve2(input: string, expected: string, varValue?: string) {
193
const snippet = new SnippetParser().parse(input)
194
.resolveVariables({ resolve(variable) { return varValue || variable.name; } });
195
196
const actual = snippet.toString();
197
assert.strictEqual(actual, expected);
198
}
199
200
test('Variable Snippet Transform', function () {
201
202
const snippet = new SnippetParser().parse('name=${TM_FILENAME/(.*)\\..+$/$1/}', true);
203
snippet.resolveVariables(resolver);
204
assert.strictEqual(snippet.toString(), 'name=text');
205
206
assertVariableResolve2('${ThisIsAVar/([A-Z]).*(Var)/$2/}', 'Var');
207
assertVariableResolve2('${ThisIsAVar/([A-Z]).*(Var)/$2-${1:/downcase}/}', 'Var-t');
208
assertVariableResolve2('${Foo/(.*)/${1:+Bar}/img}', 'Bar');
209
210
//https://github.com/microsoft/vscode/issues/33162
211
assertVariableResolve2('export default class ${TM_FILENAME/(\\w+)\\.js/$1/g}', 'export default class FooFile', 'FooFile.js');
212
213
assertVariableResolve2('${foobarfoobar/(foo)/${1:+FAR}/g}', 'FARbarFARbar'); // global
214
assertVariableResolve2('${foobarfoobar/(foo)/${1:+FAR}/}', 'FARbarfoobar'); // first match
215
assertVariableResolve2('${foobarfoobar/(bazz)/${1:+FAR}/g}', 'foobarfoobar'); // no match, no else
216
// assertVariableResolve2('${foobarfoobar/(bazz)/${1:+FAR}/g}', ''); // no match
217
218
assertVariableResolve2('${foobarfoobar/(foo)/${2:+FAR}/g}', 'barbar'); // bad group reference
219
});
220
221
test('Snippet transforms do not handle regex with alternatives or optional matches, #36089', function () {
222
223
assertVariableResolve2(
224
'${TM_FILENAME/^(.)|(?:-(.))|(\\.js)/${1:/upcase}${2:/upcase}/g}',
225
'MyClass',
226
'my-class.js'
227
);
228
229
// no hyphens
230
assertVariableResolve2(
231
'${TM_FILENAME/^(.)|(?:-(.))|(\\.js)/${1:/upcase}${2:/upcase}/g}',
232
'Myclass',
233
'myclass.js'
234
);
235
236
// none matching suffix
237
assertVariableResolve2(
238
'${TM_FILENAME/^(.)|(?:-(.))|(\\.js)/${1:/upcase}${2:/upcase}/g}',
239
'Myclass.foo',
240
'myclass.foo'
241
);
242
243
// more than one hyphen
244
assertVariableResolve2(
245
'${TM_FILENAME/^(.)|(?:-(.))|(\\.js)/${1:/upcase}${2:/upcase}/g}',
246
'ThisIsAFile',
247
'this-is-a-file.js'
248
);
249
250
// KEBAB CASE
251
assertVariableResolve2(
252
'${TM_FILENAME_BASE/([A-Z][a-z]+)([A-Z][a-z]+$)?/${1:/downcase}-${2:/downcase}/g}',
253
'capital-case',
254
'CapitalCase'
255
);
256
257
assertVariableResolve2(
258
'${TM_FILENAME_BASE/([A-Z][a-z]+)([A-Z][a-z]+$)?/${1:/downcase}-${2:/downcase}/g}',
259
'capital-case-more',
260
'CapitalCaseMore'
261
);
262
});
263
264
test('Add variable to insert value from clipboard to a snippet #40153', function () {
265
266
assertVariableResolve(new ClipboardBasedVariableResolver(() => undefined, 1, 0, true), 'CLIPBOARD', undefined);
267
268
assertVariableResolve(new ClipboardBasedVariableResolver(() => null!, 1, 0, true), 'CLIPBOARD', undefined);
269
270
assertVariableResolve(new ClipboardBasedVariableResolver(() => '', 1, 0, true), 'CLIPBOARD', undefined);
271
272
assertVariableResolve(new ClipboardBasedVariableResolver(() => 'foo', 1, 0, true), 'CLIPBOARD', 'foo');
273
274
assertVariableResolve(new ClipboardBasedVariableResolver(() => 'foo', 1, 0, true), 'foo', undefined);
275
assertVariableResolve(new ClipboardBasedVariableResolver(() => 'foo', 1, 0, true), 'cLIPBOARD', undefined);
276
});
277
278
test('Add variable to insert value from clipboard to a snippet #40153, 2', function () {
279
280
assertVariableResolve(new ClipboardBasedVariableResolver(() => 'line1', 1, 2, true), 'CLIPBOARD', 'line1');
281
assertVariableResolve(new ClipboardBasedVariableResolver(() => 'line1\nline2\nline3', 1, 2, true), 'CLIPBOARD', 'line1\nline2\nline3');
282
283
assertVariableResolve(new ClipboardBasedVariableResolver(() => 'line1\nline2', 1, 2, true), 'CLIPBOARD', 'line2');
284
resolver = new ClipboardBasedVariableResolver(() => 'line1\nline2', 0, 2, true);
285
assertVariableResolve(new ClipboardBasedVariableResolver(() => 'line1\nline2', 0, 2, true), 'CLIPBOARD', 'line1');
286
287
assertVariableResolve(new ClipboardBasedVariableResolver(() => 'line1\nline2', 0, 2, false), 'CLIPBOARD', 'line1\nline2');
288
});
289
290
291
function assertVariableResolve3(resolver: VariableResolver, varName: string) {
292
const snippet = new SnippetParser().parse(`$${varName}`);
293
const variable = <Variable>snippet.children[0];
294
295
assert.strictEqual(variable.resolve(resolver), true, `${varName} failed to resolve`);
296
}
297
298
test('Add time variables for snippets #41631, #43140', function () {
299
300
const resolver = new TimeBasedVariableResolver;
301
302
assertVariableResolve3(resolver, 'CURRENT_YEAR');
303
assertVariableResolve3(resolver, 'CURRENT_YEAR_SHORT');
304
assertVariableResolve3(resolver, 'CURRENT_MONTH');
305
assertVariableResolve3(resolver, 'CURRENT_DATE');
306
assertVariableResolve3(resolver, 'CURRENT_HOUR');
307
assertVariableResolve3(resolver, 'CURRENT_MINUTE');
308
assertVariableResolve3(resolver, 'CURRENT_SECOND');
309
assertVariableResolve3(resolver, 'CURRENT_DAY_NAME');
310
assertVariableResolve3(resolver, 'CURRENT_DAY_NAME_SHORT');
311
assertVariableResolve3(resolver, 'CURRENT_MONTH_NAME');
312
assertVariableResolve3(resolver, 'CURRENT_MONTH_NAME_SHORT');
313
assertVariableResolve3(resolver, 'CURRENT_SECONDS_UNIX');
314
assertVariableResolve3(resolver, 'CURRENT_TIMEZONE_OFFSET');
315
});
316
317
test('Time-based snippet variables resolve to the same values even as time progresses', async function () {
318
const snippetText = `
319
$CURRENT_YEAR
320
$CURRENT_YEAR_SHORT
321
$CURRENT_MONTH
322
$CURRENT_DATE
323
$CURRENT_HOUR
324
$CURRENT_MINUTE
325
$CURRENT_SECOND
326
$CURRENT_DAY_NAME
327
$CURRENT_DAY_NAME_SHORT
328
$CURRENT_MONTH_NAME
329
$CURRENT_MONTH_NAME_SHORT
330
$CURRENT_SECONDS_UNIX
331
$CURRENT_TIMEZONE_OFFSET
332
`;
333
334
const clock = sinon.useFakeTimers();
335
try {
336
const resolver = new TimeBasedVariableResolver;
337
338
const firstResolve = new SnippetParser().parse(snippetText).resolveVariables(resolver);
339
clock.tick((365 * 24 * 3600 * 1000) + (24 * 3600 * 1000) + (3661 * 1000)); // 1 year + 1 day + 1 hour + 1 minute + 1 second
340
const secondResolve = new SnippetParser().parse(snippetText).resolveVariables(resolver);
341
342
assert.strictEqual(firstResolve.toString(), secondResolve.toString(), `Time-based snippet variables resolved differently`);
343
} finally {
344
clock.restore();
345
}
346
});
347
348
test('creating snippet - format-condition doesn\'t work #53617', function () {
349
350
const snippet = new SnippetParser().parse('${TM_LINE_NUMBER/(10)/${1:?It is:It is not}/} line 10', true);
351
snippet.resolveVariables({ resolve() { return '10'; } });
352
assert.strictEqual(snippet.toString(), 'It is line 10');
353
354
snippet.resolveVariables({ resolve() { return '11'; } });
355
assert.strictEqual(snippet.toString(), 'It is not line 10');
356
});
357
358
test('Add workspace name and folder variables for snippets #68261', function () {
359
360
let workspace: IWorkspace;
361
const workspaceService = new class implements IWorkspaceContextService {
362
declare readonly _serviceBrand: undefined;
363
_throw = () => { throw new Error(); };
364
onDidChangeWorkbenchState = this._throw;
365
onDidChangeWorkspaceName = this._throw;
366
onWillChangeWorkspaceFolders = this._throw;
367
onDidChangeWorkspaceFolders = this._throw;
368
getCompleteWorkspace = this._throw;
369
getWorkspace(): IWorkspace { return workspace; }
370
getWorkbenchState = this._throw;
371
getWorkspaceFolder = this._throw;
372
isCurrentWorkspace = this._throw;
373
isInsideWorkspace = this._throw;
374
};
375
376
const resolver = new WorkspaceBasedVariableResolver(workspaceService);
377
378
// empty workspace
379
workspace = new Workspace('');
380
assertVariableResolve(resolver, 'WORKSPACE_NAME', undefined);
381
assertVariableResolve(resolver, 'WORKSPACE_FOLDER', undefined);
382
383
// single folder workspace without config
384
workspace = new Workspace('', [toWorkspaceFolder(URI.file('/folderName'))]);
385
assertVariableResolve(resolver, 'WORKSPACE_NAME', 'folderName');
386
if (!isWindows) {
387
assertVariableResolve(resolver, 'WORKSPACE_FOLDER', '/folderName');
388
}
389
390
// workspace with config
391
const workspaceConfigPath = URI.file('testWorkspace.code-workspace');
392
workspace = new Workspace('', toWorkspaceFolders([{ path: 'folderName' }], workspaceConfigPath, extUriBiasedIgnorePathCase), workspaceConfigPath);
393
assertVariableResolve(resolver, 'WORKSPACE_NAME', 'testWorkspace');
394
if (!isWindows) {
395
assertVariableResolve(resolver, 'WORKSPACE_FOLDER', '/');
396
}
397
});
398
399
test('Add RELATIVE_FILEPATH snippet variable #114208', function () {
400
401
let resolver: VariableResolver;
402
403
// Mock a label service (only coded for file uris)
404
const workspaceLabelService = ((rootPath: string): ILabelService => {
405
const labelService = new class extends mock<ILabelService>() {
406
override getUriLabel(uri: URI, options: { relative?: boolean } = {}) {
407
const rootFsPath = URI.file(rootPath).fsPath + sep;
408
const fsPath = uri.fsPath;
409
if (options.relative && rootPath && fsPath.startsWith(rootFsPath)) {
410
return fsPath.substring(rootFsPath.length);
411
}
412
return fsPath;
413
}
414
};
415
return labelService;
416
});
417
418
const model = createTextModel('', undefined, undefined, URI.parse('file:///foo/files/text.txt'));
419
420
// empty workspace
421
resolver = new ModelBasedVariableResolver(
422
workspaceLabelService(''),
423
model
424
);
425
426
if (!isWindows) {
427
assertVariableResolve(resolver, 'RELATIVE_FILEPATH', '/foo/files/text.txt');
428
} else {
429
assertVariableResolve(resolver, 'RELATIVE_FILEPATH', '\\foo\\files\\text.txt');
430
}
431
432
// single folder workspace
433
resolver = new ModelBasedVariableResolver(
434
workspaceLabelService('/foo'),
435
model
436
);
437
if (!isWindows) {
438
assertVariableResolve(resolver, 'RELATIVE_FILEPATH', 'files/text.txt');
439
} else {
440
assertVariableResolve(resolver, 'RELATIVE_FILEPATH', 'files\\text.txt');
441
}
442
443
model.dispose();
444
});
445
});
446
447