/*--------------------------------------------------------------------------------------------- * Copyright (c) Microsoft Corporation. All rights reserved. * Licensed under the MIT License. See License.txt in the project root for license information. *--------------------------------------------------------------------------------------------*/ import nls = require('vs/nls'); import lifecycle = require('vs/base/common/lifecycle'); import mime = require('vs/base/common/mime'); import ee = require('vs/base/common/eventEmitter'); import uri from 'vs/base/common/uri'; import arrays = require('vs/base/common/arrays'); import types = require('vs/base/common/types'); import errors = require('vs/base/common/errors'); import severity from 'vs/base/common/severity'; import { Promise, TPromise } from 'vs/base/common/winjs.base'; import editor = require('vs/editor/common/editorCommon'); import editorbrowser = require('vs/editor/browser/editorBrowser'); import wbeditorcommon = require('vs/workbench/common/editor'); import debug = require('vs/workbench/parts/debug/common/debug'); import session = require('vs/workbench/parts/debug/node/rawDebugSession'); import model = require('vs/workbench/parts/debug/common/debugModel'); import debuginputs = require('vs/workbench/parts/debug/browser/debugEditorInputs'); import viewmodel = require('vs/workbench/parts/debug/common/debugViewModel'); import debugactions = require('vs/workbench/parts/debug/browser/debugActions'); import { ConfigurationManager } from 'vs/workbench/parts/debug/node/debugConfigurationManager'; import { Repl } from 'vs/workbench/parts/debug/browser/replEditor'; import { Source } from 'vs/workbench/parts/debug/common/debugSource'; import { Position } from 'vs/platform/editor/common/editor'; import { ITaskService , TaskEvent, TaskType, TaskServiceEvents} from 'vs/workbench/parts/tasks/common/taskService'; import { IViewletService } from 'vs/workbench/services/viewlet/common/viewletService'; import { IPartService } from 'vs/workbench/services/part/common/partService'; import { ITextFileService } from 'vs/workbench/parts/files/common/files'; import { IWorkspaceContextService } from 'vs/workbench/services/workspace/common/contextService'; import { IConfigurationService } from 'vs/platform/configuration/common/configuration'; import { IInstantiationService } from 'vs/platform/instantiation/common/instantiation'; import { IFileService, FileChangesEvent, FileChangeType, EventType } from 'vs/platform/files/common/files'; import { IEventService } from 'vs/platform/event/common/event'; import { IMessageService, CloseAction } from 'vs/platform/message/common/message'; import { ITelemetryService } from 'vs/platform/telemetry/common/telemetry'; import { IStorageService, StorageScope } from 'vs/platform/storage/common/storage'; import { IWorkbenchEditorService } from 'vs/workbench/services/editor/common/editorService'; import { ILifecycleService } from 'vs/platform/lifecycle/common/lifecycle'; import { IPluginService, IPluginDescription } from 'vs/platform/plugins/common/plugins'; import { IOutputService } from 'vs/workbench/parts/output/common/output'; import { IKeybindingService, IKeybindingContextKey } from 'vs/platform/keybinding/common/keybindingService'; import { IWindowService, IBroadcast } from 'vs/workbench/services/window/electron-browser/windowService'; import { ILogEntry, PLUGIN_LOG_BROADCAST_CHANNEL } from 'vs/workbench/services/thread/electron-browser/threadService'; var DEBUG_BREAKPOINTS_KEY = 'debug.breakpoint'; var DEBUG_BREAKPOINTS_ACTIVATED_KEY = 'debug.breakpointactivated'; var DEBUG_FUNCTION_BREAKPOINTS_KEY = 'debug.functionbreakpoint'; var DEBUG_EXCEPTION_BREAKPOINTS_KEY = 'debug.exceptionbreakpoint'; var DEBUG_WATCH_EXPRESSIONS_KEY = 'debug.watchexpressions'; var DEBUG_SELECTED_CONFIG_NAME_KEY = 'debug.selectedconfigname'; export class DebugService extends ee.EventEmitter implements debug.IDebugService { public serviceId = debug.IDebugService; private taskService: ITaskService; private state: debug.State; private session: session.RawDebugSession; private model: model.Model; private viewModel: viewmodel.ViewModel; private configurationManager: ConfigurationManager; private debugStringEditorInputs: debuginputs.DebugStringEditorInput[]; private lastTaskEvent: TaskEvent; private toDispose: lifecycle.IDisposable[]; private inDebugMode: IKeybindingContextKey; constructor( @IStorageService private storageService: IStorageService, @IWorkbenchEditorService private editorService: IWorkbenchEditorService, @ITextFileService private textFileService: ITextFileService, @IViewletService private viewletService: IViewletService, @IFileService private fileService: IFileService, @IMessageService private messageService: IMessageService, @IPartService private partService: IPartService, @IWindowService private windowService: IWindowService, @ITelemetryService private telemetryService: ITelemetryService, @IWorkspaceContextService private contextService: IWorkspaceContextService, @IKeybindingService keybindingService: IKeybindingService, @IEventService eventService: IEventService, @ILifecycleService private lifecycleService: ILifecycleService, @IInstantiationService private instantiationService:IInstantiationService, @IPluginService private pluginService: IPluginService, @IOutputService private outputService: IOutputService ) { super(); this.toDispose = []; this.debugStringEditorInputs = []; this.session = null; this.state = debug.State.Inactive; // There is a cycle if taskService gets injected, use a workaround. this.taskService = this.instantiationService.getInstance(ITaskService); if (!this.contextService.getWorkspace()) { this.state = debug.State.Disabled; } this.configurationManager = this.instantiationService.createInstance(ConfigurationManager, this.storageService.get(DEBUG_SELECTED_CONFIG_NAME_KEY, StorageScope.WORKSPACE, 'null')); this.inDebugMode = keybindingService.createKey(debug.CONTEXT_IN_DEBUG_MODE, false); this.model = new model.Model(this.loadBreakpoints(), this.storageService.getBoolean(DEBUG_BREAKPOINTS_ACTIVATED_KEY, StorageScope.WORKSPACE, true), this.loadFunctionBreakpoints(), this.loadExceptionBreakpoints(), this.loadWatchExpressions()); this.viewModel = new viewmodel.ViewModel(); this.registerListeners(eventService, lifecycleService); } private registerListeners(eventService: IEventService, lifecycleService: ILifecycleService): void { this.toDispose.push(eventService.addListener2(EventType.FILE_CHANGES, (e: FileChangesEvent) => 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.add(this.store, this); lifecycleService.onShutdown.add(this.dispose, this); this.windowService.onBroadcast.add(this.onBroadcast, this); } private onBroadcast(broadcast: IBroadcast): void { let session = this.getActiveSession(); if (!session || session.getType() !== 'extensionHost') { return; // we are only intersted if we have an active debug session for extensionHost } // A plugin logged output, show it inside the REPL if (broadcast.channel === PLUGIN_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 (types.isObject(a) || Array.isArray(a)) { // Flush any existing simple values logged if (simpleVals.length) { this.logToRepl(simpleVals.join(' '), sev); simpleVals = []; } // Show object this.logToRepl(a, 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 += !types.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.logToRepl(simpleVals.join(' '), sev); } // Show repl this.revealRepl(true /* in background */).done(null, errors.onUnexpectedError); } } private registerSessionListeners(): void { this.toDispose.push(this.session.addListener2(debug.SessionEvents.INITIALIZED, (event: DebugProtocol.InitializedEvent) => { this.sendAllBreakpoints().done(null, errors.onUnexpectedError); this.sendExceptionBreakpoints().done(null, errors.onUnexpectedError); })); this.toDispose.push(this.session.addListener2(debug.SessionEvents.STOPPED, (event: DebugProtocol.StoppedEvent) => { this.setStateAndEmit(debug.State.Stopped, event.body.reason); var threadId = event.body.threadId; this.getThreadData(threadId).then(() => { this.session.stackTrace({ threadId: threadId, levels: 20 }).done((result) => { this.model.rawUpdate({ threadId: threadId, callStack: result.body.stackFrames, exception: event.body && event.body.reason === 'exception' }); this.windowService.getWindow().focus(); var callStack = this.model.getThreads()[threadId].callStack; if (callStack.length > 0) { this.setFocusedStackFrameAndEvaluate(callStack[0]); this.openOrRevealEditor(callStack[0].source, callStack[0].lineNumber, false, false).done(null, errors.onUnexpectedError); } else { this.setFocusedStackFrameAndEvaluate(null); } }); }, errors.onUnexpectedError); })); this.toDispose.push(this.session.addListener2(debug.SessionEvents.CONTINUED, () => { this.model.clearThreads(false); this.setFocusedStackFrameAndEvaluate(null); this.setStateAndEmit(debug.State.Running); })); this.toDispose.push(this.session.addListener2(debug.SessionEvents.THREAD, (event: DebugProtocol.ThreadEvent) => { if (event.body.reason === 'started') { this.session.threads().done((result) => { const thread = result.body.threads.filter(thread => thread.id === event.body.threadId).pop(); if (thread) { this.model.rawUpdate({ threadId: thread.id, thread: thread }); } }, errors.onUnexpectedError); } else if (event.body.reason === 'exited') { this.model.clearThreads(true, event.body.threadId); } })); this.toDispose.push(this.session.addListener2(debug.SessionEvents.DEBUGEE_TERMINATED, (event: DebugProtocol.TerminatedEvent) => { // if there is some opaque data in the body of the terminate event, just pass it to the next launch request let extensionHostData = event.body ? event.body.extensionHost : undefined; if (extensionHostData) { this.restartSession(extensionHostData).done(null, errors.onUnexpectedError); } else if (this.session) { this.session.disconnect().done(null, errors.onUnexpectedError); } })); this.toDispose.push(this.session.addListener2(debug.SessionEvents.OUTPUT, (event: DebugProtocol.OutputEvent) => { if (event.body && typeof event.body.output === 'string' && event.body.output.length > 0) { this.onOutput(event); } })); this.toDispose.push(this.session.addListener2(debug.SessionEvents.SERVER_EXIT, e => { this.onSessionEnd(); })); } private onOutput(event: DebugProtocol.OutputEvent): void { const outputSeverity = event.body.category === 'stderr' ? severity.Error : severity.Info; this.appendReplOutput(event.body.output, outputSeverity); this.revealRepl(true /* in background */).done(null, errors.onUnexpectedError); } private getThreadData(threadId: number): Promise { return this.model.getThreads()[threadId] ? Promise.as(true) : this.session.threads().then((response: DebugProtocol.ThreadsResponse) => { const thread = response.body.threads.filter(t => t.id === threadId).pop(); if (!thread) { throw new Error('Did not get a thread from debug adapter with id ' + threadId); } this.model.rawUpdate({ threadId: thread.id, thread: thread }); }); } private loadBreakpoints(): debug.IBreakpoint[] { try { return JSON.parse(this.storageService.get(DEBUG_BREAKPOINTS_KEY, StorageScope.WORKSPACE, '[]')).map((breakpoint: any) => { return new model.Breakpoint(new Source(breakpoint.source.name, breakpoint.source.uri, breakpoint.source.reference), breakpoint.desiredLineNumber || breakpoint.lineNumber, breakpoint.enabled, breakpoint.condition); }); } catch (e) { return []; } } private loadFunctionBreakpoints(): debug.IFunctionBreakpoint[] { try { return JSON.parse(this.storageService.get(DEBUG_FUNCTION_BREAKPOINTS_KEY, StorageScope.WORKSPACE, '[]')).map((fb: any) => { return new model.FunctionBreakpoint(fb.name, fb.enabled); }); } catch (e) { return []; } } private loadExceptionBreakpoints(): debug.IExceptionBreakpoint[] { var result: debug.IExceptionBreakpoint[] = null; try { result = JSON.parse(this.storageService.get(DEBUG_EXCEPTION_BREAKPOINTS_KEY, StorageScope.WORKSPACE, '[]')).map((exBreakpoint: any) => { return new model.ExceptionBreakpoint(exBreakpoint.name, exBreakpoint.enabled); }); } catch (e) { result = []; } return result.length > 0 ? result : [new model.ExceptionBreakpoint('all', false), new model.ExceptionBreakpoint('uncaught', true)]; } private loadWatchExpressions(): model.Expression[] { try { return JSON.parse(this.storageService.get(DEBUG_WATCH_EXPRESSIONS_KEY, StorageScope.WORKSPACE, '[]')).map((watch: any) => { return new model.Expression(watch.name, false, watch.id); }); } catch (e) { return []; } } public getState(): debug.State { return this.state; } private setStateAndEmit(newState: debug.State, data?: any): void { this.state = newState; this.emit(debug.ServiceEvents.STATE_CHANGED, data); } public get enabled(): boolean { return !!this.contextService.getWorkspace(); } public setFocusedStackFrameAndEvaluate(focusedStackFrame: debug.IStackFrame): void { this.viewModel.setFocusedStackFrame(focusedStackFrame); if (focusedStackFrame) { this.model.evaluateWatchExpressions(this.session, focusedStackFrame); } else { this.model.clearWatchExpressionValues(); } } public setBreakpointsForModel(modelUri: uri, rawData: debug.IRawBreakpoint[]): Promise { this.model.removeBreakpoints( this.model.getBreakpoints().filter(bp => bp.source.uri.toString() === modelUri.toString())); this.model.addBreakpoints(rawData); return this.sendBreakpoints(modelUri); } public toggleBreakpoint(rawBreakpoint: debug.IRawBreakpoint): Promise { const breakpoint = this.model.getBreakpoints().filter(bp => bp.lineNumber === rawBreakpoint.lineNumber && bp.source.uri.toString() === rawBreakpoint.uri.toString()).pop(); if (breakpoint) { this.model.removeBreakpoints([breakpoint]); } else { this.model.addBreakpoints([rawBreakpoint]); } return this.sendBreakpoints(rawBreakpoint.uri); } public enableOrDisableAllBreakpoints(enabled: boolean): Promise { this.model.enableOrDisableAllBreakpoints(enabled); return this.sendAllBreakpoints(); } public toggleEnablement(element: debug.IEnablement): Promise { this.model.toggleEnablement(element); if (element instanceof model.Breakpoint) { var breakpoint = element; return this.sendBreakpoints(breakpoint.source.uri); } else if (element instanceof model.FunctionBreakpoint) { // TODO@Isidor send function breakpoints and return } return this.sendExceptionBreakpoints(); } public removeAllBreakpoints(): Promise { const urisToClear = arrays.distinct(this.model.getBreakpoints(), bp => bp.source.uri.toString()).map(bp => bp.source.uri); this.model.removeBreakpoints(this.model.getBreakpoints()); return Promise.join(urisToClear.map(uri => this.sendBreakpoints(uri))); } public toggleBreakpointsActivated(): Promise { this.model.toggleBreakpointsActivated(); return this.sendAllBreakpoints(); } public addFunctionBreakpoint(functionName?: string): Promise { this.model.addFunctionBreakpoint(functionName); // TODO@Isidor send updated function breakpoints return Promise.as(true); } public renameFunctionBreakpoint(id: string, newFunctionName: string): Promise { this.model.renameFunctionBreakpoint(id, newFunctionName); // TODO@Isidor send updated function breakpoints return Promise.as(true); } public removeFunctionBreakpoints(id?: string): Promise { this.model.removeFunctionBreakpoints(id); // TODO@Isidor send updated function breakpoints return Promise.as(true); } public addReplExpression(name: string): Promise { return this.model.addReplExpression(this.session, this.viewModel.getFocusedStackFrame(), name); } public logToRepl(value: string, severity?: severity): void; public logToRepl(value: { [key: string]: any }, severity?: severity): void; public logToRepl(value: any, severity?: severity): void { this.model.logToRepl(value, severity); } public appendReplOutput(value: string, severity?: severity): void { this.model.appendReplOutput(value, severity); } public clearReplExpressions(): void { this.model.clearReplExpressions(); } public addWatchExpression(name: string): Promise { return this.model.addWatchExpression(this.session, this.viewModel.getFocusedStackFrame(), name); } public renameWatchExpression(id: string, newName: string): Promise { return this.model.renameWatchExpression(this.session, this.viewModel.getFocusedStackFrame(), id, newName); } public clearWatchExpressions(id?: string): void { this.model.clearWatchExpressions(id); } public createSession(extensionHostData?: any, openViewlet = true): Promise { this.textFileService.saveAll().done(null, errors.onUnexpectedError); if (!extensionHostData) { this.clearReplExpressions(); } return this.pluginService.onReady().then(() => this.configurationManager.setConfiguration(this.configurationManager.getConfigurationName(), extensionHostData)).then(() => { const configuration = this.configurationManager.getConfiguration(); if (!configuration) { return this.configurationManager.openConfigFile(false).then(openend => { if (openend) { this.messageService.show(severity.Info, nls.localize('NewLaunchConfig', "Please set up the launch configuration file to debug your application.")); } }); } const adapter = this.configurationManager.getAdapter(); if (!adapter) { return Promise.wrapError(new Error(`Configured debug type '${ configuration.type }' is not supported.`)); } return this.runPreLaunchTask(configuration).then(() => { this.session = new session.RawDebugSession(this.messageService, this.telemetryService, configuration.debugServer, adapter); this.registerSessionListeners(); return this.session.initialize({ adapterID: configuration.type, linesStartAt1: true, pathFormat: 'path' }).then((result: DebugProtocol.InitializeResponse) => { this.setStateAndEmit(debug.State.Initializing); return configuration.request === 'attach' ? this.session.attach(configuration) : this.session.launch(configuration); }).then((result: DebugProtocol.Response) => { if (openViewlet) { this.viewletService.openViewlet(debug.VIEWLET_ID); } this.partService.addClass('debugging'); this.contextService.updateOptions('editor', { glyphMargin: true }); this.inDebugMode.set(true); this.telemetryService.publicLog('debugSessionStart', { type: configuration.type, breakpointCount: this.model.getBreakpoints().length, exceptionBreakpoints: this.model.getExceptionBreakpoints() }); }).then(undefined, (error: Error) => { this.telemetryService.publicLog('debugMisconfiguration', { type: configuration ? configuration.type : undefined }); if (this.session) { this.session.disconnect(); } return Promise.wrapError(errors.create(error.message, { actions: [CloseAction, this.instantiationService.createInstance(debugactions.ConfigureAction, debugactions.ConfigureAction.ID, debugactions.ConfigureAction.LABEL)] })); }); }); }); } private runPreLaunchTask(config: debug.IConfig): Promise { // Only run the task if we are not reattaching (extensionHostData is defined). if (!config.preLaunchTask || config.extensionHostData) { return Promise.as(true); } // Run a build task before starting a debug session return this.taskService.tasks().then(descriptions => { let filteredTasks = descriptions.filter(task => task.name === config.preLaunchTask); if (filteredTasks.length !== 1) { this.messageService.show(severity.Warning, nls.localize('DebugTaskNotFound', "Could not find a unique task \'{0}\'. Make sure the task exists and that it has a unique name.", config.preLaunchTask)); return Promise.as(true); } // Task is already running - nothing to do. if (this.lastTaskEvent && this.lastTaskEvent.taskName === config.preLaunchTask) { return Promise.as(true); } if (this.lastTaskEvent) { // There is a different task running currently. return Promise.wrapError(errors.create(nls.localize('differentTaskRunning', "There is a task {0} running. Can not run pre launch task {1}.", this.lastTaskEvent.taskName, config.preLaunchTask))); } // No task running, execute the preLaunchTask. this.outputService.showOutput('Tasks', true, true); const taskPromise = this.taskService.run(filteredTasks[0].id).then(result => { this.lastTaskEvent = null; }, err => { this.lastTaskEvent = null; }); return filteredTasks[0].isWatching ? Promise.as(true) : taskPromise; }); } public restartSession(extensionHostData?: any): Promise { return this.session ? this.session.disconnect(true).then(() => { new Promise(c => { setTimeout(() => { this.createSession(extensionHostData, false).then(() => c(true)); }, 300); }); }) : this.createSession(extensionHostData, false); } public getActiveSession(): debug.IRawDebugSession { return this.session; } private onSessionEnd(): void { try { this.debugStringEditorInputs = lifecycle.disposeAll(this.debugStringEditorInputs); } catch (e) { // An internal module might be open so the dispose can throw -> ignore and continue with stop session. } if (this.session) { var bpsExist = this.model.getBreakpoints().length > 0; this.telemetryService.publicLog('debugSessionStop', { type: this.session.getType(), success: this.session.emittedStopped || !bpsExist, sessionLengthInSeconds: this.session.getLengthInSeconds(), breakpointCount: this.model.getBreakpoints().length }); } this.session = null; this.partService.removeClass('debugging'); this.contextService.updateOptions('editor', { hover: true }); this.editorService.focusEditor(); this.model.clearThreads(true); this.model.getBreakpoints().forEach(bp => this.model.updateBreakpoint(bp.getId(), bp.lineNumber, false)); this.setFocusedStackFrameAndEvaluate(null); this.setStateAndEmit(debug.State.Inactive); this.inDebugMode.reset(); } public getModel(): debug.IModel { return this.model; } public getViewModel(): debug.IViewModel { return this.viewModel; } public openOrRevealEditor(source: Source, lineNumber: number, preserveFocus: boolean, sideBySide: boolean): Promise { const visibleEditors = this.editorService.getVisibleEditors(); for (var i = 0; i < visibleEditors.length; i++) { const fileInput = wbeditorcommon.asFileEditorInput(visibleEditors[i].input); if (fileInput && fileInput.getResource().toString() === source.uri.toString()) { const control = visibleEditors[i].getControl(); if (control) { control.revealLineInCenterIfOutsideViewport(lineNumber); control.setSelection({ startLineNumber: lineNumber, startColumn: 1, endLineNumber: lineNumber, endColumn: 1 }); return this.editorService.openEditor(visibleEditors[i].input, wbeditorcommon.TextEditorOptions.create({ preserveFocus: preserveFocus, forceActive: true }), visibleEditors[i].position); } return Promise.as(null); } } if (source.inMemory) { // Internal module if (source.reference !== 0 && this.session) { return this.session.source({ sourceReference: source.reference }).then(response => { const editorInput = this.getDebugStringEditorInput(source, response.body.content, mime.guessMimeTypes(source.name)[0]); return this.editorService.openEditor(editorInput, wbeditorcommon.TextEditorOptions.create({ selection: { startLineNumber: lineNumber, startColumn: 1, endLineNumber: lineNumber, endColumn: 1 }, preserveFocus: preserveFocus }), sideBySide); }); } return this.sourceIsUnavailable(source, sideBySide); } return this.fileService.resolveFile(source.uri).then(() => this.editorService.openEditor({ resource: source.uri, options: { selection: { startLineNumber: lineNumber, startColumn: 1, endLineNumber: lineNumber, endColumn: 1 }, preserveFocus: preserveFocus } }, sideBySide), err => this.sourceIsUnavailable(source, sideBySide) ); } private sourceIsUnavailable(source: Source, sideBySide: boolean): Promise { this.model.sourceIsUnavailable(source); const editorInput = this.getDebugStringEditorInput(source, 'Source is not available.', 'text/plain'); return this.editorService.openEditor(editorInput, wbeditorcommon.TextEditorOptions.create({ preserveFocus: true }), sideBySide); } public revealRepl(inBackground: boolean = false): Promise { let editors = this.editorService.getVisibleEditors(); // First check if repl is already opened for (let i = 0; i < editors.length; i++) { let editor = editors[i]; if (editor.input instanceof debuginputs.ReplEditorInput) { if (!inBackground) { return this.editorService.focusEditor(editor); } return Promise.as(null); } } // Then find a position but try to not replace an existing file editor in any of the positions let position = Position.LEFT; let lastIndex = editors.length - 1; if (editors.length === 3) { position = wbeditorcommon.asFileEditorInput(editors[lastIndex].input, true) ? null : Position.RIGHT; } else if (editors.length === 2) { position = wbeditorcommon.asFileEditorInput(editors[lastIndex].input, true) ? Position.RIGHT : Position.CENTER; } else if (editors.length) { position = wbeditorcommon.asFileEditorInput(editors[lastIndex].input, true) ? Position.CENTER : Position.LEFT; } if (position === null) { return Promise.as(null); // could not find a good position, return } // open repl return this.editorService.openEditor(debuginputs.ReplEditorInput.getInstance(), wbeditorcommon.TextEditorOptions.create({ preserveFocus: inBackground }), position).then((editor: Repl) => { const elements = this.model.getReplElements(); if (!inBackground && elements.length > 0) { return editor.reveal(elements[elements.length - 1]); } }); } public canSetBreakpointsIn(model: editor.IModel, lineNumber: number): boolean { return this.configurationManager.canSetBreakpointsIn(model, lineNumber); } public getConfiguration(): debug.IConfig { return this.configurationManager.getConfiguration(); } public setConfiguration(name: string): Promise { return this.configurationManager.setConfiguration(name); } public openConfigFile(sideBySide: boolean): TPromise { return this.configurationManager.openConfigFile(sideBySide); } public loadLaunchConfig(): TPromise { return this.configurationManager.loadLaunchConfig(); } private getDebugStringEditorInput(source: Source, value: string, mtype: string): debuginputs.DebugStringEditorInput { var filtered = this.debugStringEditorInputs.filter(input => input.getResource().toString() === source.uri.toString()); if (filtered.length === 0) { var result = this.instantiationService.createInstance(debuginputs.DebugStringEditorInput, source.name, source.uri, 'internal module', value, mtype, void 0); this.debugStringEditorInputs.push(result); return result; } else { return filtered[0]; } } public sendAllBreakpoints(): Promise { return Promise.join(arrays.distinct(this.model.getBreakpoints(), bp => bp.source.uri.toString()).map(bp => this.sendBreakpoints(bp.source.uri))); } private sendBreakpoints(modelUri: uri): Promise { if (!this.session) { return Promise.as(null); } const breakpointsToSend = arrays.distinct( this.model.getBreakpoints().filter(bp => this.model.areBreakpointsActivated() && bp.enabled && bp.source.uri.toString() === modelUri.toString()), bp => `${ bp.desiredLineNumber }` ); return this.session.setBreakpoints({ source: Source.fromUri(modelUri).toRawSource(), lines: breakpointsToSend.map(bp => bp.desiredLineNumber) }).then(response => { let index = 0; breakpointsToSend.forEach(bp => { const rawBreakpointData = response.body.breakpoints[index++]; if (bp.lineNumber !== rawBreakpointData.line || !rawBreakpointData.verified !== bp.error) { this.model.updateBreakpoint(bp.getId(), rawBreakpointData.line, !rawBreakpointData.verified); } }); }); } private sendExceptionBreakpoints(): Promise { if (this.session) { var enabledExBreakpoints = this.model.getExceptionBreakpoints().filter(exb => exb.enabled); return this.session.setExceptionBreakpoints({ filters: enabledExBreakpoints.map(exb => exb.name) }); } } private onFileChanges(fileChangesEvent: FileChangesEvent): void { this.model.removeBreakpoints(this.model.getBreakpoints().filter(bp => fileChangesEvent.contains(bp.source.uri, FileChangeType.DELETED))); } 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.configurationManager.getConfigurationName(), StorageScope.WORKSPACE); this.storageService.store(DEBUG_WATCH_EXPRESSIONS_KEY, JSON.stringify(this.model.getWatchExpressions()), StorageScope.WORKSPACE); } public dispose(): void { if (this.session) { this.session.disconnect(); this.session = null; } this.model.dispose(); this.toDispose = lifecycle.disposeAll(this.toDispose); } }