// Copyright (c) Microsoft Corporation. All rights reserved.
// Licensed under the MIT license. See LICENSE file in the project root for details.
import * as Q from "q";
import * as path from "path";
import * as url from "url";
import * as child_process from "child_process";
import {ScriptImporter, DownloadedScript} from "./scriptImporter";
import { logger } from "vscode-chrome-debug-core";
import { ErrorHelper } from "../common/error/errorHelper";
import { IDebuggeeWorker, RNAppMessage } from "./appWorker";
import { RemoteExtension } from "../common/remoteExtension";
import { InternalErrorCode } from "../common/error/internalErrorCode";
function printDebuggingError(error: Error, reason: any) {
const nestedError = ErrorHelper.getNestedError(error, InternalErrorCode.DebuggingWontWorkReloadJSAndReconnect, reason);
logger.error(nestedError.message);
}
/** This class will run the RN App logic inside a forked Node process. The framework to run the logic is provided by the file
* debuggerWorker.js (designed to run on a WebWorker). We add a couple of tweaks (mostly to polyfill WebWorker API) to that
* file and load it inside of a process.
* On this side we listen to IPC messages and either respond to them or redirect them to packager via MultipleLifetimeAppWorker's
* instance. We also intercept packager's signal to load the bundle's code and mutate the message with path to file we've downloaded
* to let importScripts function take this file.
*/
export class ForkedAppWorker implements IDebuggeeWorker {
protected scriptImporter: ScriptImporter;
protected debuggeeProcess: child_process.ChildProcess | null = null;
/** A deferred that we use to make sure that worker has been loaded completely defore start sending IPC messages */
protected workerLoaded = Q.defer<void>();
private bundleLoaded: Q.Deferred<void>;
private remoteExtension: RemoteExtension;
constructor(
private packagerAddress: string,
private packagerPort: number,
private sourcesStoragePath: string,
private projectRootPath: string,
private postReplyToApp: (message: any) => void,
private packagerRemoteRoot?: string,
private packagerLocalRoot?: string
) {
this.scriptImporter = new ScriptImporter(this.packagerAddress, this.packagerPort, this.sourcesStoragePath, this.packagerRemoteRoot, this.packagerLocalRoot);
this.remoteExtension = RemoteExtension.atProjectRootPath(this.projectRootPath);
this.remoteExtension.api.Debugger.onShowDevMenu(() => {
this.postMessage({
method: "vscode_showDevMenu",
});
});
this.remoteExtension.api.Debugger.onReloadApp(() => {
this.postMessage({
method: "vscode_reloadApp",
});
});
}
public stop() {
if (this.debuggeeProcess) {
logger.verbose(`About to kill debuggee with pid ${this.debuggeeProcess.pid}`);
this.debuggeeProcess.kill();
this.debuggeeProcess = null;
}
}
public start(): Q.Promise<number> {
let scriptToRunPath = path.resolve(this.sourcesStoragePath, ScriptImporter.DEBUGGER_WORKER_FILENAME);
const port = Math.round(Math.random() * 40000 + 3000);
// Note that we set --debug-brk flag to pause the process on the first line - this is
// required for debug adapter to set the breakpoints BEFORE the debuggee has started.
// The adapter will continue execution once it's done with breakpoints.
const nodeArgs = [`--inspect=${port}`, "--debug-brk", scriptToRunPath];
// Start child Node process in debugging mode
// Using fork instead of spawn causes breakage of piping between app worker and VS Code debug console, e.g. console.log() in application
// wouldn't work. Please see https://github.com/Microsoft/vscode-react-native/issues/758
this.debuggeeProcess = child_process.spawn("node", nodeArgs, {
stdio: ["pipe", "pipe", "pipe", "ipc"],
})
.on("message", (message: any) => {
// 'workerLoaded' is a special message that indicates that worker is done with loading.
// We need to wait for it before doing any IPC because process.send doesn't seems to care
// about whether the message has been received or not and the first messages are often get
// discarded by spawned process
if (message && message.workerLoaded) {
this.workerLoaded.resolve(void 0);
return;
}
this.postReplyToApp(message);
})
.on("error", (error: Error) => {
printDebuggingError(ErrorHelper.getInternalError(InternalErrorCode.ReactNativeWorkerProcessThrownAnError), error);
});
// Resolve with port debugger server is listening on
// This will be sent to subscribers of MLAppWorker in "connected" event
logger.verbose(`Spawned debuggee process with pid ${this.debuggeeProcess.pid} listening to ${port}`);
return Q.resolve(port);
}
public postMessage(rnMessage: RNAppMessage): Q.Promise<RNAppMessage> {
// Before sending messages, make sure that the worker is loaded
const promise = this.workerLoaded.promise
.then(() => {
if (rnMessage.method !== "executeApplicationScript") {
// Before sending messages, make sure that the app script executed
if (this.bundleLoaded) {
return this.bundleLoaded.promise.then(() => {
return rnMessage;
});
} else {
return rnMessage;
}
} else {
this.bundleLoaded = Q.defer<void>();
// When packager asks worker to load bundle we download that bundle and
// then set url field to point to that downloaded bundle, so the worker
// will take our modified bundle
if (rnMessage.url) {
const packagerUrl = url.parse(rnMessage.url);
packagerUrl.host = `${this.packagerAddress}:${this.packagerPort}`;
rnMessage = {
...rnMessage,
url: url.format(packagerUrl),
};
logger.verbose(`Packager requested runtime to load script from ${rnMessage.url}`);
return this.scriptImporter.downloadAppScript(<string>rnMessage.url, this.projectRootPath)
.then((downloadedScript: DownloadedScript) => {
this.bundleLoaded.resolve(void 0);
return Object.assign({}, rnMessage, { url: downloadedScript.filepath });
});
} else {
throw ErrorHelper.getInternalError(InternalErrorCode.RNMessageWithMethodExecuteApplicationScriptDoesntHaveURLProperty);
}
}
});
promise.done(
(message: RNAppMessage) => {
if (this.debuggeeProcess) {
this.debuggeeProcess.send({ data: message });
}
},
(reason) => printDebuggingError(ErrorHelper.getInternalError(InternalErrorCode.CouldntImportScriptAt, rnMessage.url), reason));
return promise;
}
}microsoft/vscode-react-native
Publicmirrored fromhttps://github.com/microsoft/vscode-react-nativeAvailable
src/debugger/forkedAppWorker.ts
155lines · modepreview