debugService.ts 46.3 KB
Newer Older
E
Erich Gamma 已提交
1 2 3 4 5
/*---------------------------------------------------------------------------------------------
 *  Copyright (c) Microsoft Corporation. All rights reserved.
 *  Licensed under the MIT License. See License.txt in the project root for license information.
 *--------------------------------------------------------------------------------------------*/

I
isidor 已提交
6 7
import * as nls from 'vs/nls';
import * as lifecycle from 'vs/base/common/lifecycle';
J
Johannes Rieken 已提交
8
import Event, { Emitter } from 'vs/base/common/event';
9
import * as paths from 'vs/base/common/paths';
I
isidor 已提交
10
import * as strings from 'vs/base/common/strings';
I
isidor 已提交
11
import { generateUuid } from 'vs/base/common/uuid';
E
Erich Gamma 已提交
12
import uri from 'vs/base/common/uri';
J
Johannes Rieken 已提交
13
import { Action } from 'vs/base/common/actions';
I
isidor 已提交
14 15 16
import { first, distinct } from 'vs/base/common/arrays';
import { isObject, isUndefinedOrNull } from 'vs/base/common/types';
import * as errors from 'vs/base/common/errors';
E
Erich Gamma 已提交
17
import severity from 'vs/base/common/severity';
J
Johannes Rieken 已提交
18
import { TPromise } from 'vs/base/common/winjs.base';
I
isidor 已提交
19
import * as aria from 'vs/base/browser/ui/aria/aria';
20
import { Client as TelemetryClient } from 'vs/base/parts/ipc/node/ipc.cp';
J
Johannes Rieken 已提交
21 22 23 24 25
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';
26
import { FileChangesEvent, FileChangeType, IFileService } from 'vs/platform/files/common/files';
J
Johannes Rieken 已提交
27
import { IMessageService, CloseAction } from 'vs/platform/message/common/message';
28
import { IWindowsService } from 'vs/platform/windows/common/windows';
J
Johannes Rieken 已提交
29 30 31
import { ITelemetryService } from 'vs/platform/telemetry/common/telemetry';
import { TelemetryService } from 'vs/platform/telemetry/common/telemetryService';
import { TelemetryAppenderClient } from 'vs/platform/telemetry/common/telemetryIpc';
32
import { ICommandService } from 'vs/platform/commands/common/commands';
J
Johannes Rieken 已提交
33
import { IStorageService, StorageScope } from 'vs/platform/storage/common/storage';
I
isidor 已提交
34
import * as debug from 'vs/workbench/parts/debug/common/debug';
J
Johannes Rieken 已提交
35
import { RawDebugSession } from 'vs/workbench/parts/debug/electron-browser/rawDebugSession';
I
isidor 已提交
36
import { Model, ExceptionBreakpoint, FunctionBreakpoint, Breakpoint, Expression, OutputNameValueElement, ExpressionContainer, Process } from 'vs/workbench/parts/debug/common/debugModel';
I
isidor 已提交
37 38
import { ViewModel } from 'vs/workbench/parts/debug/common/debugViewModel';
import * as debugactions from 'vs/workbench/parts/debug/browser/debugActions';
39
import { ConfigurationManager } from 'vs/workbench/parts/debug/electron-browser/debugConfigurationManager';
I
isidor 已提交
40
import { ToggleMarkersPanelAction } from 'vs/workbench/parts/markers/browser/markersPanelActions';
J
Johannes Rieken 已提交
41 42
import { ITaskService, TaskEvent, TaskType, TaskServiceEvents, ITaskSummary } from 'vs/workbench/parts/tasks/common/taskService';
import { TaskError, TaskErrors } from 'vs/workbench/parts/tasks/common/taskSystem';
I
isidor 已提交
43
import { VIEWLET_ID as EXPLORER_VIEWLET_ID } from 'vs/workbench/parts/files/common/files';
B
Benjamin Pasero 已提交
44
import { IViewletService } from 'vs/workbench/services/viewlet/browser/viewlet';
J
Johannes Rieken 已提交
45
import { IPanelService } from 'vs/workbench/services/panel/common/panelService';
46
import { IPartService, Parts } from 'vs/workbench/services/part/common/partService';
J
Johannes Rieken 已提交
47 48 49 50
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';
J
Joao Moreno 已提交
51
import { IWindowIPCService, IBroadcast } from 'vs/workbench/services/window/electron-browser/windowService';
S
Sandeep Somavarapu 已提交
52
import { ILogEntry, EXTENSION_LOG_BROADCAST_CHANNEL, EXTENSION_ATTACH_BROADCAST_CHANNEL, EXTENSION_TERMINATE_BROADCAST_CHANNEL } from 'vs/workbench/electron-browser/extensionHost';
E
Erich Gamma 已提交
53

I
isidor 已提交
54 55 56 57 58 59
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';
E
Erich Gamma 已提交
60

61 62 63 64 65
interface StartSessionResult {
	status: 'ok' | 'initialConfiguration' | 'saveConfiguration';
	content?: string;
};

66
export class DebugService implements debug.IDebugService {
67
	public _serviceBrand: any;
E
Erich Gamma 已提交
68

I
isidor 已提交
69
	private sessionStates: Map<string, debug.State>;
70
	private _onDidChangeState: Emitter<debug.State>;
I
isidor 已提交
71 72
	private model: Model;
	private viewModel: ViewModel;
73
	private configurationManager: ConfigurationManager;
74
	private customTelemetryService: ITelemetryService;
E
Erich Gamma 已提交
75
	private lastTaskEvent: TaskEvent;
76
	private toDispose: lifecycle.IDisposable[];
I
isidor 已提交
77
	private toDisposeOnSessionEnd: Map<string, lifecycle.IDisposable[]>;
A
Alex Dima 已提交
78
	private inDebugMode: IContextKey<boolean>;
I
isidor 已提交
79
	private debugType: IContextKey<string>;
I
isidor 已提交
80
	private debugState: IContextKey<string>;
I
isidor 已提交
81
	private breakpointsToSendOnResourceSaved: Set<string>;
E
Erich Gamma 已提交
82 83 84 85

	constructor(
		@IStorageService private storageService: IStorageService,
		@IWorkbenchEditorService private editorService: IWorkbenchEditorService,
86 87
		@ITextFileService private textFileService: ITextFileService,
		@IViewletService private viewletService: IViewletService,
I
isidor 已提交
88
		@IPanelService private panelService: IPanelService,
E
Erich Gamma 已提交
89 90
		@IMessageService private messageService: IMessageService,
		@IPartService private partService: IPartService,
91
		@IWindowsService private windowsService: IWindowsService,
J
Joao Moreno 已提交
92
		@IWindowIPCService private windowService: IWindowIPCService,
E
Erich Gamma 已提交
93 94
		@ITelemetryService private telemetryService: ITelemetryService,
		@IWorkspaceContextService private contextService: IWorkspaceContextService,
95
		@IContextKeyService contextKeyService: IContextKeyService,
I
isidor 已提交
96
		@ILifecycleService lifecycleService: ILifecycleService,
I
isidor 已提交
97
		@IInstantiationService private instantiationService: IInstantiationService,
A
Alex Dima 已提交
98
		@IExtensionService private extensionService: IExtensionService,
99
		@IMarkerService private markerService: IMarkerService,
100
		@ITaskService private taskService: ITaskService,
101
		@IFileService private fileService: IFileService,
102
		@IConfigurationService private configurationService: IConfigurationService,
103
		@ICommandService private commandService: ICommandService
E
Erich Gamma 已提交
104 105
	) {
		this.toDispose = [];
I
isidor 已提交
106 107
		this.toDisposeOnSessionEnd = new Map<string, lifecycle.IDisposable[]>();
		this.breakpointsToSendOnResourceSaved = new Set<string>();
108
		this._onDidChangeState = new Emitter<debug.State>();
I
isidor 已提交
109
		this.sessionStates = new Map<string, debug.State>();
E
Erich Gamma 已提交
110

111
		this.configurationManager = this.instantiationService.createInstance(ConfigurationManager);
112
		this.inDebugMode = debug.CONTEXT_IN_DEBUG_MODE.bindTo(contextKeyService);
I
isidor 已提交
113
		this.debugType = debug.CONTEXT_DEBUG_TYPE.bindTo(contextKeyService);
I
isidor 已提交
114
		this.debugState = debug.CONTEXT_DEBUG_STATE.bindTo(contextKeyService);
E
Erich Gamma 已提交
115

I
isidor 已提交
116
		this.model = new Model(this.loadBreakpoints(), this.storageService.getBoolean(DEBUG_BREAKPOINTS_ACTIVATED_KEY, StorageScope.WORKSPACE, true), this.loadFunctionBreakpoints(),
E
Erich Gamma 已提交
117
			this.loadExceptionBreakpoints(), this.loadWatchExpressions());
I
isidor 已提交
118
		this.toDispose.push(this.model);
I
isidor 已提交
119
		this.viewModel = new ViewModel(this.storageService.get(DEBUG_SELECTED_CONFIG_NAME_KEY, StorageScope.WORKSPACE, null));
120

121
		this.registerListeners(lifecycleService);
E
Erich Gamma 已提交
122 123
	}

124 125
	private registerListeners(lifecycleService: ILifecycleService): void {
		this.toDispose.push(this.fileService.onFileChanges(e => this.onFileChanges(e)));
E
Erich Gamma 已提交
126 127 128 129 130 131 132 133 134 135 136 137 138 139 140

		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;
			}));
		}

141 142
		lifecycleService.onShutdown(this.store, this);
		lifecycleService.onShutdown(this.dispose, this);
143

I
isidor 已提交
144
		this.toDispose.push(this.windowService.onBroadcast(this.onBroadcast, this));
145
		this.toDispose.push(this.configurationService.onDidUpdateConfiguration((event) => {
I
isidor 已提交
146
			if (event.sourceConfig) {
147 148 149 150 151 152 153
				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);
				}
			}
		}));
154 155 156
	}

	private onBroadcast(broadcast: IBroadcast): void {
157

I
isidor 已提交
158
		// attach: PH is ready to be attached to
159 160 161
		// 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
I
isidor 已提交
162
		const process = this.model.getProcesses().filter(p => strings.equalsIgnoreCase(p.configuration.type, 'extensionhost')).pop();
163
		const session = process ? <RawDebugSession>process.session : null;
164
		if (broadcast.channel === EXTENSION_ATTACH_BROADCAST_CHANNEL) {
165
			this.rawAttach(session, broadcast.payload.port);
166 167
			return;
		}
168

169
		if (broadcast.channel === EXTENSION_TERMINATE_BROADCAST_CHANNEL) {
I
isidor 已提交
170 171 172
			if (session) {
				this.onSessionEnd(session);
			}
173 174 175
			return;
		}

I
isidor 已提交
176
		// from this point on we require an active session
177 178
		if (!session) {
			return;
179 180
		}

B
Benjamin Pasero 已提交
181
		// an extension logged output, show it inside the REPL
182
		if (broadcast.channel === EXTENSION_LOG_BROADCAST_CHANNEL) {
183 184 185 186 187 188 189 190 191 192 193
			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);
			}

I
isidor 已提交
194
			// add output for each argument logged
195 196 197 198
			let simpleVals: any[] = [];
			for (let i = 0; i < args.length; i++) {
				let a = args[i];

I
isidor 已提交
199
				// undefined gets printed as 'undefined'
200 201 202 203
				if (typeof a === 'undefined') {
					simpleVals.push('undefined');
				}

I
isidor 已提交
204
				// null gets printed as 'null'
205 206 207 208
				else if (a === null) {
					simpleVals.push('null');
				}

I
isidor 已提交
209
				// objects & arrays are special because we want to inspect them in the REPL
I
isidor 已提交
210
				else if (isObject(a) || Array.isArray(a)) {
211

I
isidor 已提交
212
					// flush any existing simple values logged
213
					if (simpleVals.length) {
I
isidor 已提交
214
						this.model.appendToRepl(simpleVals.join(' '), sev);
215 216 217
						simpleVals = [];
					}

I
isidor 已提交
218
					// show object
I
isidor 已提交
219
					this.model.appendToRepl(new OutputNameValueElement((<any>a).prototype, a, nls.localize('snapshotObj', "Only primitive values are shown for this object.")), sev);
220 221
				}

I
isidor 已提交
222 223
				// string: watch out for % replacement directive
				// string substitution and formatting @ https://developer.chrome.com/devtools/docs/console
224 225 226 227 228 229
				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
I
isidor 已提交
230
							buf += !isUndefinedOrNull(args[i]) ? args[i] : ''; // replace
231 232 233 234 235 236 237 238 239 240 241 242 243 244 245
							j++; // read over directive
						} else {
							buf += a[j];
						}
					}

					simpleVals.push(buf);
				}

				// number or boolean is joined together
				else {
					simpleVals.push(a);
				}
			}

I
isidor 已提交
246
			// flush simple values
247
			if (simpleVals.length) {
I
isidor 已提交
248
				this.model.appendToRepl(simpleVals.join(' '), sev);
249 250
			}
		}
E
Erich Gamma 已提交
251 252
	}

I
isidor 已提交
253
	private registerSessionListeners(process: Process, session: RawDebugSession): void {
I
isidor 已提交
254 255
		this.toDisposeOnSessionEnd.get(session.getId()).push(session);
		this.toDisposeOnSessionEnd.get(session.getId()).push(session.onDidInitialize(event => {
256
			aria.status(nls.localize('debuggingStarted', "Debugging started."));
257
			const sendConfigurationDone = () => {
258
				if (session && session.capabilities.supportsConfigurationDoneRequest) {
259
					return session.configurationDone().done(null, e => {
260
						// Disconnect the debug session on configuration done error #10596
261 262
						if (session) {
							session.disconnect().done(null, errors.onUnexpectedError);
263 264 265
						}
						this.messageService.show(severity.Error, e.message);
					});
I
isidor 已提交
266
				}
267
			};
268 269 270

			this.sendAllBreakpoints(process).then(sendConfigurationDone, sendConfigurationDone)
				.done(() => this.fetchThreads(session), errors.onUnexpectedError);
I
isidor 已提交
271
		}));
E
Erich Gamma 已提交
272

I
isidor 已提交
273
		this.toDisposeOnSessionEnd.get(session.getId()).push(session.onDidStop(event => {
I
isidor 已提交
274
			this.updateStateAndEmit(session.getId(), debug.State.Stopped);
I
isidor 已提交
275
			const threadId = event.body.threadId;
E
Erich Gamma 已提交
276

277 278 279 280 281
			session.threads().then(response => {
				if (!response || !response.body || !response.body.threads) {
					return;
				}

I
isidor 已提交
282
				const rawThread = response.body.threads.filter(t => t.id === threadId).pop();
283
				this.model.rawUpdate({
284
					sessionId: session.getId(),
I
isidor 已提交
285
					thread: rawThread,
I
isidor 已提交
286
					threadId,
287 288 289
					stoppedDetails: event.body,
					allThreadsStopped: event.body.allThreadsStopped
				});
290

I
isidor 已提交
291 292
				const thread = process && process.getThread(threadId);
				if (thread) {
293
					thread.fetchCallStack().then(callStack => {
294
						if (callStack.length > 0 && !this.viewModel.focusedStackFrame) {
295
							// focus first stack frame from top that has source location if no other stack frame is focussed
I
isidor 已提交
296
							const stackFrameToFocus = first(callStack, sf => !!sf.source, callStack[0]);
297
							this.focusStackFrameAndEvaluate(stackFrameToFocus).done(null, errors.onUnexpectedError);
I
isidor 已提交
298 299 300
							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));

I
isidor 已提交
301
							return stackFrameToFocus.openInEditor(this.editorService);
I
isidor 已提交
302
						}
303
						return undefined;
I
isidor 已提交
304 305
					});
				}
E
Erich Gamma 已提交
306 307 308
			}, errors.onUnexpectedError);
		}));

I
isidor 已提交
309
		this.toDisposeOnSessionEnd.get(session.getId()).push(session.onDidThread(event => {
E
Erich Gamma 已提交
310
			if (event.body.reason === 'started') {
311
				this.fetchThreads(session).done(undefined, errors.onUnexpectedError);
E
Erich Gamma 已提交
312
			} else if (event.body.reason === 'exited') {
313
				this.model.clearThreads(session.getId(), true, event.body.threadId);
E
Erich Gamma 已提交
314 315 316
			}
		}));

I
isidor 已提交
317
		this.toDisposeOnSessionEnd.get(session.getId()).push(session.onDidTerminateDebugee(event => {
318
			aria.status(nls.localize('debuggingStopped', "Debugging stopped."));
319
			if (session && session.getId() === event.body.sessionId) {
320
				if (event.body && typeof event.body.restart === 'boolean' && event.body.restart && process) {
I
isidor 已提交
321
					this.restartProcess(process).done(null, err => this.messageService.show(severity.Error, err.message));
322
				} else {
323
					session.disconnect().done(null, errors.onUnexpectedError);
324
				}
E
Erich Gamma 已提交
325 326 327
			}
		}));

I
isidor 已提交
328
		this.toDisposeOnSessionEnd.get(session.getId()).push(session.onDidContinued(event => {
I
isidor 已提交
329
			const threadId = event.body.allThreadsContinued !== false ? undefined : event.body.threadId;
330 331 332 333
			this.model.clearThreads(session.getId(), false, threadId);
			if (this.viewModel.focusedProcess.getId() === session.getId()) {
				this.focusStackFrameAndEvaluate(null, this.viewModel.focusedProcess).done(null, errors.onUnexpectedError);
			}
I
isidor 已提交
334
			this.updateStateAndEmit(session.getId(), debug.State.Running);
I
isidor 已提交
335 336
		}));

I
isidor 已提交
337
		this.toDisposeOnSessionEnd.get(session.getId()).push(session.onDidOutput(event => {
338 339 340 341
			if (!event.body) {
				return;
			}

342
			const outputSeverity = event.body.category === 'stderr' ? severity.Error : event.body.category === 'console' ? severity.Warning : severity.Info;
343
			if (event.body.category === 'telemetry') {
344
				// only log telemetry events from debug adapter if the adapter provided the telemetry key
I
isidor 已提交
345
				// and the user opted in telemetry
346 347
				if (this.customTelemetryService && this.telemetryService.isOptedIn) {
					this.customTelemetryService.publicLog(event.body.output, event.body.data);
348
				}
349 350 351 352 353 354
			} 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;
355
						this.model.appendToRepl(child, outputSeverity);
I
isidor 已提交
356
					});
357 358 359
				});
			} else if (typeof event.body.output === 'string') {
				this.model.appendToRepl(event.body.output, outputSeverity);
E
Erich Gamma 已提交
360 361 362
			}
		}));

I
isidor 已提交
363
		this.toDisposeOnSessionEnd.get(session.getId()).push(session.onDidBreakpoint(event => {
I
isidor 已提交
364
			const id = event.body && event.body.breakpoint ? event.body.breakpoint.id : undefined;
365
			const breakpoint = this.model.getBreakpoints().filter(bp => bp.idFromAdapter === id).pop();
I
isidor 已提交
366 367 368
			if (breakpoint) {
				this.model.updateBreakpoints({ [breakpoint.getId()]: event.body.breakpoint });
			} else {
369
				const functionBreakpoint = this.model.getFunctionBreakpoints().filter(bp => bp.idFromAdapter === id).pop();
I
isidor 已提交
370 371 372 373 374 375
				if (functionBreakpoint) {
					this.model.updateFunctionBreakpoints({ [functionBreakpoint.getId()]: event.body.breakpoint });
				}
			}
		}));

I
isidor 已提交
376
		this.toDisposeOnSessionEnd.get(session.getId()).push(session.onDidExitAdapter(event => {
377
			// 'Run without debugging' mode VSCode must terminate the extension host. More details: #3905
I
isidor 已提交
378
			const process = this.viewModel.focusedProcess;
I
isidor 已提交
379
			if (process && session && process.getId() === session.getId() && strings.equalsIgnoreCase(process.configuration.type, 'extensionhost') && this.sessionStates.get(session.getId()) === debug.State.Running &&
I
isidor 已提交
380
				process && this.contextService.getWorkspace() && process.configuration.noDebug) {
381
				this.windowsService.closeExtensionHostWindow(this.contextService.getWorkspace().resource.fsPath);
382
			}
383
			if (session && session.getId() === event.body.sessionId) {
384
				this.onSessionEnd(session);
385
			}
E
Erich Gamma 已提交
386 387 388
		}));
	}

389 390 391 392 393 394 395 396 397 398 399
	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
					}));
			}
		});
E
Erich Gamma 已提交
400 401
	}

402 403
	private loadBreakpoints(): Breakpoint[] {
		let result: Breakpoint[];
E
Erich Gamma 已提交
404
		try {
405
			result = JSON.parse(this.storageService.get(DEBUG_BREAKPOINTS_KEY, StorageScope.WORKSPACE, '[]')).map((breakpoint: any) => {
406
				return new Breakpoint(uri.parse(breakpoint.uri.external || breakpoint.source.uri.external), breakpoint.lineNumber, breakpoint.respectColumn ? breakpoint.column : undefined, breakpoint.enabled, breakpoint.condition, breakpoint.hitCondition, breakpoint.respectColumn);
E
Erich Gamma 已提交
407
			});
408 409 410
		} catch (e) { }

		return result || [];
E
Erich Gamma 已提交
411 412
	}

413 414
	private loadFunctionBreakpoints(): FunctionBreakpoint[] {
		let result: FunctionBreakpoint[];
I
isidor 已提交
415
		try {
416
			result = JSON.parse(this.storageService.get(DEBUG_FUNCTION_BREAKPOINTS_KEY, StorageScope.WORKSPACE, '[]')).map((fb: any) => {
I
isidor 已提交
417
				return new FunctionBreakpoint(fb.name, fb.enabled, fb.hitCondition);
I
isidor 已提交
418
			});
419 420 421
		} catch (e) { }

		return result || [];
I
isidor 已提交
422 423
	}

424 425
	private loadExceptionBreakpoints(): ExceptionBreakpoint[] {
		let result: ExceptionBreakpoint[];
E
Erich Gamma 已提交
426 427
		try {
			result = JSON.parse(this.storageService.get(DEBUG_EXCEPTION_BREAKPOINTS_KEY, StorageScope.WORKSPACE, '[]')).map((exBreakpoint: any) => {
I
isidor 已提交
428
				return new ExceptionBreakpoint(exBreakpoint.filter || exBreakpoint.name, exBreakpoint.label, exBreakpoint.enabled);
E
Erich Gamma 已提交
429
			});
430
		} catch (e) { }
E
Erich Gamma 已提交
431

432
		return result || [];
E
Erich Gamma 已提交
433 434
	}

I
isidor 已提交
435 436
	private loadWatchExpressions(): Expression[] {
		let result: Expression[];
E
Erich Gamma 已提交
437
		try {
J
Johannes Rieken 已提交
438
			result = JSON.parse(this.storageService.get(DEBUG_WATCH_EXPRESSIONS_KEY, StorageScope.WORKSPACE, '[]')).map((watchStoredData: { name: string, id: string }) => {
439
				return new Expression(watchStoredData.name, watchStoredData.id);
E
Erich Gamma 已提交
440
			});
441 442 443
		} catch (e) { }

		return result || [];
E
Erich Gamma 已提交
444 445
	}

I
isidor 已提交
446
	public get state(): debug.State {
I
isidor 已提交
447 448
		const focusedProcess = this.viewModel.focusedProcess;
		if (focusedProcess) {
I
isidor 已提交
449
			return this.sessionStates.get(focusedProcess.getId());
I
isidor 已提交
450
		}
I
isidor 已提交
451 452
		if (this.sessionStates.size > 0) {
			return debug.State.Initializing;
I
isidor 已提交
453 454 455
		}

		return debug.State.Inactive;
E
Erich Gamma 已提交
456 457
	}

458
	public get onDidChangeState(): Event<debug.State> {
459 460 461
		return this._onDidChangeState.event;
	}

I
isidor 已提交
462 463 464 465 466 467 468
	private updateStateAndEmit(sessionId?: string, newState?: debug.State): void {
		if (sessionId) {
			if (newState === debug.State.Inactive) {
				this.sessionStates.delete(sessionId);
			} else {
				this.sessionStates.set(sessionId, newState);
			}
I
isidor 已提交
469
		}
I
isidor 已提交
470 471 472 473

		const state = this.state;
		this.debugState.set(debug.State[state].toLowerCase());
		this._onDidChangeState.fire(state);
E
Erich Gamma 已提交
474 475 476
	}

	public get enabled(): boolean {
B
Benjamin Pasero 已提交
477
		return this.contextService.hasWorkspace();
E
Erich Gamma 已提交
478 479
	}

480
	public focusStackFrameAndEvaluate(stackFrame: debug.IStackFrame, process?: debug.IProcess): TPromise<void> {
I
isidor 已提交
481
		if (!process) {
482 483 484 485 486 487 488
			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;
I
isidor 已提交
489
		}
I
isidor 已提交
490

491
		this.viewModel.setFocusedStackFrame(stackFrame, process);
I
isidor 已提交
492
		this.updateStateAndEmit();
493

494
		return this.model.evaluateWatchExpressions(process, stackFrame);
E
Erich Gamma 已提交
495 496
	}

I
isidor 已提交
497
	public enableOrDisableBreakpoints(enable: boolean, breakpoint?: debug.IEnablement): TPromise<void> {
498 499
		if (breakpoint) {
			this.model.setEnablement(breakpoint, enable);
I
isidor 已提交
500
			if (breakpoint instanceof Breakpoint) {
501
				return this.sendBreakpoints(breakpoint.uri);
I
isidor 已提交
502
			} else if (breakpoint instanceof FunctionBreakpoint) {
503 504
				return this.sendFunctionBreakpoints();
			}
E
Erich Gamma 已提交
505

506
			return this.sendExceptionBreakpoints();
E
Erich Gamma 已提交
507 508
		}

509 510
		this.model.enableOrDisableAllBreakpoints(enable);
		return this.sendAllBreakpoints();
E
Erich Gamma 已提交
511 512
	}

513 514 515
	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)));
516

517
		return this.sendBreakpoints(uri);
518 519
	}

520 521
	public removeBreakpoints(id?: string): TPromise<any> {
		const toRemove = this.model.getBreakpoints().filter(bp => !id || bp.getId() === id);
522 523
		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);
E
Erich Gamma 已提交
524

525
		this.model.removeBreakpoints(toRemove);
I
isidor 已提交
526
		return TPromise.join(urisToClear.map(uri => this.sendBreakpoints(uri)));
E
Erich Gamma 已提交
527 528
	}

529 530
	public setBreakpointsActivated(activated: boolean): TPromise<void> {
		this.model.setBreakpointsActivated(activated);
E
Erich Gamma 已提交
531 532 533
		return this.sendAllBreakpoints();
	}

I
isidor 已提交
534 535
	public addFunctionBreakpoint(): void {
		this.model.addFunctionBreakpoint('');
536 537
	}

I
isidor 已提交
538
	public renameFunctionBreakpoint(id: string, newFunctionName: string): TPromise<void> {
I
isidor 已提交
539
		this.model.updateFunctionBreakpoints({ [id]: { name: newFunctionName } });
I
isidor 已提交
540
		return this.sendFunctionBreakpoints();
I
isidor 已提交
541 542
	}

I
isidor 已提交
543
	public removeFunctionBreakpoints(id?: string): TPromise<void> {
544
		this.model.removeFunctionBreakpoints(id);
I
isidor 已提交
545
		return this.sendFunctionBreakpoints();
I
isidor 已提交
546 547
	}

I
isidor 已提交
548
	public addReplExpression(name: string): TPromise<void> {
P
Pierson Lee 已提交
549
		this.telemetryService.publicLog('debugService/addReplExpression');
550
		return this.model.addReplExpression(this.viewModel.focusedProcess, this.viewModel.focusedStackFrame, name)
551
			// Evaluate all watch expressions and fetch variables again since repl evaluation might have changed some.
552
			.then(() => this.focusStackFrameAndEvaluate(this.viewModel.focusedStackFrame, this.viewModel.focusedProcess));
E
Erich Gamma 已提交
553 554
	}

555 556
	public removeReplExpressions(): void {
		this.model.removeReplExpressions();
E
Erich Gamma 已提交
557 558
	}

559 560
	public logToRepl(value: string, sev = severity.Info): void {
		this.model.appendToRepl(value, sev);
561 562
	}

I
isidor 已提交
563
	public addWatchExpression(name: string): TPromise<void> {
564
		return this.model.addWatchExpression(this.viewModel.focusedProcess, this.viewModel.focusedStackFrame, name);
E
Erich Gamma 已提交
565 566
	}

I
isidor 已提交
567
	public renameWatchExpression(id: string, newName: string): TPromise<void> {
568
		return this.model.renameWatchExpression(this.viewModel.focusedProcess, this.viewModel.focusedStackFrame, id, newName);
E
Erich Gamma 已提交
569 570
	}

I
isidor 已提交
571 572 573 574
	public moveWatchExpression(id: string, position: number): void {
		this.model.moveWatchExpression(id, position);
	}

575 576
	public removeWatchExpressions(id?: string): void {
		this.model.removeWatchExpressions(id);
E
Erich Gamma 已提交
577 578
	}

579
	public startDebugging(configName?: string, noDebug = false): TPromise<any> {
580 581 582 583 584
		// make sure to save all files and that the configuration is up to date
		return this.textFileService.saveAll().then(() => this.configurationService.reloadConfiguration().then(() =>
			this.extensionService.onReady().then(() => {
				if (this.model.getProcesses().length === 0) {
					this.removeReplExpressions();
585
				}
586 587 588 589 590 591 592 593 594
				const manager = this.getConfigurationManager();
				configName = configName || this.viewModel.selectedConfigurationName;
				const config = manager.getConfiguration(configName);
				const compound = manager.getCompound(configName);
				if (compound) {
					if (!compound.configurations) {
						return TPromise.wrapError(new Error(nls.localize({ key: 'compoundMustHaveConfigurations', comment: ['compound indicates a "compounds" configuration item', '"configurations" is an attribute and should not be localized'] },
							"Compound must have \"configurations\" attribute set in order to start multiple configurations.")));
					}
595

596 597 598 599
					return TPromise.join(compound.configurations.map(name => this.startDebugging(name)));
				}
				if (configName && !config) {
					return TPromise.wrapError(new Error(nls.localize('configMissing', "Configuration '{0}' is missing in 'launch.json'.", configName)));
600 601
				}

602 603 604 605 606 607 608 609 610 611 612 613 614 615 616
				return manager.getStartSessionCommand(config ? config.type : undefined).then(commandAndType => {
					if (noDebug && config) {
						config.noDebug = true;
					}
					if (commandAndType && commandAndType.command) {
						const defaultConfig = noDebug ? { noDebug: true } : {};
						return this.commandService.executeCommand(commandAndType.command, config || defaultConfig).then((result: StartSessionResult) => {
							if (this.contextService.getWorkspace()) {
								if (result && result.status === 'initialConfiguration') {
									return manager.openConfigFile(false, commandAndType.type);
								}

								if (result && result.status === 'saveConfiguration') {
									return this.fileService.updateContent(manager.configFileUri, result.content).then(() => manager.openConfigFile(false));
								}
617
							}
618 619 620
							return undefined;
						});
					}
621

622 623 624 625 626 627
					if (config) {
						return this.createProcess(config);
					}
					if (this.contextService.getWorkspace() && commandAndType) {
						return manager.openConfigFile(false, commandAndType.type);
					}
628

629 630 631 632
					return undefined;
				});
			})
		));
633
	}
634

635
	public createProcess(config: debug.IConfig): TPromise<any> {
636 637 638 639 640 641
		return this.textFileService.saveAll().then(() =>
			this.configurationManager.resloveConfiguration(config).then(resolvedConfig => {
				if (!resolvedConfig) {
					// User canceled resolving of interactive variables, silently return
					return undefined;
				}
I
isidor 已提交
642

643 644 645 646 647
				if (!this.configurationManager.getAdapter(resolvedConfig.type)) {
					const message = resolvedConfig.type ? nls.localize('debugTypeNotSupported', "Configured debug type '{0}' is not supported.", resolvedConfig.type) :
						nls.localize('debugTypeMissing', "Missing property 'type' for the chosen launch configuration.");
					return TPromise.wrapError(errors.create(message, { actions: [this.instantiationService.createInstance(debugactions.ConfigureAction, debugactions.ConfigureAction.ID, debugactions.ConfigureAction.LABEL), CloseAction] }));
				}
J
Johannes Rieken 已提交
648

649 650 651 652 653 654 655
				return this.runPreLaunchTask(resolvedConfig.preLaunchTask).then((taskSummary: ITaskSummary) => {
					const errorCount = resolvedConfig.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(resolvedConfig);
					}
I
isidor 已提交
656

657 658 659 660 661 662 663 664 665 666 667 668 669 670 671 672 673 674
					this.messageService.show(severity.Error, {
						message: errorCount > 1 ? nls.localize('preLaunchTaskErrors', "Build errors have been detected during preLaunchTask '{0}'.", resolvedConfig.preLaunchTask) :
							errorCount === 1 ? nls.localize('preLaunchTaskError', "Build error has been detected during preLaunchTask '{0}'.", resolvedConfig.preLaunchTask) :
								nls.localize('preLaunchTaskExitCode', "The preLaunchTask '{0}' terminated with exit code {1}.", resolvedConfig.preLaunchTask, taskSummary.exitCode),
						actions: [
							new Action('debug.continue', nls.localize('debugAnyway', "Debug Anyway"), null, true, () => {
								this.messageService.hideAll();
								return this.doCreateProcess(resolvedConfig);
							}),
							this.instantiationService.createInstance(ToggleMarkersPanelAction, ToggleMarkersPanelAction.ID, ToggleMarkersPanelAction.LABEL),
							CloseAction
						]
					});
					return undefined;
				}, (err: TaskError) => {
					if (err.code !== TaskErrors.NotConfigured) {
						throw err;
					}
675

676 677 678
					this.messageService.show(err.severity, {
						message: err.message,
						actions: [this.taskService.configureAction(), CloseAction]
679
					});
680 681 682 683 684
				});
			}, 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."));
				}
E
Erich Gamma 已提交
685

686 687 688 689 690 691 692 693
				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));
					}
				});
			})
		);
	}
694 695 696

	private doCreateProcess(configuration: debug.IConfig): TPromise<any> {
		const sessionId = generateUuid();
I
isidor 已提交
697
		this.updateStateAndEmit(sessionId, debug.State.Initializing);
698

J
Joao Moreno 已提交
699
		return this.telemetryService.getTelemetryInfo().then(info => {
700
			const telemetryInfo: { [key: string]: string } = Object.create(null);
701 702
			telemetryInfo['common.vscodemachineid'] = info.machineId;
			telemetryInfo['common.vscodesessionid'] = info.sessionId;
703 704
			return telemetryInfo;
		}).then(data => {
705 706 707
			const adapter = this.configurationManager.getAdapter(configuration.type);
			const { aiKey, type } = adapter;
			const publisher = adapter.extensionDescription.publisher;
708
			this.customTelemetryService = null;
709
			let client: TelemetryClient;
710 711

			if (aiKey) {
712
				client = new TelemetryClient(
713 714 715 716
					uri.parse(require.toUrl('bootstrap')).fsPath,
					{
						serverName: 'Debug Telemetry',
						timeout: 1000 * 60 * 5,
J
Johannes Rieken 已提交
717
						args: [`${publisher}.${type}`, JSON.stringify(data), aiKey],
718
						env: {
B
Benjamin Pasero 已提交
719
							ELECTRON_RUN_AS_NODE: 1,
720 721 722
							PIPE_LOGGING: 'true',
							AMD_ENTRYPOINT: 'vs/workbench/parts/debug/node/telemetryApp'
						}
723
					}
724
				);
725

726 727
				const channel = client.getChannel('telemetryAppender');
				const appender = new TelemetryAppenderClient(channel);
728

729
				this.customTelemetryService = new TelemetryService({ appender }, this.configurationService);
730 731
			}

732
			const session = this.instantiationService.createInstance(RawDebugSession, sessionId, configuration.debugServer, adapter, this.customTelemetryService);
733
			const process = this.model.addProcess(configuration, session);
734

I
isidor 已提交
735
			this.toDisposeOnSessionEnd.set(session.getId(), []);
736
			if (client) {
I
isidor 已提交
737
				this.toDisposeOnSessionEnd.get(session.getId()).push(client);
738
			}
I
isidor 已提交
739
			this.registerSessionListeners(process, session);
J
Joao Moreno 已提交
740

741
			return session.initialize({
742
				clientID: 'vscode',
J
Joao Moreno 已提交
743 744 745
				adapterID: configuration.type,
				pathFormat: 'path',
				linesStartAt1: true,
746
				columnsStartAt1: true,
I
isidor 已提交
747
				supportsVariableType: true, // #8858
748 749
				supportsVariablePaging: true, // #9537
				supportsRunInTerminalRequest: true // #10574
J
Joao Moreno 已提交
750
			}).then((result: DebugProtocol.InitializeResponse) => {
751
				this.model.setExceptionBreakpoints(session.capabilities.exceptionBreakpointFilters);
752
				return configuration.request === 'attach' ? session.attach(configuration) : session.launch(configuration);
J
Joao Moreno 已提交
753
			}).then((result: DebugProtocol.Response) => {
754
				if (session.disconnected) {
755 756
					return TPromise.as(null);
				}
757 758 759
				if (!this.viewModel.focusedProcess) {
					this.focusStackFrameAndEvaluate(null, process);
				}
J
Joao Moreno 已提交
760 761 762
				if (configuration.internalConsoleOptions === 'openOnSessionStart' || (!this.viewModel.changedWorkbenchViewState && configuration.internalConsoleOptions !== 'neverOpen')) {
					this.panelService.openPanel(debug.REPL_ID, false).done(undefined, errors.onUnexpectedError);
				}
763

I
isidor 已提交
764
				if (!this.viewModel.changedWorkbenchViewState && (this.partService.isVisible(Parts.SIDEBAR_PART) || !this.contextService.getWorkspace())) {
J
Joao Moreno 已提交
765 766 767 768
					// 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);
				}
769

J
Joao Moreno 已提交
770 771
				this.extensionService.activateByEvent(`onDebug:${configuration.type}`).done(null, errors.onUnexpectedError);
				this.inDebugMode.set(true);
I
isidor 已提交
772
				this.debugType.set(configuration.type);
773 774 775
				if (this.model.getProcesses().length > 1) {
					this.viewModel.setMultiProcessView(true);
				}
I
isidor 已提交
776
				this.updateStateAndEmit(session.getId(), debug.State.Running);
J
Joao Moreno 已提交
777

778
				return this.telemetryService.publicLog('debugSessionStart', {
J
Joao Moreno 已提交
779 780 781 782
					type: configuration.type,
					breakpointCount: this.model.getBreakpoints().length,
					exceptionBreakpoints: this.model.getExceptionBreakpoints(),
					watchExpressionsCount: this.model.getWatchExpressions().length,
783
					extensionName: `${adapter.extensionDescription.publisher}.${adapter.extensionDescription.name}`,
784 785
					isBuiltin: adapter.extensionDescription.isBuiltin,
					launchJsonExists: !!this.configurationService.getConfiguration<debug.IGlobalConfig>('launch')
J
Joao Moreno 已提交
786 787
				});
			}).then(undefined, (error: any) => {
788 789 790 791 792
				if (error instanceof Error && error.message === 'Canceled') {
					// Do not show 'canceled' error messages to the user #7906
					return TPromise.as(null);
				}

793 794
				const errorMessage = error instanceof Error ? error.message : error;
				this.telemetryService.publicLog('debugMisconfiguration', { type: configuration ? configuration.type : undefined, error: errorMessage });
I
isidor 已提交
795
				this.updateStateAndEmit(session.getId(), debug.State.Inactive);
796 797
				if (!session.disconnected) {
					session.disconnect().done(null, errors.onUnexpectedError);
J
Joao Moreno 已提交
798 799 800 801 802
				}
				// 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);
				}
I
isidor 已提交
803

J
Joao Moreno 已提交
804 805
				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];
806
				this.messageService.show(severity.Error, { message: errorMessage, actions });
807
				return undefined;
J
Joao Moreno 已提交
808
			});
E
Erich Gamma 已提交
809 810 811
		});
	}

I
isidor 已提交
812
	private runPreLaunchTask(taskName: string): TPromise<ITaskSummary> {
813
		if (!taskName) {
I
isidor 已提交
814
			return TPromise.as(null);
E
Erich Gamma 已提交
815 816
		}

817
		// run a task before starting a debug session
E
Erich Gamma 已提交
818
		return this.taskService.tasks().then(descriptions => {
819
			const filteredTasks = descriptions.filter(task => task.name === taskName);
E
Erich Gamma 已提交
820
			if (filteredTasks.length !== 1) {
821 822
				return TPromise.wrapError(errors.create(nls.localize('DebugTaskNotFound', "Could not find the preLaunchTask \'{0}\'.", taskName), {
					actions: [
823
						this.instantiationService.createInstance(debugactions.ConfigureAction, debugactions.ConfigureAction.ID, debugactions.ConfigureAction.LABEL),
824
						this.taskService.configureAction(),
825
						CloseAction
826 827
					]
				}));
E
Erich Gamma 已提交
828 829
			}

I
isidor 已提交
830
			// task is already running - nothing to do.
831
			if (this.lastTaskEvent && this.lastTaskEvent.taskName === taskName) {
I
isidor 已提交
832
				return TPromise.as(null);
E
Erich Gamma 已提交
833 834 835
			}

			if (this.lastTaskEvent) {
I
isidor 已提交
836
				// there is a different task running currently.
I
isidor 已提交
837
				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)));
E
Erich Gamma 已提交
838 839
			}

I
isidor 已提交
840
			// no task running, execute the preLaunchTask.
841
			const taskPromise = this.taskService.run(filteredTasks[0]).then(result => {
E
Erich Gamma 已提交
842
				this.lastTaskEvent = null;
I
isidor 已提交
843
				return result;
E
Erich Gamma 已提交
844 845 846
			}, err => {
				this.lastTaskEvent = null;
			});
847

848
			if (filteredTasks[0].isBackground) {
A
Alex Dima 已提交
849
				return new TPromise((c, e) => this.taskService.addOneTimeDisposableListener(TaskServiceEvents.Inactive, () => c(null)));
850 851 852
			}

			return taskPromise;
E
Erich Gamma 已提交
853 854 855
		});
	}

856 857 858
	private rawAttach(session: RawDebugSession, port: number): TPromise<any> {
		if (session) {
			return session.attach({ port });
I
isidor 已提交
859 860
		}

861 862 863 864
		const config = this.configurationManager.getConfiguration(this.viewModel.selectedConfigurationName);
		return this.configurationManager.resloveConfiguration(config).then(resolvedConfig => {
			resolvedConfig.request = 'attach';
			resolvedConfig.port = port;
I
isidor 已提交
865
			this.doCreateProcess(resolvedConfig);
I
isidor 已提交
866
		});
I
isidor 已提交
867 868
	}

I
isidor 已提交
869 870 871 872
	public deemphasizeSource(uri: uri): void {
		this.model.deemphasizeSource(uri);
	}

I
isidor 已提交
873
	public restartProcess(process: debug.IProcess): TPromise<any> {
874
		if (process.session.capabilities.supportsRestartRequest) {
875 876
			return process.session.custom('restart', null);
		}
I
isidor 已提交
877 878
		const focusedProcess = this.viewModel.focusedProcess;
		const preserveFocus = focusedProcess && process.getId() === focusedProcess.getId();
879 880

		return process.session.disconnect(true).then(() =>
881
			new TPromise<void>((c, e) => {
E
Erich Gamma 已提交
882
				setTimeout(() => {
883 884
					// Read the configuration again if a launch.json exists, if not just use the inmemory configuration #19366
					const config = this.configurationManager.getConfiguration(process.configuration.name);
885 886 887 888
					if (config) {
						// Take the type from the process since the debug extension might overwrite it #21316
						config.type = process.configuration.type;
					}
889
					this.createProcess(config || process.configuration).then(() => c(null), err => e(err));
E
Erich Gamma 已提交
890
				}, 300);
I
isidor 已提交
891
			})
892
		).then(() => {
I
isidor 已提交
893 894
			if (preserveFocus) {
				// Restart should preserve the focused process
895
				const restartedProcess = this.model.getProcesses().filter(p => p.configuration.name === process.configuration.name).pop();
I
isidor 已提交
896 897 898
				if (restartedProcess && restartedProcess !== this.viewModel.focusedProcess) {
					this.focusStackFrameAndEvaluate(null, restartedProcess);
				}
899 900
			}
		});
E
Erich Gamma 已提交
901 902
	}

903 904 905 906 907 908 909 910 911 912 913 914 915 916 917
	public stopProcess(process: debug.IProcess): TPromise<any> {
		if (process) {
			return process.session.disconnect(false, true);
		}

		const processes = this.model.getProcesses();
		if (processes.length) {
			return TPromise.join(processes.map(p => p.session.disconnect(false, true)));
		}

		this.sessionStates.clear();
		this._onDidChangeState.fire();
		return undefined;
	}

I
isidor 已提交
918
	private onSessionEnd(session: RawDebugSession): void {
I
isidor 已提交
919
		const bpsExist = this.model.getBreakpoints().length > 0;
920
		const process = this.model.getProcesses().filter(p => p.getId() === session.getId()).pop();
I
isidor 已提交
921
		this.telemetryService.publicLog('debugSessionStop', {
I
isidor 已提交
922
			type: process && process.configuration.type,
I
isidor 已提交
923 924 925 926 927
			success: session.emittedStopped || !bpsExist,
			sessionLengthInSeconds: session.getLengthInSeconds(),
			breakpointCount: this.model.getBreakpoints().length,
			watchExpressionsCount: this.model.getWatchExpressions().length
		});
928

I
isidor 已提交
929
		this.model.removeProcess(session.getId());
I
isidor 已提交
930

I
isidor 已提交
931
		this.toDisposeOnSessionEnd.set(session.getId(), lifecycle.dispose(this.toDisposeOnSessionEnd.get(session.getId())));
I
isidor 已提交
932 933
		const focusedProcess = this.viewModel.focusedProcess;
		if (focusedProcess && focusedProcess.getId() === session.getId()) {
934 935
			this.focusStackFrameAndEvaluate(null).done(null, errors.onUnexpectedError);
		}
I
isidor 已提交
936
		this.updateStateAndEmit(session.getId(), debug.State.Inactive);
937

938 939
		if (this.model.getProcesses().length === 0) {
			// set breakpoints back to unverified since the session ended.
940
			const data: { [id: string]: { line: number, verified: boolean } } = {};
941
			this.model.getBreakpoints().forEach(bp => {
942
				data[bp.getId()] = { line: bp.lineNumber, verified: false };
943 944
			});
			this.model.updateBreakpoints(data);
945

946
			this.inDebugMode.reset();
I
isidor 已提交
947
			this.debugType.reset();
948
			this.viewModel.setMultiProcessView(false);
I
isidor 已提交
949

950
			if (this.partService.isVisible(Parts.SIDEBAR_PART) && this.configurationService.getConfiguration<debug.IDebugConfiguration>('debug').openExplorerOnEnd) {
951 952
				this.viewletService.openViewlet(EXPLORER_VIEWLET_ID).done(null, errors.onUnexpectedError);
			}
I
isidor 已提交
953
		}
E
Erich Gamma 已提交
954 955 956 957 958 959 960 961 962 963
	}

	public getModel(): debug.IModel {
		return this.model;
	}

	public getViewModel(): debug.IViewModel {
		return this.viewModel;
	}

964 965
	public getConfigurationManager(): debug.IConfigurationManager {
		return this.configurationManager;
966 967
	}

968 969 970
	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))
I
isidor 已提交
971
			// send exception breakpoints at the end since some debug adapters rely on the order
972
			.then(() => this.sendExceptionBreakpoints(process));
E
Erich Gamma 已提交
973 974
	}

975 976 977 978
	private sendBreakpoints(modelUri: uri, sourceModified = false, targetProcess?: debug.IProcess): TPromise<void> {

		const sendBreakpointsToProcess = (process: debug.IProcess): TPromise<void> => {
			const session = <RawDebugSession>process.session;
I
isidor 已提交
979 980 981 982 983
			if (!session.readyForBreakpoints) {
				return TPromise.as(null);
			}
			if (this.textFileService.isDirty(modelUri)) {
				// Only send breakpoints for a file once it is not dirty #8077
I
isidor 已提交
984
				this.breakpointsToSendOnResourceSaved.add(modelUri.toString());
I
isidor 已提交
985 986
				return TPromise.as(null);
			}
987

988
			const breakpointsToSend = this.model.getBreakpoints().filter(bp => this.model.areBreakpointsActivated() && bp.enabled && bp.uri.toString() === modelUri.toString());
989

990
			const source = process.sources.get(modelUri.toString());
991
			const rawSource = source ? source.raw : { path: paths.normalize(modelUri.fsPath, true) };
I
isidor 已提交
992 993 994

			return session.setBreakpoints({
				source: rawSource,
995
				lines: breakpointsToSend.map(bp => bp.lineNumber),
996
				breakpoints: breakpointsToSend.map(bp => ({ line: bp.lineNumber, column: bp.column, condition: bp.condition, hitCondition: bp.hitCondition })),
I
isidor 已提交
997 998 999 1000 1001
				sourceModified
			}).then(response => {
				if (!response || !response.body) {
					return;
				}
1002

I
isidor 已提交
1003
				const data: { [id: string]: { line?: number, column?: number, verified: boolean } } = {};
I
isidor 已提交
1004 1005
				for (let i = 0; i < breakpointsToSend.length; i++) {
					data[breakpointsToSend[i].getId()] = response.body.breakpoints[i];
1006 1007 1008 1009
					if (!breakpointsToSend[i].column) {
						// If there was no column sent ignore the breakpoint column response from the adapter
						data[breakpointsToSend[i].getId()].column = undefined;
					}
I
isidor 已提交
1010
				}
1011

I
isidor 已提交
1012 1013 1014
				this.model.updateBreakpoints(data);
			});
		};
1015

1016
		return this.sendToOneOrAllProcesses(targetProcess, sendBreakpointsToProcess);
E
Erich Gamma 已提交
1017 1018
	}

1019 1020 1021
	private sendFunctionBreakpoints(targetProcess?: debug.IProcess): TPromise<void> {
		const sendFunctionBreakpointsToProcess = (process: debug.IProcess): TPromise<void> => {
			const session = <RawDebugSession>process.session;
1022
			if (!session.readyForBreakpoints || !session.capabilities.supportsFunctionBreakpoints) {
I
isidor 已提交
1023
				return TPromise.as(null);
1024 1025
			}

I
isidor 已提交
1026 1027 1028 1029 1030 1031 1032 1033 1034 1035 1036 1037 1038 1039 1040
			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);
			});
		};

1041
		return this.sendToOneOrAllProcesses(targetProcess, sendFunctionBreakpointsToProcess);
I
isidor 已提交
1042 1043
	}

1044 1045 1046 1047
	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) {
I
isidor 已提交
1048
				return TPromise.as(null);
I
isidor 已提交
1049 1050
			}

I
isidor 已提交
1051 1052 1053 1054
			const enabledExceptionBps = this.model.getExceptionBreakpoints().filter(exb => exb.enabled);
			return session.setExceptionBreakpoints({ filters: enabledExceptionBps.map(exb => exb.filter) });
		};

1055
		return this.sendToOneOrAllProcesses(targetProcess, sendExceptionBreakpointsToProcess);
I
isidor 已提交
1056 1057
	}

1058 1059 1060
	private sendToOneOrAllProcesses(process: debug.IProcess, send: (process: debug.IProcess) => TPromise<void>): TPromise<void> {
		if (process) {
			return send(process);
I
isidor 已提交
1061
		}
I
isidor 已提交
1062

1063
		return TPromise.join(this.model.getProcesses().map(p => send(p))).then(() => void 0);
E
Erich Gamma 已提交
1064 1065 1066
	}

	private onFileChanges(fileChangesEvent: FileChangesEvent): void {
1067
		this.model.removeBreakpoints(this.model.getBreakpoints().filter(bp =>
1068
			fileChangesEvent.contains(bp.uri, FileChangeType.DELETED)));
1069 1070

		fileChangesEvent.getUpdated().forEach(event => {
I
isidor 已提交
1071 1072
			if (this.breakpointsToSendOnResourceSaved.has(event.resource.toString())) {
				this.breakpointsToSendOnResourceSaved.delete(event.resource.toString());
1073 1074 1075
				this.sendBreakpoints(event.resource, true).done(null, errors.onUnexpectedError);
			}
		});
E
Erich Gamma 已提交
1076 1077 1078 1079 1080
	}

	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);
1081
		this.storageService.store(DEBUG_FUNCTION_BREAKPOINTS_KEY, JSON.stringify(this.model.getFunctionBreakpoints()), StorageScope.WORKSPACE);
E
Erich Gamma 已提交
1082
		this.storageService.store(DEBUG_EXCEPTION_BREAKPOINTS_KEY, JSON.stringify(this.model.getExceptionBreakpoints()), StorageScope.WORKSPACE);
1083
		this.storageService.store(DEBUG_SELECTED_CONFIG_NAME_KEY, this.viewModel.selectedConfigurationName, StorageScope.WORKSPACE);
1084
		this.storageService.store(DEBUG_WATCH_EXPRESSIONS_KEY, JSON.stringify(this.model.getWatchExpressions().map(we => ({ name: we.name, id: we.getId() }))), StorageScope.WORKSPACE);
E
Erich Gamma 已提交
1085 1086 1087
	}

	public dispose(): void {
I
isidor 已提交
1088
		this.toDisposeOnSessionEnd.forEach(toDispose => lifecycle.dispose(toDispose));
I
isidor 已提交
1089
		this.toDispose = lifecycle.dispose(this.toDispose);
E
Erich Gamma 已提交
1090 1091
	}
}