microsoft/vscode-react-native

Public

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

CodeCommitsIssuesPull requestsActionsInsightsSecurity
dev/v-zhenyuan/update-parameters

Branches

Tags

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

Clone

HTTPS

Download ZIP

src/debugger/debugSessionBase.ts

335lines · modecode

1// Copyright (c) Microsoft Corporation. All rights reserved.
2// Licensed under the MIT license. See LICENSE file in the project root for details.
3
4import * as path from "path";
5import * as fs from "fs";
6import * as vscode from "vscode";
7import { LoggingDebugSession, Logger, logger, ErrorDestination } from "@vscode/debugadapter";
8import { DebugProtocol } from "vscode-debugprotocol";
9import * as nls from "vscode-nls";
10import { stripJsonTrailingComma } from "../common/utils";
11import { getLoggingDirectory, LogHelper, LogLevel } from "../extension/log/LogHelper";
12import { ReactNativeProjectHelper } from "../common/reactNativeProjectHelper";
13import { ErrorHelper } from "../common/error/errorHelper";
14import { InternalErrorCode } from "../common/error/internalErrorCode";
15import { InternalError, NestedError } from "../common/error/internalError";
16import { ILaunchArgs, IRunOptions, PlatformType } from "../extension/launchArgs";
17import { AppLauncher } from "../extension/appLauncher";
18import { RNPackageVersions } from "../common/projectVersionHelper";
19import { SettingsHelper } from "../extension/settingsHelper";
20import { OutputChannelLogger } from "../extension/log/OutputChannelLogger";
21import { DeviceStatusIndicator } from "../extension/deviceStatusIndicator";
22import { RNSession } from "./debugSessionWrapper";
23
24nls.config({
25 messageFormat: nls.MessageFormat.bundle,
26 bundleFormat: nls.BundleFormat.standalone,
27})();
28const localize = nls.loadMessageBundle();
29
30/**
31 * Enum of possible statuses of debug session
32 */
33export enum DebugSessionStatus {
34 /** A session has been just created */
35 FirstConnection,
36 /** This status is required in order to exclude the possible creation of several debug sessions at the first start */
37 FirstConnectionPending,
38 /** This status means that an application can be reloaded */
39 ConnectionAllowed,
40 /** This status means that an application is reloading now, and we shouldn't terminate the current debug session */
41 ConnectionPending,
42 /** A debuggee connected successfully */
43 ConnectionDone,
44 /** A debuggee failed to connect */
45 ConnectionFailed,
46 /** The session is handling disconnect request now */
47 Stopping,
48 /** The session is stopped */
49 Stopped,
50}
51
52export interface TerminateEventArgs {
53 debugSession: vscode.DebugSession;
54 args: any;
55}
56
57export interface IAttachRequestArgs
58 extends DebugProtocol.AttachRequestArguments,
59 IRunOptions,
60 vscode.DebugConfiguration {
61 webkitRangeMax: number;
62 webkitRangeMin: number;
63 cwd: string /* Automatically set by VS Code to the currently opened folder */;
64 port: number;
65 url?: string;
66 useHermesEngine: boolean;
67 address?: string;
68 trace?: string;
69 skipFiles?: [];
70 sourceMaps?: boolean;
71 sourceMapPathOverrides?: { [key: string]: string };
72 jsDebugTrace?: boolean;
73 browserTarget?: string;
74}
75
76export interface ILaunchRequestArgs
77 extends DebugProtocol.LaunchRequestArguments,
78 IAttachRequestArgs {}
79
80export abstract class DebugSessionBase extends LoggingDebugSession {
81 protected static rootSessionTerminatedEventEmitter: vscode.EventEmitter<TerminateEventArgs> =
82 new vscode.EventEmitter<TerminateEventArgs>();
83 public static readonly onDidTerminateRootDebugSession =
84 DebugSessionBase.rootSessionTerminatedEventEmitter.event;
85
86 protected readonly stopCommand: string;
87 protected readonly terminateCommand: string;
88 protected readonly pwaNodeSessionName: string;
89
90 protected appLauncher!: AppLauncher;
91 protected projectRootPath!: string;
92 protected isSettingsInitialized: boolean; // used to prevent parameters reinitialization when attach is called from launch function
93 protected previousAttachArgs!: IAttachRequestArgs;
94 protected cdpProxyLogLevel!: LogLevel;
95 protected debugSessionStatus: DebugSessionStatus;
96 protected nodeSession: vscode.DebugSession | null;
97 protected rnSession: RNSession;
98 protected vsCodeDebugSession: vscode.DebugSession;
99 protected cancellationTokenSource: vscode.CancellationTokenSource;
100
101 constructor(rnSession: RNSession) {
102 super();
103
104 // constants definition
105 this.pwaNodeSessionName = "pwa-node"; // the name of node debug session created by js-debug extension
106 this.stopCommand = "workbench.action.debug.stop"; // the command which simulates a click on the "Stop" button
107 this.terminateCommand = "terminate"; // the "terminate" command is sent from the client to the debug adapter in order to give the debuggee a chance for terminating itself
108
109 // variables definition
110 this.rnSession = rnSession;
111 this.vsCodeDebugSession = rnSession.vsCodeDebugSession;
112 this.isSettingsInitialized = false;
113 this.debugSessionStatus = DebugSessionStatus.FirstConnection;
114 this.cancellationTokenSource = new vscode.CancellationTokenSource();
115 this.nodeSession = null;
116 }
117
118 protected initializeRequest(
119 response: DebugProtocol.InitializeResponse,
120 // eslint-disable-next-line @typescript-eslint/no-unused-vars
121 args: DebugProtocol.InitializeRequestArguments,
122 ): void {
123 response.body = response.body || {};
124
125 response.body.supportsConfigurationDoneRequest = true;
126 response.body.supportsEvaluateForHovers = true;
127 response.body.supportTerminateDebuggee = true;
128 response.body.supportsCancelRequest = true;
129
130 response.body.exceptionBreakpointFilters = [
131 {
132 filter: "all",
133 label: "Caught Exceptions",
134 default: false,
135 supportsCondition: true,
136 description: "Breaks on all throw errors, even if they're caught later.",
137 // eslint-disable-next-line @typescript-eslint/quotes
138 conditionDescription: 'error.name == "MyError"',
139 },
140 {
141 filter: "uncaught",
142 label: "Uncaught Exceptions",
143 default: false,
144 supportsCondition: true,
145 description: "Breaks only on errors or promise rejections that are not handled.",
146 // eslint-disable-next-line @typescript-eslint/quotes
147 conditionDescription: 'error.name == "MyError"',
148 },
149 ];
150
151 this.sendResponse(response);
152 }
153
154 protected abstract establishDebugSession(
155 attachArgs: IAttachRequestArgs,
156 resolve?: (value?: void | PromiseLike<void> | undefined) => void,
157 ): void;
158
159 protected async initializeSettings(args: any): Promise<void> {
160 if (!this.isSettingsInitialized) {
161 let chromeDebugCoreLogs = getLoggingDirectory();
162 if (chromeDebugCoreLogs) {
163 chromeDebugCoreLogs = path.join(chromeDebugCoreLogs, "DebugSessionLogs.txt");
164 }
165 let logLevel: string = args.trace;
166 if (logLevel) {
167 logLevel = logLevel.replace(logLevel[0], logLevel[0].toUpperCase());
168 logger.setup((Logger.LogLevel as any)[logLevel], chromeDebugCoreLogs || false);
169 this.cdpProxyLogLevel =
170 (LogLevel as any)[logLevel] === LogLevel.Verbose
171 ? LogLevel.Custom
172 : LogLevel.None;
173 } else {
174 logger.setup(Logger.LogLevel.Log, chromeDebugCoreLogs || false);
175 this.cdpProxyLogLevel =
176 LogHelper.LOG_LEVEL === LogLevel.Trace ? LogLevel.Custom : LogLevel.None;
177 }
178
179 if (typeof args.sourceMaps !== "boolean") {
180 args.sourceMaps = true;
181 }
182
183 if (typeof args.enableDebug !== "boolean") {
184 args.enableDebug = true;
185 }
186
187 // Now there is a problem with processing time of 'createFromSourceMap' function of js-debug
188 // So we disable this functionality by default https://github.com/microsoft/vscode-js-debug/issues/1033
189 if (typeof args.sourceMapRenames !== "boolean") {
190 args.sourceMapRenames = false;
191 }
192
193 const projectRootPath = SettingsHelper.getReactNativeProjectRoot(args.cwd);
194 const isReactProject = await ReactNativeProjectHelper.isReactNativeProject(
195 projectRootPath,
196 );
197 if (!isReactProject) {
198 throw ErrorHelper.getInternalError(InternalErrorCode.NotInReactNativeFolderError);
199 }
200
201 const appLauncher = await AppLauncher.getOrCreateAppLauncherByProjectRootPath(
202 projectRootPath,
203 );
204 this.appLauncher = appLauncher;
205 this.projectRootPath = projectRootPath;
206 this.isSettingsInitialized = true;
207 this.appLauncher.getOrUpdateNodeModulesRoot(true);
208 if (this.vsCodeDebugSession.workspaceFolder) {
209 this.appLauncher.updateDebugConfigurationRoot(
210 this.vsCodeDebugSession.workspaceFolder.uri.fsPath,
211 );
212 }
213 const settingsPort = this.appLauncher.getPackagerPort(projectRootPath);
214 if (this.appLauncher.getPackager().getPort() != settingsPort) {
215 this.appLauncher.getPackager().resetToSettingsPort();
216 }
217 }
218 }
219
220 protected async disconnectRequest(
221 response: DebugProtocol.DisconnectResponse,
222 args: DebugProtocol.DisconnectArguments,
223 // eslint-disable-next-line @typescript-eslint/no-unused-vars
224 request?: DebugProtocol.Request,
225 ): Promise<void> {
226 if (this.appLauncher) {
227 await this.appLauncher.getRnCdpProxy().stopServer();
228 }
229
230 DeviceStatusIndicator.hide();
231
232 this.cancellationTokenSource.cancel();
233 this.cancellationTokenSource.dispose();
234
235 // Then we tell the extension to stop monitoring the logcat, and then we disconnect the debugging session
236 if (this.previousAttachArgs && this.previousAttachArgs.platform === PlatformType.Android) {
237 try {
238 this.appLauncher.getMobilePlatform().dispose();
239 } catch (err) {
240 logger.warn(
241 localize(
242 "CouldNotStopMonitoringLogcat",
243 "Couldn't stop monitoring logcat: {0}",
244 err instanceof Error ? err.message : String(err),
245 ),
246 );
247 }
248 }
249
250 this.debugSessionStatus = DebugSessionStatus.Stopped;
251 await logger.dispose();
252
253 DebugSessionBase.rootSessionTerminatedEventEmitter.fire({
254 debugSession: this.vsCodeDebugSession,
255 args: {
256 forcedStop: !!(<any>args).forcedStop,
257 },
258 });
259
260 this.sendResponse(response);
261 }
262
263 protected terminateWithErrorResponse(error: Error, response: DebugProtocol.Response): void {
264 // We can't print error messages after the debugging session is stopped. This could break the extension work.
265 if (
266 (error instanceof InternalError || error instanceof NestedError) &&
267 error.errorCode === InternalErrorCode.CancellationTokenTriggered
268 ) {
269 return;
270 }
271
272 logger.error(error.message);
273
274 this.sendErrorResponse(
275 response,
276 { format: error.message, id: 1 },
277 undefined,
278 undefined,
279 ErrorDestination.User,
280 );
281 }
282
283 protected async preparePackagerBeforeAttach(
284 args: IAttachRequestArgs,
285 reactNativeVersions: RNPackageVersions,
286 ): Promise<void> {
287 if (!(await this.appLauncher.getPackager().isRunning())) {
288 const runOptions: ILaunchArgs = Object.assign(
289 { reactNativeVersions },
290 this.appLauncher.prepareBaseRunOptions(args),
291 );
292 this.appLauncher.getPackager().setRunOptions(runOptions);
293 await this.appLauncher.getPackager().start();
294 }
295 }
296
297 protected showError(error: Error): void {
298 void vscode.window.showErrorMessage(error.message, {
299 modal: true,
300 });
301 // We can't print error messages via debug session logger after the session is stopped. This could break the extension work.
302 if (this.debugSessionStatus === DebugSessionStatus.Stopped) {
303 OutputChannelLogger.getMainChannel().error(error.message);
304 return;
305 }
306 logger.error(error.message);
307 }
308
309 protected async terminate(): Promise<void> {
310 await vscode.commands.executeCommand(this.stopCommand, undefined, {
311 sessionId: this.vsCodeDebugSession.id,
312 });
313 }
314}
315
316/**
317 * Parses settings.json file for workspace root property
318 */
319export function getProjectRoot(args: any): string {
320 const vsCodeRoot = args.cwd ? path.resolve(args.cwd) : path.resolve(args.program, "../..");
321 const settingsPath = path.resolve(vsCodeRoot, ".vscode/settings.json");
322 try {
323 const settingsContent = fs.readFileSync(settingsPath, "utf8");
324 const parsedSettings = stripJsonTrailingComma(settingsContent);
325 const projectRootPath =
326 parsedSettings["react-native-tools.projectRoot"] ||
327 parsedSettings["react-native-tools"].projectRoot;
328 return path.resolve(vsCodeRoot, projectRootPath);
329 } catch (e) {
330 logger.verbose(
331 `${settingsPath} file doesn't exist or its content is incorrect. This file will be ignored.`,
332 );
333 return args.cwd ? path.resolve(args.cwd) : path.resolve(args.program, "../..");
334 }
335}
336