extensionManagementService.ts 38.8 KB
Newer Older
E
Erich Gamma 已提交
1 2 3 4 5 6 7 8 9 10
/*---------------------------------------------------------------------------------------------
 *  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 nls = require('vs/nls');
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';
S
Sandeep Somavarapu 已提交
13
import { IDisposable, dispose, toDisposable } from 'vs/base/common/lifecycle';
S
Sandeep Somavarapu 已提交
14
import { flatten, distinct, coalesce } from 'vs/base/common/arrays';
E
Erich Gamma 已提交
15
import { extract, buffer } 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 23
	IExtensionIdentifier,
	IReportedExtension
J
Joao Moreno 已提交
24
} from 'vs/platform/extensionManagement/common/extensionManagement';
S
Sandeep Somavarapu 已提交
25
import { getGalleryExtensionIdFromLocal, adoptToGalleryExtensionId, areSameExtensions, getGalleryExtensionId, groupByExtension, getMaliciousExtensionsSet, getLocalExtensionId } from 'vs/platform/extensionManagement/common/extensionManagementUtil';
26
import { localizeManifest } from '../common/extensionNls';
J
Joao Moreno 已提交
27
import { IEnvironmentService } from 'vs/platform/environment/common/environment';
E
Erich Gamma 已提交
28
import { Limiter } from 'vs/base/common/async';
J
Joao Moreno 已提交
29
import Event, { Emitter } from 'vs/base/common/event';
J
Joao Moreno 已提交
30
import * as semver from 'semver';
J
João Moreno 已提交
31
import URI from 'vs/base/common/uri';
32
import { IChoiceService, Severity } from 'vs/platform/message/common/message';
S
Sandeep Somavarapu 已提交
33
import pkg from 'vs/platform/node/package';
S
Sandeep Somavarapu 已提交
34
import { isMacintosh } from 'vs/base/common/platform';
A
Alex Dima 已提交
35
import { MANIFEST_CACHE_FOLDER, USER_MANIFEST_CACHE_FILE } from 'vs/platform/extensions/common/extensions';
36
import { ILogService } from 'vs/platform/log/common/log';
E
Erich Gamma 已提交
37

J
Joao Moreno 已提交
38
const SystemExtensionsRoot = path.normalize(path.join(URI.parse(require.toUrl('')).fsPath, '..', 'extensions'));
39 40
const ERROR_SCANNING_SYS_EXTENSIONS = 'scanningSystem';
const ERROR_SCANNING_USER_EXTENSIONS = 'scanningUser';
41
const INSTALL_ERROR_UNSET_UNINSTALLED = 'unsetUninstalled';
S
Sandeep Somavarapu 已提交
42 43 44
const INSTALL_ERROR_INCOMPATIBLE = 'incompatible';
const INSTALL_ERROR_DOWNLOADING = 'downloading';
const INSTALL_ERROR_VALIDATING = 'validating';
45 46
const INSTALL_ERROR_GALLERY = 'gallery';
const INSTALL_ERROR_LOCAL = 'local';
47
const INSTALL_ERROR_EXTRACTING = 'extracting';
48
const INSTALL_ERROR_DELETING = 'deleting';
49 50
const INSTALL_ERROR_READING_EXTENSION_FROM_DISK = 'readingExtension';
const INSTALL_ERROR_SAVING_METADATA = 'savingMetadata';
S
Sandeep Somavarapu 已提交
51 52
const INSTALL_ERROR_UNKNOWN = 'unknown';

53
export class ExtensionManagementError extends Error {
S
Sandeep Somavarapu 已提交
54 55 56 57
	constructor(message: string, readonly code: string) {
		super(message);
	}
}
J
Joao Moreno 已提交
58

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

72
export function validateLocalExtension(zipPath: string): TPromise<IExtensionManifest> {
E
Erich Gamma 已提交
73 74
	return buffer(zipPath, 'extension/package.json')
		.then(buffer => parseManifest(buffer.toString('utf8')))
75
		.then(({ manifest }) => TPromise.as(manifest));
E
Erich Gamma 已提交
76 77
}

78 79 80 81 82
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 已提交
83
			.then(null, err => err.code !== 'ENOENT' ? TPromise.wrapError<string>(err) : '{}')
84 85 86 87 88 89 90 91 92 93 94
			.then(raw => JSON.parse(raw))
	];

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

S
Sandeep Somavarapu 已提交
95 96 97
interface InstallableExtension {
	zipPath: string;
	id: string;
S
Sandeep Somavarapu 已提交
98
	metadata?: IGalleryMetadata;
S
Sandeep Somavarapu 已提交
99 100
}

J
Joao Moreno 已提交
101
export class ExtensionManagementService implements IExtensionManagementService {
E
Erich Gamma 已提交
102

103
	_serviceBrand: any;
E
Erich Gamma 已提交
104 105

	private extensionsPath: string;
106 107
	private uninstalledPath: string;
	private uninstalledFileLimiter: Limiter<void>;
108
	private reportedExtensions: TPromise<IReportedExtension[]> | undefined;
J
Joao Moreno 已提交
109
	private lastReportTimestamp = 0;
110
	private disposables: IDisposable[] = [];
E
Erich Gamma 已提交
111

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

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

S
Sandeep Somavarapu 已提交
118 119 120 121
	private readonly _onUninstallExtension = new Emitter<IExtensionIdentifier>();
	readonly onUninstallExtension: Event<IExtensionIdentifier> = this._onUninstallExtension.event;

	private readonly installingExtensions: Map<string, TPromise<ILocalExtension>> = new Map<string, TPromise<ILocalExtension>>();
E
Erich Gamma 已提交
122

S
Sandeep Somavarapu 已提交
123 124
	private _onDidUninstallExtension = new Emitter<DidUninstallExtensionEvent>();
	onDidUninstallExtension: Event<DidUninstallExtensionEvent> = this._onDidUninstallExtension.event;
E
Erich Gamma 已提交
125 126

	constructor(
J
Joao Moreno 已提交
127
		@IEnvironmentService private environmentService: IEnvironmentService,
128
		@IChoiceService private choiceService: IChoiceService,
129
		@IExtensionGalleryService private galleryService: IExtensionGalleryService,
J
Joao Moreno 已提交
130
		@ILogService private logService: ILogService
E
Erich Gamma 已提交
131
	) {
J
Joao Moreno 已提交
132
		this.extensionsPath = environmentService.extensionsPath;
133 134
		this.uninstalledPath = path.join(this.extensionsPath, '.obsolete');
		this.uninstalledFileLimiter = new Limiter(1);
S
Sandeep Somavarapu 已提交
135
		this.disposables.push(toDisposable(() => this.installingExtensions.clear()));
E
Erich Gamma 已提交
136 137
	}

A
Alex Dima 已提交
138
	private deleteExtensionsManifestCache(): void {
J
Joao Moreno 已提交
139
		const cacheFolder = path.join(this.environmentService.userDataPath, MANIFEST_CACHE_FOLDER);
A
Alex Dima 已提交
140 141 142 143 144
		const cacheFile = path.join(cacheFolder, USER_MANIFEST_CACHE_FILE);

		pfs.del(cacheFile).done(() => { }, () => { });
	}

145
	install(zipPath: string): TPromise<void> {
A
Alex Dima 已提交
146 147
		this.deleteExtensionsManifestCache();

148 149
		zipPath = path.resolve(zipPath);

S
Sandeep Somavarapu 已提交
150
		return validateLocalExtension(zipPath)
S
Sandeep Somavarapu 已提交
151
			.then(manifest => {
S
Sandeep Somavarapu 已提交
152
				const identifier = { id: getLocalExtensionIdFromManifest(manifest) };
153 154 155 156 157
				return this.unsetUninstalledAndRemove(identifier.id)
					.then(
					() => this.checkOutdated(manifest)
						.then(validated => {
							if (validated) {
158
								this.logService.info('Installing the extension:', identifier.id);
159 160 161 162
								this._onInstallExtension.fire({ identifier, zipPath });
								return this.getMetadata(getGalleryExtensionId(manifest.publisher, manifest.name))
									.then(
									metadata => this.installFromZipPath(identifier, zipPath, metadata, manifest),
163
									error => this.installFromZipPath(identifier, zipPath, null, manifest))
S
Sandeep Somavarapu 已提交
164 165 166 167 168 169
									.then(
									() => this.logService.info('Successfully installed the extension:', identifier.id),
									e => {
										this.logService.error('Failed to install the extension:', identifier.id, e.message);
										return TPromise.wrapError(e);
									});
170 171 172 173 174 175 176 177 178 179 180
							}
							return null;
						}),
					e => TPromise.wrapError(new Error(nls.localize('restartCode', "Please restart Code before reinstalling {0}.", manifest.displayName || manifest.name))));
			});
	}

	private unsetUninstalledAndRemove(id: string): TPromise<void> {
		return this.isUninstalled(id)
			.then(isUninstalled => {
				if (isUninstalled) {
181
					this.logService.trace('Removing the extension:', id);
182 183
					const extensionPath = path.join(this.extensionsPath, id);
					return pfs.rimraf(extensionPath)
184
						.then(() => this.unsetUninstalled(id))
185
						.then(() => this.logService.info('Removed the extension:', id));
186 187
				}
				return null;
S
Sandeep Somavarapu 已提交
188 189
			});
	}
190

S
Sandeep Somavarapu 已提交
191 192
	private checkOutdated(manifest: IExtensionManifest): TPromise<boolean> {
		const extensionIdentifier = { id: getGalleryExtensionId(manifest.publisher, manifest.name) };
193
		return this.getInstalled(LocalExtensionType.User)
S
Sandeep Somavarapu 已提交
194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210
			.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?");
					const options = [
						nls.localize('override', "Override"),
						nls.localize('cancel', "Cancel")
					];
					return this.choiceService.choose(Severity.Info, message, options, 1, true)
						.then<boolean>(value => {
							if (value === 0) {
								return this.uninstall(newer, true).then(() => true);
							}
							return TPromise.wrapError(errors.canceled());
						});
				}
				return true;
211
			});
S
Sandeep Somavarapu 已提交
212 213 214 215 216 217 218 219 220
	}

	private installFromZipPath(identifier: IExtensionIdentifier, zipPath: string, metadata: IGalleryMetadata, manifest: IExtensionManifest): TPromise<void> {
		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 => this.downloadAndInstallExtensions(metadata ? dependenciesToInstall.filter(d => d.identifier.uuid !== metadata.id) : dependenciesToInstall))
						.then(() => local, error => {
S
Sandeep Somavarapu 已提交
221
							this.uninstallExtension(local);
S
Sandeep Somavarapu 已提交
222
							return TPromise.wrapError(new Error(nls.localize('errorInstallingDependencies', "Error while installing dependencies. {0}", error instanceof Error ? error.message : error)));
S
Sandeep Somavarapu 已提交
223 224 225 226 227 228 229 230
						});
				}
				return local;
			})
			.then(
			local => this._onDidInstallExtension.fire({ identifier, zipPath, local }),
			error => { this._onDidInstallExtension.fire({ identifier, zipPath, error }); return TPromise.wrapError(error); }
			);
E
Erich Gamma 已提交
231 232
	}

S
Sandeep Somavarapu 已提交
233
	installFromGallery(extension: IGalleryExtension): TPromise<void> {
A
Alex Dima 已提交
234 235
		this.deleteExtensionsManifestCache();

S
Sandeep Somavarapu 已提交
236 237 238
		this.onInstallExtensions([extension]);
		return this.collectExtensionsToInstall(extension)
			.then(
S
Sandeep Somavarapu 已提交
239 240 241 242 243 244
			extensionsToInstall => {
				if (extensionsToInstall.length > 1) {
					this.onInstallExtensions(extensionsToInstall.slice(1));
				}
				return this.downloadAndInstallExtensions(extensionsToInstall)
					.then(
245 246
					locals => this.onDidInstallExtensions(extensionsToInstall, locals, []),
					errors => this.onDidInstallExtensions(extensionsToInstall, [], errors));
S
Sandeep Somavarapu 已提交
247
			},
248
			error => this.onDidInstallExtensions([extension], [], [error]));
S
Sandeep Somavarapu 已提交
249 250 251 252 253 254
	}

	private collectExtensionsToInstall(extension: IGalleryExtension): TPromise<IGalleryExtension[]> {
		return this.galleryService.loadCompatibleVersion(extension)
			.then(compatible => {
				if (!compatible) {
S
Sandeep Somavarapu 已提交
255
					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 已提交
256 257 258
				}
				return this.getDependenciesToInstall(compatible.properties.dependencies)
					.then(
259
					dependenciesToInstall => ([compatible, ...dependenciesToInstall.filter(d => d.identifier.uuid !== compatible.identifier.uuid)]),
260
					error => TPromise.wrapError<IGalleryExtension[]>(new ExtensionManagementError(this.joinErrors(error).message, INSTALL_ERROR_GALLERY)));
S
Sandeep Somavarapu 已提交
261
			},
262
			error => TPromise.wrapError<IGalleryExtension[]>(new ExtensionManagementError(this.joinErrors(error).message, INSTALL_ERROR_GALLERY)));
263 264
	}

S
Sandeep Somavarapu 已提交
265
	private downloadAndInstallExtensions(extensions: IGalleryExtension[]): TPromise<ILocalExtension[]> {
S
Sandeep Somavarapu 已提交
266 267 268 269 270 271 272
		return TPromise.join(extensions.map(extensionToInstall => this.downloadAndInstallExtension(extensionToInstall)))
			.then(null, errors => this.rollback(extensions).then(() => TPromise.wrapError(errors), () => TPromise.wrapError(errors)));
	}

	private downloadAndInstallExtension(extension: IGalleryExtension): TPromise<ILocalExtension> {
		let installingExtension = this.installingExtensions.get(extension.identifier.id);
		if (!installingExtension) {
J
Joao Moreno 已提交
273
			installingExtension = this.getExtensionsReport()
J
Joao Moreno 已提交
274 275
				.then(report => {
					if (getMaliciousExtensionsSet(report).has(extension.identifier.id)) {
J
Joao Moreno 已提交
276
						throw new Error(nls.localize('malicious extension', "Can't install extension since it was reported to be problematic."));
J
Joao Moreno 已提交
277 278 279 280
					} else {
						return extension;
					}
				})
J
Joao Moreno 已提交
281
				.then(extension => this.downloadInstallableExtension(extension))
S
Sandeep Somavarapu 已提交
282 283
				.then(installableExtension => this.installExtension(installableExtension))
				.then(
J
Joao Moreno 已提交
284 285
				local => { this.installingExtensions.delete(extension.identifier.id); return local; },
				e => { this.installingExtensions.delete(extension.identifier.id); return TPromise.wrapError(e); }
S
Sandeep Somavarapu 已提交
286 287 288
				);

			this.installingExtensions.set(extension.identifier.id, installingExtension);
S
Sandeep Somavarapu 已提交
289 290
		}
		return installingExtension;
S
Sandeep Somavarapu 已提交
291 292
	}

S
Sandeep Somavarapu 已提交
293
	private downloadInstallableExtension(extension: IGalleryExtension): TPromise<InstallableExtension> {
S
Sandeep Somavarapu 已提交
294
		const metadata = <IGalleryMetadata>{
S
Sandeep Somavarapu 已提交
295
			id: extension.identifier.uuid,
S
Sandeep Somavarapu 已提交
296 297 298
			publisherId: extension.publisherId,
			publisherDisplayName: extension.publisherDisplayName,
		};
S
Sandeep Somavarapu 已提交
299 300 301 302 303

		return this.galleryService.loadCompatibleVersion(extension)
			.then(
			compatible => {
				if (compatible) {
304
					this.logService.trace('Started downloading extension:', extension.name);
S
Sandeep Somavarapu 已提交
305 306
					return this.galleryService.download(extension)
						.then(
307
						zipPath => {
308
							this.logService.info('Downloaded extension:', extension.name);
309 310
							return validateLocalExtension(zipPath)
								.then(
S
Sandeep Somavarapu 已提交
311
								manifest => (<InstallableExtension>{ zipPath, id: getLocalExtensionIdFromManifest(manifest), metadata }),
312 313 314
								error => TPromise.wrapError(new ExtensionManagementError(this.joinErrors(error).message, INSTALL_ERROR_VALIDATING))
								);
						},
315
						error => TPromise.wrapError(new ExtensionManagementError(this.joinErrors(error).message, INSTALL_ERROR_DOWNLOADING)));
S
Sandeep Somavarapu 已提交
316
				} else {
317
					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));
S
Sandeep Somavarapu 已提交
318 319
				}
			},
320
			error => TPromise.wrapError<InstallableExtension>(new ExtensionManagementError(this.joinErrors(error).message, INSTALL_ERROR_GALLERY)));
S
Sandeep Somavarapu 已提交
321 322 323 324
	}

	private rollback(extensions: IGalleryExtension[]): TPromise<void> {
		return this.filterOutUninstalled(extensions)
S
Sandeep Somavarapu 已提交
325
			.then(installed => TPromise.join(installed.map(local => this.uninstallExtension(local))))
S
Sandeep Somavarapu 已提交
326
			.then(() => null, () => null);
327 328
	}

S
Sandeep Somavarapu 已提交
329 330
	private onInstallExtensions(extensions: IGalleryExtension[]): void {
		for (const extension of extensions) {
331
			this.logService.info('Installing extension:', extension.name);
S
Sandeep Somavarapu 已提交
332
			const id = getLocalExtensionIdFromGallery(extension, extension.version);
S
Sandeep Somavarapu 已提交
333
			this._onInstallExtension.fire({ identifier: { id, uuid: extension.identifier.uuid }, gallery: extension });
334
		}
335 336
	}

337
	private onDidInstallExtensions(extensions: IGalleryExtension[], locals: ILocalExtension[], errors: Error[]): TPromise<any> {
S
Sandeep Somavarapu 已提交
338
		extensions.forEach((gallery, index) => {
S
Sandeep Somavarapu 已提交
339
			const identifier = { id: getLocalExtensionIdFromGallery(gallery, gallery.version), uuid: gallery.identifier.uuid };
340 341 342
			const local = locals[index];
			const error = errors[index];
			if (local) {
343
				this.logService.info(`Extensions installed successfully:`, gallery.identifier.id);
344
				this._onDidInstallExtension.fire({ identifier, gallery, local });
S
Sandeep Somavarapu 已提交
345
			} else {
346
				const errorCode = error && (<ExtensionManagementError>error).code ? (<ExtensionManagementError>error).code : INSTALL_ERROR_UNKNOWN;
347
				this.logService.error(`Failed to install extension:`, gallery.identifier.id, error ? error.message : errorCode);
348
				this._onDidInstallExtension.fire({ identifier, gallery, error: errorCode });
S
Sandeep Somavarapu 已提交
349 350
			}
		});
351
		return errors.length ? TPromise.wrapError(this.joinErrors(errors)) : TPromise.as(null);
352 353
	}

S
Sandeep Somavarapu 已提交
354
	private getDependenciesToInstall(dependencies: string[]): TPromise<IGalleryExtension[]> {
S
Sandeep Somavarapu 已提交
355
		if (dependencies.length) {
S
Sandeep Somavarapu 已提交
356 357 358 359 360 361 362 363 364 365 366 367
			return this.getInstalled()
				.then(installed => {
					const uninstalledDeps = dependencies.filter(d => installed.every(i => getGalleryExtensionId(i.manifest.publisher, i.manifest.name) !== d));
					if (uninstalledDeps.length) {
						return this.galleryService.loadAllDependencies(uninstalledDeps.map(id => (<IExtensionIdentifier>{ id })))
							.then(allDependencies => allDependencies.filter(d => {
								const extensionId = getLocalExtensionIdFromGallery(d, d.version);
								return installed.every(({ identifier }) => identifier.id !== extensionId);
							}));
					}
					return [];
				});
S
Sandeep Somavarapu 已提交
368 369
		}
		return TPromise.as([]);
370 371
	}

J
Joao Moreno 已提交
372 373 374
	private filterOutUninstalled(extensions: IGalleryExtension[]): TPromise<ILocalExtension[]> {
		return this.getInstalled()
			.then(installed => installed.filter(local => !!this.getGalleryExtensionForLocalExtension(extensions, local)));
375
	}
376

377
	private getGalleryExtensionForLocalExtension(galleryExtensions: IGalleryExtension[], localExtension: ILocalExtension): IGalleryExtension {
S
Sandeep Somavarapu 已提交
378
		const filtered = galleryExtensions.filter(galleryExtension => areSameExtensions(localExtension.identifier, { id: getLocalExtensionIdFromGallery(galleryExtension, galleryExtension.version), uuid: galleryExtension.identifier.uuid }));
379 380 381
		return filtered.length ? filtered[0] : null;
	}

382 383 384 385 386 387 388 389 390 391 392
	private installExtension(installableExtension: InstallableExtension): TPromise<ILocalExtension> {
		return this.unsetUninstalledAndGetLocal(installableExtension.id)
			.then(
			local => {
				if (local) {
					return local;
				}
				return this.extractAndInstall(installableExtension);
			},
			e => {
				if (isMacintosh) {
393
					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));
394
				}
395
				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));
396 397
			});
	}
J
Joao Moreno 已提交
398

399 400 401 402
	private unsetUninstalledAndGetLocal(id: string): TPromise<ILocalExtension> {
		return this.isUninstalled(id)
			.then(isUninstalled => {
				if (isUninstalled) {
403
					this.logService.trace('Removing the extension from uninstalled list:', id);
404 405
					// If the same version of extension is marked as uninstalled, remove it from there and return the local.
					return this.unsetUninstalled(id)
406
						.then(() => {
407
							this.logService.info('Removed the extension from uninstalled list:', id);
408 409
							return this.getInstalled(LocalExtensionType.User);
						})
410 411 412 413 414 415
						.then(installed => installed.filter(i => i.identifier.id === id)[0]);
				}
				return null;
			});
	}

S
Sandeep Somavarapu 已提交
416
	private extractAndInstall({ zipPath, id, metadata }: InstallableExtension): TPromise<ILocalExtension> {
417
		const extensionPath = path.join(this.extensionsPath, id);
418 419
		return pfs.rimraf(extensionPath)
			.then(() => {
420
				this.logService.trace(`Started extracting the extension from ${zipPath} to ${extensionPath}`);
421 422
				return extract(zipPath, extensionPath, { sourcePath: 'extension', overwrite: true })
					.then(
423
					() => {
424
						this.logService.info(`Extracted extension to ${extensionPath}:`, id);
425
						return TPromise.join([readManifest(extensionPath), pfs.readdir(extensionPath)])
426
							.then(null, e => TPromise.wrapError(new ExtensionManagementError(this.joinErrors(e).message, INSTALL_ERROR_READING_EXTENSION_FROM_DISK)));
427
					},
428 429
					e => TPromise.wrapError(new ExtensionManagementError(e.message, INSTALL_ERROR_EXTRACTING)))
					.then(([{ manifest }, children]) => {
J
João Moreno 已提交
430 431 432 433 434
						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;
						const type = LocalExtensionType.User;
S
Sandeep Somavarapu 已提交
435
						const identifier = { id, uuid: metadata ? metadata.id : null };
J
João Moreno 已提交
436

S
Sandeep Somavarapu 已提交
437
						const local: ILocalExtension = { type, identifier, manifest, metadata, path: extensionPath, readmeUrl, changelogUrl };
J
João Moreno 已提交
438

439
						this.logService.trace(`Updating metadata of the extension:`, id);
440
						return this.saveMetadataForLocalExtension(local)
441
							.then(() => {
442
								this.logService.info(`Updated metadata of the extension:`, id);
443
								return local;
444
							}, e => TPromise.wrapError(new ExtensionManagementError(this.joinErrors(e).message, INSTALL_ERROR_SAVING_METADATA)));
J
João Moreno 已提交
445
					});
446
			}, e => TPromise.wrapError(new ExtensionManagementError(this.joinErrors(e).message, INSTALL_ERROR_DELETING)));
E
Erich Gamma 已提交
447 448
	}

449
	uninstall(extension: ILocalExtension, force = false): TPromise<void> {
A
Alex Dima 已提交
450 451
		this.deleteExtensionsManifestCache();

S
Sandeep Somavarapu 已提交
452 453 454 455 456 457 458
		return this.getInstalled(LocalExtensionType.User)
			.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 已提交
459 460
	}

461
	updateMetadata(local: ILocalExtension, metadata: IGalleryMetadata): TPromise<ILocalExtension> {
A
Alex Dima 已提交
462 463
		this.deleteExtensionsManifestCache();

464 465 466 467 468 469 470 471 472 473 474 475 476 477 478 479
		local.metadata = metadata;
		return this.saveMetadataForLocalExtension(local);
	}

	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 已提交
480 481 482 483 484 485 486 487
	private getMetadata(extensionName: string): TPromise<IGalleryMetadata> {
		return this.galleryService.query({ names: [extensionName], pageSize: 1 })
			.then(galleryResult => {
				const galleryExtension = galleryResult.firstPage[0];
				return galleryExtension ? <IGalleryMetadata>{ id: galleryExtension.identifier.uuid, publisherDisplayName: galleryExtension.publisherDisplayName, publisherId: galleryExtension.publisherId } : null;
			});
	}

488 489
	private joinErrors(errorOrErrors: (Error | string) | ((Error | string)[])): Error {
		const errors = Array.isArray(errorOrErrors) ? errorOrErrors : [errorOrErrors];
S
Sandeep Somavarapu 已提交
490 491 492 493
		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 已提交
494
			return new Error(`${previousValue.message}${previousValue.message ? ',' : ''}${currentValue instanceof Error ? currentValue.message : currentValue}`);
S
Sandeep Somavarapu 已提交
495
		}, new Error(''));
J
Joao Moreno 已提交
496 497
	}

498
	private checkForDependenciesAndUninstall(extension: ILocalExtension, installed: ILocalExtension[], force: boolean): TPromise<void> {
J
Joao Moreno 已提交
499
		return this.preUninstallExtension(extension)
500
			.then(() => this.hasDependencies(extension, installed) ? this.promptForDependenciesAndUninstall(extension, installed, force) : this.promptAndUninstall(extension, installed, force))
501
			.then(() => this.postUninstallExtension(extension),
502
			error => {
503
				this.postUninstallExtension(extension, INSTALL_ERROR_LOCAL);
504 505
				return TPromise.wrapError(error);
			});
S
Sandeep Somavarapu 已提交
506 507
	}

508 509
	private hasDependencies(extension: ILocalExtension, installed: ILocalExtension[]): boolean {
		if (extension.manifest.extensionDependencies && extension.manifest.extensionDependencies.length) {
S
Sandeep Somavarapu 已提交
510
			return installed.some(i => extension.manifest.extensionDependencies.indexOf(getGalleryExtensionIdFromLocal(i)) !== -1);
511 512 513 514
		}
		return false;
	}

515 516 517 518 519 520
	private promptForDependenciesAndUninstall(extension: ILocalExtension, installed: ILocalExtension[], force: boolean): TPromise<void> {
		if (force) {
			const dependencies = distinct(this.getDependenciesToUninstallRecursively(extension, installed, [])).filter(e => e !== extension);
			return this.uninstallWithDependencies(extension, dependencies, installed);
		}

521
		const message = nls.localize('uninstallDependeciesConfirmation', "Would you like to uninstall '{0}' only or its dependencies also?", extension.manifest.displayName || extension.manifest.name);
S
Sandeep Somavarapu 已提交
522 523 524 525 526
		const options = [
			nls.localize('uninstallOnly', "Only"),
			nls.localize('uninstallAll', "All"),
			nls.localize('cancel', "Cancel")
		];
527
		this.logService.info('Requesting for confirmation to uninstall extension with dependencies', extension.identifier.id);
528
		return this.choiceService.choose(Severity.Info, message, options, 2, true)
S
Sandeep Somavarapu 已提交
529 530
			.then<void>(value => {
				if (value === 0) {
531
					return this.uninstallWithDependencies(extension, [], installed);
S
Sandeep Somavarapu 已提交
532 533
				}
				if (value === 1) {
534 535
					const dependencies = distinct(this.getDependenciesToUninstallRecursively(extension, installed, [])).filter(e => e !== extension);
					return this.uninstallWithDependencies(extension, dependencies, installed);
S
Sandeep Somavarapu 已提交
536
				}
537
				this.logService.info('Cancelled uninstalling extension:', extension.identifier.id);
S
Sandeep Somavarapu 已提交
538 539 540 541
				return TPromise.wrapError(errors.canceled());
			}, error => TPromise.wrapError(errors.canceled()));
	}

542 543 544 545 546
	private promptAndUninstall(extension: ILocalExtension, installed: ILocalExtension[], force: boolean): TPromise<void> {
		if (force) {
			return this.uninstallWithDependencies(extension, [], installed);
		}

S
Sandeep Somavarapu 已提交
547 548
		const message = nls.localize('uninstallConfirmation', "Are you sure you want to uninstall '{0}'?", extension.manifest.displayName || extension.manifest.name);
		const options = [
D
David Hewson 已提交
549
			nls.localize('ok', "OK"),
S
Sandeep Somavarapu 已提交
550 551
			nls.localize('cancel', "Cancel")
		];
552
		this.logService.info('Requesting for confirmation to uninstall extension', extension.identifier.id);
553
		return this.choiceService.choose(Severity.Info, message, options, 1, true)
S
Sandeep Somavarapu 已提交
554 555 556 557
			.then<void>(value => {
				if (value === 0) {
					return this.uninstallWithDependencies(extension, [], installed);
				}
558
				this.logService.info('Cancelled uninstalling extension:', extension.identifier.id);
S
Sandeep Somavarapu 已提交
559 560 561 562
				return TPromise.wrapError(errors.canceled());
			}, error => TPromise.wrapError(errors.canceled()));
	}

563 564
	private uninstallWithDependencies(extension: ILocalExtension, dependencies: ILocalExtension[], installed: ILocalExtension[]): TPromise<void> {
		const dependenciesToUninstall = this.filterDependents(extension, dependencies, installed);
565
		let dependents = this.getDependents(extension, installed).filter(dependent => extension !== dependent && dependenciesToUninstall.indexOf(dependent) === -1);
566
		if (dependents.length) {
567
			return TPromise.wrapError<void>(new Error(this.getDependentsErrorMessage(extension, dependents)));
568
		}
S
Sandeep Somavarapu 已提交
569
		return TPromise.join([this.uninstallExtension(extension), ...dependenciesToUninstall.map(d => this.doUninstall(d))]).then(() => null);
570 571
	}

S
Sandeep Somavarapu 已提交
572 573 574 575 576 577 578 579 580 581 582 583 584
	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);
	}

585 586 587 588 589 590 591 592
	private getDependenciesToUninstallRecursively(extension: ILocalExtension, installed: ILocalExtension[], checked: ILocalExtension[]): ILocalExtension[] {
		if (checked.indexOf(extension) !== -1) {
			return [];
		}
		checked.push(extension);
		if (!extension.manifest.extensionDependencies || extension.manifest.extensionDependencies.length === 0) {
			return [];
		}
S
Sandeep Somavarapu 已提交
593
		const dependenciesToUninstall = installed.filter(i => extension.manifest.extensionDependencies.indexOf(getGalleryExtensionIdFromLocal(i)) !== -1);
594 595 596 597 598 599 600 601 602 603 604 605
		const depsOfDeps = [];
		for (const dep of dependenciesToUninstall) {
			depsOfDeps.push(...this.getDependenciesToUninstallRecursively(dep, installed, checked));
		}
		return [...dependenciesToUninstall, ...depsOfDeps];
	}

	private filterDependents(extension: ILocalExtension, dependencies: ILocalExtension[], installed: ILocalExtension[]): ILocalExtension[] {
		installed = installed.filter(i => i !== extension && i.manifest.extensionDependencies && i.manifest.extensionDependencies.length > 0);
		let result = dependencies.slice(0);
		for (let i = 0; i < dependencies.length; i++) {
			const dep = dependencies[i];
606
			const dependents = this.getDependents(dep, installed).filter(e => dependencies.indexOf(e) === -1);
607 608 609
			if (dependents.length) {
				result.splice(i - (dependencies.length - result.length), 1);
			}
S
Sandeep Somavarapu 已提交
610
		}
611
		return result;
S
Sandeep Somavarapu 已提交
612 613
	}

614
	private getDependents(extension: ILocalExtension, installed: ILocalExtension[]): ILocalExtension[] {
S
Sandeep Somavarapu 已提交
615
		return installed.filter(e => e.manifest.extensionDependencies && e.manifest.extensionDependencies.indexOf(getGalleryExtensionIdFromLocal(extension)) !== -1);
616 617
	}

J
Joao Moreno 已提交
618 619
	private doUninstall(extension: ILocalExtension): TPromise<void> {
		return this.preUninstallExtension(extension)
S
Sandeep Somavarapu 已提交
620
			.then(() => this.uninstallExtension(extension))
621
			.then(() => this.postUninstallExtension(extension),
S
Sandeep Somavarapu 已提交
622
			error => {
623
				this.postUninstallExtension(extension, INSTALL_ERROR_LOCAL);
S
Sandeep Somavarapu 已提交
624 625 626
				return TPromise.wrapError(error);
			});
	}
E
Erich Gamma 已提交
627

J
Joao Moreno 已提交
628
	private preUninstallExtension(extension: ILocalExtension): TPromise<void> {
S
Sandeep Somavarapu 已提交
629
		return pfs.exists(extension.path)
630
			.then(exists => exists ? null : TPromise.wrapError(new Error(nls.localize('notExists', "Could not find extension"))))
631
			.then(() => {
632
				this.logService.info('Uninstalling extension:', extension.identifier.id);
633 634
				this._onUninstallExtension.fire(extension.identifier);
			});
S
Sandeep Somavarapu 已提交
635 636
	}

S
Sandeep Somavarapu 已提交
637 638 639 640 641
	private uninstallExtension(local: ILocalExtension): TPromise<void> {
		const identifier = { id: getGalleryExtensionIdFromLocal(local), uuid: local.identifier.uuid };
		return this.scanUserExtensions(false) // Uninstall all extensions which are same as requested
			.then(extensions => extensions.filter(i => areSameExtensions({ id: getGalleryExtensionIdFromLocal(i), uuid: i.identifier.uuid }, identifier)))
			.then(uninstalled => this.setUninstalled(...uninstalled.map(u => u.identifier.id)));
S
Sandeep Somavarapu 已提交
642 643
	}

644
	private async postUninstallExtension(extension: ILocalExtension, error?: string): TPromise<void> {
645
		if (error) {
646
			this.logService.error('Failed to uninstall extension:', extension.identifier.id, error);
S
Sandeep Somavarapu 已提交
647 648
		} else {
			this.logService.info('Successfully uninstalled extension:', extension.identifier.id);
649 650 651 652
			// 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);
			}
653
		}
S
Sandeep Somavarapu 已提交
654
		this._onDidUninstallExtension.fire({ identifier: extension.identifier, error });
E
Erich Gamma 已提交
655 656
	}

J
Joao Moreno 已提交
657 658 659 660
	getInstalled(type: LocalExtensionType = null): TPromise<ILocalExtension[]> {
		const promises = [];

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

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

668
		return TPromise.join<ILocalExtension[]>(promises).then(flatten, errors => TPromise.wrapError<ILocalExtension[]>(this.joinErrors(errors)));
J
Joao Moreno 已提交
669 670 671
	}

	private scanSystemExtensions(): TPromise<ILocalExtension[]> {
672
		this.logService.trace('Started scanning system extensions');
673 674
		return this.scanExtensions(SystemExtensionsRoot, LocalExtensionType.System)
			.then(result => {
675
				this.logService.info('Scanned system extensions:', result.length);
676 677
				return result;
			});
J
Joao Moreno 已提交
678 679
	}

S
Sandeep Somavarapu 已提交
680
	private scanUserExtensions(excludeOutdated: boolean): TPromise<ILocalExtension[]> {
681
		this.logService.trace('Started scanning user extensions');
S
Sandeep Somavarapu 已提交
682 683 684
		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 已提交
685 686
				if (excludeOutdated) {
					const byExtension: ILocalExtension[][] = groupByExtension(extensions, e => ({ id: getGalleryExtensionIdFromLocal(e), uuid: e.identifier.uuid }));
S
Sandeep Somavarapu 已提交
687
					extensions = byExtension.map(p => p.sort((a, b) => semver.rcompare(a.manifest.version, b.manifest.version))[0]);
S
Sandeep Somavarapu 已提交
688
				}
S
Sandeep Somavarapu 已提交
689
				this.logService.info('Scanned user extensions:', extensions.length);
S
Sandeep Somavarapu 已提交
690 691
				return extensions;
			});
J
Joao Moreno 已提交
692 693
	}

J
Joao Moreno 已提交
694
	private scanExtensions(root: string, type: LocalExtensionType): TPromise<ILocalExtension[]> {
E
Erich Gamma 已提交
695
		const limiter = new Limiter(10);
S
Sandeep Somavarapu 已提交
696 697 698 699
		return pfs.readdir(root)
			.then(extensionsFolders => TPromise.join(extensionsFolders.map(extensionFolder => limiter.queue(() => this.scanExtension(extensionFolder, root, type)))))
			.then(extensions => coalesce(extensions));
	}
E
Erich Gamma 已提交
700

S
Sandeep Somavarapu 已提交
701 702 703 704 705
	private scanExtension(folderName: string, root: string, type: LocalExtensionType): TPromise<ILocalExtension> {
		const extensionPath = path.join(root, folderName);
		return pfs.readdir(extensionPath)
			.then(children => readManifest(extensionPath)
				.then<ILocalExtension>(({ manifest, metadata }) => {
J
Joao Moreno 已提交
706 707 708 709
					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;
S
Sandeep Somavarapu 已提交
710 711 712 713 714 715 716
					if (manifest.extensionDependencies) {
						manifest.extensionDependencies = manifest.extensionDependencies.map(id => adoptToGalleryExtensionId(id));
					}
					const identifier = { id: type === LocalExtensionType.System ? folderName : getLocalExtensionIdFromManifest(manifest), uuid: metadata ? metadata.id : null };
					return { type, identifier, manifest, metadata, path: extensionPath, readmeUrl, changelogUrl };
				}))
			.then(null, () => null);
E
Erich Gamma 已提交
717 718
	}

J
Joao Moreno 已提交
719
	removeDeprecatedExtensions(): TPromise<any> {
720
		return this.getUninstalledExtensions()
S
Sandeep Somavarapu 已提交
721 722 723 724 725 726 727 728 729 730 731 732 733 734 735 736 737
			.then(uninstalled => this.scanExtensions(this.extensionsPath, LocalExtensionType.User) // All user extensions
				.then(extensions => {
					const toRemove: { path: string, id: string }[] = [];
					// Uninstalled extensions
					toRemove.push(...extensions.filter(e => uninstalled[e.identifier.id]).map(e => ({ path: e.path, id: e.identifier.id })));

					// 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)))
						.map(e => ({ path: e.path, id: e.identifier.id })));

					return TPromise.join(distinct(toRemove, e => e.path).map(({ path, id }) => {
						return pfs.rimraf(path)
							.then(() => this.withUninstalledExtensions(uninstalled => delete uninstalled[id]));
					}));
				})
			);
J
Joao Moreno 已提交
738 739
	}

740 741
	private isUninstalled(id: string): TPromise<boolean> {
		return this.filterUninstalled(id).then(uninstalled => uninstalled.length === 1);
742 743
	}

744 745 746
	private filterUninstalled(...ids: string[]): TPromise<string[]> {
		return this.withUninstalledExtensions(allUninstalled => {
			const uninstalled = [];
747
			for (const id of ids) {
748 749
				if (!!allUninstalled[id]) {
					uninstalled.push(id);
750 751
				}
			}
752
			return uninstalled;
753
		});
754 755
	}

S
Sandeep Somavarapu 已提交
756 757
	private setUninstalled(...ids: string[]): TPromise<void> {
		return this.withUninstalledExtensions(uninstalled => assign(uninstalled, ids.reduce((result, id) => { result[id] = true; return result; }, {})));
758 759
	}

760 761
	private unsetUninstalled(id: string): TPromise<void> {
		return this.withUninstalledExtensions<void>(uninstalled => delete uninstalled[id]);
762 763
	}

764 765
	private getUninstalledExtensions(): TPromise<{ [id: string]: boolean; }> {
		return this.withUninstalledExtensions(uninstalled => uninstalled);
766 767
	}

768 769
	private withUninstalledExtensions<T>(fn: (uninstalled: { [id: string]: boolean; }) => T): TPromise<T> {
		return this.uninstalledFileLimiter.queue(() => {
770
			let result: T = null;
771
			return pfs.readFile(this.uninstalledPath, 'utf8')
R
Ron Buckton 已提交
772
				.then(null, err => err.code === 'ENOENT' ? TPromise.as('{}') : TPromise.wrapError(err))
J
Johannes Rieken 已提交
773
				.then<{ [id: string]: boolean }>(raw => { try { return JSON.parse(raw); } catch (e) { return {}; } })
774 775 776 777
				.then(uninstalled => { result = fn(uninstalled); return uninstalled; })
				.then(uninstalled => {
					if (Object.keys(uninstalled).length === 0) {
						return pfs.rimraf(this.uninstalledPath);
778
					} else {
779 780
						const raw = JSON.stringify(uninstalled);
						return pfs.writeFile(this.uninstalledPath, raw);
781 782 783 784 785
					}
				})
				.then(() => result);
		});
	}
786

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

790 791 792 793
		if (!this.reportedExtensions || now - this.lastReportTimestamp > 1000 * 60 * 5) { // 5 minute cache freshness
			this.reportedExtensions = this.updateReportCache();
			this.lastReportTimestamp = now;
		}
J
Joao Moreno 已提交
794

795
		return this.reportedExtensions;
J
Joao Moreno 已提交
796 797
	}

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

801
		return this.galleryService.getExtensionsReport()
J
Joao Moreno 已提交
802 803
			.then(result => {
				this.logService.trace(`ExtensionManagementService.refreshReportedCache - got ${result.length} reported extensions from service`);
804 805 806 807
				return result;
			}, err => {
				this.logService.trace('ExtensionManagementService.refreshReportedCache - failed to get extension report');
				return [];
J
Joao Moreno 已提交
808 809 810
			});
	}

811 812 813
	dispose() {
		this.disposables = dispose(this.disposables);
	}
E
Erich Gamma 已提交
814
}
S
Sandeep Somavarapu 已提交
815 816 817 818 819 820 821

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);
S
Sandeep Somavarapu 已提交
822
}