mirror of
https://github.com/microsoft/vscode.git
synced 2026-02-21 18:28:50 +00:00
* Fix removeAnsiEscapeCodes for escape sequence after : Fix #209937 * And in debug-server-ready
387 lines
12 KiB
TypeScript
387 lines
12 KiB
TypeScript
/*---------------------------------------------------------------------------------------------
|
|
* Copyright (c) Microsoft Corporation. All rights reserved.
|
|
* Licensed under the MIT License. See License.txt in the project root for license information.
|
|
*--------------------------------------------------------------------------------------------*/
|
|
|
|
import * as vscode from 'vscode';
|
|
import * as util from 'util';
|
|
import { randomUUID } from 'crypto';
|
|
|
|
const PATTERN = 'listening on.* (https?://\\S+|[0-9]+)'; // matches "listening on port 3000" or "Now listening on: https://localhost:5001"
|
|
const URI_PORT_FORMAT = 'http://localhost:%s';
|
|
const URI_FORMAT = '%s';
|
|
const WEB_ROOT = '${workspaceFolder}';
|
|
|
|
interface ServerReadyAction {
|
|
pattern: string;
|
|
action?: 'openExternally' | 'debugWithChrome' | 'debugWithEdge' | 'startDebugging';
|
|
uriFormat?: string;
|
|
webRoot?: string;
|
|
name?: string;
|
|
config?: vscode.DebugConfiguration;
|
|
killOnServerStop?: boolean;
|
|
}
|
|
|
|
// From src/vs/base/common/strings.ts
|
|
const CSI_SEQUENCE = /(?:(?:\x1b\[|\x9B)[=?>!]?[\d;:]*["$#'* ]?[a-zA-Z@^`{}|~])|(:?\x1b\].*?\x07)/g;
|
|
|
|
/**
|
|
* Froms vs/base/common/strings.ts in core
|
|
* @see https://github.com/microsoft/vscode/blob/22a2a0e833175c32a2005b977d7fbd355582e416/src/vs/base/common/strings.ts#L736
|
|
*/
|
|
function removeAnsiEscapeCodes(str: string): string {
|
|
if (str) {
|
|
str = str.replace(CSI_SEQUENCE, '');
|
|
}
|
|
|
|
return str;
|
|
}
|
|
|
|
class Trigger {
|
|
private _fired = false;
|
|
|
|
public get hasFired() {
|
|
return this._fired;
|
|
}
|
|
|
|
public fire() {
|
|
this._fired = true;
|
|
}
|
|
}
|
|
|
|
class ServerReadyDetector extends vscode.Disposable {
|
|
|
|
private static detectors = new Map<vscode.DebugSession, ServerReadyDetector>();
|
|
private static terminalDataListener: vscode.Disposable | undefined;
|
|
|
|
private readonly stoppedEmitter = new vscode.EventEmitter<void>();
|
|
private readonly onDidSessionStop = this.stoppedEmitter.event;
|
|
private readonly disposables = new Set<vscode.Disposable>([]);
|
|
private trigger: Trigger;
|
|
private shellPid?: number;
|
|
private regexp: RegExp;
|
|
|
|
static start(session: vscode.DebugSession): ServerReadyDetector | undefined {
|
|
if (session.configuration.serverReadyAction) {
|
|
let detector = ServerReadyDetector.detectors.get(session);
|
|
if (!detector) {
|
|
detector = new ServerReadyDetector(session);
|
|
ServerReadyDetector.detectors.set(session, detector);
|
|
}
|
|
return detector;
|
|
}
|
|
return undefined;
|
|
}
|
|
|
|
static stop(session: vscode.DebugSession): void {
|
|
const detector = ServerReadyDetector.detectors.get(session);
|
|
if (detector) {
|
|
ServerReadyDetector.detectors.delete(session);
|
|
detector.sessionStopped();
|
|
detector.dispose();
|
|
}
|
|
}
|
|
|
|
static rememberShellPid(session: vscode.DebugSession, pid: number) {
|
|
const detector = ServerReadyDetector.detectors.get(session);
|
|
if (detector) {
|
|
detector.shellPid = pid;
|
|
}
|
|
}
|
|
|
|
static async startListeningTerminalData() {
|
|
if (!this.terminalDataListener) {
|
|
this.terminalDataListener = vscode.window.onDidWriteTerminalData(async e => {
|
|
|
|
// first find the detector with a matching pid
|
|
const pid = await e.terminal.processId;
|
|
const str = removeAnsiEscapeCodes(e.data);
|
|
for (const [, detector] of this.detectors) {
|
|
if (detector.shellPid === pid) {
|
|
detector.detectPattern(str);
|
|
return;
|
|
}
|
|
}
|
|
|
|
// if none found, try all detectors until one matches
|
|
for (const [, detector] of this.detectors) {
|
|
if (detector.detectPattern(str)) {
|
|
return;
|
|
}
|
|
}
|
|
});
|
|
}
|
|
}
|
|
|
|
private constructor(private session: vscode.DebugSession) {
|
|
super(() => this.internalDispose());
|
|
|
|
// Re-used the triggered of the parent session, if one exists
|
|
if (session.parentSession) {
|
|
this.trigger = ServerReadyDetector.start(session.parentSession)?.trigger ?? new Trigger();
|
|
} else {
|
|
this.trigger = new Trigger();
|
|
}
|
|
|
|
this.regexp = new RegExp(session.configuration.serverReadyAction.pattern || PATTERN, 'i');
|
|
}
|
|
|
|
private internalDispose() {
|
|
this.disposables.forEach(d => d.dispose());
|
|
this.disposables.clear();
|
|
}
|
|
|
|
public sessionStopped() {
|
|
this.stoppedEmitter.fire();
|
|
}
|
|
|
|
detectPattern(s: string): boolean {
|
|
if (!this.trigger.hasFired) {
|
|
const matches = this.regexp.exec(s);
|
|
if (matches && matches.length >= 1) {
|
|
this.openExternalWithString(this.session, matches.length > 1 ? matches[1] : '');
|
|
this.trigger.fire();
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
private openExternalWithString(session: vscode.DebugSession, captureString: string) {
|
|
const args: ServerReadyAction = session.configuration.serverReadyAction;
|
|
|
|
let uri;
|
|
if (captureString === '') {
|
|
// nothing captured by reg exp -> use the uriFormat as the target uri without substitution
|
|
// verify that format does not contain '%s'
|
|
const format = args.uriFormat || '';
|
|
if (format.indexOf('%s') >= 0) {
|
|
const errMsg = vscode.l10n.t("Format uri ('{0}') uses a substitution placeholder but pattern did not capture anything.", format);
|
|
vscode.window.showErrorMessage(errMsg, { modal: true }).then(_ => undefined);
|
|
return;
|
|
}
|
|
uri = format;
|
|
} else {
|
|
// if no uriFormat is specified guess the appropriate format based on the captureString
|
|
const format = args.uriFormat || (/^[0-9]+$/.test(captureString) ? URI_PORT_FORMAT : URI_FORMAT);
|
|
// verify that format only contains a single '%s'
|
|
const s = format.split('%s');
|
|
if (s.length !== 2) {
|
|
const errMsg = vscode.l10n.t("Format uri ('{0}') must contain exactly one substitution placeholder.", format);
|
|
vscode.window.showErrorMessage(errMsg, { modal: true }).then(_ => undefined);
|
|
return;
|
|
}
|
|
uri = util.format(format, captureString);
|
|
}
|
|
|
|
this.openExternalWithUri(session, uri);
|
|
}
|
|
|
|
private async openExternalWithUri(session: vscode.DebugSession, uri: string) {
|
|
|
|
const args: ServerReadyAction = session.configuration.serverReadyAction;
|
|
switch (args.action || 'openExternally') {
|
|
|
|
case 'openExternally':
|
|
await vscode.env.openExternal(vscode.Uri.parse(uri));
|
|
break;
|
|
|
|
case 'debugWithChrome':
|
|
await this.debugWithBrowser('pwa-chrome', session, uri);
|
|
break;
|
|
|
|
case 'debugWithEdge':
|
|
await this.debugWithBrowser('pwa-msedge', session, uri);
|
|
break;
|
|
|
|
case 'startDebugging':
|
|
if (args.config) {
|
|
await this.startDebugSession(session, args.config.name, args.config);
|
|
} else {
|
|
await this.startDebugSession(session, args.name || 'unspecified');
|
|
}
|
|
break;
|
|
|
|
default:
|
|
// not supported
|
|
break;
|
|
}
|
|
}
|
|
|
|
private async debugWithBrowser(type: string, session: vscode.DebugSession, uri: string) {
|
|
const args = session.configuration.serverReadyAction as ServerReadyAction;
|
|
if (!args.killOnServerStop) {
|
|
await this.startBrowserDebugSession(type, session, uri);
|
|
return;
|
|
}
|
|
|
|
const trackerId = randomUUID();
|
|
const cts = new vscode.CancellationTokenSource();
|
|
const newSessionPromise = this.catchStartedDebugSession(session => session.configuration._debugServerReadySessionId === trackerId, cts.token);
|
|
|
|
if (!await this.startBrowserDebugSession(type, session, uri, trackerId)) {
|
|
cts.cancel();
|
|
cts.dispose();
|
|
return;
|
|
}
|
|
|
|
const createdSession = await newSessionPromise;
|
|
cts.dispose();
|
|
|
|
if (!createdSession) {
|
|
return;
|
|
}
|
|
|
|
const stopListener = this.onDidSessionStop(async () => {
|
|
stopListener.dispose();
|
|
this.disposables.delete(stopListener);
|
|
await vscode.debug.stopDebugging(createdSession);
|
|
});
|
|
this.disposables.add(stopListener);
|
|
}
|
|
|
|
private startBrowserDebugSession(type: string, session: vscode.DebugSession, uri: string, trackerId?: string) {
|
|
return vscode.debug.startDebugging(session.workspaceFolder, {
|
|
type,
|
|
name: 'Browser Debug',
|
|
request: 'launch',
|
|
url: uri,
|
|
webRoot: session.configuration.serverReadyAction.webRoot || WEB_ROOT,
|
|
_debugServerReadySessionId: trackerId,
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Starts a debug session given a debug configuration name (saved in launch.json) or a debug configuration object.
|
|
*
|
|
* @param session The parent debugSession
|
|
* @param name The name of the configuration to launch. If config it set, it assumes it is the same as config.name.
|
|
* @param config [Optional] Instead of starting a debug session by debug configuration name, use a debug configuration object instead.
|
|
*/
|
|
private async startDebugSession(session: vscode.DebugSession, name: string, config?: vscode.DebugConfiguration) {
|
|
const args = session.configuration.serverReadyAction as ServerReadyAction;
|
|
if (!args.killOnServerStop) {
|
|
await vscode.debug.startDebugging(session.workspaceFolder, config ?? name);
|
|
return;
|
|
}
|
|
|
|
const cts = new vscode.CancellationTokenSource();
|
|
const newSessionPromise = this.catchStartedDebugSession(x => x.name === name, cts.token);
|
|
|
|
if (!await vscode.debug.startDebugging(session.workspaceFolder, config ?? name)) {
|
|
cts.cancel();
|
|
cts.dispose();
|
|
return;
|
|
}
|
|
|
|
const createdSession = await newSessionPromise;
|
|
cts.dispose();
|
|
|
|
if (!createdSession) {
|
|
return;
|
|
}
|
|
|
|
const stopListener = this.onDidSessionStop(async () => {
|
|
stopListener.dispose();
|
|
this.disposables.delete(stopListener);
|
|
await vscode.debug.stopDebugging(createdSession);
|
|
});
|
|
this.disposables.add(stopListener);
|
|
}
|
|
|
|
private catchStartedDebugSession(predicate: (session: vscode.DebugSession) => boolean, cancellationToken: vscode.CancellationToken): Promise<vscode.DebugSession | undefined> {
|
|
return new Promise<vscode.DebugSession | undefined>(_resolve => {
|
|
const done = (value?: vscode.DebugSession) => {
|
|
listener.dispose();
|
|
cancellationListener.dispose();
|
|
this.disposables.delete(listener);
|
|
this.disposables.delete(cancellationListener);
|
|
_resolve(value);
|
|
};
|
|
|
|
const cancellationListener = cancellationToken.onCancellationRequested(done);
|
|
const listener = vscode.debug.onDidStartDebugSession(session => {
|
|
if (predicate(session)) {
|
|
done(session);
|
|
}
|
|
});
|
|
|
|
// In case the debug session of interest was never caught anyhow.
|
|
this.disposables.add(listener);
|
|
this.disposables.add(cancellationListener);
|
|
});
|
|
}
|
|
}
|
|
|
|
export function activate(context: vscode.ExtensionContext) {
|
|
|
|
context.subscriptions.push(vscode.debug.onDidStartDebugSession(session => {
|
|
if (session.configuration.serverReadyAction) {
|
|
const detector = ServerReadyDetector.start(session);
|
|
if (detector) {
|
|
ServerReadyDetector.startListeningTerminalData();
|
|
}
|
|
}
|
|
}));
|
|
|
|
context.subscriptions.push(vscode.debug.onDidTerminateDebugSession(session => {
|
|
ServerReadyDetector.stop(session);
|
|
}));
|
|
|
|
const trackers = new Set<string>();
|
|
|
|
context.subscriptions.push(vscode.debug.registerDebugConfigurationProvider('*', {
|
|
resolveDebugConfigurationWithSubstitutedVariables(_folder: vscode.WorkspaceFolder | undefined, debugConfiguration: vscode.DebugConfiguration) {
|
|
if (debugConfiguration.type && debugConfiguration.serverReadyAction) {
|
|
if (!trackers.has(debugConfiguration.type)) {
|
|
trackers.add(debugConfiguration.type);
|
|
startTrackerForType(context, debugConfiguration.type);
|
|
}
|
|
}
|
|
return debugConfiguration;
|
|
}
|
|
}));
|
|
}
|
|
|
|
function startTrackerForType(context: vscode.ExtensionContext, type: string) {
|
|
|
|
// scan debug console output for a PORT message
|
|
context.subscriptions.push(vscode.debug.registerDebugAdapterTrackerFactory(type, {
|
|
createDebugAdapterTracker(session: vscode.DebugSession) {
|
|
const detector = ServerReadyDetector.start(session);
|
|
if (detector) {
|
|
let runInTerminalRequestSeq: number | undefined;
|
|
return {
|
|
onDidSendMessage: m => {
|
|
if (m.type === 'event' && m.event === 'output' && m.body) {
|
|
switch (m.body.category) {
|
|
case 'console':
|
|
case 'stderr':
|
|
case 'stdout':
|
|
if (m.body.output) {
|
|
detector.detectPattern(m.body.output);
|
|
}
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
if (m.type === 'request' && m.command === 'runInTerminal' && m.arguments) {
|
|
if (m.arguments.kind === 'integrated') {
|
|
runInTerminalRequestSeq = m.seq; // remember this to find matching response
|
|
}
|
|
}
|
|
},
|
|
onWillReceiveMessage: m => {
|
|
if (runInTerminalRequestSeq && m.type === 'response' && m.command === 'runInTerminal' && m.body && runInTerminalRequestSeq === m.request_seq) {
|
|
runInTerminalRequestSeq = undefined;
|
|
ServerReadyDetector.rememberShellPid(session, m.body.shellProcessId);
|
|
}
|
|
}
|
|
};
|
|
}
|
|
return undefined;
|
|
}
|
|
}));
|
|
}
|