microsoft/vscode-react-native

Public

mirrored from https://github.com/microsoft/vscode-react-nativeAvailable

CodeCommitsIssuesPull requestsActionsInsightsSecurity
deea9923d45ca1bb2cb7e4468269372b4b0da993

Branches

Tags

  • No tags available.
0Branches0Tags
Go to file
Add file
Code

Clone

HTTPS

Download ZIP

src/common/commandExecutor.ts

213lines · modeblame

a31b007cunknown10 years ago1// Copyright (c) Microsoft Corporation. All rights reserved.
2// Licensed under the MIT license. See LICENSE file in the project root for details.
3
10873e11digeff10 years ago4import * as Q from "q";
af1474acRedMickey6 years ago5import * as path from "path";
2e15926eMeena Kunnathur Balakrishnan10 years ago6import {ChildProcess} from "child_process";
0a68f8dbArtem Egorov8 years ago7import {ILogger} from "../extension/log/LogHelper";
af1474acRedMickey6 years ago8import {NullLogger} from "../extension/log/NullLogger";
e3706a1cRedMickey6 years ago9import {ProjectVersionHelper} from "../common/projectVersionHelper";
b0061ac6Meena Kunnathur Balakrishnan10 years ago10import {Node} from "./node/node";
cb6d0922digeff10 years ago11import {ISpawnResult} from "./node/childProcess";
5c8365a6Artem Egorov8 years ago12import {HostPlatform, HostPlatformId} from "./hostPlatform";
190e393cMeena Kunnathur Balakrishnan10 years ago13import {ErrorHelper} from "./error/errorHelper";
14import {InternalErrorCode} from "./error/internalErrorCode";
e2644f37Yuri Skorokhodov7 years ago15import * as nls from "vscode-nls";
16const localize = nls.loadMessageBundle();
3fb37ad5unknown10 years ago17
831f4a85Patricio Beltran9 years ago18export enum CommandVerbosity {
19OUTPUT,
20SILENT,
21PROGRESS,
22}
23
3fb37ad5unknown10 years ago24interface EnvironmentOptions {
25REACT_DEBUGGER?: string;
26}
27
28interface Options {
29env?: EnvironmentOptions;
831f4a85Patricio Beltran9 years ago30verbosity?: CommandVerbosity;
0d827d9bJimmy Thomson9 years ago31cwd?: string;
3fb37ad5unknown10 years ago32}
33
17161993Meena Kunnathur Balakrishnan10 years ago34export enum CommandStatus {
35Start = 0,
27710197Vladimir Kotikov8 years ago36End = 1,
17161993Meena Kunnathur Balakrishnan10 years ago37}
38
3fb37ad5unknown10 years ago39export class CommandExecutor {
94cd5149Artem Egorov8 years ago40
af1474acRedMickey6 years ago41public static ReactNativeCommand: string | null;
9596aa53digeff10 years ago42private childProcess = new Node.ChildProcess();
3fb37ad5unknown10 years ago43
0a68f8dbArtem Egorov8 years ago44constructor(
45private currentWorkingDirectory: string = process.cwd(),
46private logger: ILogger = new NullLogger()
47) { }
3fb37ad5unknown10 years ago48
fb2bae06Daniel10 years ago49public execute(command: string, options: Options = {}): Q.Promise<void> {
0a68f8dbArtem Egorov8 years ago50this.logger.debug(CommandExecutor.getCommandStatusString(command, CommandStatus.Start));
9596aa53digeff10 years ago51return this.childProcess.execToString(command, { cwd: this.currentWorkingDirectory, env: options.env })
3fb37ad5unknown10 years ago52.then(stdout => {
0a68f8dbArtem Egorov8 years ago53this.logger.info(stdout);
54this.logger.debug(CommandExecutor.getCommandStatusString(command, CommandStatus.End));
3fb37ad5unknown10 years ago55},
cb6d0922digeff10 years ago56(reason: Error) =>
57this.generateRejectionForCommand(command, reason));
3fb37ad5unknown10 years ago58}
59
45944d15Meena Kunnathur Balakrishnan10 years ago60/**
61* Spawns a child process with the params passed
62* This method waits until the spawned process finishes execution
63* {command} - The command to be invoked in the child process
64* {args} - Arguments to be passed to the command
65* {options} - additional options with which the child process needs to be spawned
66*/
323a3cc0Meena Kunnathur Balakrishnan10 years ago67public spawn(command: string, args: string[], options: Options = {}): Q.Promise<any> {
9596aa53digeff10 years ago68return this.spawnChildProcess(command, args, options).outcome;
45944d15Meena Kunnathur Balakrishnan10 years ago69}
70
6126d899Meena Kunnathur Balakrishnan10 years ago71/**
72* Spawns the React Native packager in a child process.
73*/
9596aa53digeff10 years ago74public spawnReactPackager(args: string[], options: Options = {}): ISpawnResult {
75return this.spawnReactCommand("start", args, options);
6126d899Meena Kunnathur Balakrishnan10 years ago76}
77
b8a56999Patricio Beltran9 years ago78public getReactNativeVersion(): Q.Promise<string> {
e3706a1cRedMickey6 years ago79return ProjectVersionHelper.getReactNativeVersions(this.currentWorkingDirectory)
7fa90b3bRedMickey6 years ago80.then(versions => versions.reactNativeVersion);
b8a56999Patricio Beltran9 years ago81}
82
c3a987a7Meena Kunnathur Balakrishnan10 years ago83/**
84* Kills the React Native packager in a child process.
85*/
5c8365a6Artem Egorov8 years ago86public killReactPackager(packagerProcess?: ChildProcess): Q.Promise<void> {
c9b4fa6cMeena Kunnathur Balakrishnan10 years ago87if (packagerProcess) {
a1005420dlebu10 years ago88return Q({}).then(() => {
89if (HostPlatform.getPlatformId() === HostPlatformId.WINDOWS) {
9596aa53digeff10 years ago90return this.childProcess.exec("taskkill /pid " + packagerProcess.pid + " /T /F").outcome;
a1005420dlebu10 years ago91} else {
92packagerProcess.kill();
5c8365a6Artem Egorov8 years ago93return Q.resolve(void 0);
a1005420dlebu10 years ago94}
95}).then(() => {
e2644f37Yuri Skorokhodov7 years ago96this.logger.info(localize("PackagerStopped", "Packager stopped"));
a1005420dlebu10 years ago97});
98
c3a987a7Meena Kunnathur Balakrishnan10 years ago99} else {
e2644f37Yuri Skorokhodov7 years ago100this.logger.warning(localize("PackagerNotFound", "Packager not found"));
10873e11digeff10 years ago101return Q.resolve<void>(void 0);
c3a987a7Meena Kunnathur Balakrishnan10 years ago102}
103}
104
f8d32439dlebu10 years ago105/**
106* Executes a react native command and waits for its completion.
107*/
e390ca0aVladimir Kotikov8 years ago108public spawnReactCommand(command: string, args: string[] = [], options: Options = {}): ISpawnResult {
af1474acRedMickey6 years ago109const reactCommand = HostPlatform.getNpmCliCommand(this.selectReactNativeCLI());
94cd5149Artem Egorov8 years ago110return this.spawnChildProcess(reactCommand, [command, ...args], options);
5b0582f3digeff10 years ago111}
112
831f4a85Patricio Beltran9 years ago113/**
114* Spawns a child process with the params passed
115* This method has logic to do while the command is executing
116* {command} - The command to be invoked in the child process
117* {args} - Arguments to be passed to the command
118* {options} - additional options with which the child process needs to be spawned
119*/
120public spawnWithProgress(command: string, args: string[], options: Options = { verbosity: CommandVerbosity.OUTPUT }): Q.Promise<void> {
121let deferred = Q.defer<void>();
122const spawnOptions = Object.assign({}, { cwd: this.currentWorkingDirectory }, options);
123const commandWithArgs = command + " " + args.join(" ");
124const timeBetweenDots = 1500;
b0af599cJimmy Thomson9 years ago125let lastDotTime = 0;
831f4a85Patricio Beltran9 years ago126
127const printDot = () => {
b0af599cJimmy Thomson9 years ago128const now = Date.now();
129if (now - lastDotTime > timeBetweenDots) {
130lastDotTime = now;
0a68f8dbArtem Egorov8 years ago131this.logger.logStream(".", process.stdout);
b0af599cJimmy Thomson9 years ago132}
831f4a85Patricio Beltran9 years ago133};
134
135if (options.verbosity === CommandVerbosity.OUTPUT) {
0a68f8dbArtem Egorov8 years ago136this.logger.debug(CommandExecutor.getCommandStatusString(commandWithArgs, CommandStatus.Start));
831f4a85Patricio Beltran9 years ago137}
138
139const result = this.childProcess.spawn(command, args, spawnOptions);
140
141result.stdout.on("data", (data: Buffer) => {
142if (options.verbosity === CommandVerbosity.OUTPUT) {
0a68f8dbArtem Egorov8 years ago143this.logger.logStream(data, process.stdout);
b0af599cJimmy Thomson9 years ago144} else if (options.verbosity === CommandVerbosity.PROGRESS) {
831f4a85Patricio Beltran9 years ago145printDot();
146}
147});
148
149result.stderr.on("data", (data: Buffer) => {
150if (options.verbosity === CommandVerbosity.OUTPUT) {
0a68f8dbArtem Egorov8 years ago151this.logger.logStream(data, process.stderr);
b0af599cJimmy Thomson9 years ago152} else if (options.verbosity === CommandVerbosity.PROGRESS) {
831f4a85Patricio Beltran9 years ago153printDot();
154}
155});
156
157result.outcome = result.outcome.then(
158() => {
159if (options.verbosity === CommandVerbosity.OUTPUT) {
0a68f8dbArtem Egorov8 years ago160this.logger.debug(CommandExecutor.getCommandStatusString(commandWithArgs, CommandStatus.End));
831f4a85Patricio Beltran9 years ago161}
0a68f8dbArtem Egorov8 years ago162this.logger.logStream("\n", process.stdout);
831f4a85Patricio Beltran9 years ago163deferred.resolve(void 0);
164},
165reason => {
0d827d9bJimmy Thomson9 years ago166deferred.reject(reason);
831f4a85Patricio Beltran9 years ago167return this.generateRejectionForCommand(commandWithArgs, reason);
168});
169return deferred.promise;
170}
171
af1474acRedMickey6 years ago172public selectReactNativeCLI(): string {
173return CommandExecutor.ReactNativeCommand || path.resolve(this.currentWorkingDirectory, "node_modules", ".bin", "react-native");
174}
175
9596aa53digeff10 years ago176private spawnChildProcess(command: string, args: string[], options: Options = {}): ISpawnResult {
177const spawnOptions = Object.assign({}, { cwd: this.currentWorkingDirectory }, options);
178const commandWithArgs = command + " " + args.join(" ");
3fb37ad5unknown10 years ago179
0a68f8dbArtem Egorov8 years ago180this.logger.debug(CommandExecutor.getCommandStatusString(commandWithArgs, CommandStatus.Start));
9596aa53digeff10 years ago181const result = this.childProcess.spawn(command, args, spawnOptions);
3fb37ad5unknown10 years ago182
183result.stderr.on("data", (data: Buffer) => {
0a68f8dbArtem Egorov8 years ago184this.logger.logStream(data, process.stderr);
3fb37ad5unknown10 years ago185});
186
187result.stdout.on("data", (data: Buffer) => {
0a68f8dbArtem Egorov8 years ago188this.logger.logStream(data, process.stdout);
3fb37ad5unknown10 years ago189});
190
10873e11digeff10 years ago191result.outcome = result.outcome.then(
9596aa53digeff10 years ago192() =>
0a68f8dbArtem Egorov8 years ago193this.logger.debug(CommandExecutor.getCommandStatusString(commandWithArgs, CommandStatus.End)),
10873e11digeff10 years ago194reason =>
4f7c7447Meena Kunnathur Balakrishnan10 years ago195this.generateRejectionForCommand(commandWithArgs, reason));
efa076b0Meena Kunnathur Balakrishnan10 years ago196return result;
3fb37ad5unknown10 years ago197}
10873e11digeff10 years ago198
3677173cdigeff10 years ago199private generateRejectionForCommand(command: string, reason: any): Q.Promise<void> {
4f7c7447Meena Kunnathur Balakrishnan10 years ago200return Q.reject<void>(ErrorHelper.getNestedError(reason, InternalErrorCode.CommandFailed, command));
10873e11digeff10 years ago201}
0a68f8dbArtem Egorov8 years ago202
203private static getCommandStatusString(command: string, status: CommandStatus) {
204switch (status) {
205case CommandStatus.Start:
fc602bb6Yuri Skorokhodov7 years ago206return `Executing command: ${command}`;
0a68f8dbArtem Egorov8 years ago207case CommandStatus.End:
fc602bb6Yuri Skorokhodov7 years ago208return `Finished executing: ${command}`;
0a68f8dbArtem Egorov8 years ago209default:
fc602bb6Yuri Skorokhodov7 years ago210throw ErrorHelper.getInternalError(InternalErrorCode.UnsupportedCommandStatus);
0a68f8dbArtem Egorov8 years ago211}
212}
3fb37ad5unknown10 years ago213}