cli.ts 11.7 KB
Newer Older
J
Joao Moreno 已提交
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.
 *--------------------------------------------------------------------------------------------*/

6
import { spawn, ChildProcess } from 'child_process';
J
Joao Moreno 已提交
7
import { TPromise } from 'vs/base/common/winjs.base';
J
Joao Moreno 已提交
8
import { assign } from 'vs/base/common/objects';
J
Joao Moreno 已提交
9 10
import { parseCLIProcessArgv, buildHelpMessage } from 'vs/platform/environment/node/argv';
import { ParsedArgs } from 'vs/platform/environment/common/environment';
11 12
import product from 'vs/platform/node/product';
import pkg from 'vs/platform/node/package';
13 14
import * as paths from 'path';
import * as os from 'os';
15
import * as fs from 'fs';
B
Benjamin Pasero 已提交
16
import { whenDeleted } from 'vs/base/node/pfs';
17
import { findFreePort } from 'vs/base/node/ports';
18 19
import { resolveTerminalEncoding } from 'vs/base/node/encoding';
import * as iconv from 'iconv-lite';
20
import { writeFileAndFlushSync } from 'vs/base/node/extfs';
21
import { isWindows } from 'vs/base/common/platform';
22

J
Joao Moreno 已提交
23
function shouldSpawnCliProcess(argv: ParsedArgs): boolean {
24 25 26 27
	return !!argv['install-source']
		|| !!argv['list-extensions']
		|| !!argv['install-extension']
		|| !!argv['uninstall-extension'];
J
Joao Moreno 已提交
28 29 30 31 32 33
}

interface IMainCli {
	main: (argv: ParsedArgs) => TPromise<void>;
}

34
export async function main(argv: string[]): TPromise<any> {
J
Joao Moreno 已提交
35
	let args: ParsedArgs;
J
Joao Moreno 已提交
36

J
Joao Moreno 已提交
37 38 39 40 41 42 43
	try {
		args = parseCLIProcessArgv(argv);
	} catch (err) {
		console.error(err.message);
		return TPromise.as(null);
	}

B
Benjamin Pasero 已提交
44
	// Help
J
Joao Moreno 已提交
45
	if (args.help) {
J
Joao Moreno 已提交
46
		console.log(buildHelpMessage(product.nameLong, product.applicationName, pkg.version));
B
Benjamin Pasero 已提交
47 48 49 50
	}

	// Version Info
	else if (args.version) {
51
		console.log(`${pkg.version}\n${product.commit}\n${process.arch}`);
B
Benjamin Pasero 已提交
52 53 54 55
	}

	// Extensions Management
	else if (shouldSpawnCliProcess(args)) {
J
Joao Moreno 已提交
56
		const mainCli = new TPromise<IMainCli>(c => require(['vs/code/node/cliProcessMain'], c));
J
Joao Moreno 已提交
57
		return mainCli.then(cli => cli.main(args));
B
Benjamin Pasero 已提交
58 59
	}

60
	// Write Elevated
61
	else if (args['sudo-write']) {
62 63 64 65 66 67 68 69 70 71
		const source = args._[0];
		const target = args._[1];

		// Validate
		if (
			!source || !target || source === target ||					// make sure source and target are provided and are not the same
			!paths.isAbsolute(source) || !paths.isAbsolute(target) ||	// make sure both source and target are absolute paths
			!fs.existsSync(source) || !fs.statSync(source).isFile() ||	// make sure source exists as file
			!fs.existsSync(target) || !fs.statSync(target).isFile()		// make sure target exists as file
		) {
72
			return TPromise.wrapError(new Error('Using --sudo-write with invalid arguments.'));
73 74 75
		}

		try {
76 77 78 79 80 81 82 83 84 85 86 87 88

			// Check for readonly status and chmod if so if we are told so
			let targetMode: number;
			let restoreMode = false;
			if (!!args['sudo-chmod']) {
				targetMode = fs.statSync(target).mode;
				if (!(targetMode & 128) /* readonly */) {
					fs.chmodSync(target, targetMode | 128);
					restoreMode = true;
				}
			}

			// Write source to target
89 90 91 92 93 94 95 96 97 98 99 100 101 102
			const data = fs.readFileSync(source);
			try {
				writeFileAndFlushSync(target, data);
			} catch (error) {
				// On Windows and if the file exists with an EPERM error, we try a different strategy of saving the file
				// by first truncating the file and then writing with r+ mode. This helps to save hidden files on Windows
				// (see https://github.com/Microsoft/vscode/issues/931)
				if (isWindows && error.code === 'EPERM') {
					fs.truncateSync(target, 0);
					writeFileAndFlushSync(target, data, { flag: 'r+' });
				} else {
					throw error;
				}
			}
103 104 105 106 107

			// Restore previous mode as needed
			if (restoreMode) {
				fs.chmodSync(target, targetMode);
			}
108
		} catch (error) {
109
			return TPromise.wrapError(new Error(`Using --sudo-write resulted in an error: ${error}`));
110 111 112 113 114
		}

		return TPromise.as(null);
	}

B
Benjamin Pasero 已提交
115 116
	// Just Code
	else {
117
		const env = assign({}, process.env, {
118
			'VSCODE_CLI': '1', // this will signal Code that it was spawned from this module
J
Joao Moreno 已提交
119
			'ELECTRON_NO_ATTACH_CONSOLE': '1'
120
		});
B
Benjamin Pasero 已提交
121

B
Benjamin Pasero 已提交
122
		delete env['ELECTRON_RUN_AS_NODE'];
J
Joao Moreno 已提交
123

124
		const processCallbacks: ((child: ChildProcess) => Thenable<any>)[] = [];
125

126
		const verbose = args.verbose || args.status;
B
Benjamin Pasero 已提交
127
		if (verbose) {
B
Benjamin Pasero 已提交
128
			env['ELECTRON_ENABLE_LOGGING'] = '1';
129 130 131 132

			processCallbacks.push(child => {
				child.stdout.on('data', (data: Buffer) => console.log(data.toString('utf8').trim()));
				child.stderr.on('data', (data: Buffer) => console.log(data.toString('utf8').trim()));
133

134 135
				return new TPromise<void>(c => child.once('exit', () => c(null)));
			});
B
Benjamin Pasero 已提交
136 137
		}

138
		let stdinWithoutTty: boolean;
139
		try {
140
			stdinWithoutTty = !process.stdin.isTTY; // Via https://twitter.com/MylesBorins/status/782009479382626304
141 142 143 144 145
		} catch (error) {
			// Windows workaround for https://github.com/nodejs/node/issues/11656
		}

		let stdinFilePath: string;
146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167
		if (stdinWithoutTty) {

			// Read from stdin: we require a single "-" argument to be passed in order to start reading from
			// stdin. We do this because there is no reliable way to find out if data is piped to stdin. Just
			// checking for stdin being connected to a TTY is not enough (https://github.com/Microsoft/vscode/issues/40351)
			if (args._.length === 1 && args._[0] === '-') {

				// remove the "-" argument when we read from stdin
				args._ = [];
				argv = argv.filter(a => a !== '-');

				// prepare temp file to read stdin to
				stdinFilePath = paths.join(os.tmpdir(), `code-stdin-${Math.random().toString(36).replace(/[^a-z]+/g, '').substr(0, 3)}.txt`);

				// open tmp file for writing
				let stdinFileError: Error;
				let stdinFileStream: fs.WriteStream;
				try {
					stdinFileStream = fs.createWriteStream(stdinFilePath);
				} catch (error) {
					stdinFileError = error;
				}
168

169
				if (!stdinFileError) {
170

171 172 173 174 175 176 177 178 179 180 181 182 183 184
					// Pipe into tmp file using terminals encoding
					resolveTerminalEncoding(verbose).done(encoding => {
						const converterStream = iconv.decodeStream(encoding);
						process.stdin.pipe(converterStream).pipe(stdinFileStream);
					});

					// Make sure to open tmp file
					argv.push(stdinFilePath);

					// Enable --wait to get all data and ignore adding this to history
					argv.push('--wait');
					argv.push('--skip-add-to-recently-opened');
					args.wait = true;
				}
185

186 187 188 189 190 191 192
				if (verbose) {
					if (stdinFileError) {
						console.error(`Failed to create file to read via stdin: ${stdinFileError.toString()}`);
					} else {
						console.log(`Reading from stdin via: ${stdinFilePath}`);
					}
				}
193 194
			}

195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218
			// If the user pipes data via stdin but forgot to add the "-" argument, help by printing a message
			// if we detect that data flows into via stdin after a certain timeout.
			else if (args._.length === 0) {
				processCallbacks.push(child => new TPromise(c => {
					const dataListener = () => {
						if (isWindows) {
							console.log(`Run with '${product.applicationName} -' to read output from another program (e.g. 'echo Hello World | ${product.applicationName} -').`);
						} else {
							console.log(`Run with '${product.applicationName} -' to read from stdin (e.g. 'ps aux | grep code | ${product.applicationName} -').`);
						}

						c(void 0);
					};

					// wait for 1s maximum...
					setTimeout(() => {
						process.stdin.removeListener('data', dataListener);

						c(void 0);
					}, 1000);

					// ...but finish early if we detect data
					process.stdin.once('data', dataListener);
				}));
219 220 221
			}
		}

222 223 224 225 226 227 228 229 230
		// If we are started with --wait create a random temporary file
		// and pass it over to the starting instance. We can use this file
		// to wait for it to be deleted to monitor that the edited file
		// is closed and then exit the waiting process.
		let waitMarkerFilePath: string;
		if (args.wait) {
			let waitMarkerError: Error;
			const randomTmpFile = paths.join(os.tmpdir(), Math.random().toString(36).replace(/[^a-z]+/g, '').substr(0, 10));
			try {
231
				fs.writeFileSync(randomTmpFile, '');
232 233 234 235 236 237
				waitMarkerFilePath = randomTmpFile;
				argv.push('--waitMarkerFilePath', waitMarkerFilePath);
			} catch (error) {
				waitMarkerError = error;
			}

B
Benjamin Pasero 已提交
238
			if (verbose) {
239 240 241 242 243 244 245 246
				if (waitMarkerError) {
					console.error(`Failed to create marker file for --wait: ${waitMarkerError.toString()}`);
				} else {
					console.log(`Marker file for --wait created: ${waitMarkerFilePath}`);
				}
			}
		}

247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268
		// If we have been started with `--prof-startup` we need to find free ports to profile
		// the main process, the renderer, and the extension host. We also disable v8 cached data
		// to get better profile traces. Last, we listen on stdout for a signal that tells us to
		// stop profiling.
		if (args['prof-startup']) {
			const portMain = await findFreePort(9222, 10, 6000);
			const portRenderer = await findFreePort(portMain + 1, 10, 6000);
			const portExthost = await findFreePort(portRenderer + 1, 10, 6000);

			if (!portMain || !portRenderer || !portExthost) {
				console.error('Failed to find free ports for profiler to connect to do.');
				return;
			}

			const filenamePrefix = paths.join(os.homedir(), Math.random().toString(16).slice(-4));

			argv.push(`--inspect-brk=${portMain}`);
			argv.push(`--remote-debugging-port=${portRenderer}`);
			argv.push(`--inspect-brk-extensions=${portExthost}`);
			argv.push(`--prof-startup-prefix`, filenamePrefix);
			argv.push(`--no-cached-data`);

269
			fs.writeFileSync(filenamePrefix, argv.slice(-6).join('|'));
270 271 272 273 274 275 276 277 278 279 280 281 282

			processCallbacks.push(async child => {

				// load and start profiler
				const profiler = await import('v8-inspect-profiler');
				const main = await profiler.startProfiling({ port: portMain });
				const renderer = await profiler.startProfiling({ port: portRenderer, tries: 200 });
				const extHost = await profiler.startProfiling({ port: portExthost, tries: 300 });

				// wait for the renderer to delete the
				// marker file
				whenDeleted(filenamePrefix);

283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298
				let profileMain = await main.stop();
				let profileRenderer = await renderer.stop();
				let profileExtHost = await extHost.stop();
				let suffix = '';

				if (!process.env['VSCODE_DEV']) {
					// when running from a not-development-build we remove
					// absolute filenames because we don't want to reveal anything
					// about users. We also append the `.txt` suffix to make it
					// easier to attach these files to GH issues
					profileMain = profiler.rewriteAbsolutePaths(profileMain, 'piiRemoved');
					profileRenderer = profiler.rewriteAbsolutePaths(profileRenderer, 'piiRemoved');
					profileExtHost = profiler.rewriteAbsolutePaths(profileExtHost, 'piiRemoved');
					suffix = '.txt';
				}

299
				// finally stop profiling and save profiles to disk
300 301 302
				await profiler.writeProfile(profileMain, `${filenamePrefix}-main.cpuprofile${suffix}`);
				await profiler.writeProfile(profileRenderer, `${filenamePrefix}-renderer.cpuprofile${suffix}`);
				await profiler.writeProfile(profileExtHost, `${filenamePrefix}-exthost.cpuprofile${suffix}`);
303 304 305
			});
		}

B
Benjamin Pasero 已提交
306
		const options = {
D
Daniel Imms 已提交
307
			detached: true,
308
			env
309
		};
B
Benjamin Pasero 已提交
310

B
Benjamin Pasero 已提交
311
		if (!verbose) {
312 313 314
			options['stdio'] = 'ignore';
		}

J
Joao Moreno 已提交
315
		const child = spawn(process.execPath, argv.slice(2), options);
316

317 318 319 320 321 322 323
		if (args.wait && waitMarkerFilePath) {
			return new TPromise<void>(c => {

				// Complete when process exits
				child.once('exit', () => c(null));

				// Complete when wait marker file is deleted
B
Benjamin Pasero 已提交
324
				whenDeleted(waitMarkerFilePath).done(c, c);
325 326 327 328 329 330
			}).then(() => {

				// Make sure to delete the tmp stdin file if we have any
				if (stdinFilePath) {
					fs.unlinkSync(stdinFilePath);
				}
331 332
			});
		}
333 334

		return TPromise.join(processCallbacks.map(callback => callback(child)));
J
Joao Moreno 已提交
335 336
	}

J
Joao Moreno 已提交
337
	return TPromise.as(null);
J
Joao Moreno 已提交
338 339
}

340 341 342 343
function eventuallyExit(code: number): void {
	setTimeout(() => process.exit(code), 0);
}

J
Joao Moreno 已提交
344
main(process.argv)
345
	.then(() => eventuallyExit(0))
J
Joao Moreno 已提交
346 347
	.then(null, err => {
		console.error(err.stack ? err.stack : err);
348
		eventuallyExit(1);
J
Joao Moreno 已提交
349
	});