extensionManagementService.ts 46.4 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';

8
import * as nls from 'vs/nls';
E
Erich Gamma 已提交
9 10
import * as path from 'path';
import * as pfs from 'vs/base/node/pfs';
11
import * as errors from 'vs/base/common/errors';
E
Erich Gamma 已提交
12
import { assign } from 'vs/base/common/objects';
13
import { toDisposable, Disposable } from 'vs/base/common/lifecycle';
14
import { flatten } from 'vs/base/common/arrays';
S
Sandeep Somavarapu 已提交
15
import { extract, buffer, ExtractError } from 'vs/base/node/zip';
16
import { TPromise } from 'vs/base/common/winjs.base';
J
Johannes Rieken 已提交
17 18
import {
	IExtensionManagementService, IExtensionGalleryService, ILocalExtension,
19
	IGalleryExtension, IExtensionManifest, IGalleryMetadata,
J
Joao Moreno 已提交
20
	InstallExtensionEvent, DidInstallExtensionEvent, DidUninstallExtensionEvent, LocalExtensionType,
S
Sandeep Somavarapu 已提交
21
	StatisticType,
J
Joao Moreno 已提交
22
	IExtensionIdentifier,
23
	IReportedExtension,
24 25
	InstallOperation,
	IExtensionDependency
J
Joao Moreno 已提交
26
} from 'vs/platform/extensionManagement/common/extensionManagement';
S
Sandeep Somavarapu 已提交
27
import { getGalleryExtensionIdFromLocal, adoptToGalleryExtensionId, areSameExtensions, getGalleryExtensionId, groupByExtension, getMaliciousExtensionsSet, getLocalExtensionId, getGalleryExtensionTelemetryData, getLocalExtensionTelemetryData, getIdFromLocalExtensionId } from 'vs/platform/extensionManagement/common/extensionManagementUtil';
28
import { localizeManifest } from '../common/extensionNls';
J
Joao Moreno 已提交
29
import { IEnvironmentService } from 'vs/platform/environment/common/environment';
30
import { Limiter, always } from 'vs/base/common/async';
M
Matt Bierner 已提交
31
import { Event, Emitter } from 'vs/base/common/event';
J
Joao Moreno 已提交
32
import * as semver from 'semver';
J
João Moreno 已提交
33
import URI from 'vs/base/common/uri';
S
Sandeep Somavarapu 已提交
34
import pkg from 'vs/platform/node/package';
35
import { isMacintosh, isWindows } from 'vs/base/common/platform';
36
import { ILogService } from 'vs/platform/log/common/log';
37
import { ExtensionsManifestCache } from 'vs/platform/extensionManagement/node/extensionsManifestCache';
38
import { IDialogService } from 'vs/platform/dialogs/common/dialogs';
39
import Severity from 'vs/base/common/severity';
S
Sandeep Somavarapu 已提交
40
import { ExtensionsLifecycle } from 'vs/platform/extensionManagement/node/extensionLifecycle';
41
import { toErrorMessage } from 'vs/base/common/errorMessage';
42
import { ITelemetryService } from 'vs/platform/telemetry/common/telemetry';
S
Sandeep Somavarapu 已提交
43
import { isEngineValid } from 'vs/platform/extensions/node/extensionValidator';
E
Erich Gamma 已提交
44

J
Joao Moreno 已提交
45
const SystemExtensionsRoot = path.normalize(path.join(URI.parse(require.toUrl('')).fsPath, '..', 'extensions'));
46 47
const ERROR_SCANNING_SYS_EXTENSIONS = 'scanningSystem';
const ERROR_SCANNING_USER_EXTENSIONS = 'scanningUser';
48
const INSTALL_ERROR_UNSET_UNINSTALLED = 'unsetUninstalled';
S
Sandeep Somavarapu 已提交
49 50 51
const INSTALL_ERROR_INCOMPATIBLE = 'incompatible';
const INSTALL_ERROR_DOWNLOADING = 'downloading';
const INSTALL_ERROR_VALIDATING = 'validating';
52 53
const INSTALL_ERROR_GALLERY = 'gallery';
const INSTALL_ERROR_LOCAL = 'local';
54
const INSTALL_ERROR_EXTRACTING = 'extracting';
S
Sandeep Somavarapu 已提交
55
const INSTALL_ERROR_RENAMING = 'renaming';
56
const INSTALL_ERROR_DELETING = 'deleting';
57
const ERROR_UNKNOWN = 'unknown';
S
Sandeep Somavarapu 已提交
58

59
export class ExtensionManagementError extends Error {
S
Sandeep Somavarapu 已提交
60 61 62 63
	constructor(message: string, readonly code: string) {
		super(message);
	}
}
J
Joao Moreno 已提交
64

J
Joao Moreno 已提交
65
function parseManifest(raw: string): TPromise<{ manifest: IExtensionManifest; metadata: IGalleryMetadata; }> {
66
	return new TPromise((c, e) => {
E
Erich Gamma 已提交
67
		try {
J
Joao Moreno 已提交
68 69 70 71
			const manifest = JSON.parse(raw);
			const metadata = manifest.__metadata || null;
			delete manifest.__metadata;
			c({ manifest, metadata });
E
Erich Gamma 已提交
72 73 74 75 76 77
		} catch (err) {
			e(new Error(nls.localize('invalidManifest', "Extension invalid: package.json is not a JSON file.")));
		}
	});
}

78
export function validateLocalExtension(zipPath: string): TPromise<IExtensionManifest> {
E
Erich Gamma 已提交
79 80
	return buffer(zipPath, 'extension/package.json')
		.then(buffer => parseManifest(buffer.toString('utf8')))
81
		.then(({ manifest }) => TPromise.as(manifest));
E
Erich Gamma 已提交
82 83
}

84 85 86 87 88
function readManifest(extensionPath: string): TPromise<{ manifest: IExtensionManifest; metadata: IGalleryMetadata; }> {
	const promises = [
		pfs.readFile(path.join(extensionPath, 'package.json'), 'utf8')
			.then(raw => parseManifest(raw)),
		pfs.readFile(path.join(extensionPath, 'package.nls.json'), 'utf8')
R
Ron Buckton 已提交
89
			.then(null, err => err.code !== 'ENOENT' ? TPromise.wrapError<string>(err) : '{}')
90 91 92 93 94 95 96 97 98 99 100
			.then(raw => JSON.parse(raw))
	];

	return TPromise.join<any>(promises).then(([{ manifest, metadata }, translations]) => {
		return {
			manifest: localizeManifest(manifest, translations),
			metadata
		};
	});
}

S
Sandeep Somavarapu 已提交
101 102 103
interface InstallableExtension {
	zipPath: string;
	id: string;
S
Sandeep Somavarapu 已提交
104
	metadata?: IGalleryMetadata;
S
Sandeep Somavarapu 已提交
105 106
}

107
export class ExtensionManagementService extends Disposable implements IExtensionManagementService {
E
Erich Gamma 已提交
108

109
	_serviceBrand: any;
E
Erich Gamma 已提交
110 111

	private extensionsPath: string;
112 113
	private uninstalledPath: string;
	private uninstalledFileLimiter: Limiter<void>;
114
	private reportedExtensions: TPromise<IReportedExtension[]> | undefined;
J
Joao Moreno 已提交
115
	private lastReportTimestamp = 0;
116
	private readonly installationStartTime: Map<string, number> = new Map<string, number>();
117
	private readonly installingExtensions: Map<string, TPromise<ILocalExtension>> = new Map<string, TPromise<ILocalExtension>>();
S
Sandeep Somavarapu 已提交
118
	private readonly uninstallingExtensions: Map<string, TPromise<void>> = new Map<string, TPromise<void>>();
119
	private readonly manifestCache: ExtensionsManifestCache;
S
Sandeep Somavarapu 已提交
120
	private readonly extensionLifecycle: ExtensionsLifecycle;
E
Erich Gamma 已提交
121

S
Sandeep Somavarapu 已提交
122 123
	private readonly _onInstallExtension = new Emitter<InstallExtensionEvent>();
	readonly onInstallExtension: Event<InstallExtensionEvent> = this._onInstallExtension.event;
E
Erich Gamma 已提交
124

S
Sandeep Somavarapu 已提交
125 126
	private readonly _onDidInstallExtension = new Emitter<DidInstallExtensionEvent>();
	readonly onDidInstallExtension: Event<DidInstallExtensionEvent> = this._onDidInstallExtension.event;
E
Erich Gamma 已提交
127

S
Sandeep Somavarapu 已提交
128 129 130
	private readonly _onUninstallExtension = new Emitter<IExtensionIdentifier>();
	readonly onUninstallExtension: Event<IExtensionIdentifier> = this._onUninstallExtension.event;

S
Sandeep Somavarapu 已提交
131 132
	private _onDidUninstallExtension = new Emitter<DidUninstallExtensionEvent>();
	onDidUninstallExtension: Event<DidUninstallExtensionEvent> = this._onDidUninstallExtension.event;
E
Erich Gamma 已提交
133 134

	constructor(
135
		@IEnvironmentService environmentService: IEnvironmentService,
136
		@IDialogService private dialogService: IDialogService,
137
		@IExtensionGalleryService private galleryService: IExtensionGalleryService,
138 139
		@ILogService private logService: ILogService,
		@ITelemetryService private telemetryService: ITelemetryService,
E
Erich Gamma 已提交
140
	) {
141
		super();
J
Joao Moreno 已提交
142
		this.extensionsPath = environmentService.extensionsPath;
143 144
		this.uninstalledPath = path.join(this.extensionsPath, '.obsolete');
		this.uninstalledFileLimiter = new Limiter(1);
145
		this.manifestCache = this._register(new ExtensionsManifestCache(environmentService, this));
S
Sandeep Somavarapu 已提交
146
		this.extensionLifecycle = this._register(new ExtensionsLifecycle(this.logService));
S
Sandeep Somavarapu 已提交
147 148 149 150 151 152 153

		this._register(toDisposable(() => {
			this.installingExtensions.forEach(promise => promise.cancel());
			this.uninstallingExtensions.forEach(promise => promise.cancel());
			this.installingExtensions.clear();
			this.uninstallingExtensions.clear();
		}));
A
Alex Dima 已提交
154 155
	}

S
Sandeep Somavarapu 已提交
156
	install(zipPath: string): TPromise<void> {
157 158
		zipPath = path.resolve(zipPath);

S
Sandeep Somavarapu 已提交
159
		return validateLocalExtension(zipPath)
S
Sandeep Somavarapu 已提交
160
			.then(manifest => {
S
Sandeep Somavarapu 已提交
161
				const identifier = { id: getLocalExtensionIdFromManifest(manifest) };
S
Sandeep Somavarapu 已提交
162
				if (manifest.engines && manifest.engines.vscode && !isEngineValid(manifest.engines.vscode)) {
S
Sandeep Somavarapu 已提交
163
					return TPromise.wrapError<void>(new Error(nls.localize('incompatible', "Unable to install Extension '{0}' as it is not compatible with Code '{1}'.", identifier.id, pkg.version)));
S
Sandeep Somavarapu 已提交
164
				}
S
Sandeep Somavarapu 已提交
165
				return this.removeIfExists(identifier.id)
166
					.then(
M
Matt Bierner 已提交
167 168 169 170 171 172 173 174 175 176
						() => this.checkOutdated(manifest)
							.then(validated => {
								if (validated) {
									this.logService.info('Installing the extension:', identifier.id);
									this._onInstallExtension.fire({ identifier, zipPath });
									return this.getMetadata(getGalleryExtensionId(manifest.publisher, manifest.name))
										.then(
											metadata => this.installFromZipPath(identifier, zipPath, metadata, manifest),
											error => this.installFromZipPath(identifier, zipPath, null, manifest))
										.then(
S
Sandeep Somavarapu 已提交
177
											() => { this.logService.info('Successfully installed the extension:', identifier.id); },
M
Matt Bierner 已提交
178 179 180 181 182 183 184 185
											e => {
												this.logService.error('Failed to install the extension:', identifier.id, e.message);
												return TPromise.wrapError(e);
											});
								}
								return null;
							}),
						e => TPromise.wrapError(new Error(nls.localize('restartCode', "Please restart Code before reinstalling {0}.", manifest.displayName || manifest.name))));
186 187 188
			});
	}

S
Sandeep Somavarapu 已提交
189 190 191 192
	private removeIfExists(id: string): TPromise<void> {
		return this.getInstalled(LocalExtensionType.User)
			.then(installed => installed.filter(i => i.identifier.id === id)[0])
			.then(existing => existing ? this.removeExtension(existing, 'existing') : null);
S
Sandeep Somavarapu 已提交
193
	}
194

S
Sandeep Somavarapu 已提交
195 196
	private checkOutdated(manifest: IExtensionManifest): TPromise<boolean> {
		const extensionIdentifier = { id: getGalleryExtensionId(manifest.publisher, manifest.name) };
197
		return this.getInstalled(LocalExtensionType.User)
S
Sandeep Somavarapu 已提交
198 199 200 201
			.then(installedExtensions => {
				const newer = installedExtensions.filter(local => areSameExtensions(extensionIdentifier, { id: getGalleryExtensionIdFromLocal(local) }) && semver.gt(local.manifest.version, manifest.version))[0];
				if (newer) {
					const message = nls.localize('installingOutdatedExtension', "A newer version of this extension is already installed. Would you like to override this with the older version?");
202
					const buttons = [
S
Sandeep Somavarapu 已提交
203 204 205
						nls.localize('override', "Override"),
						nls.localize('cancel', "Cancel")
					];
206
					return this.dialogService.show(Severity.Info, message, buttons, { cancelId: 1 })
S
Sandeep Somavarapu 已提交
207 208 209 210 211 212 213 214
						.then<boolean>(value => {
							if (value === 0) {
								return this.uninstall(newer, true).then(() => true);
							}
							return TPromise.wrapError(errors.canceled());
						});
				}
				return true;
215
			});
S
Sandeep Somavarapu 已提交
216 217
	}

S
Sandeep Somavarapu 已提交
218
	private installFromZipPath(identifier: IExtensionIdentifier, zipPath: string, metadata: IGalleryMetadata, manifest: IExtensionManifest): TPromise<ILocalExtension> {
S
Sandeep Somavarapu 已提交
219
		return this.toNonCancellablePromise(this.getInstalled()
S
Sandeep Somavarapu 已提交
220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240
			.then(installed => {
				const operation = this.getOperation({ id: getIdFromLocalExtensionId(identifier.id), uuid: identifier.uuid }, installed);
				return this.installExtension({ zipPath, id: identifier.id, metadata })
					.then(local => {
						if (this.galleryService.isEnabled() && local.manifest.extensionDependencies && local.manifest.extensionDependencies.length) {
							return this.getDependenciesToInstall(local.manifest.extensionDependencies)
								.then(dependenciesToInstall => {
									dependenciesToInstall = metadata ? dependenciesToInstall.filter(d => d.identifier.uuid !== metadata.id) : dependenciesToInstall;
									return this.downloadAndInstallExtensions(dependenciesToInstall, dependenciesToInstall.map(d => this.getOperation(d.identifier, installed)));
								})
								.then(() => local, error => {
									this.setUninstalled(local);
									return TPromise.wrapError(new Error(nls.localize('errorInstallingDependencies', "Error while installing dependencies. {0}", error instanceof Error ? error.message : error)));
								});
						}
						return local;
					})
					.then(
						local => { this._onDidInstallExtension.fire({ identifier, zipPath, local, operation }); return local; },
						error => { this._onDidInstallExtension.fire({ identifier, zipPath, operation, error }); return TPromise.wrapError(error); }
					);
S
Sandeep Somavarapu 已提交
241
			}));
E
Erich Gamma 已提交
242 243
	}

S
Sandeep Somavarapu 已提交
244
	installFromGallery(extension: IGalleryExtension): TPromise<void> {
S
Sandeep Somavarapu 已提交
245
		this.onInstallExtensions([extension]);
S
Sandeep Somavarapu 已提交
246
		return this.toNonCancellablePromise(this.getInstalled(LocalExtensionType.User)
S
Sandeep Somavarapu 已提交
247 248 249 250 251 252
			.then(installed => this.collectExtensionsToInstall(extension)
				.then(
					extensionsToInstall => {
						if (extensionsToInstall.length > 1) {
							this.onInstallExtensions(extensionsToInstall.slice(1));
						}
S
Sandeep Somavarapu 已提交
253
						const operataions: InstallOperation[] = extensionsToInstall.map(e => this.getOperation(e.identifier, installed));
254
						return this.downloadAndInstallExtensions(extensionsToInstall, operataions)
S
Sandeep Somavarapu 已提交
255 256
							.then(
								locals => this.onDidInstallExtensions(extensionsToInstall, locals, operataions, [])
S
Sandeep Somavarapu 已提交
257
									.then(() => null),
S
Sandeep Somavarapu 已提交
258 259
								errors => this.onDidInstallExtensions(extensionsToInstall, [], operataions, errors));
					},
S
Sandeep Somavarapu 已提交
260
					error => this.onDidInstallExtensions([extension], [], [this.getOperation(extension.identifier, installed)], [error]))));
S
Sandeep Somavarapu 已提交
261 262
	}

S
Sandeep Somavarapu 已提交
263
	reinstallFromGallery(extension: ILocalExtension): TPromise<void> {
S
Sandeep Somavarapu 已提交
264
		if (!this.galleryService.isEnabled()) {
265
			return TPromise.wrapError(new Error(nls.localize('MarketPlaceDisabled', "Marketplace is not enabled")));
S
Sandeep Somavarapu 已提交
266 267 268 269
		}
		return this.findGalleryExtension(extension)
			.then(galleryExtension => {
				if (galleryExtension) {
S
Sandeep Somavarapu 已提交
270
					return this.setUninstalled(extension)
271 272
						.then(() => this.removeUninstalledExtension(extension)
							.then(
M
Matt Bierner 已提交
273 274
								() => this.installFromGallery(galleryExtension),
								e => TPromise.wrapError(new Error(nls.localize('removeError', "Error while removing the extension: {0}. Please Quit and Start VS Code before trying again.", toErrorMessage(e))))));
S
Sandeep Somavarapu 已提交
275
				}
276
				return TPromise.wrapError(new Error(nls.localize('Not a Marketplace extension', "Only Marketplace Extensions can be reinstalled")));
S
Sandeep Somavarapu 已提交
277 278 279
			});
	}

S
Sandeep Somavarapu 已提交
280 281
	private getOperation(extensionToInstall: IExtensionIdentifier, installed: ILocalExtension[]): InstallOperation {
		return installed.some(i => areSameExtensions({ id: getGalleryExtensionIdFromLocal(i), uuid: i.identifier.uuid }, extensionToInstall)) ? InstallOperation.Update : InstallOperation.Install;
S
Sandeep Somavarapu 已提交
282 283
	}

S
Sandeep Somavarapu 已提交
284 285 286 287
	private collectExtensionsToInstall(extension: IGalleryExtension): TPromise<IGalleryExtension[]> {
		return this.galleryService.loadCompatibleVersion(extension)
			.then(compatible => {
				if (!compatible) {
S
Sandeep Somavarapu 已提交
288
					return TPromise.wrapError<IGalleryExtension[]>(new ExtensionManagementError(nls.localize('notFoundCompatible', "Unable to install '{0}'; there is no available version compatible with VS Code '{1}'.", extension.identifier.id, pkg.version), INSTALL_ERROR_INCOMPATIBLE));
S
Sandeep Somavarapu 已提交
289 290 291
				}
				return this.getDependenciesToInstall(compatible.properties.dependencies)
					.then(
M
Matt Bierner 已提交
292 293
						dependenciesToInstall => ([compatible, ...dependenciesToInstall.filter(d => d.identifier.uuid !== compatible.identifier.uuid)]),
						error => TPromise.wrapError<IGalleryExtension[]>(new ExtensionManagementError(this.joinErrors(error).message, INSTALL_ERROR_GALLERY)));
S
Sandeep Somavarapu 已提交
294
			},
M
Matt Bierner 已提交
295
				error => TPromise.wrapError<IGalleryExtension[]>(new ExtensionManagementError(this.joinErrors(error).message, INSTALL_ERROR_GALLERY)));
296 297
	}

S
Sandeep Somavarapu 已提交
298
	private downloadAndInstallExtensions(extensions: IGalleryExtension[], operations: InstallOperation[]): TPromise<ILocalExtension[]> {
299
		return TPromise.join(extensions.map((extensionToInstall, index) => this.downloadAndInstallExtension(extensionToInstall, operations[index])))
S
Sandeep Somavarapu 已提交
300 301 302
			.then(null, errors => this.rollback(extensions).then(() => TPromise.wrapError(errors), () => TPromise.wrapError(errors)));
	}

S
Sandeep Somavarapu 已提交
303
	private downloadAndInstallExtension(extension: IGalleryExtension, operation: InstallOperation): TPromise<ILocalExtension> {
S
Sandeep Somavarapu 已提交
304 305
		let installingExtension = this.installingExtensions.get(extension.identifier.id);
		if (!installingExtension) {
J
Joao Moreno 已提交
306
			installingExtension = this.getExtensionsReport()
J
Joao Moreno 已提交
307 308
				.then(report => {
					if (getMaliciousExtensionsSet(report).has(extension.identifier.id)) {
J
Joao Moreno 已提交
309
						throw new Error(nls.localize('malicious extension', "Can't install extension since it was reported to be problematic."));
J
Joao Moreno 已提交
310 311 312 313
					} else {
						return extension;
					}
				})
314
				.then(extension => this.downloadInstallableExtension(extension, operation))
S
Sandeep Somavarapu 已提交
315 316
				.then(installableExtension => this.installExtension(installableExtension))
				.then(
M
Matt Bierner 已提交
317 318
					local => { this.installingExtensions.delete(extension.identifier.id); return local; },
					e => { this.installingExtensions.delete(extension.identifier.id); return TPromise.wrapError(e); }
S
Sandeep Somavarapu 已提交
319 320 321
				);

			this.installingExtensions.set(extension.identifier.id, installingExtension);
S
Sandeep Somavarapu 已提交
322 323
		}
		return installingExtension;
S
Sandeep Somavarapu 已提交
324 325
	}

S
Sandeep Somavarapu 已提交
326
	private downloadInstallableExtension(extension: IGalleryExtension, operation: InstallOperation): TPromise<InstallableExtension> {
S
Sandeep Somavarapu 已提交
327
		const metadata = <IGalleryMetadata>{
S
Sandeep Somavarapu 已提交
328
			id: extension.identifier.uuid,
S
Sandeep Somavarapu 已提交
329 330 331
			publisherId: extension.publisherId,
			publisherDisplayName: extension.publisherDisplayName,
		};
S
Sandeep Somavarapu 已提交
332 333 334

		return this.galleryService.loadCompatibleVersion(extension)
			.then(
M
Matt Bierner 已提交
335 336 337
				compatible => {
					if (compatible) {
						this.logService.trace('Started downloading extension:', extension.name);
338
						return this.galleryService.download(extension, operation)
M
Matt Bierner 已提交
339 340 341 342 343 344 345 346 347 348 349 350 351 352 353
							.then(
								zipPath => {
									this.logService.info('Downloaded extension:', extension.name);
									return validateLocalExtension(zipPath)
										.then(
											manifest => (<InstallableExtension>{ zipPath, id: getLocalExtensionIdFromManifest(manifest), metadata }),
											error => TPromise.wrapError(new ExtensionManagementError(this.joinErrors(error).message, INSTALL_ERROR_VALIDATING))
										);
								},
								error => TPromise.wrapError(new ExtensionManagementError(this.joinErrors(error).message, INSTALL_ERROR_DOWNLOADING)));
					} else {
						return TPromise.wrapError<InstallableExtension>(new ExtensionManagementError(nls.localize('notFoundCompatibleDependency', "Unable to install because, the depending extension '{0}' compatible with current version '{1}' of VS Code is not found.", extension.identifier.id, pkg.version), INSTALL_ERROR_INCOMPATIBLE));
					}
				},
				error => TPromise.wrapError<InstallableExtension>(new ExtensionManagementError(this.joinErrors(error).message, INSTALL_ERROR_GALLERY)));
S
Sandeep Somavarapu 已提交
354 355 356 357
	}

	private onInstallExtensions(extensions: IGalleryExtension[]): void {
		for (const extension of extensions) {
358
			this.logService.info('Installing extension:', extension.name);
359
			this.installationStartTime.set(extension.identifier.id, new Date().getTime());
S
Sandeep Somavarapu 已提交
360
			const id = getLocalExtensionIdFromGallery(extension, extension.version);
S
Sandeep Somavarapu 已提交
361
			this._onInstallExtension.fire({ identifier: { id, uuid: extension.identifier.uuid }, gallery: extension });
362
		}
363 364
	}

S
Sandeep Somavarapu 已提交
365
	private onDidInstallExtensions(extensions: IGalleryExtension[], locals: ILocalExtension[], operations: InstallOperation[], errors: Error[]): TPromise<any> {
S
Sandeep Somavarapu 已提交
366
		extensions.forEach((gallery, index) => {
S
Sandeep Somavarapu 已提交
367
			const identifier = { id: getLocalExtensionIdFromGallery(gallery, gallery.version), uuid: gallery.identifier.uuid };
368 369
			const local = locals[index];
			const error = errors[index];
S
Sandeep Somavarapu 已提交
370
			const operation = operations[index];
371
			if (local) {
372
				this.logService.info(`Extensions installed successfully:`, gallery.identifier.id);
S
Sandeep Somavarapu 已提交
373
				this._onDidInstallExtension.fire({ identifier, gallery, local, operation });
S
Sandeep Somavarapu 已提交
374
			} else {
375
				const errorCode = error && (<ExtensionManagementError>error).code ? (<ExtensionManagementError>error).code : ERROR_UNKNOWN;
376
				this.logService.error(`Failed to install extension:`, gallery.identifier.id, error ? error.message : errorCode);
S
Sandeep Somavarapu 已提交
377
				this._onDidInstallExtension.fire({ identifier, gallery, operation, error: errorCode });
S
Sandeep Somavarapu 已提交
378
			}
379
			const startTime = this.installationStartTime.get(gallery.identifier.id);
S
Sandeep Somavarapu 已提交
380
			this.reportTelemetry(operations[index] === InstallOperation.Update ? 'extensionGallery:update' : 'extensionGallery:install', getGalleryExtensionTelemetryData(gallery), startTime ? new Date().getTime() - startTime : void 0, error);
381
			this.installationStartTime.delete(gallery.identifier.id);
S
Sandeep Somavarapu 已提交
382
		});
383
		return errors.length ? TPromise.wrapError(this.joinErrors(errors)) : TPromise.as(null);
384 385
	}

386
	private getDependenciesToInstall(dependencies: IExtensionDependency[]): TPromise<IGalleryExtension[]> {
S
Sandeep Somavarapu 已提交
387
		if (dependencies.length) {
S
Sandeep Somavarapu 已提交
388 389
			return this.getInstalled()
				.then(installed => {
390
					const uninstalledDeps = dependencies.filter(d => installed.every(i => !areSameExtensions(i.galleryIdentifier, { id: d.id })));
S
Sandeep Somavarapu 已提交
391
					if (uninstalledDeps.length) {
392
						return this.galleryService.loadAllDependencies(uninstalledDeps.map(dep => (<IExtensionIdentifier>{ id: dep.id })))
S
Sandeep Somavarapu 已提交
393 394 395 396 397 398 399
							.then(allDependencies => allDependencies.filter(d => {
								const extensionId = getLocalExtensionIdFromGallery(d, d.version);
								return installed.every(({ identifier }) => identifier.id !== extensionId);
							}));
					}
					return [];
				});
S
Sandeep Somavarapu 已提交
400 401
		}
		return TPromise.as([]);
402 403
	}

404 405 406
	private installExtension(installableExtension: InstallableExtension): TPromise<ILocalExtension> {
		return this.unsetUninstalledAndGetLocal(installableExtension.id)
			.then(
M
Matt Bierner 已提交
407 408 409 410 411 412 413 414 415 416 417 418
				local => {
					if (local) {
						return local;
					}
					return this.extractAndInstall(installableExtension);
				},
				e => {
					if (isMacintosh) {
						return TPromise.wrapError<ILocalExtension>(new ExtensionManagementError(nls.localize('quitCode', "Unable to install the extension. Please Quit and Start VS Code before reinstalling."), INSTALL_ERROR_UNSET_UNINSTALLED));
					}
					return TPromise.wrapError<ILocalExtension>(new ExtensionManagementError(nls.localize('exitCode', "Unable to install the extension. Please Exit and Start VS Code before reinstalling."), INSTALL_ERROR_UNSET_UNINSTALLED));
				});
419
	}
J
Joao Moreno 已提交
420

421 422 423 424
	private unsetUninstalledAndGetLocal(id: string): TPromise<ILocalExtension> {
		return this.isUninstalled(id)
			.then(isUninstalled => {
				if (isUninstalled) {
425
					this.logService.trace('Removing the extension from uninstalled list:', id);
426 427
					// If the same version of extension is marked as uninstalled, remove it from there and return the local.
					return this.unsetUninstalled(id)
428
						.then(() => {
429
							this.logService.info('Removed the extension from uninstalled list:', id);
430 431
							return this.getInstalled(LocalExtensionType.User);
						})
432 433 434 435 436 437
						.then(installed => installed.filter(i => i.identifier.id === id)[0]);
				}
				return null;
			});
	}

S
Sandeep Somavarapu 已提交
438
	private extractAndInstall({ zipPath, id, metadata }: InstallableExtension): TPromise<ILocalExtension> {
S
Sandeep Somavarapu 已提交
439
		const tempPath = path.join(this.extensionsPath, `.${id}`);
440
		const extensionPath = path.join(this.extensionsPath, id);
S
Sandeep Somavarapu 已提交
441 442
		return pfs.rimraf(extensionPath)
			.then(() => this.extractAndRename(id, zipPath, tempPath, extensionPath), e => TPromise.wrapError(new ExtensionManagementError(nls.localize('errorDeleting', "Unable to delete the existing folder '{0}' while installing the extension '{1}'. Please delete the folder manually and try again", extensionPath, id), INSTALL_ERROR_DELETING)))
443 444 445 446
			.then(() => {
				this.logService.info('Installation completed.', id);
				return this.scanExtension(id, this.extensionsPath, LocalExtensionType.User);
			})
447 448 449 450 451 452 453
			.then(local => {
				if (metadata) {
					local.metadata = metadata;
					return this.saveMetadataForLocalExtension(local);
				}
				return local;
			});
E
Erich Gamma 已提交
454 455
	}

S
Sandeep Somavarapu 已提交
456 457
	private extractAndRename(id: string, zipPath: string, extractPath: string, renamePath: string): TPromise<void> {
		return this.extract(id, zipPath, extractPath)
S
Sandeep Somavarapu 已提交
458
			.then(() => this.rename(id, extractPath, renamePath, Date.now() + (2 * 60 * 1000) /* Retry for 2 minutes */)
S
Sandeep Somavarapu 已提交
459 460 461 462 463 464 465 466
				.then(
					() => this.logService.info('Renamed to', renamePath),
					e => {
						this.logService.info('Rename failed. Deleting from extracted location', extractPath);
						return always(pfs.rimraf(extractPath), () => null).then(() => TPromise.wrapError(e));
					}));
	}

467
	private extract(id: string, zipPath: string, extractPath: string): TPromise<void> {
468 469 470
		this.logService.trace(`Started extracting the extension from ${zipPath} to ${extractPath}`);
		return pfs.rimraf(extractPath)
			.then(
S
Sandeep Somavarapu 已提交
471
				() => extract(zipPath, extractPath, { sourcePath: 'extension', overwrite: true }, this.logService)
M
Matt Bierner 已提交
472 473 474
					.then(
						() => this.logService.info(`Extracted extension to ${extractPath}:`, id),
						e => always(pfs.rimraf(extractPath), () => null)
S
Sandeep Somavarapu 已提交
475
							.then(() => TPromise.wrapError(new ExtensionManagementError(e.message, e instanceof ExtractError ? e.type : INSTALL_ERROR_EXTRACTING)))),
M
Matt Bierner 已提交
476
				e => TPromise.wrapError(new ExtensionManagementError(this.joinErrors(e).message, INSTALL_ERROR_DELETING)));
477 478
	}

479 480
	private rename(id: string, extractPath: string, renamePath: string, retryUntil: number): TPromise<void> {
		return pfs.rename(extractPath, renamePath)
S
Sandeep Somavarapu 已提交
481 482 483 484 485
			.then(null, error => {
				if (isWindows && error && error.code === 'EPERM' && Date.now() < retryUntil) {
					this.logService.info(`Failed renaming ${extractPath} to ${renamePath} with 'EPERM' error. Trying again...`);
					return this.rename(id, extractPath, renamePath, retryUntil);
				}
S
Sandeep Somavarapu 已提交
486
				return TPromise.wrapError(new ExtensionManagementError(error.message || nls.localize('renameError', "Unknown error while renaming {0} to {1}", extractPath, renamePath), error.code || INSTALL_ERROR_RENAMING));
S
Sandeep Somavarapu 已提交
487
			});
S
Sandeep Somavarapu 已提交
488 489 490 491 492 493
	}

	private rollback(extensions: IGalleryExtension[]): TPromise<void> {
		return this.getInstalled(LocalExtensionType.User)
			.then(installed =>
				TPromise.join(installed.filter(local => extensions.some(galleryExtension => local.identifier.id === getLocalExtensionIdFromGallery(galleryExtension, galleryExtension.version))) // Only check id (pub.name-version) because we want to rollback the exact version
S
Sandeep Somavarapu 已提交
494
					.map(local => this.setUninstalled(local))))
S
Sandeep Somavarapu 已提交
495 496 497
			.then(() => null, () => null);
	}

498
	uninstall(extension: ILocalExtension, force = false): TPromise<void> {
S
Sandeep Somavarapu 已提交
499
		return this.toNonCancellablePromise(this.getInstalled(LocalExtensionType.User)
S
Sandeep Somavarapu 已提交
500 501 502 503 504
			.then(installed => {
				const promises = installed
					.filter(e => e.manifest.publisher === extension.manifest.publisher && e.manifest.name === extension.manifest.name)
					.map(e => this.checkForDependenciesAndUninstall(e, installed, force));
				return TPromise.join(promises).then(() => null, error => TPromise.wrapError(this.joinErrors(error)));
S
Sandeep Somavarapu 已提交
505
			}));
S
Sandeep Somavarapu 已提交
506 507
	}

508 509
	updateMetadata(local: ILocalExtension, metadata: IGalleryMetadata): TPromise<ILocalExtension> {
		local.metadata = metadata;
510 511 512 513 514
		return this.saveMetadataForLocalExtension(local)
			.then(localExtension => {
				this.manifestCache.invalidate();
				return localExtension;
			});
515 516 517 518 519 520 521 522 523 524 525 526 527 528
	}

	private saveMetadataForLocalExtension(local: ILocalExtension): TPromise<ILocalExtension> {
		if (!local.metadata) {
			return TPromise.as(local);
		}
		const manifestPath = path.join(this.extensionsPath, local.identifier.id, 'package.json');
		return pfs.readFile(manifestPath, 'utf8')
			.then(raw => parseManifest(raw))
			.then(({ manifest }) => assign(manifest, { __metadata: local.metadata }))
			.then(manifest => pfs.writeFile(manifestPath, JSON.stringify(manifest, null, '\t')))
			.then(() => local);
	}

S
Sandeep Somavarapu 已提交
529
	private getMetadata(extensionName: string): TPromise<IGalleryMetadata> {
S
Sandeep Somavarapu 已提交
530 531 532 533 534 535 536 537 538 539 540 541 542 543 544 545 546 547
		return this.findGalleryExtensionByName(extensionName)
			.then(galleryExtension => galleryExtension ? <IGalleryMetadata>{ id: galleryExtension.identifier.uuid, publisherDisplayName: galleryExtension.publisherDisplayName, publisherId: galleryExtension.publisherId } : null);
	}

	private findGalleryExtension(local: ILocalExtension): TPromise<IGalleryExtension> {
		if (local.identifier.uuid) {
			return this.findGalleryExtensionById(local.identifier.uuid)
				.then(galleryExtension => galleryExtension ? galleryExtension : this.findGalleryExtensionByName(getGalleryExtensionIdFromLocal(local)));
		}
		return this.findGalleryExtensionByName(getGalleryExtensionIdFromLocal(local));
	}

	private findGalleryExtensionById(uuid: string): TPromise<IGalleryExtension> {
		return this.galleryService.query({ ids: [uuid], pageSize: 1 }).then(galleryResult => galleryResult.firstPage[0]);
	}

	private findGalleryExtensionByName(name: string): TPromise<IGalleryExtension> {
		return this.galleryService.query({ names: [name], pageSize: 1 }).then(galleryResult => galleryResult.firstPage[0]);
S
Sandeep Somavarapu 已提交
548 549
	}

550 551
	private joinErrors(errorOrErrors: (Error | string) | ((Error | string)[])): Error {
		const errors = Array.isArray(errorOrErrors) ? errorOrErrors : [errorOrErrors];
S
Sandeep Somavarapu 已提交
552 553 554 555
		if (errors.length === 1) {
			return errors[0] instanceof Error ? <Error>errors[0] : new Error(<string>errors[0]);
		}
		return errors.reduce<Error>((previousValue: Error, currentValue: Error | string) => {
S
Sandeep Somavarapu 已提交
556
			return new Error(`${previousValue.message}${previousValue.message ? ',' : ''}${currentValue instanceof Error ? currentValue.message : currentValue}`);
S
Sandeep Somavarapu 已提交
557
		}, new Error(''));
J
Joao Moreno 已提交
558 559
	}

560
	private checkForDependenciesAndUninstall(extension: ILocalExtension, installed: ILocalExtension[], force: boolean): TPromise<void> {
J
Joao Moreno 已提交
561
		return this.preUninstallExtension(extension)
562
			.then(() => this.hasDependencies(extension, installed) ? this.promptForDependenciesAndUninstall(extension, installed, force) : this.uninstallExtensions(extension, [], installed))
563
			.then(() => this.postUninstallExtension(extension),
M
Matt Bierner 已提交
564
				error => {
565
					this.postUninstallExtension(extension, new ExtensionManagementError(error instanceof Error ? error.message : error, INSTALL_ERROR_LOCAL));
M
Matt Bierner 已提交
566 567
					return TPromise.wrapError(error);
				});
S
Sandeep Somavarapu 已提交
568 569
	}

570 571
	private hasDependencies(extension: ILocalExtension, installed: ILocalExtension[]): boolean {
		if (extension.manifest.extensionDependencies && extension.manifest.extensionDependencies.length) {
572
			return installed.some(i => extension.manifest.extensionDependencies.some(dep => areSameExtensions({ id: dep.id }, i.galleryIdentifier)));
573 574 575 576
		}
		return false;
	}

577 578
	private promptForDependenciesAndUninstall(extension: ILocalExtension, installed: ILocalExtension[], force: boolean): TPromise<void> {
		if (force) {
579
			return this.uninstallExtensionWithDependencies(extension, installed);
580 581
		}

582
		const message = nls.localize('uninstallDependeciesConfirmation', "Would you like to uninstall '{0}' only or its dependencies also?", extension.manifest.displayName || extension.manifest.name);
583
		const buttons = [
S
Sandeep Somavarapu 已提交
584 585
			nls.localize('uninstallOnly', "Extension Only"),
			nls.localize('uninstallAll', "Uninstall All"),
S
Sandeep Somavarapu 已提交
586 587
			nls.localize('cancel', "Cancel")
		];
588
		this.logService.info('Requesting for confirmation to uninstall extension with dependencies', extension.identifier.id);
589
		return this.dialogService.show(Severity.Info, message, buttons, { cancelId: 2 })
S
Sandeep Somavarapu 已提交
590 591
			.then<void>(value => {
				if (value === 0) {
592
					return this.uninstallExtensions(extension, [], installed);
S
Sandeep Somavarapu 已提交
593 594
				}
				if (value === 1) {
595
					return this.uninstallExtensionWithDependencies(extension, installed);
S
Sandeep Somavarapu 已提交
596
				}
597
				this.logService.info('Cancelled uninstalling extension:', extension.identifier.id);
S
Sandeep Somavarapu 已提交
598 599 600 601
				return TPromise.wrapError(errors.canceled());
			}, error => TPromise.wrapError(errors.canceled()));
	}

602 603 604 605 606 607 608 609 610 611 612
	private uninstallExtensionWithDependencies(extension: ILocalExtension, installed: ILocalExtension[]): TPromise<void> {
		const dependencies = this.getAllDependenciesToUninstall(extension, installed).filter(e => e !== extension);

		const dependenciesToUninstall = dependencies.slice(0);
		for (let index = 0; index < dependencies.length; index++) {
			const dep = dependencies[index];
			const dependents = this.getMandatoryDependents(dep, installed);
			// Remove the dependency from the uninstall list if there is a dependent which will not be uninstalled.
			if (dependents.some(e => e !== extension && dependencies.indexOf(e) === -1)) {
				dependenciesToUninstall.splice(index - (dependencies.length - dependenciesToUninstall.length), 1);
			}
613
		}
614 615 616 617 618 619 620 621 622 623 624 625 626

		return this.uninstallExtensions(extension, dependenciesToUninstall, installed);
	}

	private uninstallExtensions(extension: ILocalExtension, otherExtensions: ILocalExtension[], installed: ILocalExtension[]): TPromise<void> {
		const mandatoryDependents = this.getMandatoryDependents(extension, installed);
		if (mandatoryDependents.length) {
			const dependents = mandatoryDependents.filter(dependent => extension !== dependent && otherExtensions.indexOf(dependent) === -1);
			if (dependents.length) {
				return TPromise.wrapError<void>(new Error(this.getDependentsErrorMessage(extension, dependents)));
			}
		}
		return TPromise.join([this.uninstallExtension(extension), ...otherExtensions.map(d => this.doUninstall(d))]).then(() => null);
627 628
	}

S
Sandeep Somavarapu 已提交
629 630 631 632 633 634 635 636 637 638 639 640 641
	private getDependentsErrorMessage(extension: ILocalExtension, dependents: ILocalExtension[]): string {
		if (dependents.length === 1) {
			return nls.localize('singleDependentError', "Cannot uninstall extension '{0}'. Extension '{1}' depends on this.",
				extension.manifest.displayName || extension.manifest.name, dependents[0].manifest.displayName || dependents[0].manifest.name);
		}
		if (dependents.length === 2) {
			return nls.localize('twoDependentsError', "Cannot uninstall extension '{0}'. Extensions '{1}' and '{2}' depend on this.",
				extension.manifest.displayName || extension.manifest.name, dependents[0].manifest.displayName || dependents[0].manifest.name, dependents[1].manifest.displayName || dependents[1].manifest.name);
		}
		return nls.localize('multipleDependentsError', "Cannot uninstall extension '{0}'. Extensions '{1}', '{2}' and others depend on this.",
			extension.manifest.displayName || extension.manifest.name, dependents[0].manifest.displayName || dependents[0].manifest.name, dependents[1].manifest.displayName || dependents[1].manifest.name);
	}

642
	private getAllDependenciesToUninstall(extension: ILocalExtension, installed: ILocalExtension[], checked: ILocalExtension[] = []): ILocalExtension[] {
643 644 645 646 647 648 649
		if (checked.indexOf(extension) !== -1) {
			return [];
		}
		checked.push(extension);
		if (!extension.manifest.extensionDependencies || extension.manifest.extensionDependencies.length === 0) {
			return [];
		}
650
		const dependenciesToUninstall = installed.filter(i => extension.manifest.extensionDependencies.some(dep => areSameExtensions({ id: dep.id }, i.galleryIdentifier)));
651 652
		const depsOfDeps = [];
		for (const dep of dependenciesToUninstall) {
653
			depsOfDeps.push(...this.getAllDependenciesToUninstall(dep, installed, checked));
654 655 656 657
		}
		return [...dependenciesToUninstall, ...depsOfDeps];
	}

658 659
	private getMandatoryDependents(extension: ILocalExtension, installed: ILocalExtension[]): ILocalExtension[] {
		return installed.filter(e => e.manifest.extensionDependencies && e.manifest.extensionDependencies.some(dep => !dep.optional && areSameExtensions({ id: dep.id }, extension.galleryIdentifier)));
660 661
	}

J
Joao Moreno 已提交
662 663
	private doUninstall(extension: ILocalExtension): TPromise<void> {
		return this.preUninstallExtension(extension)
S
Sandeep Somavarapu 已提交
664
			.then(() => this.uninstallExtension(extension))
665
			.then(() => this.postUninstallExtension(extension),
M
Matt Bierner 已提交
666
				error => {
667
					this.postUninstallExtension(extension, new ExtensionManagementError(error instanceof Error ? error.message : error, INSTALL_ERROR_LOCAL));
M
Matt Bierner 已提交
668 669
					return TPromise.wrapError(error);
				});
S
Sandeep Somavarapu 已提交
670
	}
E
Erich Gamma 已提交
671

J
Joao Moreno 已提交
672
	private preUninstallExtension(extension: ILocalExtension): TPromise<void> {
S
Sandeep Somavarapu 已提交
673
		return pfs.exists(extension.location.fsPath)
674
			.then(exists => exists ? null : TPromise.wrapError(new Error(nls.localize('notExists', "Could not find extension"))))
675
			.then(() => {
676
				this.logService.info('Uninstalling extension:', extension.identifier.id);
677 678
				this._onUninstallExtension.fire(extension.identifier);
			});
S
Sandeep Somavarapu 已提交
679 680
	}

S
Sandeep Somavarapu 已提交
681
	private uninstallExtension(local: ILocalExtension): TPromise<void> {
S
Sandeep Somavarapu 已提交
682 683 684 685 686 687 688 689 690 691
		const id = getGalleryExtensionIdFromLocal(local);
		let promise = this.uninstallingExtensions.get(id);
		if (!promise) {
			// Set all versions of the extension as uninstalled
			promise = this.scanUserExtensions(false)
				.then(userExtensions => this.setUninstalled(...userExtensions.filter(u => areSameExtensions({ id: getGalleryExtensionIdFromLocal(u), uuid: u.identifier.uuid }, { id, uuid: local.identifier.uuid }))))
				.then(() => { this.uninstallingExtensions.delete(id); });
			this.uninstallingExtensions.set(id, promise);
		}
		return promise;
S
Sandeep Somavarapu 已提交
692 693
	}

J
Joao Moreno 已提交
694
	private async postUninstallExtension(extension: ILocalExtension, error?: Error): Promise<void> {
695
		if (error) {
696
			this.logService.error('Failed to uninstall extension:', extension.identifier.id, error.message);
S
Sandeep Somavarapu 已提交
697 698
		} else {
			this.logService.info('Successfully uninstalled extension:', extension.identifier.id);
699 700 701 702
			// only report if extension has a mapped gallery extension. UUID identifies the gallery extension.
			if (extension.identifier.uuid) {
				await this.galleryService.reportStatistic(extension.manifest.publisher, extension.manifest.name, extension.manifest.version, StatisticType.Uninstall);
			}
703
		}
704 705 706
		this.reportTelemetry('extensionGallery:uninstall', getLocalExtensionTelemetryData(extension), void 0, error);
		const errorcode = error ? error instanceof ExtensionManagementError ? error.code : ERROR_UNKNOWN : void 0;
		this._onDidUninstallExtension.fire({ identifier: extension.identifier, error: errorcode });
E
Erich Gamma 已提交
707 708
	}

J
Joao Moreno 已提交
709 710 711 712
	getInstalled(type: LocalExtensionType = null): TPromise<ILocalExtension[]> {
		const promises = [];

		if (type === null || type === LocalExtensionType.System) {
713
			promises.push(this.scanSystemExtensions().then(null, e => new ExtensionManagementError(this.joinErrors(e).message, ERROR_SCANNING_SYS_EXTENSIONS)));
J
Joao Moreno 已提交
714 715 716
		}

		if (type === null || type === LocalExtensionType.User) {
717
			promises.push(this.scanUserExtensions(true).then(null, e => new ExtensionManagementError(this.joinErrors(e).message, ERROR_SCANNING_USER_EXTENSIONS)));
J
Joao Moreno 已提交
718 719
		}

720
		return TPromise.join<ILocalExtension[]>(promises).then(flatten, errors => TPromise.wrapError<ILocalExtension[]>(this.joinErrors(errors)));
J
Joao Moreno 已提交
721 722 723
	}

	private scanSystemExtensions(): TPromise<ILocalExtension[]> {
724
		this.logService.trace('Started scanning system extensions');
725 726
		return this.scanExtensions(SystemExtensionsRoot, LocalExtensionType.System)
			.then(result => {
727
				this.logService.info('Scanned system extensions:', result.length);
728 729
				return result;
			});
J
Joao Moreno 已提交
730 731
	}

S
Sandeep Somavarapu 已提交
732
	private scanUserExtensions(excludeOutdated: boolean): TPromise<ILocalExtension[]> {
733
		this.logService.trace('Started scanning user extensions');
S
Sandeep Somavarapu 已提交
734 735 736
		return TPromise.join([this.getUninstalledExtensions(), this.scanExtensions(this.extensionsPath, LocalExtensionType.User)])
			.then(([uninstalled, extensions]) => {
				extensions = extensions.filter(e => !uninstalled[e.identifier.id]);
S
Sandeep Somavarapu 已提交
737 738
				if (excludeOutdated) {
					const byExtension: ILocalExtension[][] = groupByExtension(extensions, e => ({ id: getGalleryExtensionIdFromLocal(e), uuid: e.identifier.uuid }));
S
Sandeep Somavarapu 已提交
739
					extensions = byExtension.map(p => p.sort((a, b) => semver.rcompare(a.manifest.version, b.manifest.version))[0]);
S
Sandeep Somavarapu 已提交
740
				}
S
Sandeep Somavarapu 已提交
741
				this.logService.info('Scanned user extensions:', extensions.length);
S
Sandeep Somavarapu 已提交
742 743
				return extensions;
			});
J
Joao Moreno 已提交
744 745
	}

J
Joao Moreno 已提交
746
	private scanExtensions(root: string, type: LocalExtensionType): TPromise<ILocalExtension[]> {
E
Erich Gamma 已提交
747
		const limiter = new Limiter(10);
S
Sandeep Somavarapu 已提交
748
		return pfs.readdir(root)
S
Sandeep Somavarapu 已提交
749 750
			.then(extensionsFolders => TPromise.join<ILocalExtension>(extensionsFolders.map(extensionFolder => limiter.queue(() => this.scanExtension(extensionFolder, root, type)))))
			.then(extensions => extensions.filter(e => e && e.identifier));
S
Sandeep Somavarapu 已提交
751
	}
E
Erich Gamma 已提交
752

S
Sandeep Somavarapu 已提交
753
	private scanExtension(folderName: string, root: string, type: LocalExtensionType): TPromise<ILocalExtension> {
754 755 756
		if (type === LocalExtensionType.User && folderName.indexOf('.') === 0) { // Do not consider user exension folder starting with `.`
			return TPromise.as(null);
		}
S
Sandeep Somavarapu 已提交
757 758 759 760
		const extensionPath = path.join(root, folderName);
		return pfs.readdir(extensionPath)
			.then(children => readManifest(extensionPath)
				.then<ILocalExtension>(({ manifest, metadata }) => {
J
Joao Moreno 已提交
761 762 763 764
					const readme = children.filter(child => /^readme(\.txt|\.md|)$/i.test(child))[0];
					const readmeUrl = readme ? URI.file(path.join(extensionPath, readme)).toString() : null;
					const changelog = children.filter(child => /^changelog(\.txt|\.md|)$/i.test(child))[0];
					const changelogUrl = changelog ? URI.file(path.join(extensionPath, changelog)).toString() : null;
765
					manifest.extensionDependencies = manifest.extensionDependencies ? manifest.extensionDependencies.map(dep => typeof dep === 'string' ? { id: adoptToGalleryExtensionId(dep), optional: false } : { id: adoptToGalleryExtensionId(dep.id), optional: dep.optional }) : [];
S
Sandeep Somavarapu 已提交
766
					const identifier = { id: type === LocalExtensionType.System ? folderName : getLocalExtensionIdFromManifest(manifest), uuid: metadata ? metadata.id : null };
S
Sandeep Somavarapu 已提交
767 768
					const galleryIdentifier = { id: getGalleryExtensionId(manifest.publisher, manifest.name), uuid: identifier.uuid };
					return { type, identifier, galleryIdentifier, manifest, metadata, location: URI.file(extensionPath), readmeUrl, changelogUrl };
S
Sandeep Somavarapu 已提交
769 770
				}))
			.then(null, () => null);
E
Erich Gamma 已提交
771 772
	}

J
Joao Moreno 已提交
773
	removeDeprecatedExtensions(): TPromise<any> {
S
Sandeep Somavarapu 已提交
774 775 776 777 778
		return this.removeUninstalledExtensions()
			.then(() => this.removeOutdatedExtensions());
	}

	private removeUninstalledExtensions(): TPromise<void> {
779
		return this.getUninstalledExtensions()
S
Sandeep Somavarapu 已提交
780 781
			.then(uninstalled => this.scanExtensions(this.extensionsPath, LocalExtensionType.User) // All user extensions
				.then(extensions => {
S
Sandeep Somavarapu 已提交
782
					const toRemove: ILocalExtension[] = extensions.filter(e => uninstalled[e.identifier.id]);
783
					return TPromise.join(toRemove.map(e => this.extensionLifecycle.uninstall(e).then(() => this.removeUninstalledExtension(e))));
S
Sandeep Somavarapu 已提交
784 785 786
				})
			).then(() => null);
	}
S
Sandeep Somavarapu 已提交
787

S
Sandeep Somavarapu 已提交
788 789 790 791
	private removeOutdatedExtensions(): TPromise<void> {
		return this.scanExtensions(this.extensionsPath, LocalExtensionType.User) // All user extensions
			.then(extensions => {
				const toRemove: ILocalExtension[] = [];
S
Sandeep Somavarapu 已提交
792

S
Sandeep Somavarapu 已提交
793 794 795 796
				// Outdated extensions
				const byExtension: ILocalExtension[][] = groupByExtension(extensions, e => ({ id: getGalleryExtensionIdFromLocal(e), uuid: e.identifier.uuid }));
				toRemove.push(...flatten(byExtension.map(p => p.sort((a, b) => semver.rcompare(a.manifest.version, b.manifest.version)).slice(1))));

S
Sandeep Somavarapu 已提交
797
				return TPromise.join(toRemove.map(extension => this.removeExtension(extension, 'outdated')));
S
Sandeep Somavarapu 已提交
798 799 800 801
			}).then(() => null);
	}

	private removeUninstalledExtension(extension: ILocalExtension): TPromise<void> {
802
		return this.removeExtension(extension, 'uninstalled')
S
Sandeep Somavarapu 已提交
803 804 805 806
			.then(() => this.withUninstalledExtensions(uninstalled => delete uninstalled[extension.identifier.id]))
			.then(() => null);
	}

S
Sandeep Somavarapu 已提交
807
	private removeExtension(extension: ILocalExtension, type: string): TPromise<void> {
808
		this.logService.trace(`Deleting ${type} extension from disk`, extension.identifier.id);
S
Sandeep Somavarapu 已提交
809
		return pfs.rimraf(extension.location.fsPath).then(() => this.logService.info('Deleted from disk', extension.identifier.id));
J
Joao Moreno 已提交
810 811
	}

812 813
	private isUninstalled(id: string): TPromise<boolean> {
		return this.filterUninstalled(id).then(uninstalled => uninstalled.length === 1);
814 815
	}

816 817 818
	private filterUninstalled(...ids: string[]): TPromise<string[]> {
		return this.withUninstalledExtensions(allUninstalled => {
			const uninstalled = [];
819
			for (const id of ids) {
820 821
				if (!!allUninstalled[id]) {
					uninstalled.push(id);
822 823
				}
			}
824
			return uninstalled;
825
		});
826 827
	}

S
Sandeep Somavarapu 已提交
828 829
	private setUninstalled(...extensions: ILocalExtension[]): TPromise<void> {
		const ids = extensions.map(e => e.identifier.id);
S
Sandeep Somavarapu 已提交
830
		return this.withUninstalledExtensions(uninstalled => assign(uninstalled, ids.reduce((result, id) => { result[id] = true; return result; }, {})));
831 832
	}

833 834
	private unsetUninstalled(id: string): TPromise<void> {
		return this.withUninstalledExtensions<void>(uninstalled => delete uninstalled[id]);
835 836
	}

837 838
	private getUninstalledExtensions(): TPromise<{ [id: string]: boolean; }> {
		return this.withUninstalledExtensions(uninstalled => uninstalled);
839 840
	}

841 842
	private withUninstalledExtensions<T>(fn: (uninstalled: { [id: string]: boolean; }) => T): TPromise<T> {
		return this.uninstalledFileLimiter.queue(() => {
843
			let result: T = null;
844
			return pfs.readFile(this.uninstalledPath, 'utf8')
R
Ron Buckton 已提交
845
				.then(null, err => err.code === 'ENOENT' ? TPromise.as('{}') : TPromise.wrapError(err))
J
Johannes Rieken 已提交
846
				.then<{ [id: string]: boolean }>(raw => { try { return JSON.parse(raw); } catch (e) { return {}; } })
847 848 849 850
				.then(uninstalled => { result = fn(uninstalled); return uninstalled; })
				.then(uninstalled => {
					if (Object.keys(uninstalled).length === 0) {
						return pfs.rimraf(this.uninstalledPath);
851
					} else {
852 853
						const raw = JSON.stringify(uninstalled);
						return pfs.writeFile(this.uninstalledPath, raw);
854 855 856 857 858
					}
				})
				.then(() => result);
		});
	}
859

860 861
	getExtensionsReport(): TPromise<IReportedExtension[]> {
		const now = new Date().getTime();
J
Joao Moreno 已提交
862

863 864 865 866
		if (!this.reportedExtensions || now - this.lastReportTimestamp > 1000 * 60 * 5) { // 5 minute cache freshness
			this.reportedExtensions = this.updateReportCache();
			this.lastReportTimestamp = now;
		}
J
Joao Moreno 已提交
867

868
		return this.reportedExtensions;
J
Joao Moreno 已提交
869 870
	}

871
	private updateReportCache(): TPromise<IReportedExtension[]> {
J
Joao Moreno 已提交
872 873
		this.logService.trace('ExtensionManagementService.refreshReportedCache');

874
		return this.galleryService.getExtensionsReport()
J
Joao Moreno 已提交
875 876
			.then(result => {
				this.logService.trace(`ExtensionManagementService.refreshReportedCache - got ${result.length} reported extensions from service`);
877 878 879 880
				return result;
			}, err => {
				this.logService.trace('ExtensionManagementService.refreshReportedCache - failed to get extension report');
				return [];
J
Joao Moreno 已提交
881 882
			});
	}
883

S
Sandeep Somavarapu 已提交
884 885 886 887
	private toNonCancellablePromise<T>(promise: TPromise<T>): TPromise<T> {
		return new TPromise((c, e) => promise.then(result => c(result), error => e(error)), () => this.logService.debug('Request Cancelled'));
	}

888 889 890 891 892 893 894
	private reportTelemetry(eventName: string, extensionData: any, duration: number, error?: Error): void {
		const errorcode = error ? error instanceof ExtensionManagementError ? error.code : ERROR_UNKNOWN : void 0;
		/* __GDPR__
			"extensionGallery:install" : {
				"success": { "classification": "SystemMetaData", "purpose": "PerformanceAndHealth", "isMeasurement": true },
				"duration" : { "classification": "SystemMetaData", "purpose": "PerformanceAndHealth", "isMeasurement": true },
				"errorcode": { "classification": "CallstackOrException", "purpose": "PerformanceAndHealth" },
895
				"recommendationReason": { "retiredFromVersion": "1.23.0", "classification": "SystemMetaData", "purpose": "FeatureInsight", "isMeasurement": true },
896 897 898 899 900 901 902 903 904 905 906 907 908 909 910
				"${include}": [
					"${GalleryExtensionTelemetryData}"
				]
			}
		*/
		/* __GDPR__
			"extensionGallery:uninstall" : {
				"success": { "classification": "SystemMetaData", "purpose": "PerformanceAndHealth", "isMeasurement": true },
				"duration" : { "classification": "SystemMetaData", "purpose": "PerformanceAndHealth", "isMeasurement": true },
				"errorcode": { "classification": "CallstackOrException", "purpose": "PerformanceAndHealth" },
				"${include}": [
					"${GalleryExtensionTelemetryData}"
				]
			}
		*/
911 912 913 914 915 916 917 918 919 920
		/* __GDPR__
			"extensionGallery:update" : {
				"success": { "classification": "SystemMetaData", "purpose": "PerformanceAndHealth", "isMeasurement": true },
				"duration" : { "classification": "SystemMetaData", "purpose": "PerformanceAndHealth", "isMeasurement": true },
				"errorcode": { "classification": "CallstackOrException", "purpose": "PerformanceAndHealth" },
				"${include}": [
					"${GalleryExtensionTelemetryData}"
				]
			}
		*/
921 922
		this.telemetryService.publicLog(eventName, assign(extensionData, { success: !error, duration, errorcode }));
	}
E
Erich Gamma 已提交
923
}
S
Sandeep Somavarapu 已提交
924 925 926 927 928 929 930

export function getLocalExtensionIdFromGallery(extension: IGalleryExtension, version: string): string {
	return getLocalExtensionId(extension.identifier.id, version);
}

export function getLocalExtensionIdFromManifest(manifest: IExtensionManifest): string {
	return getLocalExtensionId(getGalleryExtensionId(manifest.publisher, manifest.name), manifest.version);
931
}