2016-02-09 14:59:11 +01:00
|
|
|
/*---------------------------------------------------------------------------------------------
|
|
|
|
* Copyright (c) Microsoft Corporation. All rights reserved.
|
|
|
|
* Licensed under the MIT License. See License.txt in the project root for license information.
|
|
|
|
*--------------------------------------------------------------------------------------------*/
|
2018-09-07 16:05:24 +02:00
|
|
|
|
|
|
|
//@ts-check
|
2016-09-19 10:22:57 +02:00
|
|
|
'use strict';
|
|
|
|
|
2018-06-13 17:13:55 +02:00
|
|
|
const perf = require('./vs/base/common/performance');
|
2019-02-20 19:26:49 +01:00
|
|
|
const lp = require('./vs/base/node/languagePacks');
|
|
|
|
|
2017-11-06 18:50:57 +01:00
|
|
|
perf.mark('main:started');
|
|
|
|
|
2018-06-13 17:13:55 +02:00
|
|
|
const path = require('path');
|
2019-10-09 17:15:01 +02:00
|
|
|
const fs = require('fs');
|
|
|
|
const os = require('os');
|
2018-09-07 12:23:09 +02:00
|
|
|
const bootstrap = require('./bootstrap');
|
|
|
|
const paths = require('./paths');
|
2020-06-03 10:22:47 +02:00
|
|
|
/** @type {any} */
|
2018-09-07 16:05:24 +02:00
|
|
|
const product = require('../product.json');
|
2019-10-10 21:26:28 +02:00
|
|
|
const { app, protocol } = require('electron');
|
2018-06-13 17:13:55 +02:00
|
|
|
|
2018-09-07 12:23:09 +02:00
|
|
|
// Enable portable support
|
|
|
|
const portable = bootstrap.configurePortable();
|
|
|
|
|
|
|
|
// Enable ASAR support
|
|
|
|
bootstrap.enableASARSupport();
|
|
|
|
|
2019-10-09 07:45:56 +02:00
|
|
|
// Set userData path before app 'ready' event
|
2018-09-07 12:23:09 +02:00
|
|
|
const args = parseCLIArgs();
|
|
|
|
const userDataPath = getUserDataPath(args);
|
|
|
|
app.setPath('userData', userDataPath);
|
|
|
|
|
2020-05-07 07:59:29 +02:00
|
|
|
// Set temp directory based on crash-reporter-directory CLI argument
|
|
|
|
// The crash reporter will store crashes in temp folder so we need
|
|
|
|
// to change that location accordingly.
|
2020-05-08 08:14:57 +02:00
|
|
|
|
|
|
|
// If a crash-reporter-directory is specified we setup the crash reporter
|
|
|
|
// right from the beginning as early as possible to monitor all processes.
|
2020-05-07 16:19:51 +02:00
|
|
|
let crashReporterDirectory = args['crash-reporter-directory'];
|
2020-05-07 07:59:29 +02:00
|
|
|
if (crashReporterDirectory) {
|
2020-05-07 16:19:51 +02:00
|
|
|
crashReporterDirectory = path.normalize(crashReporterDirectory);
|
|
|
|
|
2020-06-03 09:33:32 +02:00
|
|
|
if (!path.isAbsolute(crashReporterDirectory)) {
|
|
|
|
console.error(`The path '${crashReporterDirectory}' specified for --crash-reporter-directory must be absolute.`);
|
|
|
|
app.exit(1);
|
|
|
|
}
|
|
|
|
|
2020-05-07 07:59:29 +02:00
|
|
|
if (!fs.existsSync(crashReporterDirectory)) {
|
|
|
|
try {
|
|
|
|
fs.mkdirSync(crashReporterDirectory);
|
|
|
|
} catch (error) {
|
|
|
|
console.error(`The path '${crashReporterDirectory}' specified for --crash-reporter-directory does not seem to exist or cannot be created.`);
|
|
|
|
app.exit(1);
|
|
|
|
}
|
|
|
|
}
|
2020-05-08 08:14:57 +02:00
|
|
|
|
|
|
|
// Crashes are stored in the temp directory by default, so we
|
|
|
|
// need to change that directory to the provided one
|
2020-05-07 16:19:51 +02:00
|
|
|
console.log(`Found --crash-reporter-directory argument. Setting temp directory to be '${crashReporterDirectory}'`);
|
2020-05-07 07:59:29 +02:00
|
|
|
app.setPath('temp', crashReporterDirectory);
|
2020-05-08 08:14:57 +02:00
|
|
|
|
|
|
|
// Start crash reporter
|
|
|
|
const { crashReporter } = require('electron');
|
2020-06-03 10:22:47 +02:00
|
|
|
const productName = (product.crashReporter && product.crashReporter.productName) || product.nameShort;
|
|
|
|
const companyName = (product.crashReporter && product.crashReporter.companyName) || 'Microsoft';
|
2020-05-08 08:14:57 +02:00
|
|
|
crashReporter.start({
|
2020-06-03 10:22:47 +02:00
|
|
|
companyName: companyName,
|
|
|
|
productName: process.env['VSCODE_DEV'] ? `${productName} Dev` : productName,
|
2020-05-08 08:14:57 +02:00
|
|
|
submitURL: '',
|
|
|
|
uploadToServer: false
|
|
|
|
});
|
2020-05-07 07:59:29 +02:00
|
|
|
}
|
|
|
|
|
2019-10-14 15:14:34 +02:00
|
|
|
// Set logs path before app 'ready' event if running portable
|
|
|
|
// to ensure that no 'logs' folder is created on disk at a
|
|
|
|
// location outside of the portable directory
|
|
|
|
// (https://github.com/microsoft/vscode/issues/56651)
|
|
|
|
if (portable.isPortable) {
|
|
|
|
app.setAppLogsPath(path.join(userDataPath, 'logs'));
|
|
|
|
}
|
|
|
|
|
2018-09-07 12:23:09 +02:00
|
|
|
// Update cwd based on environment and platform
|
|
|
|
setCurrentWorkingDirectory();
|
|
|
|
|
2019-10-10 21:26:28 +02:00
|
|
|
// Register custom schemes with privileges
|
|
|
|
protocol.registerSchemesAsPrivileged([
|
2020-05-15 20:34:08 +02:00
|
|
|
{
|
2020-06-23 02:26:10 +02:00
|
|
|
scheme: 'vscode-webview',
|
2020-06-03 00:31:28 +02:00
|
|
|
privileges: {
|
2020-06-23 02:26:10 +02:00
|
|
|
standard: true,
|
2020-06-03 00:31:28 +02:00
|
|
|
secure: true,
|
|
|
|
}
|
|
|
|
}, {
|
2020-05-15 20:34:08 +02:00
|
|
|
scheme: 'vscode-webview-resource',
|
|
|
|
privileges: {
|
|
|
|
secure: true,
|
|
|
|
standard: true,
|
|
|
|
supportFetchAPI: true,
|
|
|
|
corsEnabled: true,
|
|
|
|
}
|
|
|
|
},
|
2019-10-10 21:26:28 +02:00
|
|
|
]);
|
|
|
|
|
2018-09-07 12:23:09 +02:00
|
|
|
// Global app listeners
|
|
|
|
registerListeners();
|
|
|
|
|
2019-10-10 07:37:06 +02:00
|
|
|
// Cached data
|
|
|
|
const nodeCachedDataDir = getNodeCachedDir();
|
|
|
|
|
|
|
|
// Configure static command line arguments
|
|
|
|
const argvConfig = configureCommandlineSwitchesSync(args);
|
|
|
|
|
2020-04-03 15:53:03 +02:00
|
|
|
// Remove env set by snap https://github.com/microsoft/vscode/issues/85344
|
|
|
|
if (process.env['SNAP']) {
|
|
|
|
delete process.env['GDK_PIXBUF_MODULE_FILE'];
|
|
|
|
delete process.env['GDK_PIXBUF_MODULEDIR'];
|
|
|
|
}
|
|
|
|
|
2018-09-10 07:48:52 +02:00
|
|
|
/**
|
2019-10-09 07:45:56 +02:00
|
|
|
* Support user defined locale: load it early before app('ready')
|
|
|
|
* to have more things running in parallel.
|
2018-09-10 07:48:52 +02:00
|
|
|
*
|
2019-10-09 07:45:56 +02:00
|
|
|
* @type {Promise<import('./vs/base/node/languagePacks').NLSConfiguration>} nlsConfig | undefined
|
2018-09-10 07:48:52 +02:00
|
|
|
*/
|
2019-10-09 07:45:56 +02:00
|
|
|
let nlsConfigurationPromise = undefined;
|
2019-07-04 18:08:45 +02:00
|
|
|
|
2019-10-10 07:37:06 +02:00
|
|
|
const metaDataFile = path.join(__dirname, 'nls.metadata.json');
|
|
|
|
const locale = getUserDefinedLocale(argvConfig);
|
|
|
|
if (locale) {
|
|
|
|
nlsConfigurationPromise = lp.getNLSConfiguration(product.commit, userDataPath, metaDataFile, locale);
|
|
|
|
}
|
2018-09-07 12:23:09 +02:00
|
|
|
|
|
|
|
// Load our code once ready
|
|
|
|
app.once('ready', function () {
|
2018-09-20 09:16:47 +02:00
|
|
|
if (args['trace']) {
|
|
|
|
const contentTracing = require('electron').contentTracing;
|
|
|
|
|
|
|
|
const traceOptions = {
|
|
|
|
categoryFilter: args['trace-category-filter'] || '*',
|
|
|
|
traceOptions: args['trace-options'] || 'record-until-full,enable-sampling'
|
|
|
|
};
|
|
|
|
|
2020-02-17 10:44:25 +01:00
|
|
|
contentTracing.startRecording(traceOptions).finally(() => onReady());
|
2018-09-20 09:16:47 +02:00
|
|
|
} else {
|
|
|
|
onReady();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2019-10-09 07:45:56 +02:00
|
|
|
/**
|
|
|
|
* Main startup routine
|
|
|
|
*
|
|
|
|
* @param {string | undefined} cachedDataDir
|
|
|
|
* @param {import('./vs/base/node/languagePacks').NLSConfiguration} nlsConfig
|
|
|
|
*/
|
|
|
|
function startup(cachedDataDir, nlsConfig) {
|
|
|
|
nlsConfig._languagePackSupport = true;
|
2018-09-07 12:23:09 +02:00
|
|
|
|
2019-10-09 07:45:56 +02:00
|
|
|
process.env['VSCODE_NLS_CONFIG'] = JSON.stringify(nlsConfig);
|
|
|
|
process.env['VSCODE_NODE_CACHED_DATA_DIR'] = cachedDataDir || '';
|
2018-09-07 12:23:09 +02:00
|
|
|
|
2019-10-09 07:45:56 +02:00
|
|
|
// Load main in AMD
|
|
|
|
perf.mark('willLoadMainBundle');
|
|
|
|
require('./bootstrap-amd').load('vs/code/electron-main/main', () => {
|
|
|
|
perf.mark('didLoadMainBundle');
|
|
|
|
});
|
|
|
|
}
|
2018-09-07 12:23:09 +02:00
|
|
|
|
2019-10-09 07:45:56 +02:00
|
|
|
async function onReady() {
|
|
|
|
perf.mark('main:appReady');
|
2018-09-07 12:23:09 +02:00
|
|
|
|
2019-10-09 07:45:56 +02:00
|
|
|
try {
|
2019-10-10 07:37:06 +02:00
|
|
|
const [cachedDataDir, nlsConfig] = await Promise.all([nodeCachedDataDir.ensureExists(), resolveNlsConfiguration()]);
|
2019-10-09 07:45:56 +02:00
|
|
|
|
2019-10-10 07:37:06 +02:00
|
|
|
startup(cachedDataDir, nlsConfig);
|
2019-10-09 07:45:56 +02:00
|
|
|
} catch (error) {
|
|
|
|
console.error(error);
|
|
|
|
}
|
2018-09-20 09:16:47 +02:00
|
|
|
}
|
2018-09-07 12:23:09 +02:00
|
|
|
|
2018-09-10 07:48:52 +02:00
|
|
|
/**
|
2019-08-09 09:53:57 +02:00
|
|
|
* @typedef {{ [arg: string]: any; '--'?: string[]; _: string[]; }} ParsedArgs
|
2018-09-10 07:48:52 +02:00
|
|
|
*
|
|
|
|
* @param {ParsedArgs} cliArgs
|
|
|
|
*/
|
2019-10-10 07:37:06 +02:00
|
|
|
function configureCommandlineSwitchesSync(cliArgs) {
|
|
|
|
const SUPPORTED_ELECTRON_SWITCHES = [
|
|
|
|
|
|
|
|
// alias from us for --disable-gpu
|
|
|
|
'disable-hardware-acceleration',
|
|
|
|
|
|
|
|
// provided by Electron
|
2020-02-21 08:07:34 +01:00
|
|
|
'disable-color-correct-rendering',
|
|
|
|
|
|
|
|
// override for the color profile to use
|
|
|
|
'force-color-profile'
|
2019-10-10 07:37:06 +02:00
|
|
|
];
|
2020-02-23 16:54:53 +01:00
|
|
|
|
2020-02-14 16:04:33 +01:00
|
|
|
if (process.platform === 'linux') {
|
|
|
|
SUPPORTED_ELECTRON_SWITCHES.push('force-renderer-accessibility');
|
|
|
|
}
|
2018-09-07 12:23:09 +02:00
|
|
|
|
2020-06-12 10:07:17 +02:00
|
|
|
const SUPPORTED_MAIN_PROCESS_SWITCHES = [
|
|
|
|
|
|
|
|
// Persistently enable proposed api via argv.json: https://github.com/microsoft/vscode/issues/99775
|
|
|
|
'enable-proposed-api'
|
|
|
|
];
|
|
|
|
|
2019-10-09 17:15:01 +02:00
|
|
|
// Read argv config
|
2019-10-10 07:37:06 +02:00
|
|
|
const argvConfig = readArgvConfigSync();
|
2019-10-09 17:15:01 +02:00
|
|
|
|
2019-10-10 07:37:06 +02:00
|
|
|
Object.keys(argvConfig).forEach(argvKey => {
|
|
|
|
const argvValue = argvConfig[argvKey];
|
2020-02-21 08:07:34 +01:00
|
|
|
|
2020-06-12 10:07:17 +02:00
|
|
|
// Append Electron flags to Electron
|
|
|
|
if (SUPPORTED_ELECTRON_SWITCHES.indexOf(argvKey) !== -1) {
|
|
|
|
// Color profile
|
|
|
|
if (argvKey === 'force-color-profile') {
|
|
|
|
if (argvValue) {
|
|
|
|
app.commandLine.appendSwitch(argvKey, argvValue);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Others
|
|
|
|
else if (argvValue === true || argvValue === 'true') {
|
|
|
|
if (argvKey === 'disable-hardware-acceleration') {
|
|
|
|
app.disableHardwareAcceleration(); // needs to be called explicitly
|
|
|
|
} else {
|
|
|
|
app.commandLine.appendSwitch(argvKey);
|
|
|
|
}
|
2020-02-21 08:07:34 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-06-12 10:07:17 +02:00
|
|
|
// Append main process flags to process.argv
|
|
|
|
else if (SUPPORTED_MAIN_PROCESS_SWITCHES.indexOf(argvKey) !== -1) {
|
|
|
|
if (argvKey === 'enable-proposed-api') {
|
|
|
|
if (Array.isArray(argvValue)) {
|
|
|
|
argvValue.forEach(id => id && typeof id === 'string' && process.argv.push('--enable-proposed-api', id));
|
|
|
|
} else {
|
|
|
|
console.error(`Unexpected value for \`enable-proposed-api\` in argv.json. Expected array of extension ids.`);
|
|
|
|
}
|
2019-10-09 17:15:01 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
2018-09-07 12:23:09 +02:00
|
|
|
|
|
|
|
// Support JS Flags
|
2019-07-04 18:08:45 +02:00
|
|
|
const jsFlags = getJSFlags(cliArgs);
|
2018-09-07 12:23:09 +02:00
|
|
|
if (jsFlags) {
|
2019-10-09 17:15:01 +02:00
|
|
|
app.commandLine.appendSwitch('js-flags', jsFlags);
|
2018-06-13 17:13:55 +02:00
|
|
|
}
|
2019-10-10 07:37:06 +02:00
|
|
|
|
2020-03-02 12:35:20 +01:00
|
|
|
// TODO@Deepak Electron 7 workaround for https://github.com/microsoft/vscode/issues/88873
|
2020-02-17 10:44:25 +01:00
|
|
|
app.commandLine.appendSwitch('disable-features', 'LayoutNG');
|
|
|
|
|
2019-10-10 07:37:06 +02:00
|
|
|
return argvConfig;
|
2018-06-13 17:13:55 +02:00
|
|
|
}
|
|
|
|
|
2019-10-10 07:37:06 +02:00
|
|
|
function readArgvConfigSync() {
|
2019-10-09 17:15:01 +02:00
|
|
|
|
|
|
|
// Read or create the argv.json config file sync before app('ready')
|
|
|
|
const argvConfigPath = getArgvConfigPath();
|
|
|
|
let argvConfig;
|
|
|
|
try {
|
|
|
|
argvConfig = JSON.parse(stripComments(fs.readFileSync(argvConfigPath).toString()));
|
|
|
|
} catch (error) {
|
|
|
|
if (error && error.code === 'ENOENT') {
|
2019-10-10 07:37:06 +02:00
|
|
|
createDefaultArgvConfigSync(argvConfigPath);
|
2019-10-09 17:15:01 +02:00
|
|
|
} else {
|
|
|
|
console.warn(`Unable to read argv.json configuration file in ${argvConfigPath}, falling back to defaults (${error})`);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Fallback to default
|
|
|
|
if (!argvConfig) {
|
|
|
|
argvConfig = {
|
|
|
|
'disable-color-correct-rendering': true // Force pre-Chrome-60 color profile handling (for https://github.com/Microsoft/vscode/issues/51791)
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
return argvConfig;
|
|
|
|
}
|
|
|
|
|
2019-10-10 07:37:06 +02:00
|
|
|
/**
|
|
|
|
* @param {string} argvConfigPath
|
|
|
|
*/
|
|
|
|
function createDefaultArgvConfigSync(argvConfigPath) {
|
|
|
|
try {
|
|
|
|
|
|
|
|
// Ensure argv config parent exists
|
|
|
|
const argvConfigPathDirname = path.dirname(argvConfigPath);
|
|
|
|
if (!fs.existsSync(argvConfigPathDirname)) {
|
|
|
|
fs.mkdirSync(argvConfigPathDirname);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Migrate over legacy locale
|
|
|
|
const localeConfigPath = path.join(userDataPath, 'User', 'locale.json');
|
|
|
|
const legacyLocale = getLegacyUserDefinedLocaleSync(localeConfigPath);
|
|
|
|
if (legacyLocale) {
|
|
|
|
try {
|
|
|
|
fs.unlinkSync(localeConfigPath);
|
|
|
|
} catch (error) {
|
|
|
|
//ignore
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Default argv content
|
|
|
|
const defaultArgvConfigContent = [
|
2019-10-27 12:08:51 +01:00
|
|
|
'// This configuration file allows you to pass permanent command line arguments to VS Code.',
|
2019-10-10 07:37:06 +02:00
|
|
|
'// Only a subset of arguments is currently supported to reduce the likelyhood of breaking',
|
|
|
|
'// the installation.',
|
|
|
|
'//',
|
|
|
|
'// PLEASE DO NOT CHANGE WITHOUT UNDERSTANDING THE IMPACT',
|
|
|
|
'//',
|
2019-10-27 12:08:51 +01:00
|
|
|
'// NOTE: Changing this file requires a restart of VS Code.',
|
2019-10-10 07:37:06 +02:00
|
|
|
'{',
|
2019-10-29 15:34:03 +01:00
|
|
|
' // Use software rendering instead of hardware accelerated rendering.',
|
|
|
|
' // This can help in cases where you see rendering issues in VS Code.',
|
2019-10-30 07:49:16 +01:00
|
|
|
' // "disable-hardware-acceleration": true,',
|
|
|
|
'',
|
|
|
|
' // Enabled by default by VS Code to resolve color issues in the renderer',
|
|
|
|
' // See https://github.com/Microsoft/vscode/issues/51791 for details',
|
|
|
|
' "disable-color-correct-rendering": true'
|
2019-10-10 07:37:06 +02:00
|
|
|
];
|
|
|
|
|
|
|
|
if (legacyLocale) {
|
|
|
|
defaultArgvConfigContent[defaultArgvConfigContent.length - 1] = `${defaultArgvConfigContent[defaultArgvConfigContent.length - 1]},`; // append trailing ","
|
|
|
|
|
|
|
|
defaultArgvConfigContent.push('');
|
2019-10-27 12:08:51 +01:00
|
|
|
defaultArgvConfigContent.push(' // Display language of VS Code');
|
2019-10-10 07:37:06 +02:00
|
|
|
defaultArgvConfigContent.push(` "locale": "${legacyLocale}"`);
|
|
|
|
}
|
|
|
|
|
|
|
|
defaultArgvConfigContent.push('}');
|
|
|
|
|
|
|
|
// Create initial argv.json with default content
|
|
|
|
fs.writeFileSync(argvConfigPath, defaultArgvConfigContent.join('\n'));
|
|
|
|
} catch (error) {
|
|
|
|
console.error(`Unable to create argv.json configuration file in ${argvConfigPath}, falling back to defaults (${error})`);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-10-09 17:15:01 +02:00
|
|
|
function getArgvConfigPath() {
|
|
|
|
const vscodePortable = process.env['VSCODE_PORTABLE'];
|
|
|
|
if (vscodePortable) {
|
|
|
|
return path.join(vscodePortable, 'argv.json');
|
|
|
|
}
|
|
|
|
|
|
|
|
let dataFolderName = product.dataFolderName;
|
|
|
|
if (process.env['VSCODE_DEV']) {
|
|
|
|
dataFolderName = `${dataFolderName}-dev`;
|
|
|
|
}
|
|
|
|
|
|
|
|
return path.join(os.homedir(), dataFolderName, 'argv.json');
|
|
|
|
}
|
|
|
|
|
2018-09-07 16:05:24 +02:00
|
|
|
/**
|
2018-09-10 07:48:52 +02:00
|
|
|
* @param {ParsedArgs} cliArgs
|
2018-09-07 16:05:24 +02:00
|
|
|
* @returns {string}
|
|
|
|
*/
|
2019-07-04 18:08:45 +02:00
|
|
|
function getJSFlags(cliArgs) {
|
|
|
|
const jsFlags = [];
|
2018-12-17 17:44:53 +01:00
|
|
|
|
|
|
|
// Add any existing JS flags we already got from the command line
|
2018-09-07 12:23:09 +02:00
|
|
|
if (cliArgs['js-flags']) {
|
|
|
|
jsFlags.push(cliArgs['js-flags']);
|
2018-06-27 15:27:32 +02:00
|
|
|
}
|
|
|
|
|
2018-12-17 17:44:53 +01:00
|
|
|
// Support max-memory flag
|
2018-09-07 12:23:09 +02:00
|
|
|
if (cliArgs['max-memory'] && !/max_old_space_size=(\d+)/g.exec(cliArgs['js-flags'])) {
|
|
|
|
jsFlags.push(`--max_old_space_size=${cliArgs['max-memory']}`);
|
2018-06-27 15:27:32 +02:00
|
|
|
}
|
2018-09-07 12:23:09 +02:00
|
|
|
|
|
|
|
return jsFlags.length > 0 ? jsFlags.join(' ') : null;
|
2018-06-27 15:27:32 +02:00
|
|
|
}
|
|
|
|
|
2018-09-07 16:05:24 +02:00
|
|
|
/**
|
2018-09-10 07:48:52 +02:00
|
|
|
* @param {ParsedArgs} cliArgs
|
|
|
|
*
|
2018-09-07 16:05:24 +02:00
|
|
|
* @returns {string}
|
|
|
|
*/
|
2018-09-07 12:23:09 +02:00
|
|
|
function getUserDataPath(cliArgs) {
|
|
|
|
if (portable.isPortable) {
|
|
|
|
return path.join(portable.portableDataPath, 'user-data');
|
|
|
|
}
|
2018-06-18 10:09:50 +02:00
|
|
|
|
2018-09-07 12:23:09 +02:00
|
|
|
return path.resolve(cliArgs['user-data-dir'] || paths.getDefaultUserDataPath(process.platform));
|
2018-06-13 17:13:55 +02:00
|
|
|
}
|
|
|
|
|
2018-09-10 07:48:52 +02:00
|
|
|
/**
|
|
|
|
* @returns {ParsedArgs}
|
|
|
|
*/
|
2018-09-07 12:23:09 +02:00
|
|
|
function parseCLIArgs() {
|
2020-03-30 23:08:14 +02:00
|
|
|
const minimist = require('minimist');
|
2018-09-07 16:05:24 +02:00
|
|
|
|
2018-09-07 12:23:09 +02:00
|
|
|
return minimist(process.argv, {
|
|
|
|
string: [
|
|
|
|
'user-data-dir',
|
|
|
|
'locale',
|
|
|
|
'js-flags',
|
2020-05-07 07:59:29 +02:00
|
|
|
'max-memory',
|
|
|
|
'crash-reporter-directory'
|
2018-09-07 12:23:09 +02:00
|
|
|
]
|
|
|
|
});
|
2018-06-13 17:13:55 +02:00
|
|
|
}
|
|
|
|
|
2018-09-07 12:23:09 +02:00
|
|
|
function setCurrentWorkingDirectory() {
|
|
|
|
try {
|
|
|
|
if (process.platform === 'win32') {
|
2019-05-28 10:56:09 +02:00
|
|
|
process.env['VSCODE_CWD'] = process.cwd(); // remember as environment variable
|
2018-09-07 12:23:09 +02:00
|
|
|
process.chdir(path.dirname(app.getPath('exe'))); // always set application folder as cwd
|
|
|
|
} else if (process.env['VSCODE_CWD']) {
|
|
|
|
process.chdir(process.env['VSCODE_CWD']);
|
2018-01-09 17:04:02 +01:00
|
|
|
}
|
2018-09-07 12:23:09 +02:00
|
|
|
} catch (err) {
|
|
|
|
console.error(err);
|
|
|
|
}
|
|
|
|
}
|
2018-01-09 17:04:02 +01:00
|
|
|
|
2018-09-07 12:23:09 +02:00
|
|
|
function registerListeners() {
|
2018-01-09 17:04:02 +01:00
|
|
|
|
2018-09-10 07:48:52 +02:00
|
|
|
/**
|
2020-02-23 16:54:53 +01:00
|
|
|
* macOS: when someone drops a file to the not-yet running VSCode, the open-file event fires even before
|
2018-09-10 07:48:52 +02:00
|
|
|
* the app-ready event. We listen very early for open-file and remember this upon startup as path to open.
|
|
|
|
*
|
|
|
|
* @type {string[]}
|
|
|
|
*/
|
2018-09-07 16:05:24 +02:00
|
|
|
const macOpenFiles = [];
|
|
|
|
global['macOpenFiles'] = macOpenFiles;
|
2018-09-07 12:23:09 +02:00
|
|
|
app.on('open-file', function (event, path) {
|
2018-09-07 16:05:24 +02:00
|
|
|
macOpenFiles.push(path);
|
2018-09-07 12:23:09 +02:00
|
|
|
});
|
2018-07-23 16:34:16 +02:00
|
|
|
|
2018-09-10 07:48:52 +02:00
|
|
|
/**
|
2020-02-23 16:54:53 +01:00
|
|
|
* macOS: react to open-url requests.
|
2018-09-10 07:48:52 +02:00
|
|
|
*
|
|
|
|
* @type {string[]}
|
|
|
|
*/
|
2018-09-07 12:23:09 +02:00
|
|
|
const openUrls = [];
|
|
|
|
const onOpenUrl = function (event, url) {
|
|
|
|
event.preventDefault();
|
2018-07-23 16:34:16 +02:00
|
|
|
|
2018-09-07 12:23:09 +02:00
|
|
|
openUrls.push(url);
|
|
|
|
};
|
2016-02-09 14:59:11 +01:00
|
|
|
|
2018-09-07 12:23:09 +02:00
|
|
|
app.on('will-finish-launching', function () {
|
|
|
|
app.on('open-url', onOpenUrl);
|
|
|
|
});
|
2016-08-26 16:47:57 +02:00
|
|
|
|
2018-09-07 16:05:24 +02:00
|
|
|
global['getOpenUrls'] = function () {
|
2018-09-07 12:23:09 +02:00
|
|
|
app.removeListener('open-url', onOpenUrl);
|
2018-07-09 15:04:03 +02:00
|
|
|
|
2018-09-07 12:23:09 +02:00
|
|
|
return openUrls;
|
|
|
|
};
|
2018-07-09 15:04:03 +02:00
|
|
|
}
|
|
|
|
|
2018-09-07 16:05:24 +02:00
|
|
|
/**
|
2019-10-09 07:45:56 +02:00
|
|
|
* @returns {{ ensureExists: () => Promise<string | undefined> }}
|
2018-09-07 16:05:24 +02:00
|
|
|
*/
|
2018-09-10 07:48:52 +02:00
|
|
|
function getNodeCachedDir() {
|
2018-09-07 12:23:09 +02:00
|
|
|
return new class {
|
2018-07-09 15:04:03 +02:00
|
|
|
|
2018-09-07 12:23:09 +02:00
|
|
|
constructor() {
|
|
|
|
this.value = this._compute();
|
|
|
|
}
|
|
|
|
|
2019-10-09 07:45:56 +02:00
|
|
|
async ensureExists() {
|
|
|
|
try {
|
|
|
|
await bootstrap.mkdirp(this.value);
|
|
|
|
|
|
|
|
return this.value;
|
|
|
|
} catch (error) {
|
|
|
|
// ignore
|
|
|
|
}
|
2018-09-07 12:23:09 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
_compute() {
|
|
|
|
if (process.argv.indexOf('--no-cached-data') > 0) {
|
|
|
|
return undefined;
|
|
|
|
}
|
2018-09-10 07:48:52 +02:00
|
|
|
|
2018-09-07 12:23:09 +02:00
|
|
|
// IEnvironmentService.isBuilt
|
|
|
|
if (process.env['VSCODE_DEV']) {
|
|
|
|
return undefined;
|
|
|
|
}
|
2018-09-10 07:48:52 +02:00
|
|
|
|
2018-09-07 12:23:09 +02:00
|
|
|
// find commit id
|
2018-09-10 07:48:52 +02:00
|
|
|
const commit = product.commit;
|
2018-09-07 12:23:09 +02:00
|
|
|
if (!commit) {
|
|
|
|
return undefined;
|
|
|
|
}
|
2018-09-10 07:48:52 +02:00
|
|
|
|
2018-09-07 12:23:09 +02:00
|
|
|
return path.join(userDataPath, 'CachedData', commit);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
2018-07-09 15:04:03 +02:00
|
|
|
|
2018-09-07 12:23:09 +02:00
|
|
|
//#region NLS Support
|
2020-06-03 10:38:37 +02:00
|
|
|
|
2019-10-09 07:45:56 +02:00
|
|
|
/**
|
|
|
|
* Resolve the NLS configuration
|
|
|
|
*
|
|
|
|
* @return {Promise<import('./vs/base/node/languagePacks').NLSConfiguration>}
|
|
|
|
*/
|
2019-10-10 07:37:06 +02:00
|
|
|
async function resolveNlsConfiguration() {
|
2019-10-09 07:45:56 +02:00
|
|
|
|
|
|
|
// First, we need to test a user defined locale. If it fails we try the app locale.
|
|
|
|
// If that fails we fall back to English.
|
2019-10-10 07:37:06 +02:00
|
|
|
let nlsConfiguration = nlsConfigurationPromise ? await nlsConfigurationPromise : undefined;
|
2019-10-09 07:45:56 +02:00
|
|
|
if (!nlsConfiguration) {
|
|
|
|
|
|
|
|
// Try to use the app locale. Please note that the app locale is only
|
|
|
|
// valid after we have received the app ready event. This is why the
|
|
|
|
// code is here.
|
|
|
|
let appLocale = app.getLocale();
|
|
|
|
if (!appLocale) {
|
|
|
|
nlsConfiguration = { locale: 'en', availableLanguages: {} };
|
|
|
|
} else {
|
|
|
|
|
|
|
|
// See above the comment about the loader and case sensitiviness
|
|
|
|
appLocale = appLocale.toLowerCase();
|
|
|
|
|
|
|
|
nlsConfiguration = await lp.getNLSConfiguration(product.commit, userDataPath, metaDataFile, appLocale);
|
|
|
|
if (!nlsConfiguration) {
|
|
|
|
nlsConfiguration = { locale: appLocale, availableLanguages: {} };
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// We received a valid nlsConfig from a user defined locale
|
|
|
|
}
|
|
|
|
|
|
|
|
return nlsConfiguration;
|
|
|
|
}
|
|
|
|
|
2018-09-07 16:05:24 +02:00
|
|
|
/**
|
|
|
|
* @param {string} content
|
|
|
|
* @returns {string}
|
|
|
|
*/
|
2016-03-15 16:14:59 +01:00
|
|
|
function stripComments(content) {
|
2019-02-05 23:39:23 +01:00
|
|
|
const regexp = /("(?:[^\\"]*(?:\\.)?)*")|('(?:[^\\']*(?:\\.)?)*')|(\/\*(?:\r?\n|.)*?\*\/)|(\/{2,}.*?(?:(?:\r?\n)|$))/g;
|
2018-09-10 07:48:52 +02:00
|
|
|
|
|
|
|
return content.replace(regexp, function (match, m1, m2, m3, m4) {
|
2016-03-15 16:14:59 +01:00
|
|
|
// Only one of m1, m2, m3, m4 matches
|
|
|
|
if (m3) {
|
|
|
|
// A block comment. Replace with nothing
|
|
|
|
return '';
|
2018-01-25 21:13:24 +01:00
|
|
|
} else if (m4) {
|
2016-03-15 16:14:59 +01:00
|
|
|
// A line comment. If it ends in \r?\n then keep it.
|
2018-09-10 07:48:52 +02:00
|
|
|
const length_1 = m4.length;
|
2016-03-15 16:14:59 +01:00
|
|
|
if (length_1 > 2 && m4[length_1 - 1] === '\n') {
|
|
|
|
return m4[length_1 - 2] === '\r' ? '\r\n' : '\n';
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
return '';
|
|
|
|
}
|
2018-01-25 21:13:24 +01:00
|
|
|
} else {
|
2016-03-15 16:14:59 +01:00
|
|
|
// We match a string
|
|
|
|
return match;
|
|
|
|
}
|
|
|
|
});
|
2016-07-27 09:31:07 +02:00
|
|
|
}
|
2016-03-15 16:14:59 +01:00
|
|
|
|
2018-09-07 16:05:24 +02:00
|
|
|
/**
|
2019-10-09 07:45:56 +02:00
|
|
|
* Language tags are case insensitive however an amd loader is case sensitive
|
|
|
|
* To make this work on case preserving & insensitive FS we do the following:
|
|
|
|
* the language bundles have lower case language tags and we always lower case
|
|
|
|
* the locale we receive from the user or OS.
|
|
|
|
*
|
2019-10-10 07:37:06 +02:00
|
|
|
* @param {{ locale: string | undefined; }} argvConfig
|
|
|
|
* @returns {string | undefined}
|
2018-09-07 16:05:24 +02:00
|
|
|
*/
|
2019-10-10 07:37:06 +02:00
|
|
|
function getUserDefinedLocale(argvConfig) {
|
2018-09-10 07:48:52 +02:00
|
|
|
const locale = args['locale'];
|
2018-01-25 21:13:24 +01:00
|
|
|
if (locale) {
|
2019-10-10 07:37:06 +02:00
|
|
|
return locale.toLowerCase(); // a directly provided --locale always wins
|
2018-01-25 21:13:24 +01:00
|
|
|
}
|
|
|
|
|
2019-10-10 07:37:06 +02:00
|
|
|
return argvConfig.locale && typeof argvConfig.locale === 'string' ? argvConfig.locale.toLowerCase() : undefined;
|
|
|
|
}
|
2019-10-09 07:45:56 +02:00
|
|
|
|
2019-10-10 07:37:06 +02:00
|
|
|
/**
|
|
|
|
* @param {string} localeConfigPath
|
|
|
|
* @returns {string | undefined}
|
|
|
|
*/
|
|
|
|
function getLegacyUserDefinedLocaleSync(localeConfigPath) {
|
2019-10-09 07:45:56 +02:00
|
|
|
try {
|
2019-10-10 07:37:06 +02:00
|
|
|
const content = stripComments(fs.readFileSync(localeConfigPath).toString());
|
2019-10-09 07:45:56 +02:00
|
|
|
|
|
|
|
const value = JSON.parse(content).locale;
|
|
|
|
return value && typeof value === 'string' ? value.toLowerCase() : undefined;
|
|
|
|
} catch (error) {
|
|
|
|
// ignore
|
|
|
|
}
|
2018-01-25 21:13:24 +01:00
|
|
|
}
|
2020-06-03 10:38:37 +02:00
|
|
|
|
2019-05-28 10:56:09 +02:00
|
|
|
//#endregion
|