microsoft/vscode-react-native

Public

mirrored fromhttps://github.com/microsoft/vscode-react-nativeAvailable

CodeCommitsIssuesPull requestsActionsInsightsSecurity
bb77358c8dc7ea46fae9d6aa601a11fde8eed0fd

Branches

Tags

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

Clone

HTTPS

Download ZIP

src/debugger/launcher.ts

102lines · modepreview

// 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 {TargetPlatformHelper} from "../common/targetPlatformHelper";
import {IRunOptions} from "../common/launchArgs";
import {RemoteExtension} from "../common/remoteExtension";
import {EntryPointHandler, ProcessType} from "../common/entryPointHandler";

export class Launcher {
    private projectRootPath: string;
    private remoteExtension: RemoteExtension;

    constructor(projectRootPath: string) {
        this.projectRootPath = projectRootPath;
        this.remoteExtension = RemoteExtension.atProjectRootPath(this.projectRootPath);
    }

    public launch(): void {
        // Enable telemetry
        new EntryPointHandler(ProcessType.Debugee).runApp("react-native-debug-process", () => this.getAppVersion(),
            ErrorHelper.getInternalError(InternalErrorCode.DebuggingFailed), this.projectRootPath, () => {
                return TelemetryHelper.generate("launch", (generator) => {
                    const resolver = new PlatformResolver();
                    return this.parseRunOptions().then(runOptions => {
                        const mobilePlatform = resolver.resolveMobilePlatform(runOptions.platform, runOptions);
                        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");
                            return Q({})
                                .then(() => {
                                    generator.step("checkPlatformCompatibility");
                                    TargetPlatformHelper.checkTargetPlatformSupport(runOptions.platform);
                                    generator.step("startPackager");
                                    return this.remoteExtension.startPackager();
                                })
                                .then(() => {
                                    let scriptImporter = new ScriptImporter(runOptions.packagerPort, 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 this.remoteExtension.prewarmBundleCache(runOptions.platform);
                                })
                                .then(() => {
                                    generator.step("mobilePlatform.runApp");
                                    Log.logMessage("Building and running application.");
                                    return mobilePlatform.runApp();
                                })
                                .then(() => {
                                    generator.step("Starting App Worker");
                                    Log.logMessage("Starting debugger app worker.");
                                    return new MultipleLifetimesAppWorker(runOptions.packagerPort, sourcesStoragePath, runOptions.debugAdapterPort).start();
                                }) // Start the app worker
                                .then(() => {
                                    generator.step("mobilePlatform.enableJSDebuggingMode");
                                    return mobilePlatform.enableJSDebuggingMode();
                                }).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(): Q.Promise<IRunOptions> {
        // We expect our debugAdapter to pass in arguments as [platform, debugAdapterPort, iosRelativeProjectPath, target?, logCatArguments?];
        return this.remoteExtension.getPackagerPort().then(packagerPort => {
            return {
                projectRoot: this.projectRootPath,
                platform: process.argv[2].toLowerCase(),
                debugAdapterPort: parseInt(process.argv[3], 10) || 9090,
                target: process.argv[5],
                packagerPort: packagerPort,
                iosRelativeProjectPath: process.argv[4],
                logCatArguments: process.argv[6],
            };
        });
    }
}