Book a Demo!
CoCalc Logo Icon
StoreFeaturesDocsShareSupportNewsAboutPoliciesSign UpSign In
microsoft
GitHub Repository: microsoft/vscode
Path: blob/main/build/gulpfile.vscode.js
3520 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
'use strict';
7
8
const gulp = require('gulp');
9
const fs = require('fs');
10
const path = require('path');
11
const es = require('event-stream');
12
const vfs = require('vinyl-fs');
13
const rename = require('gulp-rename');
14
const replace = require('gulp-replace');
15
const filter = require('gulp-filter');
16
const util = require('./lib/util');
17
const { getVersion } = require('./lib/getVersion');
18
const { readISODate } = require('./lib/date');
19
const task = require('./lib/task');
20
const buildfile = require('./buildfile');
21
const optimize = require('./lib/optimize');
22
const { inlineMeta } = require('./lib/inlineMeta');
23
const root = path.dirname(__dirname);
24
const commit = getVersion(root);
25
const packageJson = require('../package.json');
26
const product = require('../product.json');
27
const crypto = require('crypto');
28
const i18n = require('./lib/i18n');
29
const { getProductionDependencies } = require('./lib/dependencies');
30
const { config } = require('./lib/electron');
31
const createAsar = require('./lib/asar').createAsar;
32
const minimist = require('minimist');
33
const { compileBuildWithoutManglingTask, compileBuildWithManglingTask } = require('./gulpfile.compile');
34
const { compileNonNativeExtensionsBuildTask, compileNativeExtensionsBuildTask, compileAllExtensionsBuildTask, compileExtensionMediaBuildTask, cleanExtensionsBuildTask } = require('./gulpfile.extensions');
35
const { promisify } = require('util');
36
const glob = promisify(require('glob'));
37
const rcedit = promisify(require('rcedit'));
38
39
// Build
40
const vscodeEntryPoints = [
41
buildfile.workerEditor,
42
buildfile.workerExtensionHost,
43
buildfile.workerNotebook,
44
buildfile.workerLanguageDetection,
45
buildfile.workerLocalFileSearch,
46
buildfile.workerProfileAnalysis,
47
buildfile.workerOutputLinks,
48
buildfile.workerBackgroundTokenization,
49
buildfile.workbenchDesktop,
50
buildfile.code
51
].flat();
52
53
const vscodeResourceIncludes = [
54
55
// NLS
56
'out-build/nls.messages.json',
57
'out-build/nls.keys.json',
58
59
// Workbench
60
'out-build/vs/code/electron-browser/workbench/workbench.html',
61
62
// Electron Preload
63
'out-build/vs/base/parts/sandbox/electron-browser/preload.js',
64
'out-build/vs/base/parts/sandbox/electron-browser/preload-aux.js',
65
66
// Node Scripts
67
'out-build/vs/base/node/{terminateProcess.sh,cpuUsage.sh,ps.sh}',
68
69
// Touchbar
70
'out-build/vs/workbench/browser/parts/editor/media/*.png',
71
'out-build/vs/workbench/contrib/debug/browser/media/*.png',
72
73
// External Terminal
74
'out-build/vs/workbench/contrib/externalTerminal/**/*.scpt',
75
76
// Terminal shell integration
77
'out-build/vs/workbench/contrib/terminal/common/scripts/*.fish',
78
'out-build/vs/workbench/contrib/terminal/common/scripts/*.ps1',
79
'out-build/vs/workbench/contrib/terminal/common/scripts/*.psm1',
80
'out-build/vs/workbench/contrib/terminal/common/scripts/*.sh',
81
'out-build/vs/workbench/contrib/terminal/common/scripts/*.zsh',
82
83
// Accessibility Signals
84
'out-build/vs/platform/accessibilitySignal/browser/media/*.mp3',
85
86
// Welcome
87
'out-build/vs/workbench/contrib/welcomeGettingStarted/common/media/**/*.{svg,png}',
88
89
// Extensions
90
'out-build/vs/workbench/contrib/extensions/browser/media/{theme-icon.png,language-icon.svg}',
91
'out-build/vs/workbench/services/extensionManagement/common/media/*.{svg,png}',
92
93
// Webview
94
'out-build/vs/workbench/contrib/webview/browser/pre/*.{js,html}',
95
96
// Extension Host Worker
97
'out-build/vs/workbench/services/extensions/worker/webWorkerExtensionHostIframe.html',
98
99
// Tree Sitter highlights
100
'out-build/vs/editor/common/languages/highlights/*.scm',
101
102
// Tree Sitter injection queries
103
'out-build/vs/editor/common/languages/injections/*.scm'
104
];
105
106
const vscodeResources = [
107
108
// Includes
109
...vscodeResourceIncludes,
110
111
// Excludes
112
'!out-build/vs/code/browser/**',
113
'!out-build/vs/editor/standalone/**',
114
'!out-build/vs/code/**/*-dev.html',
115
'!out-build/vs/workbench/contrib/issue/**/*-dev.html',
116
'!**/test/**'
117
];
118
119
const bootstrapEntryPoints = [
120
'out-build/main.js',
121
'out-build/cli.js',
122
'out-build/bootstrap-fork.js'
123
];
124
125
const bundleVSCodeTask = task.define('bundle-vscode', task.series(
126
util.rimraf('out-vscode'),
127
// Optimize: bundles source files automatically based on
128
// import statements based on the passed in entry points.
129
// In addition, concat window related bootstrap files into
130
// a single file.
131
optimize.bundleTask(
132
{
133
out: 'out-vscode',
134
esm: {
135
src: 'out-build',
136
entryPoints: [
137
...vscodeEntryPoints,
138
...bootstrapEntryPoints
139
],
140
resources: vscodeResources,
141
skipTSBoilerplateRemoval: entryPoint => entryPoint === 'vs/code/electron-browser/workbench/workbench'
142
}
143
}
144
)
145
));
146
gulp.task(bundleVSCodeTask);
147
148
const sourceMappingURLBase = `https://main.vscode-cdn.net/sourcemaps/${commit}`;
149
const minifyVSCodeTask = task.define('minify-vscode', task.series(
150
bundleVSCodeTask,
151
util.rimraf('out-vscode-min'),
152
optimize.minifyTask('out-vscode', `${sourceMappingURLBase}/core`)
153
));
154
gulp.task(minifyVSCodeTask);
155
156
const coreCI = task.define('core-ci', task.series(
157
gulp.task('compile-build-with-mangling'),
158
task.parallel(
159
gulp.task('minify-vscode'),
160
gulp.task('minify-vscode-reh'),
161
gulp.task('minify-vscode-reh-web'),
162
)
163
));
164
gulp.task(coreCI);
165
166
const coreCIPR = task.define('core-ci-pr', task.series(
167
gulp.task('compile-build-without-mangling'),
168
task.parallel(
169
gulp.task('minify-vscode'),
170
gulp.task('minify-vscode-reh'),
171
gulp.task('minify-vscode-reh-web'),
172
)
173
));
174
gulp.task(coreCIPR);
175
176
/**
177
* Compute checksums for some files.
178
*
179
* @param {string} out The out folder to read the file from.
180
* @param {string[]} filenames The paths to compute a checksum for.
181
* @return {Object} A map of paths to checksums.
182
*/
183
function computeChecksums(out, filenames) {
184
const result = {};
185
filenames.forEach(function (filename) {
186
const fullPath = path.join(process.cwd(), out, filename);
187
result[filename] = computeChecksum(fullPath);
188
});
189
return result;
190
}
191
192
/**
193
* Compute checksum for a file.
194
*
195
* @param {string} filename The absolute path to a filename.
196
* @return {string} The checksum for `filename`.
197
*/
198
function computeChecksum(filename) {
199
const contents = fs.readFileSync(filename);
200
201
const hash = crypto
202
.createHash('sha256')
203
.update(contents)
204
.digest('base64')
205
.replace(/=+$/, '');
206
207
return hash;
208
}
209
210
function packageTask(platform, arch, sourceFolderName, destinationFolderName, opts) {
211
opts = opts || {};
212
213
const destination = path.join(path.dirname(root), destinationFolderName);
214
platform = platform || process.platform;
215
216
const task = () => {
217
const electron = require('@vscode/gulp-electron');
218
const json = require('gulp-json-editor');
219
220
const out = sourceFolderName;
221
222
const checksums = computeChecksums(out, [
223
'vs/base/parts/sandbox/electron-browser/preload.js',
224
'vs/workbench/workbench.desktop.main.js',
225
'vs/workbench/workbench.desktop.main.css',
226
'vs/workbench/api/node/extensionHostProcess.js',
227
'vs/code/electron-browser/workbench/workbench.html',
228
'vs/code/electron-browser/workbench/workbench.js'
229
]);
230
231
const src = gulp.src(out + '/**', { base: '.' })
232
.pipe(rename(function (path) { path.dirname = path.dirname.replace(new RegExp('^' + out), 'out'); }))
233
.pipe(util.setExecutableBit(['**/*.sh']));
234
235
const platformSpecificBuiltInExtensionsExclusions = product.builtInExtensions.filter(ext => {
236
if (!ext.platforms) {
237
return false;
238
}
239
240
const set = new Set(ext.platforms);
241
return !set.has(platform);
242
}).map(ext => `!.build/extensions/${ext.name}/**`);
243
244
const extensions = gulp.src(['.build/extensions/**', ...platformSpecificBuiltInExtensionsExclusions], { base: '.build', dot: true });
245
246
const sources = es.merge(src, extensions)
247
.pipe(filter(['**', '!**/*.{js,css}.map'], { dot: true }));
248
249
let version = packageJson.version;
250
const quality = product.quality;
251
252
if (quality && quality !== 'stable') {
253
version += '-' + quality;
254
}
255
256
const name = product.nameShort;
257
const packageJsonUpdates = { name, version };
258
259
if (platform === 'linux') {
260
packageJsonUpdates.desktopName = `${product.applicationName}.desktop`;
261
}
262
263
let packageJsonContents;
264
const packageJsonStream = gulp.src(['package.json'], { base: '.' })
265
.pipe(json(packageJsonUpdates))
266
.pipe(es.through(function (file) {
267
packageJsonContents = file.contents.toString();
268
this.emit('data', file);
269
}));
270
271
let productJsonContents;
272
const productJsonStream = gulp.src(['product.json'], { base: '.' })
273
.pipe(json({ commit, date: readISODate('out-build'), checksums, version }))
274
.pipe(es.through(function (file) {
275
productJsonContents = file.contents.toString();
276
this.emit('data', file);
277
}));
278
279
const license = gulp.src([product.licenseFileName, 'ThirdPartyNotices.txt', 'licenses/**'], { base: '.', allowEmpty: true });
280
281
// TODO the API should be copied to `out` during compile, not here
282
const api = gulp.src('src/vscode-dts/vscode.d.ts').pipe(rename('out/vscode-dts/vscode.d.ts'));
283
284
const telemetry = gulp.src('.build/telemetry/**', { base: '.build/telemetry', dot: true });
285
286
const jsFilter = util.filter(data => !data.isDirectory() && /\.js$/.test(data.path));
287
const root = path.resolve(path.join(__dirname, '..'));
288
const productionDependencies = getProductionDependencies(root);
289
const dependenciesSrc = productionDependencies.map(d => path.relative(root, d)).map(d => [`${d}/**`, `!${d}/**/{test,tests}/**`]).flat().concat('!**/*.mk');
290
291
const deps = gulp.src(dependenciesSrc, { base: '.', dot: true })
292
.pipe(filter(['**', `!**/${config.version}/**`, '!**/bin/darwin-arm64-87/**', '!**/package-lock.json', '!**/yarn.lock', '!**/*.{js,css}.map']))
293
.pipe(util.cleanNodeModules(path.join(__dirname, '.moduleignore')))
294
.pipe(util.cleanNodeModules(path.join(__dirname, `.moduleignore.${process.platform}`)))
295
.pipe(jsFilter)
296
.pipe(util.rewriteSourceMappingURL(sourceMappingURLBase))
297
.pipe(jsFilter.restore)
298
.pipe(createAsar(path.join(process.cwd(), 'node_modules'), [
299
'**/*.node',
300
'**/@vscode/ripgrep/bin/*',
301
'**/node-pty/build/Release/*',
302
'**/node-pty/build/Release/conpty/*',
303
'**/node-pty/lib/worker/conoutSocketWorker.js',
304
'**/node-pty/lib/shared/conout.js',
305
'**/*.wasm',
306
'**/@vscode/vsce-sign/bin/*',
307
], [
308
'**/*.mk',
309
'!node_modules/vsda/**' // stay compatible with extensions that depend on us shipping `vsda` into ASAR
310
], [
311
'node_modules/vsda/**' // retain copy of `vsda` in node_modules for internal use
312
], 'node_modules.asar'));
313
314
let all = es.merge(
315
packageJsonStream,
316
productJsonStream,
317
license,
318
api,
319
telemetry,
320
sources,
321
deps
322
);
323
324
if (platform === 'win32') {
325
all = es.merge(all, gulp.src([
326
'resources/win32/bower.ico',
327
'resources/win32/c.ico',
328
'resources/win32/config.ico',
329
'resources/win32/cpp.ico',
330
'resources/win32/csharp.ico',
331
'resources/win32/css.ico',
332
'resources/win32/default.ico',
333
'resources/win32/go.ico',
334
'resources/win32/html.ico',
335
'resources/win32/jade.ico',
336
'resources/win32/java.ico',
337
'resources/win32/javascript.ico',
338
'resources/win32/json.ico',
339
'resources/win32/less.ico',
340
'resources/win32/markdown.ico',
341
'resources/win32/php.ico',
342
'resources/win32/powershell.ico',
343
'resources/win32/python.ico',
344
'resources/win32/react.ico',
345
'resources/win32/ruby.ico',
346
'resources/win32/sass.ico',
347
'resources/win32/shell.ico',
348
'resources/win32/sql.ico',
349
'resources/win32/typescript.ico',
350
'resources/win32/vue.ico',
351
'resources/win32/xml.ico',
352
'resources/win32/yaml.ico',
353
'resources/win32/code_70x70.png',
354
'resources/win32/code_150x150.png'
355
], { base: '.' }));
356
} else if (platform === 'linux') {
357
all = es.merge(all, gulp.src('resources/linux/code.png', { base: '.' }));
358
} else if (platform === 'darwin') {
359
const shortcut = gulp.src('resources/darwin/bin/code.sh')
360
.pipe(replace('@@APPNAME@@', product.applicationName))
361
.pipe(rename('bin/code'));
362
const policyDest = gulp.src('.build/policies/darwin/**', { base: '.build/policies/darwin' })
363
.pipe(rename(f => f.dirname = `policies/${f.dirname}`));
364
all = es.merge(all, shortcut, policyDest);
365
}
366
367
let result = all
368
.pipe(util.skipDirectories())
369
.pipe(util.fixWin32DirectoryPermissions())
370
.pipe(filter(['**', '!**/.github/**'], { dot: true })) // https://github.com/microsoft/vscode/issues/116523
371
.pipe(electron({ ...config, platform, arch: arch === 'armhf' ? 'arm' : arch, ffmpegChromium: false }))
372
.pipe(filter(['**', '!LICENSE', '!version'], { dot: true }));
373
374
if (platform === 'linux') {
375
result = es.merge(result, gulp.src('resources/completions/bash/code', { base: '.' })
376
.pipe(replace('@@APPNAME@@', product.applicationName))
377
.pipe(rename(function (f) { f.basename = product.applicationName; })));
378
379
result = es.merge(result, gulp.src('resources/completions/zsh/_code', { base: '.' })
380
.pipe(replace('@@APPNAME@@', product.applicationName))
381
.pipe(rename(function (f) { f.basename = '_' + product.applicationName; })));
382
}
383
384
if (platform === 'win32') {
385
result = es.merge(result, gulp.src('resources/win32/bin/code.js', { base: 'resources/win32', allowEmpty: true }));
386
387
result = es.merge(result, gulp.src('resources/win32/bin/code.cmd', { base: 'resources/win32' })
388
.pipe(replace('@@NAME@@', product.nameShort))
389
.pipe(rename(function (f) { f.basename = product.applicationName; })));
390
391
result = es.merge(result, gulp.src('resources/win32/bin/code.sh', { base: 'resources/win32' })
392
.pipe(replace('@@NAME@@', product.nameShort))
393
.pipe(replace('@@PRODNAME@@', product.nameLong))
394
.pipe(replace('@@VERSION@@', version))
395
.pipe(replace('@@COMMIT@@', commit))
396
.pipe(replace('@@APPNAME@@', product.applicationName))
397
.pipe(replace('@@SERVERDATAFOLDER@@', product.serverDataFolderName || '.vscode-remote'))
398
.pipe(replace('@@QUALITY@@', quality))
399
.pipe(rename(function (f) { f.basename = product.applicationName; f.extname = ''; })));
400
401
result = es.merge(result, gulp.src('resources/win32/VisualElementsManifest.xml', { base: 'resources/win32' })
402
.pipe(rename(product.nameShort + '.VisualElementsManifest.xml')));
403
404
result = es.merge(result, gulp.src('.build/policies/win32/**', { base: '.build/policies/win32' })
405
.pipe(rename(f => f.dirname = `policies/${f.dirname}`)));
406
407
if (quality === 'stable' || quality === 'insider') {
408
result = es.merge(result, gulp.src('.build/win32/appx/**', { base: '.build/win32' }));
409
const rawVersion = version.replace(/-\w+$/, '').split('.');
410
const appxVersion = `${rawVersion[0]}.0.${rawVersion[1]}.${rawVersion[2]}`;
411
result = es.merge(result, gulp.src('resources/win32/appx/AppxManifest.xml', { base: '.' })
412
.pipe(replace('@@AppxPackageName@@', product.win32AppUserModelId))
413
.pipe(replace('@@AppxPackageVersion@@', appxVersion))
414
.pipe(replace('@@AppxPackageDisplayName@@', product.nameLong))
415
.pipe(replace('@@AppxPackageDescription@@', product.win32NameVersion))
416
.pipe(replace('@@ApplicationIdShort@@', product.win32RegValueName))
417
.pipe(replace('@@ApplicationExe@@', product.nameShort + '.exe'))
418
.pipe(replace('@@FileExplorerContextMenuID@@', quality === 'stable' ? 'OpenWithCode' : 'OpenWithCodeInsiders'))
419
.pipe(replace('@@FileExplorerContextMenuCLSID@@', product.win32ContextMenu[arch].clsid))
420
.pipe(replace('@@FileExplorerContextMenuDLL@@', `${quality === 'stable' ? 'code' : 'code_insider'}_explorer_command_${arch}.dll`))
421
.pipe(rename(f => f.dirname = `appx/manifest`)));
422
}
423
} else if (platform === 'linux') {
424
result = es.merge(result, gulp.src('resources/linux/bin/code.sh', { base: '.' })
425
.pipe(replace('@@PRODNAME@@', product.nameLong))
426
.pipe(replace('@@APPNAME@@', product.applicationName))
427
.pipe(rename('bin/' + product.applicationName)));
428
}
429
430
result = inlineMeta(result, {
431
targetPaths: bootstrapEntryPoints,
432
packageJsonFn: () => packageJsonContents,
433
productJsonFn: () => productJsonContents
434
});
435
436
return result.pipe(vfs.dest(destination));
437
};
438
task.taskName = `package-${platform}-${arch}`;
439
return task;
440
}
441
442
function patchWin32DependenciesTask(destinationFolderName) {
443
const cwd = path.join(path.dirname(root), destinationFolderName);
444
445
return async () => {
446
const deps = await glob('**/*.node', { cwd, ignore: 'extensions/node_modules/@parcel/watcher/**' });
447
const packageJson = JSON.parse(await fs.promises.readFile(path.join(cwd, 'resources', 'app', 'package.json'), 'utf8'));
448
const product = JSON.parse(await fs.promises.readFile(path.join(cwd, 'resources', 'app', 'product.json'), 'utf8'));
449
const baseVersion = packageJson.version.replace(/-.*$/, '');
450
451
await Promise.all(deps.map(async dep => {
452
const basename = path.basename(dep);
453
454
await rcedit(path.join(cwd, dep), {
455
'file-version': baseVersion,
456
'version-string': {
457
'CompanyName': 'Microsoft Corporation',
458
'FileDescription': product.nameLong,
459
'FileVersion': packageJson.version,
460
'InternalName': basename,
461
'LegalCopyright': 'Copyright (C) 2022 Microsoft. All rights reserved',
462
'OriginalFilename': basename,
463
'ProductName': product.nameLong,
464
'ProductVersion': packageJson.version,
465
}
466
});
467
}));
468
};
469
}
470
471
const buildRoot = path.dirname(root);
472
473
const BUILD_TARGETS = [
474
{ platform: 'win32', arch: 'x64' },
475
{ platform: 'win32', arch: 'arm64' },
476
{ platform: 'darwin', arch: 'x64', opts: { stats: true } },
477
{ platform: 'darwin', arch: 'arm64', opts: { stats: true } },
478
{ platform: 'linux', arch: 'x64' },
479
{ platform: 'linux', arch: 'armhf' },
480
{ platform: 'linux', arch: 'arm64' },
481
];
482
BUILD_TARGETS.forEach(buildTarget => {
483
const dashed = (str) => (str ? `-${str}` : ``);
484
const platform = buildTarget.platform;
485
const arch = buildTarget.arch;
486
const opts = buildTarget.opts;
487
488
const [vscode, vscodeMin] = ['', 'min'].map(minified => {
489
const sourceFolderName = `out-vscode${dashed(minified)}`;
490
const destinationFolderName = `VSCode${dashed(platform)}${dashed(arch)}`;
491
492
const tasks = [
493
compileNativeExtensionsBuildTask,
494
util.rimraf(path.join(buildRoot, destinationFolderName)),
495
packageTask(platform, arch, sourceFolderName, destinationFolderName, opts)
496
];
497
498
if (platform === 'win32') {
499
tasks.push(patchWin32DependenciesTask(destinationFolderName));
500
}
501
502
const vscodeTaskCI = task.define(`vscode${dashed(platform)}${dashed(arch)}${dashed(minified)}-ci`, task.series(...tasks));
503
gulp.task(vscodeTaskCI);
504
505
const vscodeTask = task.define(`vscode${dashed(platform)}${dashed(arch)}${dashed(minified)}`, task.series(
506
minified ? compileBuildWithManglingTask : compileBuildWithoutManglingTask,
507
cleanExtensionsBuildTask,
508
compileNonNativeExtensionsBuildTask,
509
compileExtensionMediaBuildTask,
510
minified ? minifyVSCodeTask : bundleVSCodeTask,
511
vscodeTaskCI
512
));
513
gulp.task(vscodeTask);
514
515
return vscodeTask;
516
});
517
518
if (process.platform === platform && process.arch === arch) {
519
gulp.task(task.define('vscode', task.series(vscode)));
520
gulp.task(task.define('vscode-min', task.series(vscodeMin)));
521
}
522
});
523
524
// #region nls
525
526
const innoSetupConfig = {
527
'zh-cn': { codePage: 'CP936', defaultInfo: { name: 'Simplified Chinese', id: '$0804', } },
528
'zh-tw': { codePage: 'CP950', defaultInfo: { name: 'Traditional Chinese', id: '$0404' } },
529
'ko': { codePage: 'CP949', defaultInfo: { name: 'Korean', id: '$0412' } },
530
'ja': { codePage: 'CP932' },
531
'de': { codePage: 'CP1252' },
532
'fr': { codePage: 'CP1252' },
533
'es': { codePage: 'CP1252' },
534
'ru': { codePage: 'CP1251' },
535
'it': { codePage: 'CP1252' },
536
'pt-br': { codePage: 'CP1252' },
537
'hu': { codePage: 'CP1250' },
538
'tr': { codePage: 'CP1254' }
539
};
540
541
gulp.task(task.define(
542
'vscode-translations-export',
543
task.series(
544
coreCI,
545
compileAllExtensionsBuildTask,
546
function () {
547
const pathToMetadata = './out-build/nls.metadata.json';
548
const pathToExtensions = '.build/extensions/*';
549
const pathToSetup = 'build/win32/i18n/messages.en.isl';
550
551
return es.merge(
552
gulp.src(pathToMetadata).pipe(i18n.createXlfFilesForCoreBundle()),
553
gulp.src(pathToSetup).pipe(i18n.createXlfFilesForIsl()),
554
gulp.src(pathToExtensions).pipe(i18n.createXlfFilesForExtensions())
555
).pipe(vfs.dest('../vscode-translations-export'));
556
}
557
)
558
));
559
560
gulp.task('vscode-translations-import', function () {
561
const options = minimist(process.argv.slice(2), {
562
string: 'location',
563
default: {
564
location: '../vscode-translations-import'
565
}
566
});
567
return es.merge([...i18n.defaultLanguages, ...i18n.extraLanguages].map(language => {
568
const id = language.id;
569
return gulp.src(`${options.location}/${id}/vscode-setup/messages.xlf`)
570
.pipe(i18n.prepareIslFiles(language, innoSetupConfig[language.id]))
571
.pipe(vfs.dest(`./build/win32/i18n`));
572
}));
573
});
574
575
// #endregion
576
577