searchModel.test.ts 12.2 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 12 13
import { PPromise } from 'vs/base/common/winjs.base';
import { nullEvent } from 'vs/base/common/timer';
import { SearchModel } from 'vs/workbench/parts/search/common/searchModel';
J
Johannes Rieken 已提交
14
import URI from 'vs/base/common/uri';
J
Joao Moreno 已提交
15
import { IFileMatch, ILineMatch, ISearchService, ISearchComplete, ISearchProgressItem, IUncachedSearchStats } from 'vs/platform/search/common/search';
16
import { ITelemetryService, NullTelemetryService } from 'vs/platform/telemetry/common/telemetry';
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

23
suite('SearchModel', () => {
E
Erich Gamma 已提交
24

25 26
	let instantiationService: TestInstantiationService;
	let restoreStubs;
E
Erich Gamma 已提交
27

28 29 30
	const testSearchStats: IUncachedSearchStats = {
		fromCache: false,
		resultCount: 4,
C
Christof Marti 已提交
31 32
		traversal: 'node',
		errors: [],
C
chrmarti 已提交
33 34 35 36 37 38
		fileWalkStartTime: 0,
		fileWalkResultTime: 1,
		directoriesWalked: 2,
		filesWalked: 3
	};

E
Erich Gamma 已提交
39
	setup(() => {
J
Johannes Rieken 已提交
40 41
		restoreStubs = [];
		instantiationService = new TestInstantiationService();
42
		instantiationService.stub(ITelemetryService, NullTelemetryService);
43
		instantiationService.stub(IModelService, stubModelService(instantiationService));
44 45
		instantiationService.stub(ISearchService, {});
		instantiationService.stub(ISearchService, 'search', PPromise.as({ results: [] }));
46 47 48 49 50 51 52 53 54
	});

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

	test('Search Model: Search adds to results', function () {
J
Johannes Rieken 已提交
55 56
		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 }));
57

J
Johannes Rieken 已提交
58 59
		let testObject = instantiationService.createInstance(SearchModel);
		testObject.search({ contentPattern: { pattern: 'somestring' }, type: 1 });
60

J
Johannes Rieken 已提交
61
		let actual = testObject.searchResult.matches();
62 63 64 65

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

J
Johannes Rieken 已提交
66
		let actuaMatches = actual[0].matches();
67 68 69 70 71 72
		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 已提交
73
		actuaMatches = actual[1].matches();
74 75 76 77 78 79
		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 adds to results during progress', function (done) {
J
Johannes Rieken 已提交
80 81 82
		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);
83

J
Johannes Rieken 已提交
84 85
		let testObject = instantiationService.createInstance(SearchModel);
		let result = testObject.search({ contentPattern: { pattern: 'somestring' }, type: 1 });
86 87 88

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

		result.done(() => {
J
Johannes Rieken 已提交
92
			let actual = testObject.searchResult.matches();
93 94 95 96

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

J
Johannes Rieken 已提交
97
			let actuaMatches = actual[0].matches();
98 99 100 101 102 103
			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 已提交
104
			actuaMatches = actual[1].matches();
105 106 107 108 109 110 111 112 113
			assert.equal(1, actuaMatches.length);
			assert.equal('preview 2', actuaMatches[0].text());
			assert.ok(new Range(2, 1, 2, 2).equalsRange(actuaMatches[0].range()));

			done();
		});
	});

	test('Search Model: Search reports telemetry on search completed', function () {
J
Johannes Rieken 已提交
114 115 116
		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 }));
117

J
Johannes Rieken 已提交
118 119
		let testObject = instantiationService.createInstance(SearchModel);
		testObject.search({ contentPattern: { pattern: 'somestring' }, type: 1 });
120 121

		assert.ok(target.calledOnce);
122 123 124
		const data = target.args[0];
		data[1].duration = -1;
		assert.deepEqual(['searchResultsShown', { count: 3, fileCount: 2, options: {}, duration: -1 }], data);
125 126
	});

J
Joao Moreno 已提交
127
	test('Search Model: Search reports timed telemetry on search when progress is not called', function (done) {
J
Johannes Rieken 已提交
128
		let target2 = sinon.spy();
129
		stub(nullEvent, 'stop', target2);
J
Johannes Rieken 已提交
130
		let target1 = sinon.stub().returns(nullEvent);
J
Joao Moreno 已提交
131
		instantiationService.stub(ITelemetryService, 'publicLog', target1);
132

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

J
Johannes Rieken 已提交
135 136
		let testObject = instantiationService.createInstance(SearchModel);
		const result = testObject.search({ contentPattern: { pattern: 'somestring' }, type: 1 });
J
Joao Moreno 已提交
137 138 139 140 141

		setTimeout(() => {
			result.done(() => {
				assert.ok(target1.calledWith('searchResultsFirstRender'));
				assert.ok(target1.calledWith('searchResultsFinished'));
142

J
Joao Moreno 已提交
143 144 145
				done();
			});
		}, 0);
146 147 148
	});

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

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

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

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

J
Joao Moreno 已提交
163 164 165 166 167
		setTimeout(() => {
			result.done(() => {
				assert.ok(target1.calledWith('searchResultsFirstRender'));
				assert.ok(target1.calledWith('searchResultsFinished'));
				// assert.equal(1, target2.callCount);
168

J
Joao Moreno 已提交
169 170 171
				done();
			});
		}, 0);
172 173 174
	});

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

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

J
Johannes Rieken 已提交
183 184
		let testObject = instantiationService.createInstance(SearchModel);
		let result = testObject.search({ contentPattern: { pattern: 'somestring' }, type: 1 });
185 186 187

		promise.error('error');

J
Joao Moreno 已提交
188
		setTimeout(() => {
J
Johannes Rieken 已提交
189
			result.done(() => { }, () => {
J
Joao Moreno 已提交
190 191 192
				assert.ok(target1.calledWith('searchResultsFirstRender'));
				assert.ok(target1.calledWith('searchResultsFinished'));
				// assert.ok(target2.calledOnce);
193

J
Joao Moreno 已提交
194 195 196
				done();
			});
		}, 0);
197 198 199
	});

	test('Search Model: Search reports timed telemetry on search when error is cancelled error', function (done) {
J
Johannes Rieken 已提交
200
		let target2 = sinon.spy();
201
		stub(nullEvent, 'stop', target2);
J
Johannes Rieken 已提交
202
		let target1 = sinon.stub().returns(nullEvent);
J
Joao Moreno 已提交
203
		instantiationService.stub(ITelemetryService, 'publicLog', target1);
204

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

J
Johannes Rieken 已提交
208 209
		let testObject = instantiationService.createInstance(SearchModel);
		let result = testObject.search({ contentPattern: { pattern: 'somestring' }, type: 1 });
210 211 212

		promise.cancel();

J
Joao Moreno 已提交
213
		setTimeout(() => {
J
Johannes Rieken 已提交
214
			result.done(() => { }, () => {
J
Joao Moreno 已提交
215 216 217 218 219 220
				assert.ok(target1.calledWith('searchResultsFirstRender'));
				assert.ok(target1.calledWith('searchResultsFinished'));
				// assert.ok(target2.calledOnce);
				done();
			});
		}, 0);
221 222 223
	});

	test('Search Model: Search results are cleared during search', function () {
J
Johannes Rieken 已提交
224 225 226 227
		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);
		testObject.search({ contentPattern: { pattern: 'somestring' }, type: 1 });
228 229
		assert.ok(!testObject.searchResult.isEmpty());

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

J
Johannes Rieken 已提交
232
		testObject.search({ contentPattern: { pattern: 'somestring' }, type: 1 });
233 234 235 236
		assert.ok(testObject.searchResult.isEmpty());
	});

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

J
Johannes Rieken 已提交
241 242 243
		testObject.search({ contentPattern: { pattern: 'somestring' }, type: 1 });
		instantiationService.stub(ISearchService, 'search', new DeferredPPromise<ISearchComplete, ISearchProgressItem>());
		testObject.search({ contentPattern: { pattern: 'somestring' }, type: 1 });
244 245 246 247

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

S
Sandeep Somavarapu 已提交
248
	test('getReplaceString returns proper replace string for regExpressions', function () {
J
Johannes Rieken 已提交
249 250
		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 已提交
251

J
Johannes Rieken 已提交
252
		let testObject: SearchModel = instantiationService.createInstance(SearchModel);
S
Sandeep Somavarapu 已提交
253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275
		testObject.search({ contentPattern: { pattern: 're' }, type: 1 });
		testObject.replaceString = 'hello';
		let match = testObject.searchResult.matches()[0].matches()[0];
		assert.equal('hello', match.replaceString);

		testObject.search({ contentPattern: { pattern: 're', isRegExp: true }, type: 1 });
		match = testObject.searchResult.matches()[0].matches()[0];
		assert.equal('hello', match.replaceString);

		testObject.search({ contentPattern: { pattern: 're(?:vi)', isRegExp: true }, type: 1 });
		match = testObject.searchResult.matches()[0].matches()[0];
		assert.equal('hello', match.replaceString);

		testObject.search({ contentPattern: { pattern: 'r(e)(?:vi)', isRegExp: true }, type: 1 });
		match = testObject.searchResult.matches()[0].matches()[0];
		assert.equal('hello', match.replaceString);

		testObject.search({ contentPattern: { pattern: 'r(e)(?:vi)', isRegExp: true }, type: 1 });
		testObject.replaceString = 'hello$1';
		match = testObject.searchResult.matches()[0].matches()[0];
		assert.equal('helloe', match.replaceString);
	});

276 277 278 279 280 281 282 283
	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 };
	}

J
Johannes Rieken 已提交
284 285
	function stub(arg1, arg2, arg3): sinon.SinonStub {
		const stub = sinon.stub(arg1, arg2, arg3);
286 287 288 289
		restoreStubs.push(stub);
		return stub;
	}

290 291 292 293 294
	function stubModelService(instantiationService: TestInstantiationService): IModelService {
		instantiationService.stub(IConfigurationService, new TestConfigurationService());
		return instantiationService.createInstance(ModelServiceImpl);
	}

E
Erich Gamma 已提交
295
});