Book a Demo!
CoCalc Logo Icon
StoreFeaturesDocsShareSupportNewsAboutPoliciesSign UpSign In
microsoft
GitHub Repository: microsoft/vscode
Path: blob/main/src/vs/code/electron-utility/sharedProcess/contrib/languagePackCachedDataCleaner.ts
3296 views
1
/*---------------------------------------------------------------------------------------------
2
* Copyright (c) Microsoft Corporation. All rights reserved.
3
* Licensed under the MIT License. See License.txt in the project root for license information.
4
*--------------------------------------------------------------------------------------------*/
5
6
import { promises } from 'fs';
7
import { RunOnceScheduler } from '../../../../base/common/async.js';
8
import { IStringDictionary } from '../../../../base/common/collections.js';
9
import { onUnexpectedError } from '../../../../base/common/errors.js';
10
import { Disposable } from '../../../../base/common/lifecycle.js';
11
import { join } from '../../../../base/common/path.js';
12
import { Promises } from '../../../../base/node/pfs.js';
13
import { INativeEnvironmentService } from '../../../../platform/environment/common/environment.js';
14
import { ILogService } from '../../../../platform/log/common/log.js';
15
import { IProductService } from '../../../../platform/product/common/productService.js';
16
17
interface IExtensionEntry {
18
version: string;
19
extensionIdentifier: {
20
id: string;
21
uuid: string;
22
};
23
}
24
25
interface ILanguagePackEntry {
26
hash: string;
27
extensions: IExtensionEntry[];
28
}
29
30
interface ILanguagePackFile {
31
[locale: string]: ILanguagePackEntry;
32
}
33
34
export class LanguagePackCachedDataCleaner extends Disposable {
35
36
private readonly dataMaxAge: number;
37
38
constructor(
39
@INativeEnvironmentService private readonly environmentService: INativeEnvironmentService,
40
@ILogService private readonly logService: ILogService,
41
@IProductService productService: IProductService
42
) {
43
super();
44
45
this.dataMaxAge = productService.quality !== 'stable'
46
? 1000 * 60 * 60 * 24 * 7 // roughly 1 week (insiders)
47
: 1000 * 60 * 60 * 24 * 30 * 3; // roughly 3 months (stable)
48
49
// We have no Language pack support for dev version (run from source)
50
// So only cleanup when we have a build version.
51
if (this.environmentService.isBuilt) {
52
const scheduler = this._register(new RunOnceScheduler(() => {
53
this.cleanUpLanguagePackCache();
54
}, 40 * 1000 /* after 40s */));
55
scheduler.schedule();
56
}
57
}
58
59
private async cleanUpLanguagePackCache(): Promise<void> {
60
this.logService.trace('[language pack cache cleanup]: Starting to clean up unused language packs.');
61
62
try {
63
const installed: IStringDictionary<boolean> = Object.create(null);
64
const metaData: ILanguagePackFile = JSON.parse(await promises.readFile(join(this.environmentService.userDataPath, 'languagepacks.json'), 'utf8'));
65
for (const locale of Object.keys(metaData)) {
66
const entry = metaData[locale];
67
installed[`${entry.hash}.${locale}`] = true;
68
}
69
70
// Cleanup entries for language packs that aren't installed anymore
71
const cacheDir = join(this.environmentService.userDataPath, 'clp');
72
const cacheDirExists = await Promises.exists(cacheDir);
73
if (!cacheDirExists) {
74
return;
75
}
76
77
const entries = await Promises.readdir(cacheDir);
78
for (const entry of entries) {
79
if (installed[entry]) {
80
this.logService.trace(`[language pack cache cleanup]: Skipping folder ${entry}. Language pack still in use.`);
81
continue;
82
}
83
84
this.logService.trace(`[language pack cache cleanup]: Removing unused language pack: ${entry}`);
85
86
await Promises.rm(join(cacheDir, entry));
87
}
88
89
const now = Date.now();
90
for (const packEntry of Object.keys(installed)) {
91
const folder = join(cacheDir, packEntry);
92
const entries = await Promises.readdir(folder);
93
for (const entry of entries) {
94
if (entry === 'tcf.json') {
95
continue;
96
}
97
98
const candidate = join(folder, entry);
99
const stat = await promises.stat(candidate);
100
if (stat.isDirectory() && (now - stat.mtime.getTime()) > this.dataMaxAge) {
101
this.logService.trace(`[language pack cache cleanup]: Removing language pack cache folder: ${join(packEntry, entry)}`);
102
103
await Promises.rm(candidate);
104
}
105
}
106
}
107
} catch (error) {
108
onUnexpectedError(error);
109
}
110
}
111
}
112
113