// Copyright (c) Microsoft Corporation. All rights reserved.
// Licensed under the MIT license. See LICENSE file in the project root for details.
import * as fs from "fs";
import * as path from "path";
import * as Q from "q";
import {MultipleLifetimesAppWorker} from "./appWorker";
import {Log} from "../common/log/log";
import {ErrorHelper} from "../common/error/errorHelper";
import {InternalErrorCode} from "../common/error/internalErrorCode";
import {ScriptImporter} from "./scriptImporter";
import {PlatformResolver} from "./platformResolver";
import {TelemetryHelper} from "../common/telemetryHelper";
import {IRunOptions} from "../common/launchArgs";
import * as em from "../common/extensionMessaging";
import {EntryPointHandler} from "../common/entryPointHandler";
export class Launcher {
private projectRootPath: string;
constructor(projectRootPath: string) {
this.projectRootPath = projectRootPath;
}
public launch(): void {
// Enable telemetry
new EntryPointHandler(true).runApp("react-native-debug-process", () => this.getAppVersion(),
ErrorHelper.getInternalError(InternalErrorCode.DebuggingFailed), () => {
return TelemetryHelper.generate("launch", (generator) => {
const resolver = new PlatformResolver();
const runOptions = this.parseRunOptions();
const mobilePlatform = resolver.resolveMobilePlatform(runOptions.platform);
if (!mobilePlatform) {
throw new RangeError("The target platform could not be read. Did you forget to add it to the launch.json configuration arguments?");
} else {
const sourcesStoragePath = path.join(this.projectRootPath, ".vscode", ".react");
let extensionMessageSender = new em.ExtensionMessageSender();
return Q({})
.then(() => {
generator.step("startPackager");
return extensionMessageSender.sendMessage(em.ExtensionMessage.START_PACKAGER);
})
.then(() => {
let scriptImporter = new ScriptImporter(sourcesStoragePath);
return scriptImporter.downloadDebuggerWorker(sourcesStoragePath).then(() => {
Log.logMessage("Downloaded debuggerWorker.js (Logic to run the React Native app) from the Packager.");
});
})
// We've seen that if we don't prewarm the bundle cache, the app fails on the first attempt to connect to the debugger logic
// and the user needs to Reload JS manually. We prewarm it to prevent that issue
.then(() => {
generator.step("prewarmBundleCache");
Log.logMessage("Prewarming bundle cache. This may take a while ...");
return extensionMessageSender.sendMessage(em.ExtensionMessage.PREWARM_BUNDLE_CACHE, [runOptions.platform]);
})
.then(() => {
generator.step("mobilePlatform.runApp");
Log.logMessage("Building and running application.");
return mobilePlatform.runApp(runOptions);
})
.then(() => {
generator.step("Starting App Worker");
Log.logMessage("Starting debugger app worker.");
return new MultipleLifetimesAppWorker(sourcesStoragePath, runOptions.debugAdapterPort).start();
}) // Start the app worker
.then(() => {
generator.step("mobilePlatform.enableJSDebuggingMode");
return mobilePlatform.enableJSDebuggingMode(runOptions);
}).then(() =>
Log.logMessage("Debugging session started successfully."));
}
});
});
}
private getAppVersion() {
return JSON.parse(fs.readFileSync(path.join(__dirname, "..", "..", "package.json"), "utf-8")).version;
}
/**
* Parses the launch arguments set in the launch configuration.
*/
private parseRunOptions(): IRunOptions {
const result: IRunOptions = { projectRoot: this.projectRootPath };
// We expect our debugAdapter to pass in arguments as [platform, debugAdapterPort, target?];
result.platform = process.argv[2].toLowerCase();
result.debugAdapterPort = parseInt(process.argv[3], 10) || 9090;
result.target = process.argv[4];
result.logCatArguments = process.argv[5];
return result;
}
}microsoft/vscode-react-native
Publicmirrored from https://github.com/microsoft/vscode-react-nativeAvailable
src/debugger/launcher.ts
94lines · modepreview