gulpfile.vscode.js 26.0 KB
Newer Older
I
isidor 已提交
1 2 3 4 5
/*---------------------------------------------------------------------------------------------
 *  Copyright (c) Microsoft Corporation. All rights reserved.
 *  Licensed under the MIT License. See License.txt in the project root for license information.
 *--------------------------------------------------------------------------------------------*/

J
Joao Moreno 已提交
6 7 8 9
'use strict';

const gulp = require('gulp');
const fs = require('fs');
10 11
const os = require('os');
const cp = require('child_process');
J
Joao Moreno 已提交
12 13 14 15
const path = require('path');
const es = require('event-stream');
const azure = require('gulp-azure-storage');
const electron = require('gulp-atom-electron');
J
Joao Moreno 已提交
16
const vfs = require('vinyl-fs');
J
Joao Moreno 已提交
17 18 19 20 21 22
const rename = require('gulp-rename');
const replace = require('gulp-replace');
const filter = require('gulp-filter');
const json = require('gulp-json-editor');
const _ = require('underscore');
const util = require('./lib/util');
J
Joao Moreno 已提交
23
const ext = require('./lib/extensions');
J
Joao Moreno 已提交
24
const buildfile = require('../src/buildfile');
25
const common = require('./lib/optimize');
J
Joao Moreno 已提交
26 27
const root = path.dirname(__dirname);
const commit = util.getVersion(root);
J
Joao Moreno 已提交
28 29
const packageJson = require('../package.json');
const product = require('../product.json');
A
Alex Dima 已提交
30
const crypto = require('crypto');
31
const i18n = require('./lib/i18n');
32
const deps = require('./dependencies');
J
Joao Moreno 已提交
33
const getElectronVersion = require('./lib/electron').getElectronVersion;
A
Alex Dima 已提交
34
const createAsar = require('./lib/asar').createAsar;
35
const minimist = require('minimist');
J
Joao Moreno 已提交
36

37
const productionDependencies = deps.getProductionDependencies(path.dirname(__dirname));
J
Joao Moreno 已提交
38
// @ts-ignore
J
Joao Moreno 已提交
39 40
const baseModules = Object.keys(process.binding('natives')).filter(n => !/^_|\//.test(n));
const nodeModules = ['electron', 'original-fs']
41
	// @ts-ignore JSON checking: dependencies property is optional
42 43
	.concat(Object.keys(product.dependencies || {}))
	.concat(_.uniq(productionDependencies.map(d => d.name)))
J
Joao Moreno 已提交
44
	.concat(baseModules);
I
isidor 已提交
45 46

// Build
J
Joao Moreno 已提交
47
const vscodeEntryPoints = _.flatten([
48
	buildfile.entrypoint('vs/workbench/workbench.main'),
I
isidor 已提交
49
	buildfile.base,
50 51
	buildfile.workbench,
	buildfile.code
I
isidor 已提交
52 53
]);

J
Joao Moreno 已提交
54
const vscodeResources = [
J
Joao Moreno 已提交
55
	'out-build/main.js',
J
Joao Moreno 已提交
56
	'out-build/cli.js',
J
Joao Moreno 已提交
57
	'out-build/driver.js',
I
isidor 已提交
58
	'out-build/bootstrap.js',
59
	'out-build/bootstrap-fork.js',
J
Joao Moreno 已提交
60
	'out-build/bootstrap-amd.js',
61
	'out-build/bootstrap-window.js',
J
Joao Moreno 已提交
62
	'out-build/paths.js',
J
Joao Moreno 已提交
63
	'out-build/vs/**/*.{svg,png,cur,html}',
64
	'out-build/vs/base/common/performance.js',
65
	'out-build/vs/base/node/{stdForkStart.js,terminateProcess.sh,cpuUsage.sh}',
66
	'out-build/vs/base/browser/ui/octiconLabel/octicons/**',
I
isidor 已提交
67 68 69
	'out-build/vs/workbench/browser/media/*-theme.css',
	'out-build/vs/workbench/parts/debug/**/*.json',
	'out-build/vs/workbench/parts/execution/**/*.scpt',
70
	'out-build/vs/workbench/parts/webview/electron-browser/webview-pre.js',
J
Joao Moreno 已提交
71
	'out-build/vs/**/markdown.css',
I
isidor 已提交
72
	'out-build/vs/workbench/parts/tasks/**/*.json',
73
	'out-build/vs/workbench/parts/welcome/walkThrough/**/*.md',
I
isidor 已提交
74 75
	'out-build/vs/workbench/services/files/**/*.exe',
	'out-build/vs/workbench/services/files/**/*.md',
76
	'out-build/vs/code/electron-browser/workbench/**',
77
	'out-build/vs/code/electron-browser/sharedProcess/sharedProcess.js',
78
	'out-build/vs/code/electron-browser/issue/issueReporter.js',
79
	'out-build/vs/code/electron-browser/processExplorer/processExplorer.js',
I
isidor 已提交
80 81 82
	'!**/test/**'
];

J
Joao Moreno 已提交
83
const BUNDLED_FILE_HEADER = [
I
isidor 已提交
84 85 86 87 88 89
	'/*!--------------------------------------------------------',
	' * Copyright (C) Microsoft Corporation. All rights reserved.',
	' *--------------------------------------------------------*/'
].join('\n');

gulp.task('clean-optimized-vscode', util.rimraf('out-vscode'));
J
Joao Moreno 已提交
90
gulp.task('optimize-vscode', ['clean-optimized-vscode', 'compile-build', 'compile-extensions-build'], common.optimizeTask({
91
	src: 'out-build',
I
isidor 已提交
92 93 94
	entryPoints: vscodeEntryPoints,
	otherSources: [],
	resources: vscodeResources,
95
	loaderConfig: common.loaderConfig(nodeModules),
I
isidor 已提交
96
	header: BUNDLED_FILE_HEADER,
97
	out: 'out-vscode',
E
Erich Gamma 已提交
98
	bundleInfo: undefined
I
isidor 已提交
99 100
}));

101 102

gulp.task('optimize-index-js', ['optimize-vscode'], () => {
J
Johannes Rieken 已提交
103
	const fullpath = path.join(process.cwd(), 'out-vscode/bootstrap-window.js');
104 105 106
	const contents = fs.readFileSync(fullpath).toString();
	const newContents = contents.replace('[/*BUILD->INSERT_NODE_MODULES*/]', JSON.stringify(nodeModules));
	fs.writeFileSync(fullpath, newContents);
B
Benjamin Pasero 已提交
107
});
108

109
const sourceMappingURLBase = `https://ticino.blob.core.windows.net/sourcemaps/${commit}`;
I
isidor 已提交
110
gulp.task('clean-minified-vscode', util.rimraf('out-vscode-min'));
111
gulp.task('minify-vscode', ['clean-minified-vscode', 'optimize-index-js'], common.minifyTask('out-vscode', `${sourceMappingURLBase}/core`));
I
isidor 已提交
112 113

// Package
114 115

// @ts-ignore JSON checking: darwinCredits is optional
J
Joao Moreno 已提交
116
const darwinCreditsTemplate = product.darwinCredits && _.template(fs.readFileSync(path.join(root, product.darwinCredits), 'utf8'));
I
isidor 已提交
117

118 119 120 121 122 123 124 125 126 127
function darwinBundleDocumentType(extensions, icon) {
	return {
		name: product.nameLong + ' document',
		role: 'Editor',
		ostypes: ["TEXT", "utxt", "TUTX", "****"],
		extensions: extensions,
		iconFile: icon
	};
}

J
Joao Moreno 已提交
128
const config = {
J
Joao Moreno 已提交
129
	version: getElectronVersion(),
I
isidor 已提交
130
	productAppName: product.nameLong,
J
Joao Moreno 已提交
131
	companyName: 'Microsoft Corporation',
H
Henk Mollema 已提交
132
	copyright: 'Copyright (C) 2019 Microsoft. All rights reserved',
J
Joao Moreno 已提交
133
	darwinIcon: 'resources/darwin/code.icns',
I
isidor 已提交
134
	darwinBundleIdentifier: product.darwinBundleIdentifier,
J
Joao Moreno 已提交
135
	darwinApplicationCategoryType: 'public.app-category.developer-tools',
136 137
	darwinHelpBookFolder: 'VS Code HelpBook',
	darwinHelpBookName: 'VS Code HelpBook',
138 139 140 141 142 143 144 145 146
	darwinBundleDocumentTypes: [
		darwinBundleDocumentType(["bat", "cmd"], 'resources/darwin/bat.icns'),
		darwinBundleDocumentType(["bowerrc"], 'resources/darwin/bower.icns'),
		darwinBundleDocumentType(["c", "h"], 'resources/darwin/c.icns'),
		darwinBundleDocumentType(["config", "editorconfig", "gitattributes", "gitconfig", "gitignore", "ini"], 'resources/darwin/config.icns'),
		darwinBundleDocumentType(["cc", "cpp", "cxx", "hh", "hpp", "hxx"], 'resources/darwin/cpp.icns'),
		darwinBundleDocumentType(["cs", "csx"], 'resources/darwin/csharp.icns'),
		darwinBundleDocumentType(["css"], 'resources/darwin/css.icns'),
		darwinBundleDocumentType(["go"], 'resources/darwin/go.icns'),
B
Benjamin Pasero 已提交
147
		darwinBundleDocumentType(["asp", "aspx", "cshtml", "htm", "html", "jshtm", "jsp", "phtml", "shtml"], 'resources/darwin/html.icns'),
148 149
		darwinBundleDocumentType(["jade"], 'resources/darwin/jade.icns'),
		darwinBundleDocumentType(["jav", "java"], 'resources/darwin/java.icns'),
B
Benjamin Pasero 已提交
150
		darwinBundleDocumentType(["js", "jscsrc", "jshintrc", "mjs"], 'resources/darwin/javascript.icns'),
151 152 153 154 155 156 157
		darwinBundleDocumentType(["json"], 'resources/darwin/json.icns'),
		darwinBundleDocumentType(["less"], 'resources/darwin/less.icns'),
		darwinBundleDocumentType(["markdown", "md", "mdoc", "mdown", "mdtext", "mdtxt", "mdwn", "mkd", "mkdn"], 'resources/darwin/markdown.icns'),
		darwinBundleDocumentType(["php"], 'resources/darwin/php.icns'),
		darwinBundleDocumentType(["ps1", "psd1", "psm1"], 'resources/darwin/powershell.icns'),
		darwinBundleDocumentType(["py"], 'resources/darwin/python.icns'),
		darwinBundleDocumentType(["gemspec", "rb"], 'resources/darwin/ruby.icns'),
M
Miguel Solorio 已提交
158
		darwinBundleDocumentType(["scss"], 'resources/darwin/sass.icns'),
159 160 161 162 163 164 165
		darwinBundleDocumentType(["bash", "bash_login", "bash_logout", "bash_profile", "bashrc", "profile", "rhistory", "rprofile", "sh", "zlogin", "zlogout", "zprofile", "zsh", "zshenv", "zshrc"], 'resources/darwin/shell.icns'),
		darwinBundleDocumentType(["sql"], 'resources/darwin/sql.icns'),
		darwinBundleDocumentType(["ts"], 'resources/darwin/typescript.icns'),
		darwinBundleDocumentType(["tsx", "jsx"], 'resources/darwin/react.icns'),
		darwinBundleDocumentType(["vue"], 'resources/darwin/vue.icns'),
		darwinBundleDocumentType(["ascx", "csproj", "dtd", "wxi", "wxl", "wxs", "xml", "xaml"], 'resources/darwin/xml.icns'),
		darwinBundleDocumentType(["eyaml", "eyml", "yaml", "yml"], 'resources/darwin/yaml.icns'),
B
Benjamin Pasero 已提交
166
		darwinBundleDocumentType(["clj", "cljs", "cljx", "clojure", "code-workspace", "coffee", "ctp", "dockerfile", "dot", "edn", "fs", "fsi", "fsscript", "fsx", "handlebars", "hbs", "lua", "m", "makefile", "ml", "mli", "pl", "pl6", "pm", "pm6", "pod", "pp", "properties", "psgi", "pug", "r", "rs", "rt", "svg", "svgz", "t", "txt", "vb", "xcodeproj", "xcworkspace"], 'resources/darwin/default.icns')
167
	],
J
Joao Moreno 已提交
168 169 170
	darwinBundleURLTypes: [{
		role: 'Viewer',
		name: product.nameLong,
J
Joao Moreno 已提交
171
		urlSchemes: [product.urlProtocol]
J
Joao Moreno 已提交
172
	}],
173
	darwinForceDarkModeSupport: true,
R
Rob Lourens 已提交
174
	darwinCredits: darwinCreditsTemplate ? Buffer.from(darwinCreditsTemplate({ commit: commit, date: new Date().toISOString() })) : undefined,
175
	linuxExecutableName: product.applicationName,
J
Joao Moreno 已提交
176
	winIcon: 'resources/win32/code.ico',
R
Rob Lourens 已提交
177
	token: process.env['VSCODE_MIXIN_PASSWORD'] || process.env['GITHUB_TOKEN'] || undefined,
178 179

	// @ts-ignore JSON checking: electronRepository is optional
R
Rob Lourens 已提交
180
	repo: product.electronRepository || undefined
I
isidor 已提交
181 182
};

J
fix es6  
Joao Moreno 已提交
183
function getElectron(arch) {
J
Joao Moreno 已提交
184 185 186 187 188 189 190 191 192 193 194 195 196 197 198
	return () => {
		const electronOpts = _.extend({}, config, {
			platform: process.platform,
			arch,
			ffmpegChromium: true,
			keepDefaultApp: true
		});

		return gulp.src('package.json')
			.pipe(json({ name: product.nameShort }))
			.pipe(electron(electronOpts))
			.pipe(filter(['**', '!**/app/package.json']))
			.pipe(vfs.dest('.build/electron'));
	};
}
199

J
Joao Moreno 已提交
200
gulp.task('clean-electron', util.rimraf('.build/electron'));
J
fix es6  
Joao Moreno 已提交
201
gulp.task('electron', ['clean-electron'], getElectron(process.arch));
J
Joao Moreno 已提交
202 203
gulp.task('electron-ia32', ['clean-electron'], getElectron('ia32'));
gulp.task('electron-x64', ['clean-electron'], getElectron('x64'));
204 205
gulp.task('electron-arm', ['clean-electron'], getElectron('arm'));
gulp.task('electron-arm64', ['clean-electron'], getElectron('arm64'));
I
isidor 已提交
206 207


A
Alex Dima 已提交
208 209 210 211 212 213 214 215 216
/**
 * Compute checksums for some files.
 *
 * @param {string} out The out folder to read the file from.
 * @param {string[]} filenames The paths to compute a checksum for.
 * @return {Object} A map of paths to checksums.
 */
function computeChecksums(out, filenames) {
	var result = {};
217
	filenames.forEach(function (filename) {
A
Alex Dima 已提交
218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241
		var fullPath = path.join(process.cwd(), out, filename);
		result[filename] = computeChecksum(fullPath);
	});
	return result;
}

/**
 * Compute checksum for a file.
 *
 * @param {string} filename The absolute path to a filename.
 * @return {string} The checksum for `filename`.
 */
function computeChecksum(filename) {
	var contents = fs.readFileSync(filename);

	var hash = crypto
		.createHash('md5')
		.update(contents)
		.digest('base64')
		.replace(/=+$/, '');

	return hash;
}

J
fix es6  
Joao Moreno 已提交
242 243 244
function packageTask(platform, arch, opts) {
	opts = opts || {};

J
Joao Moreno 已提交
245
	const destination = path.join(path.dirname(root), 'VSCode') + (platform ? '-' + platform : '') + (arch ? '-' + arch : '');
I
isidor 已提交
246 247
	platform = platform || process.platform;

J
Joao Moreno 已提交
248 249
	return () => {
		const out = opts.minified ? 'out-vscode-min' : 'out-vscode';
I
isidor 已提交
250

A
Alex Dima 已提交
251
		const checksums = computeChecksums(out, [
252 253
			'vs/workbench/workbench.main.js',
			'vs/workbench/workbench.main.css',
254 255
			'vs/code/electron-browser/workbench/workbench.html',
			'vs/code/electron-browser/workbench/workbench.js'
A
Alex Dima 已提交
256 257
		]);

J
Joao Moreno 已提交
258
		const src = gulp.src(out + '/**', { base: '.' })
259 260 261
			.pipe(rename(function (path) { path.dirname = path.dirname.replace(new RegExp('^' + out), 'out'); }))
			.pipe(util.setExecutableBit(['**/*.sh']))
			.pipe(filter(['**', '!**/*.js.map']));
I
isidor 已提交
262

263 264
		const root = path.resolve(path.join(__dirname, '..'));

265 266
		const sources = es.merge(src, ext.packageExtensionsStream({
			sourceMappingURLBase: sourceMappingURLBase
267 268
		}));

J
Joao Moreno 已提交
269
		let version = packageJson.version;
270
		// @ts-ignore JSON checking: quality is optional
J
Joao Moreno 已提交
271
		const quality = product.quality;
J
Joao Moreno 已提交
272 273 274 275 276

		if (quality && quality !== 'stable') {
			version += '-' + quality;
		}

J
Joao Moreno 已提交
277
		const name = product.nameShort;
278 279 280 281 282 283 284
		const packageJsonUpdates = { name, version };

		// for linux url handling
		if (platform === 'linux') {
			packageJsonUpdates.desktopName = `${product.applicationName}-url-handler.desktop`;
		}

J
Joao Moreno 已提交
285
		const packageJsonStream = gulp.src(['package.json'], { base: '.' })
286
			.pipe(json(packageJsonUpdates));
J
Joao Moreno 已提交
287 288

		const date = new Date().toISOString();
289 290
		const productJsonUpdate = { commit, date, checksums };

291
		if (shouldSetupSettingsSearch()) {
292 293 294
			productJsonUpdate.settingsSearchBuildId = getSettingsSearchBuildId(packageJson);
		}

J
Joao Moreno 已提交
295
		const productJsonStream = gulp.src(['product.json'], { base: '.' })
296
			.pipe(json(productJsonUpdate));
I
isidor 已提交
297

298
		const license = gulp.src(['LICENSES.chromium.html', 'LICENSE.txt', 'ThirdPartyNotices.txt', 'licenses/**'], { base: '.' });
I
isidor 已提交
299

300 301
		const watermark = gulp.src(['resources/letterpress.svg', 'resources/letterpress-dark.svg', 'resources/letterpress-hc.svg'], { base: '.' });

J
Joao Moreno 已提交
302 303 304
		// TODO the API should be copied to `out` during compile, not here
		const api = gulp.src('src/vs/vscode.d.ts').pipe(rename('out/vs/vscode.d.ts'));

305 306
		const depsSrc = [
			..._.flatten(productionDependencies.map(d => path.relative(root, d.path)).map(d => [`${d}/**`, `!${d}/**/{test,tests}/**`])),
307
			// @ts-ignore JSON checking: dependencies is optional
308 309
			..._.flatten(Object.keys(product.dependencies || {}).map(d => [`node_modules/${d}/**`, `!node_modules/${d}/**/{test,tests}/**`]))
		];
I
isidor 已提交
310

J
Joao Moreno 已提交
311
		const deps = gulp.src(depsSrc, { base: '.', dot: true })
J
Joao Moreno 已提交
312
			.pipe(filter(['**', '!**/package-lock.json']))
J
Joao Moreno 已提交
313
			.pipe(util.cleanNodeModule('fsevents', ['binding.gyp', 'fsevents.cc', 'build/**', 'src/**', 'test/**'], ['**/*.node']))
314 315
			.pipe(util.cleanNodeModule('vscode-sqlite3', ['binding.gyp', 'benchmark/**', 'cloudformation/**', 'deps/**', 'test/**', 'build/**', 'src/**'], ['build/Release/*.node']))
			.pipe(util.cleanNodeModule('oniguruma', ['binding.gyp', 'build/**', 'src/**', 'deps/**'], ['build/Release/*.node', 'src/*.js']))
J
Joao Moreno 已提交
316
			.pipe(util.cleanNodeModule('windows-mutex', ['binding.gyp', 'build/**', 'src/**'], ['**/*.node']))
317 318 319 320
			.pipe(util.cleanNodeModule('native-keymap', ['binding.gyp', 'build/**', 'src/**', 'deps/**'], ['build/Release/*.node']))
			.pipe(util.cleanNodeModule('native-is-elevated', ['binding.gyp', 'build/**', 'src/**', 'deps/**'], ['build/Release/*.node']))
			.pipe(util.cleanNodeModule('native-watchdog', ['binding.gyp', 'build/**', 'src/**'], ['build/Release/*.node']))
			.pipe(util.cleanNodeModule('spdlog', ['binding.gyp', 'build/**', 'deps/**', 'src/**', 'test/**'], ['build/Release/*.node']))
B
Benjamin Pasero 已提交
321
			.pipe(util.cleanNodeModule('jschardet', ['dist/**']))
322
			.pipe(util.cleanNodeModule('windows-foreground-love', ['binding.gyp', 'build/**', 'src/**'], ['**/*.node']))
A
Amy Qiu 已提交
323
			.pipe(util.cleanNodeModule('windows-process-tree', ['binding.gyp', 'build/**', 'src/**'], ['**/*.node']))
324
			.pipe(util.cleanNodeModule('gc-signals', ['binding.gyp', 'build/**', 'src/**', 'deps/**'], ['build/Release/*.node', 'src/index.js']))
C
Christof Marti 已提交
325
			.pipe(util.cleanNodeModule('keytar', ['binding.gyp', 'build/**', 'src/**', 'script/**', 'node_modules/**'], ['**/*.node']))
326
			.pipe(util.cleanNodeModule('node-pty', ['binding.gyp', 'build/**', 'src/**', 'tools/**'], ['build/Release/*.exe', 'build/Release/*.dll', 'build/Release/*.node']))
327
			.pipe(util.cleanNodeModule('vscode-nsfw', ['binding.gyp', 'build/**', 'src/**', 'openpa/**', 'includes/**'], ['build/Release/*.node', '**/*.a']))
A
Alex Dima 已提交
328 329
			.pipe(util.cleanNodeModule('vsda', ['binding.gyp', 'README.md', 'build/**', '*.bat', '*.sh', '*.cpp', '*.h'], ['build/Release/vsda.node']))
			.pipe(createAsar(path.join(process.cwd(), 'node_modules'), ['**/*.node', '**/vscode-ripgrep/bin/*', '**/node-pty/build/Release/*'], 'app/node_modules.asar'));
I
isidor 已提交
330

J
Joao Moreno 已提交
331
		let all = es.merge(
J
Joao Moreno 已提交
332
			packageJsonStream,
J
Joao Moreno 已提交
333
			productJsonStream,
I
isidor 已提交
334
			license,
335
			watermark,
J
Joao Moreno 已提交
336
			api,
I
isidor 已提交
337
			sources,
J
Joao Moreno 已提交
338
			deps
J
Joao Moreno 已提交
339 340 341
		);

		if (platform === 'win32') {
J
Joao Moreno 已提交
342
			all = es.merge(all, gulp.src(['resources/win32/*.ico', 'resources/win32/code_*.png',], { base: '.' }));
J
Joao Moreno 已提交
343 344
		} else if (platform === 'linux') {
			all = es.merge(all, gulp.src('resources/linux/code.png', { base: '.' }));
J
Joao Moreno 已提交
345
		} else if (platform === 'darwin') {
J
Joao Moreno 已提交
346
			const shortcut = gulp.src('resources/darwin/bin/code.sh')
J
Joao Moreno 已提交
347 348 349
				.pipe(rename('bin/code'));

			all = es.merge(all, shortcut);
J
Joao Moreno 已提交
350
		}
I
isidor 已提交
351

J
Joao Moreno 已提交
352
		let result = all
J
Joao Moreno 已提交
353
			.pipe(util.skipDirectories())
I
isidor 已提交
354
			.pipe(util.fixWin32DirectoryPermissions())
355
			.pipe(electron(_.extend({}, config, { platform, arch, ffmpegChromium: true })))
356
			.pipe(filter(['**', '!LICENSE', '!LICENSES.chromium.html', '!version']));
I
isidor 已提交
357

358 359
		result = es.merge(result, gulp.src('resources/completions/**', { base: '.' }));

I
isidor 已提交
360
		if (platform === 'win32') {
J
typo  
João Moreno 已提交
361
			result = es.merge(result, gulp.src('resources/win32/bin/code.js', { base: 'resources/win32' }));
J
Joao Moreno 已提交
362

J
typo  
João Moreno 已提交
363
			result = es.merge(result, gulp.src('resources/win32/bin/code.cmd', { base: 'resources/win32' })
J
Joao Moreno 已提交
364
				.pipe(replace('@@NAME@@', product.nameShort))
J
Joao Moreno 已提交
365 366
				.pipe(rename(function (f) { f.basename = product.applicationName; })));

J
typo  
João Moreno 已提交
367
			result = es.merge(result, gulp.src('resources/win32/bin/code.sh', { base: 'resources/win32' })
J
Joao Moreno 已提交
368
				.pipe(replace('@@NAME@@', product.nameShort))
A
Alex Dima 已提交
369 370
				.pipe(replace('@@COMMIT@@', commit))
				.pipe(replace('@@APPNAME@@', product.applicationName))
J
Joao Moreno 已提交
371
				.pipe(rename(function (f) { f.basename = product.applicationName; f.extname = ''; })));
372 373

			result = es.merge(result, gulp.src('resources/win32/VisualElementsManifest.xml', { base: 'resources/win32' })
374
				.pipe(rename(product.nameShort + '.VisualElementsManifest.xml')));
375 376 377 378
		} else if (platform === 'linux') {
			result = es.merge(result, gulp.src('resources/linux/bin/code.sh', { base: '.' })
				.pipe(replace('@@NAME@@', product.applicationName))
				.pipe(rename('bin/' + product.applicationName)));
I
isidor 已提交
379 380
		}

J
Johannes Rieken 已提交
381 382 383 384 385
		// submit all stats that have been collected
		// during the build phase
		if (opts.stats) {
			result.on('end', () => {
				const { submitAllStats } = require('./lib/stats');
J
Johannes Rieken 已提交
386
				submitAllStats(product, commit).then(() => console.log('Submitted bundle stats!'));
J
Johannes Rieken 已提交
387 388 389
			});
		}

J
Joao Moreno 已提交
390
		return result.pipe(vfs.dest(destination));
I
isidor 已提交
391 392 393
	};
}

J
Joao Moreno 已提交
394 395
const buildRoot = path.dirname(root);

J
Joao Moreno 已提交
396 397
gulp.task('clean-vscode-win32-ia32', util.rimraf(path.join(buildRoot, 'VSCode-win32-ia32')));
gulp.task('clean-vscode-win32-x64', util.rimraf(path.join(buildRoot, 'VSCode-win32-x64')));
J
Joao Moreno 已提交
398 399 400 401
gulp.task('clean-vscode-darwin', util.rimraf(path.join(buildRoot, 'VSCode-darwin')));
gulp.task('clean-vscode-linux-ia32', util.rimraf(path.join(buildRoot, 'VSCode-linux-ia32')));
gulp.task('clean-vscode-linux-x64', util.rimraf(path.join(buildRoot, 'VSCode-linux-x64')));
gulp.task('clean-vscode-linux-arm', util.rimraf(path.join(buildRoot, 'VSCode-linux-arm')));
A
ava1ar 已提交
402
gulp.task('clean-vscode-linux-arm64', util.rimraf(path.join(buildRoot, 'VSCode-linux-arm64')));
I
isidor 已提交
403

J
Joao Moreno 已提交
404 405
gulp.task('vscode-win32-ia32', ['optimize-vscode', 'clean-vscode-win32-ia32'], packageTask('win32', 'ia32'));
gulp.task('vscode-win32-x64', ['optimize-vscode', 'clean-vscode-win32-x64'], packageTask('win32', 'x64'));
406
gulp.task('vscode-darwin', ['optimize-vscode', 'clean-vscode-darwin'], packageTask('darwin', null, { stats: true }));
I
isidor 已提交
407 408
gulp.task('vscode-linux-ia32', ['optimize-vscode', 'clean-vscode-linux-ia32'], packageTask('linux', 'ia32'));
gulp.task('vscode-linux-x64', ['optimize-vscode', 'clean-vscode-linux-x64'], packageTask('linux', 'x64'));
409
gulp.task('vscode-linux-arm', ['optimize-vscode', 'clean-vscode-linux-arm'], packageTask('linux', 'arm'));
A
ava1ar 已提交
410
gulp.task('vscode-linux-arm64', ['optimize-vscode', 'clean-vscode-linux-arm64'], packageTask('linux', 'arm64'));
I
isidor 已提交
411

J
Joao Moreno 已提交
412 413
gulp.task('vscode-win32-ia32-min', ['minify-vscode', 'clean-vscode-win32-ia32'], packageTask('win32', 'ia32', { minified: true }));
gulp.task('vscode-win32-x64-min', ['minify-vscode', 'clean-vscode-win32-x64'], packageTask('win32', 'x64', { minified: true }));
414
gulp.task('vscode-darwin-min', ['minify-vscode', 'clean-vscode-darwin'], packageTask('darwin', null, { minified: true, stats: true }));
I
isidor 已提交
415 416
gulp.task('vscode-linux-ia32-min', ['minify-vscode', 'clean-vscode-linux-ia32'], packageTask('linux', 'ia32', { minified: true }));
gulp.task('vscode-linux-x64-min', ['minify-vscode', 'clean-vscode-linux-x64'], packageTask('linux', 'x64', { minified: true }));
417
gulp.task('vscode-linux-arm-min', ['minify-vscode', 'clean-vscode-linux-arm'], packageTask('linux', 'arm', { minified: true }));
A
ava1ar 已提交
418
gulp.task('vscode-linux-arm64-min', ['minify-vscode', 'clean-vscode-linux-arm64'], packageTask('linux', 'arm64', { minified: true }));
I
isidor 已提交
419

420
// Transifex Localizations
D
Dirk Baeumer 已提交
421 422 423 424 425 426 427 428 429 430 431 432 433 434 435

const innoSetupConfig = {
	'zh-cn': { codePage: 'CP936', defaultInfo: { name: 'Simplified Chinese', id: '$0804', } },
	'zh-tw': { codePage: 'CP950', defaultInfo: { name: 'Traditional Chinese', id: '$0404' } },
	'ko': { codePage: 'CP949', defaultInfo: { name: 'Korean', id: '$0412' } },
	'ja': { codePage: 'CP932' },
	'de': { codePage: 'CP1252' },
	'fr': { codePage: 'CP1252' },
	'es': { codePage: 'CP1252' },
	'ru': { codePage: 'CP1251' },
	'it': { codePage: 'CP1252' },
	'pt-br': { codePage: 'CP1252' },
	'hu': { codePage: 'CP1250' },
	'tr': { codePage: 'CP1254' }
};
436

437
const apiHostname = process.env.TRANSIFEX_API_URL;
438 439 440
const apiName = process.env.TRANSIFEX_API_NAME;
const apiToken = process.env.TRANSIFEX_API_TOKEN;

J
Joao Moreno 已提交
441
gulp.task('vscode-translations-push', ['optimize-vscode'], function () {
442
	const pathToMetadata = './out-vscode/nls.metadata.json';
D
Dirk Baeumer 已提交
443
	const pathToExtensions = './extensions/*';
444 445
	const pathToSetup = 'build/win32/**/{Default.isl,messages.en.isl}';

446
	return es.merge(
D
Dirk Baeumer 已提交
447 448 449
		gulp.src(pathToMetadata).pipe(i18n.createXlfFilesForCoreBundle()),
		gulp.src(pathToSetup).pipe(i18n.createXlfFilesForIsl()),
		gulp.src(pathToExtensions).pipe(i18n.createXlfFilesForExtensions())
450
	).pipe(i18n.findObsoleteResources(apiHostname, apiName, apiToken)
J
Joao Moreno 已提交
451
	).pipe(i18n.pushXlfFiles(apiHostname, apiName, apiToken));
452 453
});

454
gulp.task('vscode-translations-export', ['optimize-vscode'], function () {
D
Dirk Baeumer 已提交
455 456 457 458
	const pathToMetadata = './out-vscode/nls.metadata.json';
	const pathToExtensions = './extensions/*';
	const pathToSetup = 'build/win32/**/{Default.isl,messages.en.isl}';

459
	return es.merge(
D
Dirk Baeumer 已提交
460 461 462
		gulp.src(pathToMetadata).pipe(i18n.createXlfFilesForCoreBundle()),
		gulp.src(pathToSetup).pipe(i18n.createXlfFilesForIsl()),
		gulp.src(pathToExtensions).pipe(i18n.createXlfFilesForExtensions())
463
	).pipe(vfs.dest('../vscode-translations-export'));
D
Dirk Baeumer 已提交
464 465 466
});

gulp.task('vscode-translations-pull', function () {
467
	return es.merge([...i18n.defaultLanguages, ...i18n.extraLanguages].map(language => {
D
Dirk Baeumer 已提交
468
		let includeDefault = !!innoSetupConfig[language.id].defaultInfo;
469
		return i18n.pullSetupXlfFiles(apiHostname, apiName, apiToken, language, includeDefault).pipe(vfs.dest(`../vscode-translations-import/${language.id}/setup`));
470
	}));
471 472
});

473
gulp.task('vscode-translations-import', function () {
474 475 476 477 478 479
	var options = minimist(process.argv.slice(2), {
		string: 'location',
		default: {
			location: '../vscode-translations-import'
		}
	});
480
	return es.merge([...i18n.defaultLanguages, ...i18n.extraLanguages].map(language => {
481 482
		let id = language.transifexId || language.id;
		return gulp.src(`${options.location}/${id}/setup/*/*.xlf`)
D
Dirk Baeumer 已提交
483 484
			.pipe(i18n.prepareIslFiles(language, innoSetupConfig[language.id]))
			.pipe(vfs.dest(`./build/win32/i18n`));
485
	}));
486 487
});

I
isidor 已提交
488 489
// Sourcemaps

J
Johannes Rieken 已提交
490
gulp.task('upload-vscode-sourcemaps', ['vscode-darwin-min', 'minify-vscode'], () => {
J
Joao Moreno 已提交
491 492
	const vs = gulp.src('out-vscode-min/**/*.map', { base: 'out-vscode-min' })
		.pipe(es.mapSync(f => {
493
			f.path = `${f.base}/core/${f.relative}`;
J
Joao Moreno 已提交
494 495 496
			return f;
		}));

497 498
	const extensionsOut = gulp.src('extensions/**/out/**/*.map', { base: '.' });
	const extensionsDist = gulp.src('extensions/**/dist/**/*.map', { base: '.' });
J
Joao Moreno 已提交
499

500
	return es.merge(vs, extensionsOut, extensionsDist)
J
Johannes Rieken 已提交
501 502 503 504 505
		.pipe(es.through(function (data) {
			// debug
			console.log('Uploading Sourcemap', data.relative);
			this.emit('data', data);
		}))
I
isidor 已提交
506 507 508 509
		.pipe(azure.upload({
			account: process.env.AZURE_STORAGE_ACCOUNT,
			key: process.env.AZURE_STORAGE_ACCESS_KEY,
			container: 'sourcemaps',
J
Joao Moreno 已提交
510
			prefix: commit + '/'
I
isidor 已提交
511
		}));
512
});
513 514 515

const allConfigDetailsPath = path.join(os.tmpdir(), 'configuration.json');
gulp.task('upload-vscode-configuration', ['generate-vscode-configuration'], () => {
516 517
	if (!shouldSetupSettingsSearch()) {
		const branch = process.env.BUILD_SOURCEBRANCH;
518 519 520 521
		console.log(`Only runs on master and release branches, not ${branch}`);
		return;
	}

522
	if (!fs.existsSync(allConfigDetailsPath)) {
523
		throw new Error(`configuration file at ${allConfigDetailsPath} does not exist`);
524 525
	}

526
	const settingsSearchBuildId = getSettingsSearchBuildId(packageJson);
527
	if (!settingsSearchBuildId) {
528
		throw new Error('Failed to compute build number');
R
Rob Lourens 已提交
529 530
	}

531 532 533 534 535
	return gulp.src(allConfigDetailsPath)
		.pipe(azure.upload({
			account: process.env.AZURE_STORAGE_ACCOUNT,
			key: process.env.AZURE_STORAGE_ACCESS_KEY,
			container: 'configuration',
536
			prefix: `${settingsSearchBuildId}/${commit}/`
537 538 539
		}));
});

540 541 542 543
function shouldSetupSettingsSearch() {
	const branch = process.env.BUILD_SOURCEBRANCH;
	return branch && (/\/master$/.test(branch) || branch.indexOf('/release/') >= 0);
}
R
Rob Lourens 已提交
544

545
function getSettingsSearchBuildId(packageJson) {
R
Rob Lourens 已提交
546
	try {
547 548 549
		const branch = process.env.BUILD_SOURCEBRANCH;
		const branchId = branch.indexOf('/release/') >= 0 ? 0 :
			/\/master$/.test(branch) ? 1 :
550
				2; // Some unexpected branch
551

R
Rob Lourens 已提交
552
		const out = cp.execSync(`git rev-list HEAD --count`);
R
Rob Lourens 已提交
553
		const count = parseInt(out.toString());
554 555 556 557

		// <version number><commit count><branchId (avoid unlikely conflicts)>
		// 1.25.1, 1,234,567 commits, master = 1250112345671
		return util.versionStringToNumber(packageJson.version) * 1e8 + count * 10 + branchId;
R
Rob Lourens 已提交
558
	} catch (e) {
559
		throw new Error('Could not determine build number: ' + e.toString());
R
Rob Lourens 已提交
560 561 562
	}
}

563
// This task is only run for the MacOS build
564 565 566 567 568 569 570
gulp.task('generate-vscode-configuration', () => {
	return new Promise((resolve, reject) => {
		const buildDir = process.env['AGENT_BUILDDIRECTORY'];
		if (!buildDir) {
			return reject(new Error('$AGENT_BUILDDIRECTORY not set'));
		}

J
Joao Moreno 已提交
571 572 573 574
		if (process.env.VSCODE_QUALITY !== 'insider' && process.env.VSCODE_QUALITY !== 'stable') {
			return resolve();
		}

575 576
		const userDataDir = path.join(os.tmpdir(), 'tmpuserdata');
		const extensionsDir = path.join(os.tmpdir(), 'tmpextdir');
577 578
		const appName = process.env.VSCODE_QUALITY === 'insider' ? 'Visual\\ Studio\\ Code\\ -\\ Insiders.app' : 'Visual\\ Studio\\ Code.app';
		const appPath = path.join(buildDir, `VSCode-darwin/${appName}/Contents/Resources/app/bin/code`);
579
		const codeProc = cp.exec(`${appPath} --export-default-configuration='${allConfigDetailsPath}' --wait --user-data-dir='${userDataDir}' --extensions-dir='${extensionsDir}'`);
580 581 582

		const timer = setTimeout(() => {
			codeProc.kill();
583
			reject(new Error('export-default-configuration process timed out'));
584 585 586 587 588 589 590 591 592 593 594 595 596 597 598 599
		}, 10 * 1000);

		codeProc.stdout.on('data', d => console.log(d.toString()));
		codeProc.stderr.on('data', d => console.log(d.toString()));

		codeProc.on('exit', () => {
			clearTimeout(timer);
			resolve();
		});

		codeProc.on('error', err => {
			clearTimeout(timer);
			reject(err);
		});
	});
});