@theia/debug 1.53.0-next.5 → 1.53.0-next.55
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +62 -62
- package/lib/browser/debug-configuration-manager.js +6 -6
- package/lib/browser/debug-frontend-application-contribution.d.ts.map +1 -1
- package/lib/browser/debug-frontend-application-contribution.js.map +1 -1
- package/lib/common/inline-debug-adapter.d.ts +1 -0
- package/lib/common/inline-debug-adapter.d.ts.map +1 -1
- package/package.json +16 -16
- package/src/browser/breakpoint/breakpoint-manager.ts +369 -369
- package/src/browser/breakpoint/breakpoint-marker.ts +104 -104
- package/src/browser/console/debug-console-contribution.tsx +240 -240
- package/src/browser/console/debug-console-items.tsx +384 -384
- package/src/browser/console/debug-console-session.ts +205 -205
- package/src/browser/debug-call-stack-item-type-key.ts +20 -20
- package/src/browser/debug-configuration-manager.ts +591 -591
- package/src/browser/debug-configuration-model.ts +100 -100
- package/src/browser/debug-contribution.ts +43 -43
- package/src/browser/debug-frontend-application-contribution.ts +1551 -1551
- package/src/browser/debug-frontend-module.ts +133 -133
- package/src/browser/debug-package.spec.ts +20 -20
- package/src/browser/debug-preferences.ts +98 -98
- package/src/browser/debug-prefix-configuration.ts +195 -195
- package/src/browser/debug-resource.ts +59 -59
- package/src/browser/debug-schema-updater.ts +149 -149
- package/src/browser/debug-session-connection.ts +357 -357
- package/src/browser/debug-session-contribution.ts +157 -157
- package/src/browser/debug-session-manager.ts +683 -683
- package/src/browser/debug-session-options.ts +120 -120
- package/src/browser/debug-session.tsx +974 -974
- package/src/browser/debug-tab-bar-decorator.ts +57 -57
- package/src/browser/debug-watch-manager.ts +93 -93
- package/src/browser/disassembly-view/disassembly-view-accessibility-provider.ts +43 -43
- package/src/browser/disassembly-view/disassembly-view-breakpoint-renderer.ts +119 -119
- package/src/browser/disassembly-view/disassembly-view-contribution.ts +109 -109
- package/src/browser/disassembly-view/disassembly-view-instruction-renderer.ts +245 -245
- package/src/browser/disassembly-view/disassembly-view-table-delegate.ts +39 -39
- package/src/browser/disassembly-view/disassembly-view-utilities.ts +55 -55
- package/src/browser/disassembly-view/disassembly-view-widget.ts +463 -463
- package/src/browser/editor/debug-breakpoint-widget.tsx +293 -293
- package/src/browser/editor/debug-editor-model.ts +529 -529
- package/src/browser/editor/debug-editor-service.ts +192 -192
- package/src/browser/editor/debug-editor.ts +20 -20
- package/src/browser/editor/debug-exception-widget.tsx +122 -122
- package/src/browser/editor/debug-expression-provider.ts +78 -78
- package/src/browser/editor/debug-hover-source.tsx +105 -105
- package/src/browser/editor/debug-hover-widget.ts +298 -298
- package/src/browser/editor/debug-inline-value-decorator.ts +373 -373
- package/src/browser/model/debug-breakpoint.tsx +151 -151
- package/src/browser/model/debug-function-breakpoint.tsx +101 -101
- package/src/browser/model/debug-instruction-breakpoint.tsx +68 -68
- package/src/browser/model/debug-source-breakpoint.tsx +237 -237
- package/src/browser/model/debug-source.ts +93 -93
- package/src/browser/model/debug-stack-frame.tsx +177 -177
- package/src/browser/model/debug-thread.tsx +292 -292
- package/src/browser/preferences/launch-preferences.ts +38 -38
- package/src/browser/style/index.css +453 -453
- package/src/browser/view/debug-action.tsx +57 -57
- package/src/browser/view/debug-breakpoints-source.tsx +53 -53
- package/src/browser/view/debug-breakpoints-widget.ts +71 -71
- package/src/browser/view/debug-configuration-select.tsx +269 -269
- package/src/browser/view/debug-configuration-widget.tsx +121 -121
- package/src/browser/view/debug-exception-breakpoint.tsx +68 -68
- package/src/browser/view/debug-session-widget.ts +124 -124
- package/src/browser/view/debug-stack-frames-source.tsx +75 -75
- package/src/browser/view/debug-stack-frames-widget.ts +135 -135
- package/src/browser/view/debug-threads-source.tsx +48 -48
- package/src/browser/view/debug-threads-widget.ts +126 -126
- package/src/browser/view/debug-toolbar-widget.tsx +145 -145
- package/src/browser/view/debug-variables-source.ts +43 -43
- package/src/browser/view/debug-variables-widget.ts +61 -61
- package/src/browser/view/debug-view-model.ts +230 -230
- package/src/browser/view/debug-watch-expression.tsx +88 -88
- package/src/browser/view/debug-watch-source.ts +41 -41
- package/src/browser/view/debug-watch-widget.ts +61 -61
- package/src/browser/view/debug-widget.ts +97 -97
- package/src/common/debug-adapter-contribution-registry.ts +206 -206
- package/src/common/debug-adapter-session.ts +102 -102
- package/src/common/debug-common.ts +19 -19
- package/src/common/debug-compound.ts +33 -33
- package/src/common/debug-configuration.ts +112 -112
- package/src/common/debug-model.ts +200 -200
- package/src/common/debug-service.ts +184 -184
- package/src/common/debug-uri-utils.ts +24 -24
- package/src/common/inline-debug-adapter.ts +47 -47
- package/src/node/debug-adapter-factory.ts +107 -107
- package/src/node/debug-adapter-session-manager.ts +106 -106
- package/src/node/debug-backend-module.ts +57 -57
- package/src/node/debug-service-impl.ts +119 -119
- package/src/node/stream-debug-adapter.ts +126 -126
|
@@ -1,292 +1,292 @@
|
|
|
1
|
-
// *****************************************************************************
|
|
2
|
-
// Copyright (C) 2018 TypeFox and others.
|
|
3
|
-
//
|
|
4
|
-
// This program and the accompanying materials are made available under the
|
|
5
|
-
// terms of the Eclipse Public License v. 2.0 which is available at
|
|
6
|
-
// http://www.eclipse.org/legal/epl-2.0.
|
|
7
|
-
//
|
|
8
|
-
// This Source Code may also be made available under the following Secondary
|
|
9
|
-
// Licenses when the conditions for such availability set forth in the Eclipse
|
|
10
|
-
// Public License v. 2.0 are satisfied: GNU General Public License, version 2
|
|
11
|
-
// with the GNU Classpath Exception which is available at
|
|
12
|
-
// https://www.gnu.org/software/classpath/license.html.
|
|
13
|
-
//
|
|
14
|
-
// SPDX-License-Identifier: EPL-2.0 OR GPL-2.0-only WITH Classpath-exception-2.0
|
|
15
|
-
// *****************************************************************************
|
|
16
|
-
|
|
17
|
-
import * as React from '@theia/core/shared/react';
|
|
18
|
-
import { CancellationTokenSource, Emitter, Event, nls } from '@theia/core';
|
|
19
|
-
import { DebugProtocol } from '@vscode/debugprotocol/lib/debugProtocol';
|
|
20
|
-
import { TreeElement } from '@theia/core/lib/browser/source-tree';
|
|
21
|
-
import { DebugStackFrame } from './debug-stack-frame';
|
|
22
|
-
import { DebugSession } from '../debug-session';
|
|
23
|
-
|
|
24
|
-
export type StoppedDetails = DebugProtocol.StoppedEvent['body'] & {
|
|
25
|
-
framesErrorMessage?: string
|
|
26
|
-
totalFrames?: number
|
|
27
|
-
};
|
|
28
|
-
|
|
29
|
-
export class DebugThreadData {
|
|
30
|
-
readonly raw: DebugProtocol.Thread;
|
|
31
|
-
readonly stoppedDetails: StoppedDetails | undefined;
|
|
32
|
-
}
|
|
33
|
-
|
|
34
|
-
export interface DebugExceptionInfo {
|
|
35
|
-
id?: string
|
|
36
|
-
description?: string
|
|
37
|
-
details?: DebugProtocol.ExceptionDetails
|
|
38
|
-
}
|
|
39
|
-
|
|
40
|
-
export class DebugThread extends DebugThreadData implements TreeElement {
|
|
41
|
-
|
|
42
|
-
protected readonly onDidChangedEmitter = new Emitter<void>();
|
|
43
|
-
readonly onDidChanged: Event<void> = this.onDidChangedEmitter.event;
|
|
44
|
-
protected readonly onDidFocusStackFrameEmitter = new Emitter<DebugStackFrame | undefined>();
|
|
45
|
-
get onDidFocusStackFrame(): Event<DebugStackFrame | undefined> {
|
|
46
|
-
return this.onDidFocusStackFrameEmitter.event;
|
|
47
|
-
}
|
|
48
|
-
|
|
49
|
-
constructor(
|
|
50
|
-
readonly session: DebugSession
|
|
51
|
-
) {
|
|
52
|
-
super();
|
|
53
|
-
}
|
|
54
|
-
|
|
55
|
-
get id(): string {
|
|
56
|
-
return this.session.id + ':' + this.raw.id;
|
|
57
|
-
}
|
|
58
|
-
|
|
59
|
-
get threadId(): number {
|
|
60
|
-
return this.raw.id;
|
|
61
|
-
}
|
|
62
|
-
|
|
63
|
-
protected _currentFrame: DebugStackFrame | undefined;
|
|
64
|
-
get currentFrame(): DebugStackFrame | undefined {
|
|
65
|
-
return this._currentFrame;
|
|
66
|
-
}
|
|
67
|
-
set currentFrame(frame: DebugStackFrame | undefined) {
|
|
68
|
-
if (this._currentFrame?.id === frame?.id) {
|
|
69
|
-
return;
|
|
70
|
-
}
|
|
71
|
-
this._currentFrame = frame;
|
|
72
|
-
this.onDidChangedEmitter.fire(undefined);
|
|
73
|
-
this.onDidFocusStackFrameEmitter.fire(frame);
|
|
74
|
-
}
|
|
75
|
-
|
|
76
|
-
get stopped(): boolean {
|
|
77
|
-
return !!this.stoppedDetails;
|
|
78
|
-
}
|
|
79
|
-
|
|
80
|
-
update(data: Partial<DebugThreadData>): void {
|
|
81
|
-
Object.assign(this, data);
|
|
82
|
-
if ('stoppedDetails' in data) {
|
|
83
|
-
this.clearFrames();
|
|
84
|
-
}
|
|
85
|
-
}
|
|
86
|
-
|
|
87
|
-
clear(): void {
|
|
88
|
-
this.update({
|
|
89
|
-
raw: this.raw,
|
|
90
|
-
stoppedDetails: undefined
|
|
91
|
-
});
|
|
92
|
-
}
|
|
93
|
-
|
|
94
|
-
continue(): Promise<DebugProtocol.ContinueResponse> {
|
|
95
|
-
return this.session.sendRequest('continue', this.toArgs());
|
|
96
|
-
}
|
|
97
|
-
|
|
98
|
-
stepOver(): Promise<DebugProtocol.NextResponse> {
|
|
99
|
-
return this.session.sendRequest('next', this.toArgs());
|
|
100
|
-
}
|
|
101
|
-
|
|
102
|
-
stepIn(): Promise<DebugProtocol.StepInResponse> {
|
|
103
|
-
return this.session.sendRequest('stepIn', this.toArgs());
|
|
104
|
-
}
|
|
105
|
-
|
|
106
|
-
stepOut(): Promise<DebugProtocol.StepOutResponse> {
|
|
107
|
-
return this.session.sendRequest('stepOut', this.toArgs());
|
|
108
|
-
}
|
|
109
|
-
|
|
110
|
-
pause(): Promise<DebugProtocol.PauseResponse> {
|
|
111
|
-
return this.session.sendRequest('pause', this.toArgs());
|
|
112
|
-
}
|
|
113
|
-
|
|
114
|
-
async getExceptionInfo(): Promise<DebugExceptionInfo | undefined> {
|
|
115
|
-
if (this.stoppedDetails && this.stoppedDetails.reason === 'exception') {
|
|
116
|
-
if (this.session.capabilities.supportsExceptionInfoRequest) {
|
|
117
|
-
const response = await this.session.sendRequest('exceptionInfo', this.toArgs());
|
|
118
|
-
return {
|
|
119
|
-
id: response.body.exceptionId,
|
|
120
|
-
description: response.body.description,
|
|
121
|
-
details: response.body.details
|
|
122
|
-
};
|
|
123
|
-
}
|
|
124
|
-
return {
|
|
125
|
-
description: this.stoppedDetails.text
|
|
126
|
-
};
|
|
127
|
-
}
|
|
128
|
-
return undefined;
|
|
129
|
-
}
|
|
130
|
-
|
|
131
|
-
get supportsTerminate(): boolean {
|
|
132
|
-
return !!this.session.capabilities.supportsTerminateThreadsRequest;
|
|
133
|
-
}
|
|
134
|
-
|
|
135
|
-
async terminate(): Promise<void> {
|
|
136
|
-
if (this.supportsTerminate) {
|
|
137
|
-
await this.session.sendRequest('terminateThreads', {
|
|
138
|
-
threadIds: [this.raw.id]
|
|
139
|
-
});
|
|
140
|
-
}
|
|
141
|
-
}
|
|
142
|
-
|
|
143
|
-
protected readonly _frames = new Map<number, DebugStackFrame>();
|
|
144
|
-
get frames(): IterableIterator<DebugStackFrame> {
|
|
145
|
-
return this._frames.values();
|
|
146
|
-
}
|
|
147
|
-
get topFrame(): DebugStackFrame | undefined {
|
|
148
|
-
return this.frames.next().value;
|
|
149
|
-
}
|
|
150
|
-
get frameCount(): number {
|
|
151
|
-
return this._frames.size;
|
|
152
|
-
}
|
|
153
|
-
|
|
154
|
-
protected pendingFetch = Promise.resolve<DebugStackFrame[]>([]);
|
|
155
|
-
protected _pendingFetchCount: number = 0;
|
|
156
|
-
protected pendingFetchCancel = new CancellationTokenSource();
|
|
157
|
-
async fetchFrames(levels: number = 20): Promise<DebugStackFrame[]> {
|
|
158
|
-
const cancel = this.pendingFetchCancel.token;
|
|
159
|
-
this._pendingFetchCount += 1;
|
|
160
|
-
|
|
161
|
-
return this.pendingFetch = this.pendingFetch.then(async () => {
|
|
162
|
-
try {
|
|
163
|
-
const start = this.frameCount;
|
|
164
|
-
const frames = await this.doFetchFrames(start, levels);
|
|
165
|
-
if (cancel.isCancellationRequested) {
|
|
166
|
-
return [];
|
|
167
|
-
}
|
|
168
|
-
return this.doUpdateFrames(frames);
|
|
169
|
-
} catch (e) {
|
|
170
|
-
console.error('fetchFrames failed:', e);
|
|
171
|
-
return [];
|
|
172
|
-
} finally {
|
|
173
|
-
if (!cancel.isCancellationRequested) {
|
|
174
|
-
this._pendingFetchCount -= 1;
|
|
175
|
-
}
|
|
176
|
-
}
|
|
177
|
-
});
|
|
178
|
-
}
|
|
179
|
-
get pendingFrameCount(): number {
|
|
180
|
-
return this._pendingFetchCount;
|
|
181
|
-
}
|
|
182
|
-
protected async doFetchFrames(startFrame: number, levels: number): Promise<DebugProtocol.StackFrame[]> {
|
|
183
|
-
try {
|
|
184
|
-
const response = await this.session.sendRequest('stackTrace',
|
|
185
|
-
this.toArgs<Partial<DebugProtocol.StackTraceArguments>>({ startFrame, levels })
|
|
186
|
-
);
|
|
187
|
-
if (this.stoppedDetails) {
|
|
188
|
-
this.stoppedDetails.totalFrames = response.body.totalFrames;
|
|
189
|
-
}
|
|
190
|
-
return response.body.stackFrames;
|
|
191
|
-
} catch (e) {
|
|
192
|
-
if (this.stoppedDetails) {
|
|
193
|
-
this.stoppedDetails.framesErrorMessage = e.message;
|
|
194
|
-
}
|
|
195
|
-
return [];
|
|
196
|
-
}
|
|
197
|
-
}
|
|
198
|
-
protected doUpdateFrames(frames: DebugProtocol.StackFrame[]): DebugStackFrame[] {
|
|
199
|
-
const result = new Set<DebugStackFrame>();
|
|
200
|
-
for (const raw of frames) {
|
|
201
|
-
const id = raw.id;
|
|
202
|
-
const frame = this._frames.get(id) || new DebugStackFrame(this, this.session);
|
|
203
|
-
this._frames.set(id, frame);
|
|
204
|
-
frame.update({ raw });
|
|
205
|
-
result.add(frame);
|
|
206
|
-
}
|
|
207
|
-
this.updateCurrentFrame();
|
|
208
|
-
return [...result.values()];
|
|
209
|
-
}
|
|
210
|
-
protected clearFrames(): void {
|
|
211
|
-
// Clear all frames
|
|
212
|
-
this._frames.clear();
|
|
213
|
-
|
|
214
|
-
// Cancel all request promises
|
|
215
|
-
this.pendingFetchCancel.cancel();
|
|
216
|
-
this.pendingFetchCancel = new CancellationTokenSource();
|
|
217
|
-
|
|
218
|
-
// Empty all current requests
|
|
219
|
-
this.pendingFetch = Promise.resolve([]);
|
|
220
|
-
this._pendingFetchCount = 0;
|
|
221
|
-
|
|
222
|
-
this.updateCurrentFrame();
|
|
223
|
-
}
|
|
224
|
-
protected updateCurrentFrame(): void {
|
|
225
|
-
const { currentFrame } = this;
|
|
226
|
-
const frameId = currentFrame && currentFrame.raw.id;
|
|
227
|
-
this.currentFrame = typeof frameId === 'number' &&
|
|
228
|
-
this._frames.get(frameId) ||
|
|
229
|
-
this._frames.values().next().value;
|
|
230
|
-
}
|
|
231
|
-
|
|
232
|
-
protected toArgs<T extends object>(arg?: T): { threadId: number } & T {
|
|
233
|
-
return Object.assign({}, arg, {
|
|
234
|
-
threadId: this.raw.id
|
|
235
|
-
});
|
|
236
|
-
}
|
|
237
|
-
|
|
238
|
-
render(): React.ReactNode {
|
|
239
|
-
return (
|
|
240
|
-
<div className="theia-debug-thread" title={nls.localizeByDefault('Session')}>
|
|
241
|
-
<span className="label">{this.raw.name}</span>
|
|
242
|
-
<span className="status">{this.threadStatus()}</span>
|
|
243
|
-
</div>
|
|
244
|
-
);
|
|
245
|
-
}
|
|
246
|
-
|
|
247
|
-
protected threadStatus(): string {
|
|
248
|
-
|
|
249
|
-
if (!this.stoppedDetails) {
|
|
250
|
-
return nls.localizeByDefault('Running');
|
|
251
|
-
}
|
|
252
|
-
|
|
253
|
-
const description = this.stoppedDetails.description;
|
|
254
|
-
|
|
255
|
-
if (description) {
|
|
256
|
-
// According to DAP we must show description as is. Translation is made by debug adapter
|
|
257
|
-
return description;
|
|
258
|
-
}
|
|
259
|
-
|
|
260
|
-
const reason = this.stoppedDetails.reason;
|
|
261
|
-
const localizedReason = this.getLocalizedReason(reason);
|
|
262
|
-
|
|
263
|
-
return reason
|
|
264
|
-
? nls.localizeByDefault('Paused on {0}', localizedReason)
|
|
265
|
-
: nls.localizeByDefault('Paused');
|
|
266
|
-
}
|
|
267
|
-
|
|
268
|
-
protected getLocalizedReason(reason: string | undefined): string {
|
|
269
|
-
switch (reason) {
|
|
270
|
-
case 'step':
|
|
271
|
-
return nls.localize('theia/debug/step', 'step');
|
|
272
|
-
case 'breakpoint':
|
|
273
|
-
return nls.localize('theia/debug/breakpoint', 'breakpoint');
|
|
274
|
-
case 'exception':
|
|
275
|
-
return nls.localize('theia/debug/exception', 'exception');
|
|
276
|
-
case 'pause':
|
|
277
|
-
return nls.localize('theia/debug/pause', 'pause');
|
|
278
|
-
case 'entry':
|
|
279
|
-
return nls.localize('theia/debug/entry', 'entry');
|
|
280
|
-
case 'goto':
|
|
281
|
-
return nls.localize('theia/debug/goto', 'goto');
|
|
282
|
-
case 'function breakpoint':
|
|
283
|
-
return nls.localize('theia/debug/functionBreakpoint', 'function breakpoint');
|
|
284
|
-
case 'data breakpoint':
|
|
285
|
-
return nls.localize('theia/debug/dataBreakpoint', 'data breakpoint');
|
|
286
|
-
case 'instruction breakpoint':
|
|
287
|
-
return nls.localize('theia/debug/instructionBreakpoint', 'instruction breakpoint');
|
|
288
|
-
default:
|
|
289
|
-
return '';
|
|
290
|
-
}
|
|
291
|
-
}
|
|
292
|
-
}
|
|
1
|
+
// *****************************************************************************
|
|
2
|
+
// Copyright (C) 2018 TypeFox and others.
|
|
3
|
+
//
|
|
4
|
+
// This program and the accompanying materials are made available under the
|
|
5
|
+
// terms of the Eclipse Public License v. 2.0 which is available at
|
|
6
|
+
// http://www.eclipse.org/legal/epl-2.0.
|
|
7
|
+
//
|
|
8
|
+
// This Source Code may also be made available under the following Secondary
|
|
9
|
+
// Licenses when the conditions for such availability set forth in the Eclipse
|
|
10
|
+
// Public License v. 2.0 are satisfied: GNU General Public License, version 2
|
|
11
|
+
// with the GNU Classpath Exception which is available at
|
|
12
|
+
// https://www.gnu.org/software/classpath/license.html.
|
|
13
|
+
//
|
|
14
|
+
// SPDX-License-Identifier: EPL-2.0 OR GPL-2.0-only WITH Classpath-exception-2.0
|
|
15
|
+
// *****************************************************************************
|
|
16
|
+
|
|
17
|
+
import * as React from '@theia/core/shared/react';
|
|
18
|
+
import { CancellationTokenSource, Emitter, Event, nls } from '@theia/core';
|
|
19
|
+
import { DebugProtocol } from '@vscode/debugprotocol/lib/debugProtocol';
|
|
20
|
+
import { TreeElement } from '@theia/core/lib/browser/source-tree';
|
|
21
|
+
import { DebugStackFrame } from './debug-stack-frame';
|
|
22
|
+
import { DebugSession } from '../debug-session';
|
|
23
|
+
|
|
24
|
+
export type StoppedDetails = DebugProtocol.StoppedEvent['body'] & {
|
|
25
|
+
framesErrorMessage?: string
|
|
26
|
+
totalFrames?: number
|
|
27
|
+
};
|
|
28
|
+
|
|
29
|
+
export class DebugThreadData {
|
|
30
|
+
readonly raw: DebugProtocol.Thread;
|
|
31
|
+
readonly stoppedDetails: StoppedDetails | undefined;
|
|
32
|
+
}
|
|
33
|
+
|
|
34
|
+
export interface DebugExceptionInfo {
|
|
35
|
+
id?: string
|
|
36
|
+
description?: string
|
|
37
|
+
details?: DebugProtocol.ExceptionDetails
|
|
38
|
+
}
|
|
39
|
+
|
|
40
|
+
export class DebugThread extends DebugThreadData implements TreeElement {
|
|
41
|
+
|
|
42
|
+
protected readonly onDidChangedEmitter = new Emitter<void>();
|
|
43
|
+
readonly onDidChanged: Event<void> = this.onDidChangedEmitter.event;
|
|
44
|
+
protected readonly onDidFocusStackFrameEmitter = new Emitter<DebugStackFrame | undefined>();
|
|
45
|
+
get onDidFocusStackFrame(): Event<DebugStackFrame | undefined> {
|
|
46
|
+
return this.onDidFocusStackFrameEmitter.event;
|
|
47
|
+
}
|
|
48
|
+
|
|
49
|
+
constructor(
|
|
50
|
+
readonly session: DebugSession
|
|
51
|
+
) {
|
|
52
|
+
super();
|
|
53
|
+
}
|
|
54
|
+
|
|
55
|
+
get id(): string {
|
|
56
|
+
return this.session.id + ':' + this.raw.id;
|
|
57
|
+
}
|
|
58
|
+
|
|
59
|
+
get threadId(): number {
|
|
60
|
+
return this.raw.id;
|
|
61
|
+
}
|
|
62
|
+
|
|
63
|
+
protected _currentFrame: DebugStackFrame | undefined;
|
|
64
|
+
get currentFrame(): DebugStackFrame | undefined {
|
|
65
|
+
return this._currentFrame;
|
|
66
|
+
}
|
|
67
|
+
set currentFrame(frame: DebugStackFrame | undefined) {
|
|
68
|
+
if (this._currentFrame?.id === frame?.id) {
|
|
69
|
+
return;
|
|
70
|
+
}
|
|
71
|
+
this._currentFrame = frame;
|
|
72
|
+
this.onDidChangedEmitter.fire(undefined);
|
|
73
|
+
this.onDidFocusStackFrameEmitter.fire(frame);
|
|
74
|
+
}
|
|
75
|
+
|
|
76
|
+
get stopped(): boolean {
|
|
77
|
+
return !!this.stoppedDetails;
|
|
78
|
+
}
|
|
79
|
+
|
|
80
|
+
update(data: Partial<DebugThreadData>): void {
|
|
81
|
+
Object.assign(this, data);
|
|
82
|
+
if ('stoppedDetails' in data) {
|
|
83
|
+
this.clearFrames();
|
|
84
|
+
}
|
|
85
|
+
}
|
|
86
|
+
|
|
87
|
+
clear(): void {
|
|
88
|
+
this.update({
|
|
89
|
+
raw: this.raw,
|
|
90
|
+
stoppedDetails: undefined
|
|
91
|
+
});
|
|
92
|
+
}
|
|
93
|
+
|
|
94
|
+
continue(): Promise<DebugProtocol.ContinueResponse> {
|
|
95
|
+
return this.session.sendRequest('continue', this.toArgs());
|
|
96
|
+
}
|
|
97
|
+
|
|
98
|
+
stepOver(): Promise<DebugProtocol.NextResponse> {
|
|
99
|
+
return this.session.sendRequest('next', this.toArgs());
|
|
100
|
+
}
|
|
101
|
+
|
|
102
|
+
stepIn(): Promise<DebugProtocol.StepInResponse> {
|
|
103
|
+
return this.session.sendRequest('stepIn', this.toArgs());
|
|
104
|
+
}
|
|
105
|
+
|
|
106
|
+
stepOut(): Promise<DebugProtocol.StepOutResponse> {
|
|
107
|
+
return this.session.sendRequest('stepOut', this.toArgs());
|
|
108
|
+
}
|
|
109
|
+
|
|
110
|
+
pause(): Promise<DebugProtocol.PauseResponse> {
|
|
111
|
+
return this.session.sendRequest('pause', this.toArgs());
|
|
112
|
+
}
|
|
113
|
+
|
|
114
|
+
async getExceptionInfo(): Promise<DebugExceptionInfo | undefined> {
|
|
115
|
+
if (this.stoppedDetails && this.stoppedDetails.reason === 'exception') {
|
|
116
|
+
if (this.session.capabilities.supportsExceptionInfoRequest) {
|
|
117
|
+
const response = await this.session.sendRequest('exceptionInfo', this.toArgs());
|
|
118
|
+
return {
|
|
119
|
+
id: response.body.exceptionId,
|
|
120
|
+
description: response.body.description,
|
|
121
|
+
details: response.body.details
|
|
122
|
+
};
|
|
123
|
+
}
|
|
124
|
+
return {
|
|
125
|
+
description: this.stoppedDetails.text
|
|
126
|
+
};
|
|
127
|
+
}
|
|
128
|
+
return undefined;
|
|
129
|
+
}
|
|
130
|
+
|
|
131
|
+
get supportsTerminate(): boolean {
|
|
132
|
+
return !!this.session.capabilities.supportsTerminateThreadsRequest;
|
|
133
|
+
}
|
|
134
|
+
|
|
135
|
+
async terminate(): Promise<void> {
|
|
136
|
+
if (this.supportsTerminate) {
|
|
137
|
+
await this.session.sendRequest('terminateThreads', {
|
|
138
|
+
threadIds: [this.raw.id]
|
|
139
|
+
});
|
|
140
|
+
}
|
|
141
|
+
}
|
|
142
|
+
|
|
143
|
+
protected readonly _frames = new Map<number, DebugStackFrame>();
|
|
144
|
+
get frames(): IterableIterator<DebugStackFrame> {
|
|
145
|
+
return this._frames.values();
|
|
146
|
+
}
|
|
147
|
+
get topFrame(): DebugStackFrame | undefined {
|
|
148
|
+
return this.frames.next().value;
|
|
149
|
+
}
|
|
150
|
+
get frameCount(): number {
|
|
151
|
+
return this._frames.size;
|
|
152
|
+
}
|
|
153
|
+
|
|
154
|
+
protected pendingFetch = Promise.resolve<DebugStackFrame[]>([]);
|
|
155
|
+
protected _pendingFetchCount: number = 0;
|
|
156
|
+
protected pendingFetchCancel = new CancellationTokenSource();
|
|
157
|
+
async fetchFrames(levels: number = 20): Promise<DebugStackFrame[]> {
|
|
158
|
+
const cancel = this.pendingFetchCancel.token;
|
|
159
|
+
this._pendingFetchCount += 1;
|
|
160
|
+
|
|
161
|
+
return this.pendingFetch = this.pendingFetch.then(async () => {
|
|
162
|
+
try {
|
|
163
|
+
const start = this.frameCount;
|
|
164
|
+
const frames = await this.doFetchFrames(start, levels);
|
|
165
|
+
if (cancel.isCancellationRequested) {
|
|
166
|
+
return [];
|
|
167
|
+
}
|
|
168
|
+
return this.doUpdateFrames(frames);
|
|
169
|
+
} catch (e) {
|
|
170
|
+
console.error('fetchFrames failed:', e);
|
|
171
|
+
return [];
|
|
172
|
+
} finally {
|
|
173
|
+
if (!cancel.isCancellationRequested) {
|
|
174
|
+
this._pendingFetchCount -= 1;
|
|
175
|
+
}
|
|
176
|
+
}
|
|
177
|
+
});
|
|
178
|
+
}
|
|
179
|
+
get pendingFrameCount(): number {
|
|
180
|
+
return this._pendingFetchCount;
|
|
181
|
+
}
|
|
182
|
+
protected async doFetchFrames(startFrame: number, levels: number): Promise<DebugProtocol.StackFrame[]> {
|
|
183
|
+
try {
|
|
184
|
+
const response = await this.session.sendRequest('stackTrace',
|
|
185
|
+
this.toArgs<Partial<DebugProtocol.StackTraceArguments>>({ startFrame, levels })
|
|
186
|
+
);
|
|
187
|
+
if (this.stoppedDetails) {
|
|
188
|
+
this.stoppedDetails.totalFrames = response.body.totalFrames;
|
|
189
|
+
}
|
|
190
|
+
return response.body.stackFrames;
|
|
191
|
+
} catch (e) {
|
|
192
|
+
if (this.stoppedDetails) {
|
|
193
|
+
this.stoppedDetails.framesErrorMessage = e.message;
|
|
194
|
+
}
|
|
195
|
+
return [];
|
|
196
|
+
}
|
|
197
|
+
}
|
|
198
|
+
protected doUpdateFrames(frames: DebugProtocol.StackFrame[]): DebugStackFrame[] {
|
|
199
|
+
const result = new Set<DebugStackFrame>();
|
|
200
|
+
for (const raw of frames) {
|
|
201
|
+
const id = raw.id;
|
|
202
|
+
const frame = this._frames.get(id) || new DebugStackFrame(this, this.session);
|
|
203
|
+
this._frames.set(id, frame);
|
|
204
|
+
frame.update({ raw });
|
|
205
|
+
result.add(frame);
|
|
206
|
+
}
|
|
207
|
+
this.updateCurrentFrame();
|
|
208
|
+
return [...result.values()];
|
|
209
|
+
}
|
|
210
|
+
protected clearFrames(): void {
|
|
211
|
+
// Clear all frames
|
|
212
|
+
this._frames.clear();
|
|
213
|
+
|
|
214
|
+
// Cancel all request promises
|
|
215
|
+
this.pendingFetchCancel.cancel();
|
|
216
|
+
this.pendingFetchCancel = new CancellationTokenSource();
|
|
217
|
+
|
|
218
|
+
// Empty all current requests
|
|
219
|
+
this.pendingFetch = Promise.resolve([]);
|
|
220
|
+
this._pendingFetchCount = 0;
|
|
221
|
+
|
|
222
|
+
this.updateCurrentFrame();
|
|
223
|
+
}
|
|
224
|
+
protected updateCurrentFrame(): void {
|
|
225
|
+
const { currentFrame } = this;
|
|
226
|
+
const frameId = currentFrame && currentFrame.raw.id;
|
|
227
|
+
this.currentFrame = typeof frameId === 'number' &&
|
|
228
|
+
this._frames.get(frameId) ||
|
|
229
|
+
this._frames.values().next().value;
|
|
230
|
+
}
|
|
231
|
+
|
|
232
|
+
protected toArgs<T extends object>(arg?: T): { threadId: number } & T {
|
|
233
|
+
return Object.assign({}, arg, {
|
|
234
|
+
threadId: this.raw.id
|
|
235
|
+
});
|
|
236
|
+
}
|
|
237
|
+
|
|
238
|
+
render(): React.ReactNode {
|
|
239
|
+
return (
|
|
240
|
+
<div className="theia-debug-thread" title={nls.localizeByDefault('Session')}>
|
|
241
|
+
<span className="label">{this.raw.name}</span>
|
|
242
|
+
<span className="status">{this.threadStatus()}</span>
|
|
243
|
+
</div>
|
|
244
|
+
);
|
|
245
|
+
}
|
|
246
|
+
|
|
247
|
+
protected threadStatus(): string {
|
|
248
|
+
|
|
249
|
+
if (!this.stoppedDetails) {
|
|
250
|
+
return nls.localizeByDefault('Running');
|
|
251
|
+
}
|
|
252
|
+
|
|
253
|
+
const description = this.stoppedDetails.description;
|
|
254
|
+
|
|
255
|
+
if (description) {
|
|
256
|
+
// According to DAP we must show description as is. Translation is made by debug adapter
|
|
257
|
+
return description;
|
|
258
|
+
}
|
|
259
|
+
|
|
260
|
+
const reason = this.stoppedDetails.reason;
|
|
261
|
+
const localizedReason = this.getLocalizedReason(reason);
|
|
262
|
+
|
|
263
|
+
return reason
|
|
264
|
+
? nls.localizeByDefault('Paused on {0}', localizedReason)
|
|
265
|
+
: nls.localizeByDefault('Paused');
|
|
266
|
+
}
|
|
267
|
+
|
|
268
|
+
protected getLocalizedReason(reason: string | undefined): string {
|
|
269
|
+
switch (reason) {
|
|
270
|
+
case 'step':
|
|
271
|
+
return nls.localize('theia/debug/step', 'step');
|
|
272
|
+
case 'breakpoint':
|
|
273
|
+
return nls.localize('theia/debug/breakpoint', 'breakpoint');
|
|
274
|
+
case 'exception':
|
|
275
|
+
return nls.localize('theia/debug/exception', 'exception');
|
|
276
|
+
case 'pause':
|
|
277
|
+
return nls.localize('theia/debug/pause', 'pause');
|
|
278
|
+
case 'entry':
|
|
279
|
+
return nls.localize('theia/debug/entry', 'entry');
|
|
280
|
+
case 'goto':
|
|
281
|
+
return nls.localize('theia/debug/goto', 'goto');
|
|
282
|
+
case 'function breakpoint':
|
|
283
|
+
return nls.localize('theia/debug/functionBreakpoint', 'function breakpoint');
|
|
284
|
+
case 'data breakpoint':
|
|
285
|
+
return nls.localize('theia/debug/dataBreakpoint', 'data breakpoint');
|
|
286
|
+
case 'instruction breakpoint':
|
|
287
|
+
return nls.localize('theia/debug/instructionBreakpoint', 'instruction breakpoint');
|
|
288
|
+
default:
|
|
289
|
+
return '';
|
|
290
|
+
}
|
|
291
|
+
}
|
|
292
|
+
}
|
|
@@ -1,38 +1,38 @@
|
|
|
1
|
-
// *****************************************************************************
|
|
2
|
-
// Copyright (C) 2019 TypeFox and others.
|
|
3
|
-
//
|
|
4
|
-
// This program and the accompanying materials are made available under the
|
|
5
|
-
// terms of the Eclipse Public License v. 2.0 which is available at
|
|
6
|
-
// http://www.eclipse.org/legal/epl-2.0.
|
|
7
|
-
//
|
|
8
|
-
// This Source Code may also be made available under the following Secondary
|
|
9
|
-
// Licenses when the conditions for such availability set forth in the Eclipse
|
|
10
|
-
// Public License v. 2.0 are satisfied: GNU General Public License, version 2
|
|
11
|
-
// with the GNU Classpath Exception which is available at
|
|
12
|
-
// https://www.gnu.org/software/classpath/license.html.
|
|
13
|
-
//
|
|
14
|
-
// SPDX-License-Identifier: EPL-2.0 OR GPL-2.0-only WITH Classpath-exception-2.0
|
|
15
|
-
// *****************************************************************************
|
|
16
|
-
|
|
17
|
-
import { interfaces } from '@theia/core/shared/inversify';
|
|
18
|
-
import { PreferenceContribution, PreferenceSchema } from '@theia/core/lib/browser/preferences/preference-contribution';
|
|
19
|
-
import { launchSchemaId } from '../debug-schema-updater';
|
|
20
|
-
import { PreferenceConfiguration } from '@theia/core/lib/browser/preferences/preference-configurations';
|
|
21
|
-
import { nls } from '@theia/core/lib/common/nls';
|
|
22
|
-
|
|
23
|
-
export const launchPreferencesSchema: PreferenceSchema = {
|
|
24
|
-
type: 'object',
|
|
25
|
-
scope: 'resource',
|
|
26
|
-
properties: {
|
|
27
|
-
'launch': {
|
|
28
|
-
$ref: launchSchemaId,
|
|
29
|
-
description: nls.localizeByDefault("Global debug launch configuration. Should be used as an alternative to 'launch.json' that is shared across workspaces."),
|
|
30
|
-
defaultValue: { configurations: [], compounds: [] }
|
|
31
|
-
}
|
|
32
|
-
}
|
|
33
|
-
};
|
|
34
|
-
|
|
35
|
-
export function bindLaunchPreferences(bind: interfaces.Bind): void {
|
|
36
|
-
bind(PreferenceContribution).toConstantValue({ schema: launchPreferencesSchema });
|
|
37
|
-
bind(PreferenceConfiguration).toConstantValue({ name: 'launch' });
|
|
38
|
-
}
|
|
1
|
+
// *****************************************************************************
|
|
2
|
+
// Copyright (C) 2019 TypeFox and others.
|
|
3
|
+
//
|
|
4
|
+
// This program and the accompanying materials are made available under the
|
|
5
|
+
// terms of the Eclipse Public License v. 2.0 which is available at
|
|
6
|
+
// http://www.eclipse.org/legal/epl-2.0.
|
|
7
|
+
//
|
|
8
|
+
// This Source Code may also be made available under the following Secondary
|
|
9
|
+
// Licenses when the conditions for such availability set forth in the Eclipse
|
|
10
|
+
// Public License v. 2.0 are satisfied: GNU General Public License, version 2
|
|
11
|
+
// with the GNU Classpath Exception which is available at
|
|
12
|
+
// https://www.gnu.org/software/classpath/license.html.
|
|
13
|
+
//
|
|
14
|
+
// SPDX-License-Identifier: EPL-2.0 OR GPL-2.0-only WITH Classpath-exception-2.0
|
|
15
|
+
// *****************************************************************************
|
|
16
|
+
|
|
17
|
+
import { interfaces } from '@theia/core/shared/inversify';
|
|
18
|
+
import { PreferenceContribution, PreferenceSchema } from '@theia/core/lib/browser/preferences/preference-contribution';
|
|
19
|
+
import { launchSchemaId } from '../debug-schema-updater';
|
|
20
|
+
import { PreferenceConfiguration } from '@theia/core/lib/browser/preferences/preference-configurations';
|
|
21
|
+
import { nls } from '@theia/core/lib/common/nls';
|
|
22
|
+
|
|
23
|
+
export const launchPreferencesSchema: PreferenceSchema = {
|
|
24
|
+
type: 'object',
|
|
25
|
+
scope: 'resource',
|
|
26
|
+
properties: {
|
|
27
|
+
'launch': {
|
|
28
|
+
$ref: launchSchemaId,
|
|
29
|
+
description: nls.localizeByDefault("Global debug launch configuration. Should be used as an alternative to 'launch.json' that is shared across workspaces."),
|
|
30
|
+
defaultValue: { configurations: [], compounds: [] }
|
|
31
|
+
}
|
|
32
|
+
}
|
|
33
|
+
};
|
|
34
|
+
|
|
35
|
+
export function bindLaunchPreferences(bind: interfaces.Bind): void {
|
|
36
|
+
bind(PreferenceContribution).toConstantValue({ schema: launchPreferencesSchema });
|
|
37
|
+
bind(PreferenceConfiguration).toConstantValue({ name: 'launch' });
|
|
38
|
+
}
|