searchModel.test.ts 12.9 KB
Newer Older
E
Erich Gamma 已提交
1 2 3 4 5 6 7
/*---------------------------------------------------------------------------------------------
 *  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';
8
import * as sinon from 'sinon';
9
import { TestInstantiationService } from 'vs/platform/instantiation/test/common/instantiationServiceMock';
10
import { DeferredPPromise } from 'vs/base/test/common/utils';
11
import { PPromise } from 'vs/base/common/winjs.base';
12
import { SearchModel } from 'vs/workbench/parts/search/common/searchModel';
J
Johannes Rieken 已提交
13
import URI from 'vs/base/common/uri';
14
import { IFileMatch, IFolderQuery, ILineMatch, ISearchService, ISearchComplete, ISearchProgressItem, IUncachedSearchStats } from 'vs/platform/search/common/search';
15 16
import { ITelemetryService } from 'vs/platform/telemetry/common/telemetry';
import { NullTelemetryService } from 'vs/platform/telemetry/common/telemetryUtils';
17
import { Range } from 'vs/editor/common/core/range';
18
import { IModelService } from 'vs/editor/common/services/modelService';
19 20 21
import { IConfigurationService } from 'vs/platform/configuration/common/configuration';
import { TestConfigurationService } from 'vs/platform/configuration/test/common/testConfigurationService';
import { ModelServiceImpl } from 'vs/editor/common/services/modelServiceImpl';
22
import { timeout } from 'vs/base/common/async';
23

24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44
const nullEvent = new class {

	public id: number;
	public topic: string;
	public name: string;
	public description: string;
	public data: any;

	public startTime: Date;
	public stopTime: Date;

	public stop(): void {
		return;
	}

	public timeTaken(): number {
		return -1;
	}
};


45
suite('SearchModel', () => {
E
Erich Gamma 已提交
46

47
	let instantiationService: TestInstantiationService;
48
	let restoreStubs: sinon.SinonStub[];
E
Erich Gamma 已提交
49

50 51 52
	const testSearchStats: IUncachedSearchStats = {
		fromCache: false,
		resultCount: 4,
C
Christof Marti 已提交
53 54
		traversal: 'node',
		errors: [],
C
chrmarti 已提交
55 56 57 58 59 60
		fileWalkStartTime: 0,
		fileWalkResultTime: 1,
		directoriesWalked: 2,
		filesWalked: 3
	};

61 62 63 64
	const folderQueries: IFolderQuery[] = [
		{ folder: URI.parse('file://c:/') }
	];

E
Erich Gamma 已提交
65
	setup(() => {
J
Johannes Rieken 已提交
66 67
		restoreStubs = [];
		instantiationService = new TestInstantiationService();
68
		instantiationService.stub(ITelemetryService, NullTelemetryService);
69
		instantiationService.stub(IModelService, stubModelService(instantiationService));
70 71
		instantiationService.stub(ISearchService, {});
		instantiationService.stub(ISearchService, 'search', PPromise.as({ results: [] }));
72 73 74 75 76 77 78 79 80
	});

	teardown(() => {
		restoreStubs.forEach(element => {
			element.restore();
		});
	});

	test('Search Model: Search adds to results', function () {
J
Johannes Rieken 已提交
81 82
		let results = [aRawMatch('file://c:/1', aLineMatch('preview 1', 1, [[1, 3], [4, 7]])), aRawMatch('file://c:/2', aLineMatch('preview 2'))];
		instantiationService.stub(ISearchService, 'search', PPromise.as({ results: results }));
83

84 85
		let testObject: SearchModel = instantiationService.createInstance(SearchModel);
		testObject.search({ contentPattern: { pattern: 'somestring' }, type: 1, folderQueries });
86

J
Johannes Rieken 已提交
87
		let actual = testObject.searchResult.matches();
88 89 90 91

		assert.equal(2, actual.length);
		assert.equal('file://c:/1', actual[0].resource().toString());

J
Johannes Rieken 已提交
92
		let actuaMatches = actual[0].matches();
93 94 95 96 97 98
		assert.equal(2, actuaMatches.length);
		assert.equal('preview 1', actuaMatches[0].text());
		assert.ok(new Range(2, 2, 2, 5).equalsRange(actuaMatches[0].range()));
		assert.equal('preview 1', actuaMatches[1].text());
		assert.ok(new Range(2, 5, 2, 12).equalsRange(actuaMatches[1].range()));

J
Johannes Rieken 已提交
99
		actuaMatches = actual[1].matches();
100 101 102 103 104
		assert.equal(1, actuaMatches.length);
		assert.equal('preview 2', actuaMatches[0].text());
		assert.ok(new Range(2, 1, 2, 2).equalsRange(actuaMatches[0].range()));
	});

105
	test('Search Model: Search adds to results during progress', function () {
J
Johannes Rieken 已提交
106 107 108
		let results = [aRawMatch('file://c:/1', aLineMatch('preview 1', 1, [[1, 3], [4, 7]])), aRawMatch('file://c:/2', aLineMatch('preview 2'))];
		let promise = new DeferredPPromise<ISearchComplete, ISearchProgressItem>();
		instantiationService.stub(ISearchService, 'search', promise);
109

J
Johannes Rieken 已提交
110
		let testObject = instantiationService.createInstance(SearchModel);
111
		let result = testObject.search({ contentPattern: { pattern: 'somestring' }, type: 1, folderQueries });
112 113 114

		promise.progress(results[0]);
		promise.progress(results[1]);
J
Johannes Rieken 已提交
115
		promise.complete({ results: [], stats: testSearchStats });
116

117
		return result.then(() => {
J
Johannes Rieken 已提交
118
			let actual = testObject.searchResult.matches();
119 120 121 122

			assert.equal(2, actual.length);
			assert.equal('file://c:/1', actual[0].resource().toString());

J
Johannes Rieken 已提交
123
			let actuaMatches = actual[0].matches();
124 125 126 127 128 129
			assert.equal(2, actuaMatches.length);
			assert.equal('preview 1', actuaMatches[0].text());
			assert.ok(new Range(2, 2, 2, 5).equalsRange(actuaMatches[0].range()));
			assert.equal('preview 1', actuaMatches[1].text());
			assert.ok(new Range(2, 5, 2, 12).equalsRange(actuaMatches[1].range()));

J
Johannes Rieken 已提交
130
			actuaMatches = actual[1].matches();
131 132 133 134 135 136 137
			assert.equal(1, actuaMatches.length);
			assert.equal('preview 2', actuaMatches[0].text());
			assert.ok(new Range(2, 1, 2, 2).equalsRange(actuaMatches[0].range()));
		});
	});

	test('Search Model: Search reports telemetry on search completed', function () {
J
Johannes Rieken 已提交
138 139 140
		let target = instantiationService.spy(ITelemetryService, 'publicLog');
		let results = [aRawMatch('file://c:/1', aLineMatch('preview 1', 1, [[1, 3], [4, 7]])), aRawMatch('file://c:/2', aLineMatch('preview 2'))];
		instantiationService.stub(ISearchService, 'search', PPromise.as({ results: results }));
141

J
Johannes Rieken 已提交
142
		let testObject = instantiationService.createInstance(SearchModel);
143
		testObject.search({ contentPattern: { pattern: 'somestring' }, type: 1, folderQueries });
144 145

		assert.ok(target.calledOnce);
146 147
		const data = target.args[0];
		data[1].duration = -1;
B
Benjamin Pasero 已提交
148
		assert.deepEqual(['searchResultsShown', { count: 3, fileCount: 2, options: {}, duration: -1, useRipgrep: undefined }], data);
149 150
	});

151
	test('Search Model: Search reports timed telemetry on search when progress is not called', function () {
J
Johannes Rieken 已提交
152
		let target2 = sinon.spy();
153
		stub(nullEvent, 'stop', target2);
J
Johannes Rieken 已提交
154
		let target1 = sinon.stub().returns(nullEvent);
J
Joao Moreno 已提交
155
		instantiationService.stub(ITelemetryService, 'publicLog', target1);
156

J
Johannes Rieken 已提交
157
		instantiationService.stub(ISearchService, 'search', PPromise.as({ results: [] }));
158

J
Johannes Rieken 已提交
159
		let testObject = instantiationService.createInstance(SearchModel);
160
		const result = testObject.search({ contentPattern: { pattern: 'somestring' }, type: 1, folderQueries });
J
Joao Moreno 已提交
161

162
		return timeout(1).then(() => {
163
			return result.then(() => {
J
Joao Moreno 已提交
164 165 166
				assert.ok(target1.calledWith('searchResultsFirstRender'));
				assert.ok(target1.calledWith('searchResultsFinished'));
			});
167 168

		});
169 170
	});

171
	test('Search Model: Search reports timed telemetry on search when progress is called', function () {
J
Johannes Rieken 已提交
172
		let target2 = sinon.spy();
173
		stub(nullEvent, 'stop', target2);
J
Johannes Rieken 已提交
174
		let target1 = sinon.stub().returns(nullEvent);
J
Joao Moreno 已提交
175
		instantiationService.stub(ITelemetryService, 'publicLog', target1);
176

J
Johannes Rieken 已提交
177 178
		let promise = new DeferredPPromise<ISearchComplete, ISearchProgressItem>();
		instantiationService.stub(ISearchService, 'search', promise);
179

J
Johannes Rieken 已提交
180
		let testObject = instantiationService.createInstance(SearchModel);
181
		let result = testObject.search({ contentPattern: { pattern: 'somestring' }, type: 1, folderQueries });
182 183

		promise.progress(aRawMatch('file://c:/1', aLineMatch('some preview')));
J
Johannes Rieken 已提交
184
		promise.complete({ results: [], stats: testSearchStats });
185

186
		return timeout(1).then(() => {
187
			return result.then(() => {
J
Joao Moreno 已提交
188 189 190 191
				assert.ok(target1.calledWith('searchResultsFirstRender'));
				assert.ok(target1.calledWith('searchResultsFinished'));
				// assert.equal(1, target2.callCount);
			});
192
		});
193 194
	});

195
	test('Search Model: Search reports timed telemetry on search when error is called', function () {
J
Johannes Rieken 已提交
196
		let target2 = sinon.spy();
197
		stub(nullEvent, 'stop', target2);
J
Johannes Rieken 已提交
198
		let target1 = sinon.stub().returns(nullEvent);
J
Joao Moreno 已提交
199
		instantiationService.stub(ITelemetryService, 'publicLog', target1);
200

J
Johannes Rieken 已提交
201 202
		let promise = new DeferredPPromise<ISearchComplete, ISearchProgressItem>();
		instantiationService.stub(ISearchService, 'search', promise);
203

J
Johannes Rieken 已提交
204
		let testObject = instantiationService.createInstance(SearchModel);
205
		let result = testObject.search({ contentPattern: { pattern: 'somestring' }, type: 1, folderQueries });
206 207 208

		promise.error('error');

209
		return timeout(1).then(() => {
210
			return result.then(() => { }, () => {
J
Joao Moreno 已提交
211 212 213 214
				assert.ok(target1.calledWith('searchResultsFirstRender'));
				assert.ok(target1.calledWith('searchResultsFinished'));
				// assert.ok(target2.calledOnce);
			});
215
		});
216 217
	});

218
	test('Search Model: Search reports timed telemetry on search when error is cancelled error', function () {
J
Johannes Rieken 已提交
219
		let target2 = sinon.spy();
220
		stub(nullEvent, 'stop', target2);
J
Johannes Rieken 已提交
221
		let target1 = sinon.stub().returns(nullEvent);
J
Joao Moreno 已提交
222
		instantiationService.stub(ITelemetryService, 'publicLog', target1);
223

J
Johannes Rieken 已提交
224 225
		let promise = new DeferredPPromise<ISearchComplete, ISearchProgressItem>();
		instantiationService.stub(ISearchService, 'search', promise);
226

J
Johannes Rieken 已提交
227
		let testObject = instantiationService.createInstance(SearchModel);
228
		let result = testObject.search({ contentPattern: { pattern: 'somestring' }, type: 1, folderQueries });
229 230 231

		promise.cancel();

232
		return timeout(1).then(() => {
233
			return result.then(() => { }, () => {
J
Joao Moreno 已提交
234 235 236 237
				assert.ok(target1.calledWith('searchResultsFirstRender'));
				assert.ok(target1.calledWith('searchResultsFinished'));
				// assert.ok(target2.calledOnce);
			});
238
		});
239 240 241
	});

	test('Search Model: Search results are cleared during search', function () {
J
Johannes Rieken 已提交
242 243 244
		let results = [aRawMatch('file://c:/1', aLineMatch('preview 1', 1, [[1, 3], [4, 7]])), aRawMatch('file://c:/2', aLineMatch('preview 2'))];
		instantiationService.stub(ISearchService, 'search', PPromise.as({ results: results }));
		let testObject: SearchModel = instantiationService.createInstance(SearchModel);
245
		testObject.search({ contentPattern: { pattern: 'somestring' }, type: 1, folderQueries });
246 247
		assert.ok(!testObject.searchResult.isEmpty());

J
Johannes Rieken 已提交
248
		instantiationService.stub(ISearchService, 'search', new DeferredPPromise<ISearchComplete, ISearchProgressItem>());
249

250
		testObject.search({ contentPattern: { pattern: 'somestring' }, type: 1, folderQueries });
251 252 253 254
		assert.ok(testObject.searchResult.isEmpty());
	});

	test('Search Model: Previous search is cancelled when new search is called', function () {
J
Johannes Rieken 已提交
255 256 257
		let target = sinon.spy();
		instantiationService.stub(ISearchService, 'search', new DeferredPPromise((c, e, p) => { }, target));
		let testObject: SearchModel = instantiationService.createInstance(SearchModel);
258

259
		testObject.search({ contentPattern: { pattern: 'somestring' }, type: 1, folderQueries });
J
Johannes Rieken 已提交
260
		instantiationService.stub(ISearchService, 'search', new DeferredPPromise<ISearchComplete, ISearchProgressItem>());
261
		testObject.search({ contentPattern: { pattern: 'somestring' }, type: 1, folderQueries });
262 263 264 265

		assert.ok(target.calledOnce);
	});

S
Sandeep Somavarapu 已提交
266
	test('getReplaceString returns proper replace string for regExpressions', function () {
J
Johannes Rieken 已提交
267 268
		let results = [aRawMatch('file://c:/1', aLineMatch('preview 1', 1, [[1, 3], [4, 7]]))];
		instantiationService.stub(ISearchService, 'search', PPromise.as({ results: results }));
S
Sandeep Somavarapu 已提交
269

J
Johannes Rieken 已提交
270
		let testObject: SearchModel = instantiationService.createInstance(SearchModel);
271
		testObject.search({ contentPattern: { pattern: 're' }, type: 1, folderQueries });
S
Sandeep Somavarapu 已提交
272 273 274 275
		testObject.replaceString = 'hello';
		let match = testObject.searchResult.matches()[0].matches()[0];
		assert.equal('hello', match.replaceString);

276
		testObject.search({ contentPattern: { pattern: 're', isRegExp: true }, type: 1, folderQueries });
S
Sandeep Somavarapu 已提交
277 278 279
		match = testObject.searchResult.matches()[0].matches()[0];
		assert.equal('hello', match.replaceString);

280
		testObject.search({ contentPattern: { pattern: 're(?:vi)', isRegExp: true }, type: 1, folderQueries });
S
Sandeep Somavarapu 已提交
281 282 283
		match = testObject.searchResult.matches()[0].matches()[0];
		assert.equal('hello', match.replaceString);

284
		testObject.search({ contentPattern: { pattern: 'r(e)(?:vi)', isRegExp: true }, type: 1, folderQueries });
S
Sandeep Somavarapu 已提交
285 286 287
		match = testObject.searchResult.matches()[0].matches()[0];
		assert.equal('hello', match.replaceString);

288
		testObject.search({ contentPattern: { pattern: 'r(e)(?:vi)', isRegExp: true }, type: 1, folderQueries });
S
Sandeep Somavarapu 已提交
289 290 291 292 293
		testObject.replaceString = 'hello$1';
		match = testObject.searchResult.matches()[0].matches()[0];
		assert.equal('helloe', match.replaceString);
	});

294 295 296 297 298 299 300 301
	function aRawMatch(resource: string, ...lineMatches: ILineMatch[]): IFileMatch {
		return { resource: URI.parse(resource), lineMatches };
	}

	function aLineMatch(preview: string, lineNumber: number = 1, offsetAndLengths: number[][] = [[0, 1]]): ILineMatch {
		return { preview, lineNumber, offsetAndLengths };
	}

302
	function stub(arg1: any, arg2: any, arg3: any): sinon.SinonStub {
J
Johannes Rieken 已提交
303
		const stub = sinon.stub(arg1, arg2, arg3);
304 305 306 307
		restoreStubs.push(stub);
		return stub;
	}

308 309 310 311 312
	function stubModelService(instantiationService: TestInstantiationService): IModelService {
		instantiationService.stub(IConfigurationService, new TestConfigurationService());
		return instantiationService.createInstance(ModelServiceImpl);
	}

313
});