mirror of
https://github.com/microsoft/vscode.git
synced 2026-05-08 09:08:48 +01:00
1019 lines
44 KiB
TypeScript
1019 lines
44 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 nls from 'vs/nls';
|
|
import * as lifecycle from 'vs/base/common/lifecycle';
|
|
import Event, { Emitter } from 'vs/base/common/event';
|
|
import * as paths from 'vs/base/common/paths';
|
|
import { generateUuid } from 'vs/base/common/uuid';
|
|
import uri from 'vs/base/common/uri';
|
|
import { Action } from 'vs/base/common/actions';
|
|
import { first, distinct } from 'vs/base/common/arrays';
|
|
import { isObject, isUndefinedOrNull } from 'vs/base/common/types';
|
|
import * as errors from 'vs/base/common/errors';
|
|
import severity from 'vs/base/common/severity';
|
|
import { TPromise } from 'vs/base/common/winjs.base';
|
|
import * as aria from 'vs/base/browser/ui/aria/aria';
|
|
import { Client as TelemetryClient } from 'vs/base/parts/ipc/node/ipc.cp';
|
|
import { IContextKeyService, IContextKey } from 'vs/platform/contextkey/common/contextkey';
|
|
import { IMarkerService } from 'vs/platform/markers/common/markers';
|
|
import { ILifecycleService } from 'vs/platform/lifecycle/common/lifecycle';
|
|
import { IExtensionService } from 'vs/platform/extensions/common/extensions';
|
|
import { IInstantiationService } from 'vs/platform/instantiation/common/instantiation';
|
|
import { FileChangesEvent, FileChangeType, IFileService } from 'vs/platform/files/common/files';
|
|
import { IMessageService, CloseAction } from 'vs/platform/message/common/message';
|
|
import { IWindowsService } from 'vs/platform/windows/common/windows';
|
|
import { ITelemetryService } from 'vs/platform/telemetry/common/telemetry';
|
|
import { TelemetryService } from 'vs/platform/telemetry/common/telemetryService';
|
|
import { TelemetryAppenderClient } from 'vs/platform/telemetry/common/telemetryIpc';
|
|
import { IStorageService, StorageScope } from 'vs/platform/storage/common/storage';
|
|
import * as debug from 'vs/workbench/parts/debug/common/debug';
|
|
import { RawDebugSession } from 'vs/workbench/parts/debug/electron-browser/rawDebugSession';
|
|
import { Model, ExceptionBreakpoint, FunctionBreakpoint, Breakpoint, Expression, OutputNameValueElement, ExpressionContainer, Process } from 'vs/workbench/parts/debug/common/debugModel';
|
|
import { ViewModel } from 'vs/workbench/parts/debug/common/debugViewModel';
|
|
import * as debugactions from 'vs/workbench/parts/debug/browser/debugActions';
|
|
import { ConfigurationManager } from 'vs/workbench/parts/debug/electron-browser/debugConfigurationManager';
|
|
import { ToggleMarkersPanelAction } from 'vs/workbench/parts/markers/browser/markersPanelActions';
|
|
import { ITaskService, TaskEvent, TaskType, TaskServiceEvents, ITaskSummary } from 'vs/workbench/parts/tasks/common/taskService';
|
|
import { TaskError, TaskErrors } from 'vs/workbench/parts/tasks/common/taskSystem';
|
|
import { VIEWLET_ID as EXPLORER_VIEWLET_ID } from 'vs/workbench/parts/files/common/files';
|
|
import { IViewletService } from 'vs/workbench/services/viewlet/browser/viewlet';
|
|
import { IPanelService } from 'vs/workbench/services/panel/common/panelService';
|
|
import { IPartService, Parts } from 'vs/workbench/services/part/common/partService';
|
|
import { ITextFileService } from 'vs/workbench/services/textfile/common/textfiles';
|
|
import { IConfigurationService } from 'vs/platform/configuration/common/configuration';
|
|
import { IWorkspaceContextService } from 'vs/platform/workspace/common/workspace';
|
|
import { IWorkbenchEditorService } from 'vs/workbench/services/editor/common/editorService';
|
|
import { IWindowIPCService, IBroadcast } from 'vs/workbench/services/window/electron-browser/windowService';
|
|
import { ILogEntry, EXTENSION_LOG_BROADCAST_CHANNEL, EXTENSION_ATTACH_BROADCAST_CHANNEL, EXTENSION_TERMINATE_BROADCAST_CHANNEL } from 'vs/workbench/electron-browser/extensionHost';
|
|
|
|
const DEBUG_BREAKPOINTS_KEY = 'debug.breakpoint';
|
|
const DEBUG_BREAKPOINTS_ACTIVATED_KEY = 'debug.breakpointactivated';
|
|
const DEBUG_FUNCTION_BREAKPOINTS_KEY = 'debug.functionbreakpoint';
|
|
const DEBUG_EXCEPTION_BREAKPOINTS_KEY = 'debug.exceptionbreakpoint';
|
|
const DEBUG_WATCH_EXPRESSIONS_KEY = 'debug.watchexpressions';
|
|
const DEBUG_SELECTED_CONFIG_NAME_KEY = 'debug.selectedconfigname';
|
|
|
|
export class DebugService implements debug.IDebugService {
|
|
public _serviceBrand: any;
|
|
|
|
private sessionStates: Map<string, debug.State>;
|
|
private _onDidChangeState: Emitter<void>;
|
|
private model: Model;
|
|
private viewModel: ViewModel;
|
|
private configurationManager: ConfigurationManager;
|
|
private customTelemetryService: ITelemetryService;
|
|
private lastTaskEvent: TaskEvent;
|
|
private toDispose: lifecycle.IDisposable[];
|
|
private toDisposeOnSessionEnd: Map<string, lifecycle.IDisposable[]>;
|
|
private inDebugMode: IContextKey<boolean>;
|
|
private breakpointsToSendOnResourceSaved: Set<string>;
|
|
|
|
constructor(
|
|
@IStorageService private storageService: IStorageService,
|
|
@IWorkbenchEditorService private editorService: IWorkbenchEditorService,
|
|
@ITextFileService private textFileService: ITextFileService,
|
|
@IViewletService private viewletService: IViewletService,
|
|
@IPanelService private panelService: IPanelService,
|
|
@IMessageService private messageService: IMessageService,
|
|
@IPartService private partService: IPartService,
|
|
@IWindowsService private windowsService: IWindowsService,
|
|
@IWindowIPCService private windowService: IWindowIPCService,
|
|
@ITelemetryService private telemetryService: ITelemetryService,
|
|
@IWorkspaceContextService private contextService: IWorkspaceContextService,
|
|
@IContextKeyService contextKeyService: IContextKeyService,
|
|
@ILifecycleService lifecycleService: ILifecycleService,
|
|
@IInstantiationService private instantiationService: IInstantiationService,
|
|
@IExtensionService private extensionService: IExtensionService,
|
|
@IMarkerService private markerService: IMarkerService,
|
|
@ITaskService private taskService: ITaskService,
|
|
@IFileService private fileService: IFileService,
|
|
@IConfigurationService private configurationService: IConfigurationService
|
|
) {
|
|
this.toDispose = [];
|
|
this.toDisposeOnSessionEnd = new Map<string, lifecycle.IDisposable[]>();
|
|
this.breakpointsToSendOnResourceSaved = new Set<string>();
|
|
this._onDidChangeState = new Emitter<void>();
|
|
this.sessionStates = new Map<string, debug.State>();
|
|
|
|
this.configurationManager = this.instantiationService.createInstance(ConfigurationManager);
|
|
this.inDebugMode = debug.CONTEXT_IN_DEBUG_MODE.bindTo(contextKeyService);
|
|
|
|
this.model = new Model(this.loadBreakpoints(), this.storageService.getBoolean(DEBUG_BREAKPOINTS_ACTIVATED_KEY, StorageScope.WORKSPACE, true), this.loadFunctionBreakpoints(),
|
|
this.loadExceptionBreakpoints(), this.loadWatchExpressions());
|
|
this.toDispose.push(this.model);
|
|
this.viewModel = new ViewModel(this.storageService.get(DEBUG_SELECTED_CONFIG_NAME_KEY, StorageScope.WORKSPACE, null));
|
|
|
|
this.registerListeners(lifecycleService);
|
|
}
|
|
|
|
private registerListeners(lifecycleService: ILifecycleService): void {
|
|
this.toDispose.push(this.fileService.onFileChanges(e => this.onFileChanges(e)));
|
|
|
|
if (this.taskService) {
|
|
this.toDispose.push(this.taskService.addListener2(TaskServiceEvents.Active, (e: TaskEvent) => {
|
|
this.lastTaskEvent = e;
|
|
}));
|
|
this.toDispose.push(this.taskService.addListener2(TaskServiceEvents.Inactive, (e: TaskEvent) => {
|
|
if (e.type === TaskType.SingleRun) {
|
|
this.lastTaskEvent = null;
|
|
}
|
|
}));
|
|
this.toDispose.push(this.taskService.addListener2(TaskServiceEvents.Terminated, (e: TaskEvent) => {
|
|
this.lastTaskEvent = null;
|
|
}));
|
|
}
|
|
|
|
lifecycleService.onShutdown(this.store, this);
|
|
lifecycleService.onShutdown(this.dispose, this);
|
|
|
|
this.toDispose.push(this.windowService.onBroadcast(this.onBroadcast, this));
|
|
this.toDispose.push(this.configurationService.onDidUpdateConfiguration((event) => {
|
|
if (event.sourceConfig) {
|
|
const names = this.configurationManager.getConfigurationNames();
|
|
if (names.every(name => name !== this.viewModel.selectedConfigurationName)) {
|
|
// Current selected configuration no longer exists - take the first configuration instead.
|
|
this.viewModel.setSelectedConfigurationName(names.length ? names[0] : undefined);
|
|
}
|
|
}
|
|
}));
|
|
}
|
|
|
|
private onBroadcast(broadcast: IBroadcast): void {
|
|
|
|
// attach: PH is ready to be attached to
|
|
// TODO@Isidor this is a hack to just get any 'extensionHost' session.
|
|
// Optimally the broadcast would contain the id of the session
|
|
// We are only intersted if we have an active debug session for extensionHost
|
|
const session = <RawDebugSession>this.model.getProcesses().map(p => p.session).filter(s => s.configuration.type === 'extensionHost').pop();
|
|
if (broadcast.channel === EXTENSION_ATTACH_BROADCAST_CHANNEL) {
|
|
this.rawAttach(session, broadcast.payload.port);
|
|
return;
|
|
}
|
|
|
|
if (broadcast.channel === EXTENSION_TERMINATE_BROADCAST_CHANNEL) {
|
|
this.onSessionEnd(session);
|
|
return;
|
|
}
|
|
|
|
// from this point on we require an active session
|
|
if (!session) {
|
|
return;
|
|
}
|
|
|
|
// an extension logged output, show it inside the REPL
|
|
if (broadcast.channel === EXTENSION_LOG_BROADCAST_CHANNEL) {
|
|
let extensionOutput: ILogEntry = broadcast.payload;
|
|
let sev = extensionOutput.severity === 'warn' ? severity.Warning : extensionOutput.severity === 'error' ? severity.Error : severity.Info;
|
|
|
|
let args: any[] = [];
|
|
try {
|
|
let parsed = JSON.parse(extensionOutput.arguments);
|
|
args.push(...Object.getOwnPropertyNames(parsed).map(o => parsed[o]));
|
|
} catch (error) {
|
|
args.push(extensionOutput.arguments);
|
|
}
|
|
|
|
// add output for each argument logged
|
|
let simpleVals: any[] = [];
|
|
for (let i = 0; i < args.length; i++) {
|
|
let a = args[i];
|
|
|
|
// undefined gets printed as 'undefined'
|
|
if (typeof a === 'undefined') {
|
|
simpleVals.push('undefined');
|
|
}
|
|
|
|
// null gets printed as 'null'
|
|
else if (a === null) {
|
|
simpleVals.push('null');
|
|
}
|
|
|
|
// objects & arrays are special because we want to inspect them in the REPL
|
|
else if (isObject(a) || Array.isArray(a)) {
|
|
|
|
// flush any existing simple values logged
|
|
if (simpleVals.length) {
|
|
this.model.appendToRepl(simpleVals.join(' '), sev);
|
|
simpleVals = [];
|
|
}
|
|
|
|
// show object
|
|
this.model.appendToRepl(new OutputNameValueElement((<any>a).prototype, a, nls.localize('snapshotObj', "Only primitive values are shown for this object.")), sev);
|
|
}
|
|
|
|
// string: watch out for % replacement directive
|
|
// string substitution and formatting @ https://developer.chrome.com/devtools/docs/console
|
|
else if (typeof a === 'string') {
|
|
let buf = '';
|
|
|
|
for (let j = 0, len = a.length; j < len; j++) {
|
|
if (a[j] === '%' && (a[j + 1] === 's' || a[j + 1] === 'i' || a[j + 1] === 'd')) {
|
|
i++; // read over substitution
|
|
buf += !isUndefinedOrNull(args[i]) ? args[i] : ''; // replace
|
|
j++; // read over directive
|
|
} else {
|
|
buf += a[j];
|
|
}
|
|
}
|
|
|
|
simpleVals.push(buf);
|
|
}
|
|
|
|
// number or boolean is joined together
|
|
else {
|
|
simpleVals.push(a);
|
|
}
|
|
}
|
|
|
|
// flush simple values
|
|
if (simpleVals.length) {
|
|
this.model.appendToRepl(simpleVals.join(' '), sev);
|
|
}
|
|
}
|
|
}
|
|
|
|
private registerSessionListeners(process: Process, session: RawDebugSession): void {
|
|
this.toDisposeOnSessionEnd.get(session.getId()).push(session);
|
|
this.toDisposeOnSessionEnd.get(session.getId()).push(session.onDidInitialize(event => {
|
|
aria.status(nls.localize('debuggingStarted', "Debugging started."));
|
|
const sendConfigurationDone = () => {
|
|
if (session && session.configuration.capabilities.supportsConfigurationDoneRequest) {
|
|
return session.configurationDone().done(null, e => {
|
|
// Disconnect the debug session on configuration done error #10596
|
|
if (session) {
|
|
session.disconnect().done(null, errors.onUnexpectedError);
|
|
}
|
|
this.messageService.show(severity.Error, e.message);
|
|
});
|
|
}
|
|
};
|
|
|
|
this.sendAllBreakpoints(process).then(sendConfigurationDone, sendConfigurationDone)
|
|
.done(() => this.fetchThreads(session), errors.onUnexpectedError);
|
|
}));
|
|
|
|
this.toDisposeOnSessionEnd.get(session.getId()).push(session.onDidStop(event => {
|
|
this.setStateAndEmit(session.getId(), debug.State.Stopped);
|
|
const threadId = event.body.threadId;
|
|
|
|
session.threads().then(response => {
|
|
if (!response || !response.body || !response.body.threads) {
|
|
return;
|
|
}
|
|
|
|
const rawThread = response.body.threads.filter(t => t.id === threadId).pop();
|
|
this.model.rawUpdate({
|
|
sessionId: session.getId(),
|
|
thread: rawThread,
|
|
threadId,
|
|
stoppedDetails: event.body,
|
|
allThreadsStopped: event.body.allThreadsStopped
|
|
});
|
|
|
|
const thread = process && process.getThread(threadId);
|
|
if (thread) {
|
|
thread.fetchCallStack().then(callStack => {
|
|
if (callStack.length > 0 && !this.viewModel.focusedStackFrame) {
|
|
// focus first stack frame from top that has source location if no other stack frame is focussed
|
|
const stackFrameToFocus = first(callStack, sf => sf.source && !sf.source.deemphasize, callStack[0]);
|
|
this.focusStackFrameAndEvaluate(stackFrameToFocus).done(null, errors.onUnexpectedError);
|
|
this.windowService.getWindow().focus();
|
|
aria.alert(nls.localize('debuggingPaused', "Debugging paused, reason {0}, {1} {2}", event.body.reason, stackFrameToFocus.source ? stackFrameToFocus.source.name : '', stackFrameToFocus.lineNumber));
|
|
|
|
return stackFrameToFocus.openInEditor(this.editorService);
|
|
}
|
|
});
|
|
}
|
|
}, errors.onUnexpectedError);
|
|
}));
|
|
|
|
this.toDisposeOnSessionEnd.get(session.getId()).push(session.onDidThread(event => {
|
|
if (event.body.reason === 'started') {
|
|
this.fetchThreads(session).done(undefined, errors.onUnexpectedError);
|
|
} else if (event.body.reason === 'exited') {
|
|
this.model.clearThreads(session.getId(), true, event.body.threadId);
|
|
}
|
|
}));
|
|
|
|
this.toDisposeOnSessionEnd.get(session.getId()).push(session.onDidTerminateDebugee(event => {
|
|
aria.status(nls.localize('debuggingStopped', "Debugging stopped."));
|
|
if (session && session.getId() === event.body.sessionId) {
|
|
if (event.body && typeof event.body.restart === 'boolean' && event.body.restart) {
|
|
this.restartProcess(process).done(null, err => this.messageService.show(severity.Error, err.message));
|
|
} else {
|
|
session.disconnect().done(null, errors.onUnexpectedError);
|
|
}
|
|
}
|
|
}));
|
|
|
|
this.toDisposeOnSessionEnd.get(session.getId()).push(session.onDidContinued(event => {
|
|
const threadId = event.body.allThreadsContinued !== false ? undefined : event.body.threadId;
|
|
this.model.clearThreads(session.getId(), false, threadId);
|
|
if (this.viewModel.focusedProcess.getId() === session.getId()) {
|
|
this.focusStackFrameAndEvaluate(null, this.viewModel.focusedProcess).done(null, errors.onUnexpectedError);
|
|
}
|
|
this.setStateAndEmit(session.getId(), session.requestType === debug.SessionRequestType.LAUNCH_NO_DEBUG ? debug.State.RunningNoDebug : debug.State.Running);
|
|
}));
|
|
|
|
this.toDisposeOnSessionEnd.get(session.getId()).push(session.onDidOutput(event => {
|
|
if (!event.body) {
|
|
return;
|
|
}
|
|
|
|
if (event.body.category === 'telemetry') {
|
|
// only log telemetry events from debug adapter if the adapter provided the telemetry key
|
|
// and the user opted in telemetry
|
|
if (this.customTelemetryService && this.telemetryService.isOptedIn) {
|
|
this.customTelemetryService.publicLog(event.body.output, event.body.data);
|
|
}
|
|
} else if (event.body.variablesReference) {
|
|
const container = new ExpressionContainer(process, event.body.variablesReference, generateUuid());
|
|
container.getChildren().then(children => {
|
|
children.forEach(child => {
|
|
// Since we can not display multiple trees in a row, we are displaying these variables one after the other (ignoring their names)
|
|
child.name = null;
|
|
this.model.appendToRepl(child, null);
|
|
});
|
|
});
|
|
} else if (typeof event.body.output === 'string') {
|
|
const outputSeverity = event.body.category === 'stderr' ? severity.Error : event.body.category === 'console' ? severity.Warning : severity.Info;
|
|
this.model.appendToRepl(event.body.output, outputSeverity);
|
|
}
|
|
}));
|
|
|
|
this.toDisposeOnSessionEnd.get(session.getId()).push(session.onDidBreakpoint(event => {
|
|
const id = event.body && event.body.breakpoint ? event.body.breakpoint.id : undefined;
|
|
const breakpoint = this.model.getBreakpoints().filter(bp => bp.idFromAdapter === id).pop();
|
|
if (breakpoint) {
|
|
this.model.updateBreakpoints({ [breakpoint.getId()]: event.body.breakpoint });
|
|
} else {
|
|
const functionBreakpoint = this.model.getFunctionBreakpoints().filter(bp => bp.idFromAdapter === id).pop();
|
|
if (functionBreakpoint) {
|
|
this.model.updateFunctionBreakpoints({ [functionBreakpoint.getId()]: event.body.breakpoint });
|
|
}
|
|
}
|
|
}));
|
|
|
|
this.toDisposeOnSessionEnd.get(session.getId()).push(session.onDidExitAdapter(event => {
|
|
// 'Run without debugging' mode VSCode must terminate the extension host. More details: #3905
|
|
if (session && session.configuration.type === 'extensionHost' && this.sessionStates.get(session.getId()) === debug.State.RunningNoDebug && this.contextService.getWorkspace()) {
|
|
this.windowsService.closeExtensionHostWindow(this.contextService.getWorkspace().resource.fsPath);
|
|
}
|
|
if (session && session.getId() === event.body.sessionId) {
|
|
this.onSessionEnd(session);
|
|
}
|
|
}));
|
|
}
|
|
|
|
private fetchThreads(session: RawDebugSession): TPromise<any> {
|
|
return session.threads().then(response => {
|
|
if (response && response.body && response.body.threads) {
|
|
response.body.threads.forEach(thread =>
|
|
this.model.rawUpdate({
|
|
sessionId: session.getId(),
|
|
threadId: thread.id,
|
|
thread
|
|
}));
|
|
}
|
|
});
|
|
}
|
|
|
|
private loadBreakpoints(): Breakpoint[] {
|
|
let result: Breakpoint[];
|
|
try {
|
|
result = JSON.parse(this.storageService.get(DEBUG_BREAKPOINTS_KEY, StorageScope.WORKSPACE, '[]')).map((breakpoint: any) => {
|
|
return new Breakpoint(uri.parse(breakpoint.uri.external || breakpoint.source.uri.external), breakpoint.lineNumber, breakpoint.column, breakpoint.enabled, breakpoint.condition, breakpoint.hitCondition);
|
|
});
|
|
} catch (e) { }
|
|
|
|
return result || [];
|
|
}
|
|
|
|
private loadFunctionBreakpoints(): FunctionBreakpoint[] {
|
|
let result: FunctionBreakpoint[];
|
|
try {
|
|
result = JSON.parse(this.storageService.get(DEBUG_FUNCTION_BREAKPOINTS_KEY, StorageScope.WORKSPACE, '[]')).map((fb: any) => {
|
|
return new FunctionBreakpoint(fb.name, fb.enabled, fb.hitCondition);
|
|
});
|
|
} catch (e) { }
|
|
|
|
return result || [];
|
|
}
|
|
|
|
private loadExceptionBreakpoints(): ExceptionBreakpoint[] {
|
|
let result: ExceptionBreakpoint[];
|
|
try {
|
|
result = JSON.parse(this.storageService.get(DEBUG_EXCEPTION_BREAKPOINTS_KEY, StorageScope.WORKSPACE, '[]')).map((exBreakpoint: any) => {
|
|
return new ExceptionBreakpoint(exBreakpoint.filter || exBreakpoint.name, exBreakpoint.label, exBreakpoint.enabled);
|
|
});
|
|
} catch (e) { }
|
|
|
|
return result || [];
|
|
}
|
|
|
|
private loadWatchExpressions(): Expression[] {
|
|
let result: Expression[];
|
|
try {
|
|
result = JSON.parse(this.storageService.get(DEBUG_WATCH_EXPRESSIONS_KEY, StorageScope.WORKSPACE, '[]')).map((watchStoredData: { name: string, id: string }) => {
|
|
return new Expression(watchStoredData.name, watchStoredData.id);
|
|
});
|
|
} catch (e) { }
|
|
|
|
return result || [];
|
|
}
|
|
|
|
public get state(): debug.State {
|
|
const focusedProcess = this.viewModel.focusedProcess;
|
|
if (focusedProcess) {
|
|
return this.sessionStates.get(focusedProcess.getId());
|
|
}
|
|
const processes = this.model.getProcesses();
|
|
if (processes.length > 0) {
|
|
return this.sessionStates.get(processes[0].getId());
|
|
}
|
|
|
|
return debug.State.Inactive;
|
|
}
|
|
|
|
public get onDidChangeState(): Event<void> {
|
|
return this._onDidChangeState.event;
|
|
}
|
|
|
|
private setStateAndEmit(sessionId: string, newState: debug.State): void {
|
|
this.sessionStates.set(sessionId, newState);
|
|
this._onDidChangeState.fire();
|
|
}
|
|
|
|
public get enabled(): boolean {
|
|
return this.contextService.hasWorkspace();
|
|
}
|
|
|
|
public focusStackFrameAndEvaluate(stackFrame: debug.IStackFrame, process?: debug.IProcess): TPromise<void> {
|
|
if (!process) {
|
|
const processes = this.model.getProcesses();
|
|
process = stackFrame ? stackFrame.thread.process : processes.length ? processes[0] : null;
|
|
}
|
|
if (!stackFrame) {
|
|
const threads = process ? process.getAllThreads() : null;
|
|
const callStack = threads && threads.length ? threads[0].getCallStack() : null;
|
|
stackFrame = callStack && callStack.length ? callStack[0] : null;
|
|
}
|
|
|
|
this.viewModel.setFocusedStackFrame(stackFrame, process);
|
|
this._onDidChangeState.fire();
|
|
|
|
return this.model.evaluateWatchExpressions(process, stackFrame);
|
|
}
|
|
|
|
public enableOrDisableBreakpoints(enable: boolean, breakpoint?: debug.IEnablement): TPromise<void> {
|
|
if (breakpoint) {
|
|
this.model.setEnablement(breakpoint, enable);
|
|
if (breakpoint instanceof Breakpoint) {
|
|
return this.sendBreakpoints(breakpoint.uri);
|
|
} else if (breakpoint instanceof FunctionBreakpoint) {
|
|
return this.sendFunctionBreakpoints();
|
|
}
|
|
|
|
return this.sendExceptionBreakpoints();
|
|
}
|
|
|
|
this.model.enableOrDisableAllBreakpoints(enable);
|
|
return this.sendAllBreakpoints();
|
|
}
|
|
|
|
public addBreakpoints(uri: uri, rawBreakpoints: debug.IRawBreakpoint[]): TPromise<void> {
|
|
this.model.addBreakpoints(uri, rawBreakpoints);
|
|
rawBreakpoints.forEach(rbp => aria.status(nls.localize('breakpointAdded', "Added breakpoint, line {0}, file {1}", rbp.lineNumber, uri.fsPath)));
|
|
|
|
return this.sendBreakpoints(uri);
|
|
}
|
|
|
|
public removeBreakpoints(id?: string): TPromise<any> {
|
|
const toRemove = this.model.getBreakpoints().filter(bp => !id || bp.getId() === id);
|
|
toRemove.forEach(bp => aria.status(nls.localize('breakpointRemoved', "Removed breakpoint, line {0}, file {1}", bp.lineNumber, bp.uri.fsPath)));
|
|
const urisToClear = distinct(toRemove, bp => bp.uri.toString()).map(bp => bp.uri);
|
|
|
|
this.model.removeBreakpoints(toRemove);
|
|
return TPromise.join(urisToClear.map(uri => this.sendBreakpoints(uri)));
|
|
}
|
|
|
|
public setBreakpointsActivated(activated: boolean): TPromise<void> {
|
|
this.model.setBreakpointsActivated(activated);
|
|
return this.sendAllBreakpoints();
|
|
}
|
|
|
|
public addFunctionBreakpoint(): void {
|
|
this.model.addFunctionBreakpoint('');
|
|
}
|
|
|
|
public renameFunctionBreakpoint(id: string, newFunctionName: string): TPromise<void> {
|
|
this.model.updateFunctionBreakpoints({ [id]: { name: newFunctionName } });
|
|
return this.sendFunctionBreakpoints();
|
|
}
|
|
|
|
public removeFunctionBreakpoints(id?: string): TPromise<void> {
|
|
this.model.removeFunctionBreakpoints(id);
|
|
return this.sendFunctionBreakpoints();
|
|
}
|
|
|
|
public addReplExpression(name: string): TPromise<void> {
|
|
this.telemetryService.publicLog('debugService/addReplExpression');
|
|
return this.model.addReplExpression(this.viewModel.focusedProcess, this.viewModel.focusedStackFrame, name)
|
|
// Evaluate all watch expressions and fetch variables again since repl evaluation might have changed some.
|
|
.then(() => this.focusStackFrameAndEvaluate(this.viewModel.focusedStackFrame));
|
|
}
|
|
|
|
public removeReplExpressions(): void {
|
|
this.model.removeReplExpressions();
|
|
}
|
|
|
|
public addWatchExpression(name: string): TPromise<void> {
|
|
return this.model.addWatchExpression(this.viewModel.focusedProcess, this.viewModel.focusedStackFrame, name);
|
|
}
|
|
|
|
public renameWatchExpression(id: string, newName: string): TPromise<void> {
|
|
return this.model.renameWatchExpression(this.viewModel.focusedProcess, this.viewModel.focusedStackFrame, id, newName)
|
|
// Evaluate all watch expressions and fetch variables again since watch expression evaluation might have changed some.
|
|
.then(() => this.focusStackFrameAndEvaluate(this.viewModel.focusedStackFrame));
|
|
}
|
|
|
|
public moveWatchExpression(id: string, position: number): void {
|
|
this.model.moveWatchExpression(id, position);
|
|
}
|
|
|
|
public removeWatchExpressions(id?: string): void {
|
|
this.model.removeWatchExpressions(id);
|
|
}
|
|
|
|
public createProcess(configurationOrName: debug.IConfig | string): TPromise<any> {
|
|
if (this.model.getProcesses().length === 0) {
|
|
// Repl shouldn't be cleared if a process is already running since the repl is shared.
|
|
this.removeReplExpressions();
|
|
}
|
|
|
|
const sessionId = generateUuid();
|
|
this.setStateAndEmit(sessionId, debug.State.Initializing);
|
|
|
|
return this.textFileService.saveAll() // make sure all dirty files are saved
|
|
.then(() => this.configurationService.reloadConfiguration() // make sure configuration is up to date
|
|
.then(() => this.extensionService.onReady()
|
|
.then(() => {
|
|
const compound = typeof configurationOrName === 'string' ? this.configurationManager.getCompound(configurationOrName) : null;
|
|
if (compound) {
|
|
if (!compound.configurations) {
|
|
return TPromise.wrapError(new Error(nls.localize({ key: 'compoundMustHaveConfigurations', comment: ['compound indicates a "compounds" configuration item'] },
|
|
"Compound must have \"configurations\" attribute set in order to start multiple configurations.")));
|
|
}
|
|
|
|
return TPromise.join(compound.configurations.map(name => this.createProcess(name)));
|
|
}
|
|
|
|
return this.configurationManager.getConfiguration(configurationOrName).then(configuration => {
|
|
if (!this.configurationManager.getAdapter(configuration.type)) {
|
|
return configuration.type ? TPromise.wrapError(new Error(nls.localize('debugTypeNotSupported', "Configured debug type '{0}' is not supported.", configuration.type)))
|
|
: TPromise.wrapError(errors.create(nls.localize('debugTypeMissing', "Missing property 'type' for the chosen launch configuration."),
|
|
{ actions: [this.instantiationService.createInstance(debugactions.ConfigureAction, debugactions.ConfigureAction.ID, debugactions.ConfigureAction.LABEL), CloseAction] }));
|
|
}
|
|
|
|
return this.runPreLaunchTask(configuration.preLaunchTask).then((taskSummary: ITaskSummary) => {
|
|
const errorCount = configuration.preLaunchTask ? this.markerService.getStatistics().errors : 0;
|
|
const successExitCode = taskSummary && taskSummary.exitCode === 0;
|
|
const failureExitCode = taskSummary && taskSummary.exitCode !== undefined && taskSummary.exitCode !== 0;
|
|
if (successExitCode || (errorCount === 0 && !failureExitCode)) {
|
|
return this.doCreateProcess(sessionId, configuration);
|
|
}
|
|
|
|
this.messageService.show(severity.Error, {
|
|
message: errorCount > 1 ? nls.localize('preLaunchTaskErrors', "Build errors have been detected during preLaunchTask '{0}'.", configuration.preLaunchTask) :
|
|
errorCount === 1 ? nls.localize('preLaunchTaskError', "Build error has been detected during preLaunchTask '{0}'.", configuration.preLaunchTask) :
|
|
nls.localize('preLaunchTaskExitCode', "The preLaunchTask '{0}' terminated with exit code {1}.", configuration.preLaunchTask, taskSummary.exitCode),
|
|
actions: [
|
|
new Action('debug.continue', nls.localize('debugAnyway', "Debug Anyway"), null, true, () => {
|
|
this.messageService.hideAll();
|
|
return this.doCreateProcess(sessionId, configuration);
|
|
}),
|
|
this.instantiationService.createInstance(ToggleMarkersPanelAction, ToggleMarkersPanelAction.ID, ToggleMarkersPanelAction.LABEL),
|
|
CloseAction
|
|
]
|
|
});
|
|
}, (err: TaskError) => {
|
|
if (err.code !== TaskErrors.NotConfigured) {
|
|
throw err;
|
|
}
|
|
|
|
this.messageService.show(err.severity, {
|
|
message: err.message,
|
|
actions: [this.taskService.configureAction(), CloseAction]
|
|
});
|
|
});
|
|
}, err => {
|
|
if (!this.contextService.getWorkspace()) {
|
|
return this.messageService.show(severity.Error, nls.localize('noFolderWorkspaceDebugError', "The active file can not be debugged. Make sure it is saved on disk and that you have a debug extension installed for that file type."));
|
|
}
|
|
|
|
return this.configurationManager.openConfigFile(false).then(openend => {
|
|
if (openend) {
|
|
this.messageService.show(severity.Info, nls.localize('NewLaunchConfig', "Please set up the launch configuration file for your application. {0}", err.message));
|
|
}
|
|
});
|
|
});
|
|
})));
|
|
}
|
|
|
|
private doCreateProcess(sessionId: string, configuration: debug.IConfig): TPromise<any> {
|
|
|
|
return this.telemetryService.getTelemetryInfo().then(info => {
|
|
const telemetryInfo: { [key: string]: string } = Object.create(null);
|
|
telemetryInfo['common.vscodemachineid'] = info.machineId;
|
|
telemetryInfo['common.vscodesessionid'] = info.sessionId;
|
|
return telemetryInfo;
|
|
}).then(data => {
|
|
const adapter = this.configurationManager.getAdapter(configuration.type);
|
|
const { aiKey, type } = adapter;
|
|
const publisher = adapter.extensionDescription.publisher;
|
|
this.customTelemetryService = null;
|
|
let client: TelemetryClient;
|
|
|
|
if (aiKey) {
|
|
client = new TelemetryClient(
|
|
uri.parse(require.toUrl('bootstrap')).fsPath,
|
|
{
|
|
serverName: 'Debug Telemetry',
|
|
timeout: 1000 * 60 * 5,
|
|
args: [`${publisher}.${type}`, JSON.stringify(data), aiKey],
|
|
env: {
|
|
ELECTRON_RUN_AS_NODE: 1,
|
|
PIPE_LOGGING: 'true',
|
|
AMD_ENTRYPOINT: 'vs/workbench/parts/debug/node/telemetryApp'
|
|
}
|
|
}
|
|
);
|
|
|
|
const channel = client.getChannel('telemetryAppender');
|
|
const appender = new TelemetryAppenderClient(channel);
|
|
|
|
this.customTelemetryService = new TelemetryService({ appender }, this.configurationService);
|
|
}
|
|
|
|
const session = this.instantiationService.createInstance(RawDebugSession, sessionId, configuration.debugServer, adapter, this.customTelemetryService);
|
|
const process = this.model.addProcess(configuration.name, session);
|
|
|
|
if (!this.viewModel.focusedProcess) {
|
|
this.focusStackFrameAndEvaluate(null, process);
|
|
}
|
|
this.toDisposeOnSessionEnd.set(session.getId(), []);
|
|
if (client) {
|
|
this.toDisposeOnSessionEnd.get(session.getId()).push(client);
|
|
}
|
|
this.registerSessionListeners(process, session);
|
|
|
|
return session.initialize({
|
|
adapterID: configuration.type,
|
|
pathFormat: 'path',
|
|
linesStartAt1: true,
|
|
columnsStartAt1: true,
|
|
supportsVariableType: true, // #8858
|
|
supportsVariablePaging: true, // #9537
|
|
supportsRunInTerminalRequest: true // #10574
|
|
}).then((result: DebugProtocol.InitializeResponse) => {
|
|
if (session.disconnected) {
|
|
return TPromise.wrapError(new Error(nls.localize('debugAdapterCrash', "Debug adapter process has terminated unexpectedly")));
|
|
}
|
|
|
|
this.model.setExceptionBreakpoints(session.configuration.capabilities.exceptionBreakpointFilters);
|
|
return configuration.request === 'attach' ? session.attach(configuration) : session.launch(configuration);
|
|
}).then((result: DebugProtocol.Response) => {
|
|
if (session.disconnected) {
|
|
return TPromise.as(null);
|
|
}
|
|
|
|
if (configuration.internalConsoleOptions === 'openOnSessionStart' || (!this.viewModel.changedWorkbenchViewState && configuration.internalConsoleOptions !== 'neverOpen')) {
|
|
this.panelService.openPanel(debug.REPL_ID, false).done(undefined, errors.onUnexpectedError);
|
|
}
|
|
|
|
if (!this.viewModel.changedWorkbenchViewState && (this.partService.isVisible(Parts.SIDEBAR_PART) || !this.contextService.getWorkspace())) {
|
|
// We only want to change the workbench view state on the first debug session #5738 and if the side bar is not hidden
|
|
this.viewModel.changedWorkbenchViewState = true;
|
|
this.viewletService.openViewlet(debug.VIEWLET_ID);
|
|
}
|
|
|
|
// Do not change status bar to orange if we are just running without debug.
|
|
if (!configuration.noDebug) {
|
|
this.partService.addClass('debugging');
|
|
}
|
|
this.extensionService.activateByEvent(`onDebug:${configuration.type}`).done(null, errors.onUnexpectedError);
|
|
this.inDebugMode.set(true);
|
|
this.setStateAndEmit(session.getId(), session.requestType === debug.SessionRequestType.LAUNCH_NO_DEBUG ? debug.State.RunningNoDebug : debug.State.Running);
|
|
if (this.model.getProcesses().length > 1) {
|
|
this.viewModel.setMultiProcessView(true);
|
|
}
|
|
|
|
this.telemetryService.publicLog('debugSessionStart', {
|
|
type: configuration.type,
|
|
breakpointCount: this.model.getBreakpoints().length,
|
|
exceptionBreakpoints: this.model.getExceptionBreakpoints(),
|
|
watchExpressionsCount: this.model.getWatchExpressions().length,
|
|
extensionName: `${adapter.extensionDescription.publisher}.${adapter.extensionDescription.name}`,
|
|
isBuiltin: adapter.extensionDescription.isBuiltin
|
|
});
|
|
}).then(undefined, (error: any) => {
|
|
if (error instanceof Error && error.message === 'Canceled') {
|
|
// Do not show 'canceled' error messages to the user #7906
|
|
return TPromise.as(null);
|
|
}
|
|
|
|
const errorMessage = error instanceof Error ? error.message : error;
|
|
this.telemetryService.publicLog('debugMisconfiguration', { type: configuration ? configuration.type : undefined, error: errorMessage });
|
|
this.setStateAndEmit(session.getId(), debug.State.Inactive);
|
|
if (!session.disconnected) {
|
|
session.disconnect().done(null, errors.onUnexpectedError);
|
|
}
|
|
// Show the repl if some error got logged there #5870
|
|
if (this.model.getReplElements().length > 0) {
|
|
this.panelService.openPanel(debug.REPL_ID, false).done(undefined, errors.onUnexpectedError);
|
|
}
|
|
|
|
const configureAction = this.instantiationService.createInstance(debugactions.ConfigureAction, debugactions.ConfigureAction.ID, debugactions.ConfigureAction.LABEL);
|
|
const actions = (error.actions && error.actions.length) ? error.actions.concat([configureAction]) : [CloseAction, configureAction];
|
|
this.messageService.show(severity.Error, { message: errorMessage, actions });
|
|
});
|
|
});
|
|
}
|
|
|
|
private runPreLaunchTask(taskName: string): TPromise<ITaskSummary> {
|
|
if (!taskName) {
|
|
return TPromise.as(null);
|
|
}
|
|
|
|
// run a task before starting a debug session
|
|
return this.taskService.tasks().then(descriptions => {
|
|
const filteredTasks = descriptions.filter(task => task.name === taskName);
|
|
if (filteredTasks.length !== 1) {
|
|
return TPromise.wrapError(errors.create(nls.localize('DebugTaskNotFound', "Could not find the preLaunchTask \'{0}\'.", taskName), {
|
|
actions: [
|
|
this.instantiationService.createInstance(debugactions.ConfigureAction, debugactions.ConfigureAction.ID, debugactions.ConfigureAction.LABEL),
|
|
this.taskService.configureAction(),
|
|
CloseAction
|
|
]
|
|
}));
|
|
}
|
|
|
|
// task is already running - nothing to do.
|
|
if (this.lastTaskEvent && this.lastTaskEvent.taskName === taskName) {
|
|
return TPromise.as(null);
|
|
}
|
|
|
|
if (this.lastTaskEvent) {
|
|
// there is a different task running currently.
|
|
return TPromise.wrapError(errors.create(nls.localize('differentTaskRunning', "There is a task {0} running. Can not run pre launch task {1}.", this.lastTaskEvent.taskName, taskName)));
|
|
}
|
|
|
|
// no task running, execute the preLaunchTask.
|
|
const taskPromise = this.taskService.run(filteredTasks[0].id).then(result => {
|
|
this.lastTaskEvent = null;
|
|
return result;
|
|
}, err => {
|
|
this.lastTaskEvent = null;
|
|
});
|
|
|
|
if (filteredTasks[0].isBackground) {
|
|
return new TPromise((c, e) => this.taskService.addOneTimeDisposableListener(TaskServiceEvents.Inactive, () => c(null)));
|
|
}
|
|
|
|
return taskPromise;
|
|
});
|
|
}
|
|
|
|
private rawAttach(session: RawDebugSession, port: number): TPromise<any> {
|
|
if (session) {
|
|
return session.attach({ port });
|
|
}
|
|
|
|
const sessionId = generateUuid();
|
|
this.setStateAndEmit(sessionId, debug.State.Initializing);
|
|
return this.configurationManager.getConfiguration(this.viewModel.selectedConfigurationName).then(config => {
|
|
config.request = 'attach';
|
|
config.port = port;
|
|
this.doCreateProcess(sessionId, config);
|
|
});
|
|
}
|
|
|
|
public deemphasizeSource(uri: uri): void {
|
|
this.model.deemphasizeSource(uri);
|
|
}
|
|
|
|
public restartProcess(process: debug.IProcess): TPromise<any> {
|
|
if (!process) {
|
|
return this.createProcess(this.viewModel.selectedConfigurationName);
|
|
}
|
|
|
|
if (process.session.configuration.capabilities.supportsRestartRequest) {
|
|
return process.session.custom('restart', null);
|
|
}
|
|
const preserveFocus = process.getId() === this.viewModel.focusedProcess.getId();
|
|
|
|
return process.session.disconnect(true).then(() =>
|
|
new TPromise<void>((c, e) => {
|
|
setTimeout(() => {
|
|
this.createProcess(process.name).then(() => c(null), err => e(err));
|
|
}, 300);
|
|
})
|
|
).then(() => {
|
|
if (preserveFocus) {
|
|
// Restart should preserve the focused process
|
|
const restartedProcess = this.model.getProcesses().filter(p => p.name === process.name).pop();
|
|
if (restartedProcess && restartedProcess !== this.viewModel.focusedProcess) {
|
|
this.focusStackFrameAndEvaluate(null, restartedProcess);
|
|
}
|
|
}
|
|
});
|
|
}
|
|
|
|
private onSessionEnd(session: RawDebugSession): void {
|
|
const bpsExist = this.model.getBreakpoints().length > 0;
|
|
this.telemetryService.publicLog('debugSessionStop', {
|
|
type: session.configuration.type,
|
|
success: session.emittedStopped || !bpsExist,
|
|
sessionLengthInSeconds: session.getLengthInSeconds(),
|
|
breakpointCount: this.model.getBreakpoints().length,
|
|
watchExpressionsCount: this.model.getWatchExpressions().length
|
|
});
|
|
|
|
try {
|
|
this.toDisposeOnSessionEnd.set(session.getId(), lifecycle.dispose(this.toDisposeOnSessionEnd.get(session.getId())));
|
|
} catch (e) {
|
|
// an internal module might be open so the dispose can throw -> ignore and continue with stop session.
|
|
}
|
|
|
|
this.model.removeProcess(session.getId());
|
|
if (this.viewModel.focusedProcess.getId() === session.getId()) {
|
|
this.focusStackFrameAndEvaluate(null).done(null, errors.onUnexpectedError);
|
|
}
|
|
this.setStateAndEmit(session.getId(), debug.State.Inactive);
|
|
|
|
if (this.model.getProcesses().length === 0) {
|
|
this.partService.removeClass('debugging');
|
|
// set breakpoints back to unverified since the session ended.
|
|
const data: { [id: string]: { line: number, verified: boolean } } = {};
|
|
this.model.getBreakpoints().forEach(bp => {
|
|
data[bp.getId()] = { line: bp.lineNumber, verified: false };
|
|
});
|
|
this.model.updateBreakpoints(data);
|
|
|
|
this.inDebugMode.reset();
|
|
this.viewModel.setMultiProcessView(false);
|
|
|
|
if (this.partService.isVisible(Parts.SIDEBAR_PART) && this.configurationService.getConfiguration<debug.IDebugConfiguration>('debug').openExplorerOnEnd) {
|
|
this.viewletService.openViewlet(EXPLORER_VIEWLET_ID).done(null, errors.onUnexpectedError);
|
|
}
|
|
}
|
|
}
|
|
|
|
public getModel(): debug.IModel {
|
|
return this.model;
|
|
}
|
|
|
|
public getViewModel(): debug.IViewModel {
|
|
return this.viewModel;
|
|
}
|
|
|
|
public getConfigurationManager(): debug.IConfigurationManager {
|
|
return this.configurationManager;
|
|
}
|
|
|
|
private sendAllBreakpoints(process?: debug.IProcess): TPromise<any> {
|
|
return TPromise.join(distinct(this.model.getBreakpoints(), bp => bp.uri.toString()).map(bp => this.sendBreakpoints(bp.uri, false, process)))
|
|
.then(() => this.sendFunctionBreakpoints(process))
|
|
// send exception breakpoints at the end since some debug adapters rely on the order
|
|
.then(() => this.sendExceptionBreakpoints(process));
|
|
}
|
|
|
|
private sendBreakpoints(modelUri: uri, sourceModified = false, targetProcess?: debug.IProcess): TPromise<void> {
|
|
|
|
const sendBreakpointsToProcess = (process: debug.IProcess): TPromise<void> => {
|
|
const session = <RawDebugSession>process.session;
|
|
if (!session.readyForBreakpoints) {
|
|
return TPromise.as(null);
|
|
}
|
|
if (this.textFileService.isDirty(modelUri)) {
|
|
// Only send breakpoints for a file once it is not dirty #8077
|
|
this.breakpointsToSendOnResourceSaved.add(modelUri.toString());
|
|
return TPromise.as(null);
|
|
}
|
|
|
|
const breakpointsToSend = distinct(this.model.getBreakpoints().filter(bp => this.model.areBreakpointsActivated() && bp.enabled && bp.uri.toString() === modelUri.toString()),
|
|
bp => bp.lineNumber.toString());
|
|
|
|
let rawSource: DebugProtocol.Source;
|
|
for (let t of process.getAllThreads()) {
|
|
const callStack = t.getCallStack();
|
|
if (callStack) {
|
|
for (let sf of callStack) {
|
|
if (sf.source.uri.toString() === modelUri.toString()) {
|
|
rawSource = sf.source.raw;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
rawSource = rawSource || { path: paths.normalize(modelUri.fsPath, true), name: paths.basename(modelUri.fsPath) };
|
|
|
|
return session.setBreakpoints({
|
|
source: rawSource,
|
|
lines: breakpointsToSend.map(bp => bp.lineNumber),
|
|
breakpoints: breakpointsToSend.map(bp => ({ line: bp.lineNumber, condition: bp.condition, hitCondition: bp.hitCondition })),
|
|
sourceModified
|
|
}).then(response => {
|
|
if (!response || !response.body) {
|
|
return;
|
|
}
|
|
|
|
const data: { [id: string]: { line?: number, verified: boolean } } = {};
|
|
for (let i = 0; i < breakpointsToSend.length; i++) {
|
|
data[breakpointsToSend[i].getId()] = response.body.breakpoints[i];
|
|
}
|
|
|
|
this.model.updateBreakpoints(data);
|
|
});
|
|
};
|
|
|
|
return this.sendToOneOrAllProcesses(targetProcess, sendBreakpointsToProcess);
|
|
}
|
|
|
|
private sendFunctionBreakpoints(targetProcess?: debug.IProcess): TPromise<void> {
|
|
const sendFunctionBreakpointsToProcess = (process: debug.IProcess): TPromise<void> => {
|
|
const session = <RawDebugSession>process.session;
|
|
if (!session.readyForBreakpoints || !session.configuration.capabilities.supportsFunctionBreakpoints) {
|
|
return TPromise.as(null);
|
|
}
|
|
|
|
const breakpointsToSend = this.model.getFunctionBreakpoints().filter(fbp => fbp.enabled && this.model.areBreakpointsActivated());
|
|
return session.setFunctionBreakpoints({ breakpoints: breakpointsToSend }).then(response => {
|
|
if (!response || !response.body) {
|
|
return;
|
|
}
|
|
|
|
const data: { [id: string]: { name?: string, verified?: boolean } } = {};
|
|
for (let i = 0; i < breakpointsToSend.length; i++) {
|
|
data[breakpointsToSend[i].getId()] = response.body.breakpoints[i];
|
|
}
|
|
|
|
this.model.updateFunctionBreakpoints(data);
|
|
});
|
|
};
|
|
|
|
return this.sendToOneOrAllProcesses(targetProcess, sendFunctionBreakpointsToProcess);
|
|
}
|
|
|
|
private sendExceptionBreakpoints(targetProcess?: debug.IProcess): TPromise<void> {
|
|
const sendExceptionBreakpointsToProcess = (process: debug.IProcess): TPromise<any> => {
|
|
const session = <RawDebugSession>process.session;
|
|
if (!session.readyForBreakpoints || this.model.getExceptionBreakpoints().length === 0) {
|
|
return TPromise.as(null);
|
|
}
|
|
|
|
const enabledExceptionBps = this.model.getExceptionBreakpoints().filter(exb => exb.enabled);
|
|
return session.setExceptionBreakpoints({ filters: enabledExceptionBps.map(exb => exb.filter) });
|
|
};
|
|
|
|
return this.sendToOneOrAllProcesses(targetProcess, sendExceptionBreakpointsToProcess);
|
|
}
|
|
|
|
private sendToOneOrAllProcesses(process: debug.IProcess, send: (process: debug.IProcess) => TPromise<void>): TPromise<void> {
|
|
if (process) {
|
|
return send(process);
|
|
}
|
|
|
|
return TPromise.join(this.model.getProcesses().map(p => send(p))).then(() => void 0);
|
|
}
|
|
|
|
private onFileChanges(fileChangesEvent: FileChangesEvent): void {
|
|
this.model.removeBreakpoints(this.model.getBreakpoints().filter(bp =>
|
|
fileChangesEvent.contains(bp.uri, FileChangeType.DELETED)));
|
|
|
|
fileChangesEvent.getUpdated().forEach(event => {
|
|
if (this.breakpointsToSendOnResourceSaved.has(event.resource.toString())) {
|
|
this.breakpointsToSendOnResourceSaved.delete(event.resource.toString());
|
|
this.sendBreakpoints(event.resource, true).done(null, errors.onUnexpectedError);
|
|
}
|
|
});
|
|
}
|
|
|
|
private store(): void {
|
|
this.storageService.store(DEBUG_BREAKPOINTS_KEY, JSON.stringify(this.model.getBreakpoints()), StorageScope.WORKSPACE);
|
|
this.storageService.store(DEBUG_BREAKPOINTS_ACTIVATED_KEY, this.model.areBreakpointsActivated() ? 'true' : 'false', StorageScope.WORKSPACE);
|
|
this.storageService.store(DEBUG_FUNCTION_BREAKPOINTS_KEY, JSON.stringify(this.model.getFunctionBreakpoints()), StorageScope.WORKSPACE);
|
|
this.storageService.store(DEBUG_EXCEPTION_BREAKPOINTS_KEY, JSON.stringify(this.model.getExceptionBreakpoints()), StorageScope.WORKSPACE);
|
|
this.storageService.store(DEBUG_SELECTED_CONFIG_NAME_KEY, this.viewModel.selectedConfigurationName, StorageScope.WORKSPACE);
|
|
this.storageService.store(DEBUG_WATCH_EXPRESSIONS_KEY, JSON.stringify(this.model.getWatchExpressions().map(we => ({ name: we.name, id: we.getId() }))), StorageScope.WORKSPACE);
|
|
}
|
|
|
|
public dispose(): void {
|
|
this.toDisposeOnSessionEnd.forEach(toDispose => lifecycle.dispose(toDispose));
|
|
this.toDispose = lifecycle.dispose(this.toDispose);
|
|
}
|
|
}
|