tunnelView.ts 43.9 KB
Newer Older
A
Alex Ross 已提交
1 2 3 4 5 6 7 8
/*---------------------------------------------------------------------------------------------
 *  Copyright (c) Microsoft Corporation. All rights reserved.
 *  Licensed under the MIT License. See License.txt in the project root for license information.
 *--------------------------------------------------------------------------------------------*/

import 'vs/css!./media/tunnelView';
import * as nls from 'vs/nls';
import * as dom from 'vs/base/browser/dom';
9
import { IViewDescriptor, IEditableData, IViewsService, IViewDescriptorService } from 'vs/workbench/common/views';
10
import { WorkbenchAsyncDataTree } from 'vs/platform/list/browser/listService';
A
Alex Ross 已提交
11
import { IKeybindingService } from 'vs/platform/keybinding/common/keybinding';
12
import { IContextMenuService, IContextViewService } from 'vs/platform/contextview/browser/contextView';
A
Alex Ross 已提交
13 14 15 16
import { IContextKeyService, IContextKey, RawContextKey, ContextKeyExpr } from 'vs/platform/contextkey/common/contextkey';
import { IConfigurationService } from 'vs/platform/configuration/common/configuration';
import { IInstantiationService } from 'vs/platform/instantiation/common/instantiation';
import { IOpenerService } from 'vs/platform/opener/common/opener';
A
Alex Ross 已提交
17
import { IQuickInputService, IQuickPickItem, QuickPickInput } from 'vs/platform/quickinput/common/quickInput';
A
Alex Ross 已提交
18 19 20
import { ICommandService, ICommandHandler, CommandsRegistry } from 'vs/platform/commands/common/commands';
import { Event, Emitter } from 'vs/base/common/event';
import { IListVirtualDelegate } from 'vs/base/browser/ui/list/list';
21
import { ITreeRenderer, ITreeNode, IAsyncDataSource, ITreeContextMenuEvent, ITreeMouseEvent } from 'vs/base/browser/ui/tree/tree';
A
Alex Ross 已提交
22
import { IWorkbenchEnvironmentService } from 'vs/workbench/services/environment/common/environmentService';
23
import { Disposable, IDisposable, toDisposable, MutableDisposable, dispose, DisposableStore } from 'vs/base/common/lifecycle';
24
import { ActionBar } from 'vs/base/browser/ui/actionbar/actionbar';
A
Alex Ross 已提交
25
import { IconLabel } from 'vs/base/browser/ui/iconLabel/iconLabel';
J
João Moreno 已提交
26
import { ActionRunner, IAction } from 'vs/base/common/actions';
27 28
import { IMenuService, MenuId, IMenu, MenuRegistry, MenuItemAction, ILocalizedString, SubmenuItemAction } from 'vs/platform/actions/common/actions';
import { createAndFillInContextMenuActions, createAndFillInActionBarActions, MenuEntryActionViewItem, SubmenuEntryActionViewItem } from 'vs/platform/actions/browser/menuEntryActionViewItem';
A
Alex Ross 已提交
29
import { IRemoteExplorerService, TunnelModel, makeAddress, TunnelType, ITunnelItem, Tunnel, mapHasAddressLocalhostOrAllInterfaces, TUNNEL_VIEW_ID, parseAddress } from 'vs/workbench/services/remote/common/remoteExplorerService';
A
Alex Ross 已提交
30
import { IClipboardService } from 'vs/platform/clipboard/common/clipboardService';
31
import { INotificationService, Severity } from 'vs/platform/notification/common/notification';
32 33 34
import { InputBox, MessageType } from 'vs/base/browser/ui/inputbox/inputBox';
import { attachInputBoxStyler } from 'vs/platform/theme/common/styler';
import { once } from 'vs/base/common/functional';
35
import { KeyCode, KeyMod } from 'vs/base/common/keyCodes';
36 37
import { IThemeService } from 'vs/platform/theme/common/themeService';
import { IKeyboardEvent } from 'vs/base/browser/keyboardEvent';
S
SteVen Batten 已提交
38
import { ViewPane, IViewPaneOptions } from 'vs/workbench/browser/parts/views/viewPaneContainer';
A
Alex Ross 已提交
39
import { URI } from 'vs/base/common/uri';
A
Alex Ross 已提交
40
import { RemoteTunnel } from 'vs/platform/remote/common/tunnel';
S
Sandeep Somavarapu 已提交
41
import { SyncDescriptor } from 'vs/platform/instantiation/common/descriptors';
42
import { KeybindingsRegistry, KeybindingWeight } from 'vs/platform/keybinding/common/keybindingsRegistry';
43
import { ITelemetryService } from 'vs/platform/telemetry/common/telemetry';
44
import { ActionViewItem } from 'vs/base/browser/ui/actionbar/actionViewItems';
45
import { Codicon } from 'vs/base/common/codicons';
A
Alex Ross 已提交
46 47

export const forwardedPortsViewEnabled = new RawContextKey<boolean>('forwardedPortsViewEnabled', false);
A
Alex Ross 已提交
48 49 50 51 52 53 54 55 56 57 58 59 60 61

class TunnelTreeVirtualDelegate implements IListVirtualDelegate<ITunnelItem> {
	getHeight(element: ITunnelItem): number {
		return 22;
	}

	getTemplateId(element: ITunnelItem): string {
		return 'tunnelItemTemplate';
	}
}

export interface ITunnelViewModel {
	onForwardedPortsChanged: Event<void>;
	readonly forwarded: TunnelItem[];
A
Alex Ross 已提交
62
	readonly detected: TunnelItem[];
63
	readonly candidates: TunnelItem[];
64
	readonly input: TunnelItem;
A
Alex Ross 已提交
65
	groups(): Promise<ITunnelGroup[]>;
A
Alex Ross 已提交
66 67 68 69 70 71
}

export class TunnelViewModel extends Disposable implements ITunnelViewModel {
	private _onForwardedPortsChanged: Emitter<void> = new Emitter();
	public onForwardedPortsChanged: Event<void> = this._onForwardedPortsChanged.event;
	private model: TunnelModel;
72
	private _input: TunnelItem;
73
	private _candidates: Map<string, { host: string, port: number, detail: string }> = new Map();
A
Alex Ross 已提交
74 75

	constructor(
76
		@IRemoteExplorerService private readonly remoteExplorerService: IRemoteExplorerService) {
A
Alex Ross 已提交
77 78 79 80 81
		super();
		this.model = remoteExplorerService.tunnelModel;
		this._register(this.model.onForwardPort(() => this._onForwardedPortsChanged.fire()));
		this._register(this.model.onClosePort(() => this._onForwardedPortsChanged.fire()));
		this._register(this.model.onPortName(() => this._onForwardedPortsChanged.fire()));
82
		this._register(this.model.onCandidatesChanged(() => this._onForwardedPortsChanged.fire()));
83 84 85 86 87 88 89
		this._input = {
			label: nls.localize('remote.tunnelsView.add', "Forward a Port..."),
			tunnelType: TunnelType.Add,
			remoteHost: 'localhost',
			remotePort: 0,
			description: ''
		};
A
Alex Ross 已提交
90 91
	}

A
Alex Ross 已提交
92
	async groups(): Promise<ITunnelGroup[]> {
A
Alex Ross 已提交
93
		const groups: ITunnelGroup[] = [];
94 95
		this._candidates = new Map();
		(await this.model.candidates).forEach(candidate => {
A
Alex Ross 已提交
96
			this._candidates.set(makeAddress(candidate.host, candidate.port), candidate);
97
		});
98
		if ((this.model.forwarded.size > 0) || this.remoteExplorerService.getEditableData(undefined)) {
A
Alex Ross 已提交
99 100 101 102 103 104
			groups.push({
				label: nls.localize('remote.tunnelsView.forwarded', "Forwarded"),
				tunnelType: TunnelType.Forwarded,
				items: this.forwarded
			});
		}
A
Alex Ross 已提交
105
		if (this.model.detected.size > 0) {
A
Alex Ross 已提交
106
			groups.push({
107
				label: nls.localize('remote.tunnelsView.detected', "Existing Tunnels"),
A
Alex Ross 已提交
108 109
				tunnelType: TunnelType.Detected,
				items: this.detected
A
Alex Ross 已提交
110 111
			});
		}
A
Alex Ross 已提交
112 113
		const candidates = await this.candidates;
		if (candidates.length > 0) {
A
Alex Ross 已提交
114
			groups.push({
115
				label: nls.localize('remote.tunnelsView.candidates', "Not Forwarded"),
A
Alex Ross 已提交
116 117 118 119
				tunnelType: TunnelType.Candidate,
				items: candidates
			});
		}
120 121
		if (groups.length === 0) {
			groups.push(this._input);
122
		}
A
Alex Ross 已提交
123 124 125
		return groups;
	}

126
	private addProcessInfoFromCandidate(tunnelItem: ITunnelItem) {
A
Alex Ross 已提交
127
		const key = makeAddress(tunnelItem.remoteHost, tunnelItem.remotePort);
128 129 130 131 132
		if (this._candidates.has(key)) {
			tunnelItem.description = this._candidates.get(key)!.detail;
		}
	}

A
Alex Ross 已提交
133
	get forwarded(): TunnelItem[] {
134
		const forwarded = Array.from(this.model.forwarded.values()).map(tunnel => {
135 136 137
			const tunnelItem = TunnelItem.createFromTunnel(tunnel);
			this.addProcessInfoFromCandidate(tunnelItem);
			return tunnelItem;
A
Alex Ross 已提交
138 139 140 141 142 143
		}).sort((a: TunnelItem, b: TunnelItem) => {
			if (a.remotePort === b.remotePort) {
				return a.remoteHost < b.remoteHost ? -1 : 1;
			} else {
				return a.remotePort < b.remotePort ? -1 : 1;
			}
A
Alex Ross 已提交
144
		});
145 146 147 148
		if (this.remoteExplorerService.getEditableData(undefined)) {
			forwarded.push(this._input);
		}
		return forwarded;
A
Alex Ross 已提交
149 150
	}

A
Alex Ross 已提交
151 152
	get detected(): TunnelItem[] {
		return Array.from(this.model.detected.values()).map(tunnel => {
153 154 155
			const tunnelItem = TunnelItem.createFromTunnel(tunnel, TunnelType.Detected, false);
			this.addProcessInfoFromCandidate(tunnelItem);
			return tunnelItem;
A
Alex Ross 已提交
156 157 158
		});
	}

159 160 161
	get candidates(): TunnelItem[] {
		const candidates: TunnelItem[] = [];
		this._candidates.forEach(value => {
A
Alex Ross 已提交
162 163
			if (!mapHasAddressLocalhostOrAllInterfaces(this.model.forwarded, value.host, value.port) &&
				!mapHasAddressLocalhostOrAllInterfaces(this.model.detected, value.host, value.port)) {
164
				candidates.push(new TunnelItem(TunnelType.Candidate, value.host, value.port, undefined, undefined, false, undefined, value.detail));
165
			}
A
Alex Ross 已提交
166
		});
167
		return candidates;
A
Alex Ross 已提交
168 169
	}

170
	get input(): TunnelItem {
171 172 173
		return this._input;
	}

A
Alex Ross 已提交
174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189
	dispose() {
		super.dispose();
	}
}

interface ITunnelTemplateData {
	elementDisposable: IDisposable;
	container: HTMLElement;
	iconLabel: IconLabel;
	actionBar: ActionBar;
}

class TunnelTreeRenderer extends Disposable implements ITreeRenderer<ITunnelGroup | ITunnelItem, ITunnelItem, ITunnelTemplateData> {
	static readonly ITEM_HEIGHT = 22;
	static readonly TREE_TEMPLATE_ID = 'tunnelItemTemplate';

A
Alex Ross 已提交
190
	private inputDone?: (success: boolean, finishEditing: boolean) => void;
A
Alex Ross 已提交
191 192 193 194 195 196
	private _actionRunner: ActionRunner | undefined;

	constructor(
		private readonly viewId: string,
		@IMenuService private readonly menuService: IMenuService,
		@IContextKeyService private readonly contextKeyService: IContextKeyService,
197 198 199 200
		@IInstantiationService private readonly instantiationService: IInstantiationService,
		@IContextViewService private readonly contextViewService: IContextViewService,
		@IThemeService private readonly themeService: IThemeService,
		@IRemoteExplorerService private readonly remoteExplorerService: IRemoteExplorerService
A
Alex Ross 已提交
201 202 203 204 205 206 207 208 209 210 211 212 213
	) {
		super();
	}

	set actionRunner(actionRunner: ActionRunner) {
		this._actionRunner = actionRunner;
	}

	get templateId(): string {
		return TunnelTreeRenderer.TREE_TEMPLATE_ID;
	}

	renderTemplate(container: HTMLElement): ITunnelTemplateData {
214
		container.classList.add('custom-view-tree-node-item');
A
Alex Ross 已提交
215 216 217 218 219 220
		const iconLabel = new IconLabel(container, { supportHighlights: true });
		// dom.addClass(iconLabel.element, 'tunnel-view-label');
		const actionsContainer = dom.append(iconLabel.element, dom.$('.actions'));
		const actionBar = new ActionBar(actionsContainer, {
			actionViewItemProvider: (action: IAction) => {
				if (action instanceof MenuItemAction) {
221
					return this.instantiationService.createInstance(MenuEntryActionViewItem, action);
222 223
				} else if (action instanceof SubmenuItemAction) {
					return this.instantiationService.createInstance(SubmenuEntryActionViewItem, action);
A
Alex Ross 已提交
224 225 226 227 228 229 230 231 232 233
				}

				return undefined;
			}
		});

		return { iconLabel, actionBar, container, elementDisposable: Disposable.None };
	}

	private isTunnelItem(item: ITunnelGroup | ITunnelItem): item is ITunnelItem {
234
		return !!((<ITunnelItem>item).remotePort);
A
Alex Ross 已提交
235 236 237 238 239
	}

	renderElement(element: ITreeNode<ITunnelGroup | ITunnelItem, ITunnelGroup | ITunnelItem>, index: number, templateData: ITunnelTemplateData): void {
		templateData.elementDisposable.dispose();
		const node = element.element;
240

A
Alex Ross 已提交
241 242
		// reset
		templateData.actionBar.clear();
A
Alex Ross 已提交
243
		let editableData: IEditableData | undefined;
A
Alex Ross 已提交
244
		if (this.isTunnelItem(node)) {
245
			editableData = this.remoteExplorerService.getEditableData(node);
246 247
			if (editableData) {
				templateData.iconLabel.element.style.display = 'none';
A
Alex Ross 已提交
248
				this.renderInputBox(templateData.container, editableData);
249 250 251
			} else {
				templateData.iconLabel.element.style.display = 'flex';
				this.renderTunnel(node, templateData);
A
Alex Ross 已提交
252
			}
253
		} else if ((node.tunnelType === TunnelType.Add) && (editableData = this.remoteExplorerService.getEditableData(undefined))) {
A
Alex Ross 已提交
254 255
			templateData.iconLabel.element.style.display = 'none';
			this.renderInputBox(templateData.container, editableData);
A
Alex Ross 已提交
256
		} else {
A
Alex Ross 已提交
257
			templateData.iconLabel.element.style.display = 'flex';
A
Alex Ross 已提交
258 259 260 261
			templateData.iconLabel.setLabel(node.label);
		}
	}

262
	private renderTunnel(node: ITunnelItem, templateData: ITunnelTemplateData) {
263 264
		const label = node.label + (node.description ? (' - ' + node.description) : '');
		templateData.iconLabel.setLabel(node.label, node.description, { title: label, extraClasses: ['tunnel-view-label'] });
265
		templateData.actionBar.context = node;
266
		const contextKeyService = this._register(this.contextKeyService.createScoped());
267 268 269
		contextKeyService.createKey('view', this.viewId);
		contextKeyService.createKey('tunnelType', node.tunnelType);
		contextKeyService.createKey('tunnelCloseable', node.closeable);
270 271 272
		const disposableStore = new DisposableStore();
		templateData.elementDisposable = disposableStore;
		const menu = disposableStore.add(this.menuService.createMenu(MenuId.TunnelInline, contextKeyService));
273
		const actions: IAction[] = [];
274
		disposableStore.add(createAndFillInActionBarActions(menu, { shouldForwardArgs: true }, actions));
275 276 277 278 279 280 281 282
		if (actions) {
			templateData.actionBar.push(actions, { icon: true, label: false });
			if (this._actionRunner) {
				templateData.actionBar.actionRunner = this._actionRunner;
			}
		}
	}

A
Alex Ross 已提交
283
	private renderInputBox(container: HTMLElement, editableData: IEditableData): IDisposable {
A
Alex Ross 已提交
284 285 286 287 288
		// Required for FireFox. The blur event doesn't fire on FireFox when you just mash the "+" button to forward a port.
		if (this.inputDone) {
			this.inputDone(false, false);
			this.inputDone = undefined;
		}
289 290 291 292 293
		const value = editableData.startingValue || '';
		const inputBox = new InputBox(container, this.contextViewService, {
			ariaLabel: nls.localize('remote.tunnelsView.input', "Press Enter to confirm or Escape to cancel."),
			validationOptions: {
				validation: (value) => {
294 295
					const message = editableData.validationMessage(value);
					if (!message || message.severity !== Severity.Error) {
296 297 298 299
						return null;
					}

					return {
300
						content: message.content,
301 302 303 304 305 306 307 308 309 310 311
						formatContent: true,
						type: MessageType.ERROR
					};
				}
			},
			placeholder: editableData.placeholder || ''
		});
		const styler = attachInputBoxStyler(inputBox, this.themeService);

		inputBox.value = value;
		inputBox.focus();
312
		inputBox.select({ start: 0, end: editableData.startingValue ? editableData.startingValue.length : 0 });
313 314

		const done = once((success: boolean, finishEditing: boolean) => {
A
Alex Ross 已提交
315 316 317
			if (this.inputDone) {
				this.inputDone = undefined;
			}
318
			inputBox.element.style.display = 'none';
A
Alex Ross 已提交
319
			const inputValue = inputBox.value;
320 321
			dispose(toDispose);
			if (finishEditing) {
A
Alex Ross 已提交
322
				editableData.onFinish(inputValue, success);
323 324
			}
		});
A
Alex Ross 已提交
325
		this.inputDone = done;
326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348

		const toDispose = [
			inputBox,
			dom.addStandardDisposableListener(inputBox.inputElement, dom.EventType.KEY_DOWN, (e: IKeyboardEvent) => {
				if (e.equals(KeyCode.Enter)) {
					if (inputBox.validate()) {
						done(true, true);
					}
				} else if (e.equals(KeyCode.Escape)) {
					done(false, true);
				}
			}),
			dom.addDisposableListener(inputBox.inputElement, dom.EventType.BLUR, () => {
				done(inputBox.isInputValid(), true);
			}),
			styler
		];

		return toDisposable(() => {
			done(false, false);
		});
	}

A
Alex Ross 已提交
349 350 351 352 353 354 355 356 357 358 359 360 361 362 363 364 365 366 367 368 369 370 371 372
	disposeElement(resource: ITreeNode<ITunnelGroup | ITunnelItem, ITunnelGroup | ITunnelItem>, index: number, templateData: ITunnelTemplateData): void {
		templateData.elementDisposable.dispose();
	}

	disposeTemplate(templateData: ITunnelTemplateData): void {
		templateData.actionBar.dispose();
		templateData.elementDisposable.dispose();
	}
}

class TunnelDataSource implements IAsyncDataSource<ITunnelViewModel, ITunnelItem | ITunnelGroup> {
	hasChildren(element: ITunnelViewModel | ITunnelItem | ITunnelGroup) {
		if (element instanceof TunnelViewModel) {
			return true;
		} else if (element instanceof TunnelItem) {
			return false;
		} else if ((<ITunnelGroup>element).items) {
			return true;
		}
		return false;
	}

	getChildren(element: ITunnelViewModel | ITunnelItem | ITunnelGroup) {
		if (element instanceof TunnelViewModel) {
A
Alex Ross 已提交
373
			return element.groups();
A
Alex Ross 已提交
374 375 376 377 378 379 380 381 382 383 384 385
		} else if (element instanceof TunnelItem) {
			return [];
		} else if ((<ITunnelGroup>element).items) {
			return (<ITunnelGroup>element).items!;
		}
		return [];
	}
}

interface ITunnelGroup {
	tunnelType: TunnelType;
	label: string;
A
Alex Ross 已提交
386
	items?: ITunnelItem[] | Promise<ITunnelItem[]>;
A
Alex Ross 已提交
387 388 389
}

class TunnelItem implements ITunnelItem {
A
Alex Ross 已提交
390
	static createFromTunnel(tunnel: Tunnel, type: TunnelType = TunnelType.Forwarded, closeable?: boolean) {
A
Alex Ross 已提交
391
		return new TunnelItem(type, tunnel.remoteHost, tunnel.remotePort, tunnel.localAddress, tunnel.localPort, closeable === undefined ? tunnel.closeable : closeable, tunnel.name, tunnel.description ?? tunnel.runningProcess);
A
Alex Ross 已提交
392 393
	}

A
Alex Ross 已提交
394 395
	constructor(
		public tunnelType: TunnelType,
396 397
		public remoteHost: string,
		public remotePort: number,
A
Alex Ross 已提交
398
		public localAddress?: string,
399
		public localPort?: number,
A
Alex Ross 已提交
400 401 402 403 404
		public closeable?: boolean,
		public name?: string,
		private _description?: string,
	) { }
	get label(): string {
A
Alex Ross 已提交
405
		if (this.name) {
A
Alex Ross 已提交
406
			return nls.localize('remote.tunnelsView.forwardedPortLabel0', "{0}", this.name);
A
Alex Ross 已提交
407
		} else if (this.localAddress) {
A
Alex Ross 已提交
408
			return nls.localize('remote.tunnelsView.forwardedPortLabel1', "{0} \u2192 {1}", this.remotePort, TunnelItem.compactLongAddress(this.localAddress));
A
Alex Ross 已提交
409
		} else {
A
Alex Ross 已提交
410
			return nls.localize('remote.tunnelsView.forwardedPortLabel2', "{0}", this.remotePort);
A
Alex Ross 已提交
411 412 413
		}
	}

A
Alex Ross 已提交
414
	private static compactLongAddress(address: string): string {
415
		if (address.length < 16) {
A
Alex Ross 已提交
416 417
			return address;
		}
418
		let displayAddress: string = address;
419
		try {
420
			if (!address.startsWith('http')) {
A
Alex Ross 已提交
421
				address = `http://${address}`;
422 423 424 425 426 427 428 429 430
			}
			const url = new URL(address);
			if (url && url.host) {
				const lastDotIndex = url.host.lastIndexOf('.');
				const secondLastDotIndex = lastDotIndex !== -1 ? url.host.substring(0, lastDotIndex).lastIndexOf('.') : -1;
				if (secondLastDotIndex !== -1) {
					displayAddress = `${url.protocol}//...${url.host.substring(secondLastDotIndex + 1)}`;
				}
			}
431 432 433
		} catch (e) {
			// Address isn't a valid url and can't be compacted.
		}
434
		return displayAddress;
A
Alex Ross 已提交
435 436
	}

437 438 439 440
	set description(description: string | undefined) {
		this._description = description;
	}

A
Alex Ross 已提交
441
	get description(): string | undefined {
A
Alex Ross 已提交
442
		if (this._description) {
A
Alex Ross 已提交
443
			return this._description;
444 445
		} else if (this.name && this.localAddress) {
			return nls.localize('remote.tunnelsView.forwardedPortDescription0', "{0} \u2192 {1}", this.remotePort, TunnelItem.compactLongAddress(this.localAddress));
A
Alex Ross 已提交
446
		}
A
Alex Ross 已提交
447
		return undefined;
A
Alex Ross 已提交
448 449 450 451 452
	}
}

export const TunnelTypeContextKey = new RawContextKey<TunnelType>('tunnelType', TunnelType.Add);
export const TunnelCloseableContextKey = new RawContextKey<boolean>('tunnelCloseable', false);
453 454 455
const TunnelViewFocusContextKey = new RawContextKey<boolean>('tunnelViewFocus', false);
const TunnelViewSelectionKeyName = 'tunnelViewSelection';
const TunnelViewSelectionContextKey = new RawContextKey<ITunnelItem | undefined>(TunnelViewSelectionKeyName, undefined);
456
const PortChangableContextKey = new RawContextKey<boolean>('portChangable', false);
A
Alex Ross 已提交
457

M
Matt Bierner 已提交
458 459
class TunnelDataTree extends WorkbenchAsyncDataTree<any, any, any> { }

S
SteVen Batten 已提交
460
export class TunnelPanel extends ViewPane {
A
Alex Ross 已提交
461
	static readonly ID = TUNNEL_VIEW_ID;
A
Alex Ross 已提交
462
	static readonly TITLE = nls.localize('remote.tunnel', "Ports");
M
Matt Bierner 已提交
463
	private tree!: TunnelDataTree;
A
Alex Ross 已提交
464 465
	private tunnelTypeContext: IContextKey<TunnelType>;
	private tunnelCloseableContext: IContextKey<boolean>;
466 467
	private tunnelViewFocusContext: IContextKey<boolean>;
	private tunnelViewSelectionContext: IContextKey<ITunnelItem | undefined>;
468
	private portChangableContextKey: IContextKey<boolean>;
A
Alex Ross 已提交
469
	private isEditing: boolean = false;
A
Alex Ross 已提交
470 471 472 473 474
	private titleActions: IAction[] = [];
	private readonly titleActionsDisposable = this._register(new MutableDisposable());

	constructor(
		protected viewModel: ITunnelViewModel,
S
SteVen Batten 已提交
475
		options: IViewPaneOptions,
A
Alex Ross 已提交
476 477 478 479 480
		@IKeybindingService protected keybindingService: IKeybindingService,
		@IContextMenuService protected contextMenuService: IContextMenuService,
		@IContextKeyService protected contextKeyService: IContextKeyService,
		@IConfigurationService protected configurationService: IConfigurationService,
		@IInstantiationService protected readonly instantiationService: IInstantiationService,
481
		@IViewDescriptorService viewDescriptorService: IViewDescriptorService,
482
		@IOpenerService openerService: IOpenerService,
A
Alex Ross 已提交
483 484 485
		@IQuickInputService protected quickInputService: IQuickInputService,
		@ICommandService protected commandService: ICommandService,
		@IMenuService private readonly menuService: IMenuService,
486
		@IContextViewService private readonly contextViewService: IContextViewService,
487
		@IThemeService themeService: IThemeService,
488 489
		@IRemoteExplorerService private readonly remoteExplorerService: IRemoteExplorerService,
		@ITelemetryService telemetryService: ITelemetryService,
A
Alex Ross 已提交
490
	) {
491
		super(options, keybindingService, contextMenuService, configurationService, contextKeyService, viewDescriptorService, instantiationService, openerService, themeService, telemetryService);
A
Alex Ross 已提交
492 493
		this.tunnelTypeContext = TunnelTypeContextKey.bindTo(contextKeyService);
		this.tunnelCloseableContext = TunnelCloseableContextKey.bindTo(contextKeyService);
494 495
		this.tunnelViewFocusContext = TunnelViewFocusContextKey.bindTo(contextKeyService);
		this.tunnelViewSelectionContext = TunnelViewSelectionContextKey.bindTo(contextKeyService);
496
		this.portChangableContextKey = PortChangableContextKey.bindTo(contextKeyService);
A
Alex Ross 已提交
497 498 499 500 501 502 503 504 505 506 507 508 509 510 511 512 513 514 515 516

		const scopedContextKeyService = this._register(this.contextKeyService.createScoped());
		scopedContextKeyService.createKey('view', TunnelPanel.ID);

		const titleMenu = this._register(this.menuService.createMenu(MenuId.TunnelTitle, scopedContextKeyService));
		const updateActions = () => {
			this.titleActions = [];
			this.titleActionsDisposable.value = createAndFillInActionBarActions(titleMenu, undefined, this.titleActions);
			this.updateActions();
		};

		this._register(titleMenu.onDidChange(updateActions));
		updateActions();

		this._register(toDisposable(() => {
			this.titleActions = [];
		}));
	}

	protected renderBody(container: HTMLElement): void {
J
Joao Moreno 已提交
517 518
		super.renderBody(container);

A
Alex Ross 已提交
519 520
		const panelContainer = dom.append(container, dom.$('.tree-explorer-viewlet-tree-view'));
		const treeContainer = dom.append(panelContainer, dom.$('.customview-tree'));
521
		treeContainer.classList.add('file-icon-themable-tree', 'show-file-icons');
A
Alex Ross 已提交
522

523
		const renderer = new TunnelTreeRenderer(TunnelPanel.ID, this.menuService, this.contextKeyService, this.instantiationService, this.contextViewService, this.themeService, this.remoteExplorerService);
M
Matt Bierner 已提交
524
		this.tree = this.instantiationService.createInstance(TunnelDataTree,
A
Alex Ross 已提交
525 526 527 528 529 530 531 532 533 534 535 536 537 538
			'RemoteTunnels',
			treeContainer,
			new TunnelTreeVirtualDelegate(),
			[renderer],
			new TunnelDataSource(),
			{
				collapseByDefault: (e: ITunnelItem | ITunnelGroup): boolean => {
					return false;
				},
				keyboardNavigationLabelProvider: {
					getKeyboardNavigationLabel: (item: ITunnelItem | ITunnelGroup) => {
						return item.label;
					}
				},
539 540 541 542 543 544 545 546 547 548 549 550
				multipleSelectionSupport: false,
				accessibilityProvider: {
					getAriaLabel: (item: ITunnelItem | ITunnelGroup) => {
						if (item instanceof TunnelItem) {
							if (item.localAddress) {
								return nls.localize('remote.tunnel.ariaLabelForwarded', "Remote port {0}:{1} forwarded to local address {2}", item.remoteHost, item.remotePort, item.localAddress);
							} else {
								return nls.localize('remote.tunnel.ariaLabelCandidate', "Remote port {0}:{1} not forwarded", item.remoteHost, item.remotePort);
							}
						} else {
							return item.label;
						}
I
isidor 已提交
551 552
					},
					getWidgetAriaLabel: () => nls.localize('tunnelView', "Tunnel View")
J
João Moreno 已提交
553
				}
A
Alex Ross 已提交
554 555 556 557 558 559
			}
		);
		const actionRunner: ActionRunner = new ActionRunner();
		renderer.actionRunner = actionRunner;

		this._register(this.tree.onContextMenu(e => this.onContextMenu(e, actionRunner)));
560
		this._register(this.tree.onMouseDblClick(e => this.onMouseDblClick(e)));
561 562 563
		this._register(this.tree.onDidChangeFocus(e => this.onFocusChanged(e.elements)));
		this._register(this.tree.onDidFocus(() => this.tunnelViewFocusContext.set(true)));
		this._register(this.tree.onDidBlur(() => this.tunnelViewFocusContext.set(false)));
A
Alex Ross 已提交
564 565 566

		this.tree.setInput(this.viewModel);
		this._register(this.viewModel.onForwardedPortsChanged(() => {
A
Alex Ross 已提交
567
			this._onDidChangeViewWelcomeState.fire();
A
Alex Ross 已提交
568 569 570
			this.tree.updateChildren(undefined, true);
		}));

571
		this._register(Event.debounce(this.tree.onDidOpen, (last, event) => event, 75, true)(e => {
572
			if (e.element && (e.element.tunnelType === TunnelType.Add)) {
573
				this.commandService.executeCommand(ForwardPortAction.INLINE_ID);
A
Alex Ross 已提交
574 575
			}
		}));
576 577

		this._register(this.remoteExplorerService.onDidChangeEditable(async e => {
A
Alex Ross 已提交
578 579
			this.isEditing = !!this.remoteExplorerService.getEditableData(e);
			this._onDidChangeViewWelcomeState.fire();
580

A
Alex Ross 已提交
581
			if (!this.isEditing) {
582
				treeContainer.classList.remove('highlight');
583 584 585 586
			}

			await this.tree.updateChildren(undefined, false);

A
Alex Ross 已提交
587
			if (this.isEditing) {
588
				treeContainer.classList.add('highlight');
A
Alex Ross 已提交
589 590 591 592
				if (!e) {
					// When we are in editing mode for a new forward, rather than updating an existing one we need to reveal the input box since it might be out of view.
					this.tree.reveal(this.viewModel.input);
				}
593 594 595 596
			} else {
				this.tree.domFocus();
			}
		}));
A
Alex Ross 已提交
597 598 599
	}

	private get contributedContextMenu(): IMenu {
600
		const contributedContextMenu = this._register(this.menuService.createMenu(MenuId.TunnelContext, this.tree.contextKeyService));
A
Alex Ross 已提交
601 602 603 604 605 606 607
		return contributedContextMenu;
	}

	getActions(): IAction[] {
		return this.titleActions;
	}

A
Alex Ross 已提交
608
	shouldShowWelcome(): boolean {
609 610
		return (this.viewModel.forwarded.length === 0) && (this.viewModel.candidates.length === 0) &&
			(this.viewModel.detected.length === 0) && !this.isEditing;
A
Alex Ross 已提交
611 612
	}

A
Alex Ross 已提交
613 614 615 616 617
	focus(): void {
		super.focus();
		this.tree.domFocus();
	}

618 619 620 621
	private onFocusChanged(elements: ITunnelItem[]) {
		const item = elements && elements.length ? elements[0] : undefined;
		if (item) {
			this.tunnelViewSelectionContext.set(item);
622
			this.tunnelTypeContext.set(item.tunnelType);
623
			this.tunnelCloseableContext.set(!!item.closeable);
624
			this.portChangableContextKey.set(!!item.localPort);
625
		} else {
626
			this.tunnelTypeContext.reset();
627 628
			this.tunnelViewSelectionContext.reset();
			this.tunnelCloseableContext.reset();
629
			this.portChangableContextKey.reset();
630 631 632
		}
	}

A
Alex Ross 已提交
633
	private onContextMenu(treeEvent: ITreeContextMenuEvent<ITunnelItem | ITunnelGroup>, actionRunner: ActionRunner): void {
634
		if ((treeEvent.element !== null) && !(treeEvent.element instanceof TunnelItem)) {
A
Alex Ross 已提交
635 636 637 638 639 640 641 642
			return;
		}
		const node: ITunnelItem | null = treeEvent.element;
		const event: UIEvent = treeEvent.browserEvent;

		event.preventDefault();
		event.stopPropagation();

643 644 645 646
		if (node) {
			this.tree!.setFocus([node]);
			this.tunnelTypeContext.set(node.tunnelType);
			this.tunnelCloseableContext.set(!!node.closeable);
647
			this.portChangableContextKey.set(!!node.localPort);
648 649
		} else {
			this.tunnelTypeContext.set(TunnelType.Add);
650
			this.tunnelCloseableContext.set(false);
651
			this.portChangableContextKey.set(false);
652
		}
A
Alex Ross 已提交
653 654

		const actions: IAction[] = [];
655
		this._register(createAndFillInContextMenuActions(this.contributedContextMenu, { shouldForwardArgs: true }, actions));
A
Alex Ross 已提交
656 657 658 659 660 661 662 663 664 665 666 667 668 669 670 671 672 673 674 675 676

		this.contextMenuService.showContextMenu({
			getAnchor: () => treeEvent.anchor,
			getActions: () => actions,
			getActionViewItem: (action) => {
				const keybinding = this.keybindingService.lookupKeybinding(action.id);
				if (keybinding) {
					return new ActionViewItem(action, action, { label: true, keybinding: keybinding.getLabel() });
				}
				return undefined;
			},
			onHide: (wasCancelled?: boolean) => {
				if (wasCancelled) {
					this.tree!.domFocus();
				}
			},
			getActionsContext: () => node,
			actionRunner
		});
	}

677 678 679 680 681 682
	private onMouseDblClick(e: ITreeMouseEvent<ITunnelGroup | ITunnelItem | null>): void {
		if (!e.element) {
			this.commandService.executeCommand(ForwardPortAction.INLINE_ID);
		}
	}

A
Alex Ross 已提交
683
	protected layoutBody(height: number, width: number): void {
J
João Moreno 已提交
684
		super.layoutBody(height, width);
A
Alex Ross 已提交
685 686 687 688 689 690 691
		this.tree.layout(height, width);
	}
}

export class TunnelPanelDescriptor implements IViewDescriptor {
	readonly id = TunnelPanel.ID;
	readonly name = TunnelPanel.TITLE;
S
Sandeep Somavarapu 已提交
692
	readonly ctorDescriptor: SyncDescriptor<TunnelPanel>;
A
Alex Ross 已提交
693 694 695 696 697
	readonly canToggleVisibility = true;
	readonly hideByDefault = false;
	readonly workspace = true;
	readonly group = 'details@0';
	readonly remoteAuthority?: string | string[];
A
Alex Ross 已提交
698
	readonly canMoveView = true;
A
Alex Ross 已提交
699 700

	constructor(viewModel: ITunnelViewModel, environmentService: IWorkbenchEnvironmentService) {
S
Sandeep Somavarapu 已提交
701
		this.ctorDescriptor = new SyncDescriptor(TunnelPanel, [viewModel]);
702
		this.remoteAuthority = environmentService.remoteAuthority ? environmentService.remoteAuthority.split('+')[0] : undefined;
A
Alex Ross 已提交
703 704 705
	}
}

J
jeanp413 已提交
706 707 708 709 710 711 712 713 714 715 716
function validationMessage(validationString: string | null): { content: string, severity: Severity } | null {
	if (!validationString) {
		return null;
	}

	return {
		content: validationString,
		severity: Severity.Error
	};
}

717 718 719
namespace LabelTunnelAction {
	export const ID = 'remote.tunnel.label';
	export const LABEL = nls.localize('remote.tunnel.label', "Set Label");
A
Alex Ross 已提交
720 721 722

	export function handler(): ICommandHandler {
		return async (accessor, arg) => {
723 724
			const context = (arg !== undefined || arg instanceof TunnelItem) ? arg : accessor.get(IContextKeyService).getContextKeyValue(TunnelViewSelectionKeyName);
			if (context instanceof TunnelItem) {
A
Alex Ross 已提交
725
				const remoteExplorerService = accessor.get(IRemoteExplorerService);
726
				remoteExplorerService.setEditable(context, {
727
					onFinish: async (value, success) => {
728
						if (success) {
729
							remoteExplorerService.tunnelModel.name(context.remoteHost, context.remotePort, value);
730
						}
731
						remoteExplorerService.setEditable(context, null);
732 733
					},
					validationMessage: () => null,
734
					placeholder: nls.localize('remote.tunnelsView.labelPlaceholder', "Port label"),
735
					startingValue: context.name
736
				});
A
Alex Ross 已提交
737 738 739 740 741 742
			}
			return;
		};
	}
}

743 744 745
const invalidPortString: string = nls.localize('remote.tunnelsView.portNumberValid', "Forwarded port is invalid.");
const maxPortNumber: number = 65536;
const invalidPortNumberString: string = nls.localize('remote.tunnelsView.portNumberToHigh', "Port number must be \u2265 0 and < {0}.", maxPortNumber);
746

A
Alex Ross 已提交
747
export namespace ForwardPortAction {
748 749
	export const INLINE_ID = 'remote.tunnel.forwardInline';
	export const COMMANDPALETTE_ID = 'remote.tunnel.forwardCommandPalette';
750
	export const LABEL: ILocalizedString = { value: nls.localize('remote.tunnel.forward', "Forward a Port"), original: 'Forward a Port' };
751
	export const TREEITEM_LABEL = nls.localize('remote.tunnel.forwardItem', "Forward Port");
A
Alex Ross 已提交
752
	const forwardPrompt = nls.localize('remote.tunnel.forwardPrompt', "Port number or address (eg. 3000 or 10.10.10.10:2000).");
A
Alex Ross 已提交
753

A
Alex Ross 已提交
754
	function validateInput(value: string): string | null {
A
Alex Ross 已提交
755
		const parsed = parseAddress(value);
756
		if (!parsed) {
757
			return invalidPortString;
758 759
		} else if (parsed.port >= maxPortNumber) {
			return invalidPortNumberString;
A
Alex Ross 已提交
760 761 762 763
		}
		return null;
	}

A
Alex Ross 已提交
764 765
	function error(notificationService: INotificationService, tunnel: RemoteTunnel | void, host: string, port: number) {
		if (!tunnel) {
766
			notificationService.warn(nls.localize('remote.tunnel.forwardError', "Unable to forward {0}:{1}. The host may not be available or that remote port may already be forwarded", host, port));
A
Alex Ross 已提交
767 768 769
		}
	}

770
	export function inlineHandler(): ICommandHandler {
A
Alex Ross 已提交
771 772
		return async (accessor, arg) => {
			const remoteExplorerService = accessor.get(IRemoteExplorerService);
A
Alex Ross 已提交
773
			const notificationService = accessor.get(INotificationService);
A
Alex Ross 已提交
774
			if (arg instanceof TunnelItem) {
A
Alex Ross 已提交
775
				remoteExplorerService.forward({ host: arg.remoteHost, port: arg.remotePort }).then(tunnel => error(notificationService, tunnel, arg.remoteHost, arg.remotePort));
776 777
			} else {
				remoteExplorerService.setEditable(undefined, {
778
					onFinish: async (value, success) => {
779
						let parsed: { host: string, port: number } | undefined;
A
Alex Ross 已提交
780
						if (success && (parsed = parseAddress(value))) {
A
Alex Ross 已提交
781
							remoteExplorerService.forward({ host: parsed.host, port: parsed.port }).then(tunnel => error(notificationService, tunnel, parsed!.host, parsed!.port));
A
Alex Ross 已提交
782
						}
783
						remoteExplorerService.setEditable(undefined, null);
A
Alex Ross 已提交
784
					},
J
jeanp413 已提交
785
					validationMessage: (value) => validationMessage(validateInput(value)),
A
Alex Ross 已提交
786
					placeholder: forwardPrompt
A
Alex Ross 已提交
787
				});
788 789 790 791 792 793 794
			}
		};
	}

	export function commandPaletteHandler(): ICommandHandler {
		return async (accessor, arg) => {
			const remoteExplorerService = accessor.get(IRemoteExplorerService);
A
Alex Ross 已提交
795
			const notificationService = accessor.get(INotificationService);
796 797 798 799 800 801 802 803
			const viewsService = accessor.get(IViewsService);
			const quickInputService = accessor.get(IQuickInputService);
			await viewsService.openView(TunnelPanel.ID, true);
			const value = await quickInputService.input({
				prompt: forwardPrompt,
				validateInput: (value) => Promise.resolve(validateInput(value))
			});
			let parsed: { host: string, port: number } | undefined;
A
Alex Ross 已提交
804
			if (value && (parsed = parseAddress(value))) {
A
Alex Ross 已提交
805
				remoteExplorerService.forward({ host: parsed.host, port: parsed.port }).then(tunnel => error(notificationService, tunnel, parsed!.host, parsed!.port));
A
Alex Ross 已提交
806 807 808 809 810
			}
		};
	}
}

A
Alex Ross 已提交
811
interface QuickPickTunnel extends IQuickPickItem {
A
Alex Ross 已提交
812
	tunnel?: ITunnelItem
A
Alex Ross 已提交
813 814
}

A
Alex Ross 已提交
815 816 817 818 819 820 821 822
function makeTunnelPicks(tunnels: Tunnel[]): QuickPickInput<QuickPickTunnel>[] {
	const picks: QuickPickInput<QuickPickTunnel>[] = tunnels.map(forwarded => {
		const item = TunnelItem.createFromTunnel(forwarded);
		return {
			label: item.label,
			description: item.description,
			tunnel: item
		};
A
Alex Ross 已提交
823 824 825
	});
	if (picks.length === 0) {
		picks.push({
826
			label: nls.localize('remote.tunnel.closeNoPorts', "No ports currently forwarded. Try running the {0} command", ForwardPortAction.LABEL.value)
A
Alex Ross 已提交
827 828 829 830 831
		});
	}
	return picks;
}

A
Alex Ross 已提交
832
namespace ClosePortAction {
A
Alex Ross 已提交
833 834
	export const INLINE_ID = 'remote.tunnel.closeInline';
	export const COMMANDPALETTE_ID = 'remote.tunnel.closeCommandPalette';
835
	export const LABEL: ILocalizedString = { value: nls.localize('remote.tunnel.close', "Stop Forwarding Port"), original: 'Stop Forwarding Port' };
A
Alex Ross 已提交
836

A
Alex Ross 已提交
837
	export function inlineHandler(): ICommandHandler {
A
Alex Ross 已提交
838
		return async (accessor, arg) => {
839
			const context = (arg !== undefined || arg instanceof TunnelItem) ? arg : accessor.get(IContextKeyService).getContextKeyValue(TunnelViewSelectionKeyName);
840
			if (context instanceof TunnelItem) {
A
Alex Ross 已提交
841
				const remoteExplorerService = accessor.get(IRemoteExplorerService);
842
				await remoteExplorerService.close({ host: context.remoteHost, port: context.remotePort });
A
Alex Ross 已提交
843 844 845
			}
		};
	}
A
Alex Ross 已提交
846 847 848 849 850 851 852

	export function commandPaletteHandler(): ICommandHandler {
		return async (accessor) => {
			const quickInputService = accessor.get(IQuickInputService);
			const remoteExplorerService = accessor.get(IRemoteExplorerService);
			const commandService = accessor.get(ICommandService);

A
Alex Ross 已提交
853
			const picks: QuickPickInput<QuickPickTunnel>[] = makeTunnelPicks(Array.from(remoteExplorerService.tunnelModel.forwarded.values()).filter(tunnel => tunnel.closeable));
A
Alex Ross 已提交
854 855 856 857 858 859 860 861
			const result = await quickInputService.pick(picks, { placeHolder: nls.localize('remote.tunnel.closePlaceholder', "Choose a port to stop forwarding") });
			if (result && result.tunnel) {
				await remoteExplorerService.close({ host: result.tunnel.remoteHost, port: result.tunnel.remotePort });
			} else if (result) {
				await commandService.executeCommand(ForwardPortAction.COMMANDPALETTE_ID);
			}
		};
	}
A
Alex Ross 已提交
862 863
}

864
export namespace OpenPortInBrowserAction {
A
Alex Ross 已提交
865 866 867 868 869
	export const ID = 'remote.tunnel.open';
	export const LABEL = nls.localize('remote.tunnel.open', "Open in Browser");

	export function handler(): ICommandHandler {
		return async (accessor, arg) => {
A
Alex Ross 已提交
870
			let key: string | undefined;
A
Alex Ross 已提交
871
			if (arg instanceof TunnelItem) {
A
Alex Ross 已提交
872 873 874 875 876
				key = makeAddress(arg.remoteHost, arg.remotePort);
			} else if (arg.tunnelRemoteHost && arg.tunnelRemotePort) {
				key = makeAddress(arg.tunnelRemoteHost, arg.tunnelRemotePort);
			}
			if (key) {
A
Alex Ross 已提交
877 878
				const model = accessor.get(IRemoteExplorerService).tunnelModel;
				const openerService = accessor.get(IOpenerService);
879
				return run(model, openerService, key);
A
Alex Ross 已提交
880 881 882
			}
		};
	}
883 884 885 886 887

	export function run(model: TunnelModel, openerService: IOpenerService, key: string) {
		const tunnel = model.forwarded.get(key) || model.detected.get(key);
		let address: string | undefined;
		if (tunnel && tunnel.localAddress && (address = model.address(tunnel.remoteHost, tunnel.remotePort))) {
888
			if (!address.startsWith('http')) {
A
Alex Ross 已提交
889
				address = `http://${address}`;
890 891
			}
			return openerService.open(URI.parse(address));
892 893 894
		}
		return Promise.resolve();
	}
A
Alex Ross 已提交
895 896 897
}

namespace CopyAddressAction {
A
Alex Ross 已提交
898 899 900 901 902 903 904 905 906 907 908
	export const INLINE_ID = 'remote.tunnel.copyAddressInline';
	export const COMMANDPALETTE_ID = 'remote.tunnel.copyAddressCommandPalette';
	export const INLINE_LABEL = nls.localize('remote.tunnel.copyAddressInline', "Copy Address");
	export const COMMANDPALETTE_LABEL = nls.localize('remote.tunnel.copyAddressCommandPalette', "Copy Forwarded Port Address");

	async function copyAddress(remoteExplorerService: IRemoteExplorerService, clipboardService: IClipboardService, tunnelItem: ITunnelItem) {
		const address = remoteExplorerService.tunnelModel.address(tunnelItem.remoteHost, tunnelItem.remotePort);
		if (address) {
			await clipboardService.writeText(address.toString());
		}
	}
A
Alex Ross 已提交
909

A
Alex Ross 已提交
910
	export function inlineHandler(): ICommandHandler {
A
Alex Ross 已提交
911
		return async (accessor, arg) => {
912 913 914
			const context = (arg !== undefined || arg instanceof TunnelItem) ? arg : accessor.get(IContextKeyService).getContextKeyValue(TunnelViewSelectionKeyName);
			if (context instanceof TunnelItem) {
				return copyAddress(accessor.get(IRemoteExplorerService), accessor.get(IClipboardService), context);
A
Alex Ross 已提交
915 916 917 918 919 920 921 922 923 924 925 926 927 928 929 930 931
			}
		};
	}

	export function commandPaletteHandler(): ICommandHandler {
		return async (accessor, arg) => {
			const quickInputService = accessor.get(IQuickInputService);
			const remoteExplorerService = accessor.get(IRemoteExplorerService);
			const commandService = accessor.get(ICommandService);
			const clipboardService = accessor.get(IClipboardService);

			const tunnels = Array.from(remoteExplorerService.tunnelModel.forwarded.values()).concat(Array.from(remoteExplorerService.tunnelModel.detected.values()));
			const result = await quickInputService.pick(makeTunnelPicks(tunnels), { placeHolder: nls.localize('remote.tunnel.copyAddressPlaceholdter', "Choose a forwarded port") });
			if (result && result.tunnel) {
				await copyAddress(remoteExplorerService, clipboardService, result.tunnel);
			} else if (result) {
				await commandService.executeCommand(ForwardPortAction.COMMANDPALETTE_ID);
A
Alex Ross 已提交
932 933 934 935 936
			}
		};
	}
}

937 938 939 940 941 942 943
namespace ChangeLocalPortAction {
	export const ID = 'remote.tunnel.changeLocalPort';
	export const LABEL = nls.localize('remote.tunnel.changeLocalPort', "Change Local Port");

	function validateInput(value: string): string | null {
		if (!value.match(/^[0-9]+$/)) {
			return invalidPortString;
944 945
		} else if (Number(value) >= maxPortNumber) {
			return invalidPortNumberString;
946 947 948 949 950 951 952
		}
		return null;
	}

	export function handler(): ICommandHandler {
		return async (accessor, arg) => {
			const remoteExplorerService = accessor.get(IRemoteExplorerService);
953
			const notificationService = accessor.get(INotificationService);
954 955 956 957 958 959 960
			const context = (arg !== undefined || arg instanceof TunnelItem) ? arg : accessor.get(IContextKeyService).getContextKeyValue(TunnelViewSelectionKeyName);
			if (context instanceof TunnelItem) {
				remoteExplorerService.setEditable(context, {
					onFinish: async (value, success) => {
						remoteExplorerService.setEditable(context, null);
						if (success) {
							await remoteExplorerService.close({ host: context.remoteHost, port: context.remotePort });
961
							const numberValue = Number(value);
962
							const newForward = await remoteExplorerService.forward({ host: context.remoteHost, port: context.remotePort }, numberValue, context.name);
963 964 965
							if (newForward && newForward.tunnelLocalPort !== numberValue) {
								notificationService.warn(nls.localize('remote.tunnel.changeLocalPortNumber', "The local port {0} is not available. Port number {1} has been used instead", value, newForward.tunnelLocalPort));
							}
966 967
						}
					},
J
jeanp413 已提交
968
					validationMessage: (value) => validationMessage(validateInput(value)),
969 970 971 972 973 974 975
					placeholder: nls.localize('remote.tunnelsView.changePort', "New local port")
				});
			}
		};
	}
}

976 977
const tunnelViewCommandsWeightBonus = 10; // give our commands a little bit more weight over other default list/tree commands

978 979
KeybindingsRegistry.registerCommandAndKeybindingRule({
	id: LabelTunnelAction.ID,
980
	weight: KeybindingWeight.WorkbenchContrib + tunnelViewCommandsWeightBonus,
981 982 983 984 985 986 987
	when: ContextKeyExpr.and(TunnelViewFocusContextKey, TunnelTypeContextKey.isEqualTo(TunnelType.Forwarded)),
	primary: KeyCode.F2,
	mac: {
		primary: KeyCode.Enter
	},
	handler: LabelTunnelAction.handler()
});
988 989
CommandsRegistry.registerCommand(ForwardPortAction.INLINE_ID, ForwardPortAction.inlineHandler());
CommandsRegistry.registerCommand(ForwardPortAction.COMMANDPALETTE_ID, ForwardPortAction.commandPaletteHandler());
990 991
KeybindingsRegistry.registerCommandAndKeybindingRule({
	id: ClosePortAction.INLINE_ID,
992
	weight: KeybindingWeight.WorkbenchContrib + tunnelViewCommandsWeightBonus,
993 994 995 996 997 998 999 1000
	when: ContextKeyExpr.and(TunnelCloseableContextKey, TunnelViewFocusContextKey),
	primary: KeyCode.Delete,
	mac: {
		primary: KeyMod.CtrlCmd | KeyCode.Backspace
	},
	handler: ClosePortAction.inlineHandler()
});

A
Alex Ross 已提交
1001
CommandsRegistry.registerCommand(ClosePortAction.COMMANDPALETTE_ID, ClosePortAction.commandPaletteHandler());
A
Alex Ross 已提交
1002
CommandsRegistry.registerCommand(OpenPortInBrowserAction.ID, OpenPortInBrowserAction.handler());
1003 1004
KeybindingsRegistry.registerCommandAndKeybindingRule({
	id: CopyAddressAction.INLINE_ID,
1005
	weight: KeybindingWeight.WorkbenchContrib + tunnelViewCommandsWeightBonus,
1006 1007 1008 1009
	when: ContextKeyExpr.or(ContextKeyExpr.and(TunnelViewFocusContextKey, TunnelTypeContextKey.isEqualTo(TunnelType.Forwarded)), ContextKeyExpr.and(TunnelViewFocusContextKey, TunnelTypeContextKey.isEqualTo(TunnelType.Detected))),
	primary: KeyMod.CtrlCmd | KeyCode.KEY_C,
	handler: CopyAddressAction.inlineHandler()
});
A
Alex Ross 已提交
1010
CommandsRegistry.registerCommand(CopyAddressAction.COMMANDPALETTE_ID, CopyAddressAction.commandPaletteHandler());
1011
CommandsRegistry.registerCommand(ChangeLocalPortAction.ID, ChangeLocalPortAction.handler());
A
Alex Ross 已提交
1012

A
Alex Ross 已提交
1013 1014 1015 1016 1017 1018 1019
MenuRegistry.appendMenuItem(MenuId.CommandPalette, ({
	command: {
		id: ClosePortAction.COMMANDPALETTE_ID,
		title: ClosePortAction.LABEL
	},
	when: forwardedPortsViewEnabled
}));
1020 1021
MenuRegistry.appendMenuItem(MenuId.CommandPalette, ({
	command: {
1022
		id: ForwardPortAction.COMMANDPALETTE_ID,
1023 1024 1025 1026
		title: ForwardPortAction.LABEL
	},
	when: forwardedPortsViewEnabled
}));
A
Alex Ross 已提交
1027 1028 1029 1030 1031 1032 1033
MenuRegistry.appendMenuItem(MenuId.CommandPalette, ({
	command: {
		id: CopyAddressAction.COMMANDPALETTE_ID,
		title: CopyAddressAction.COMMANDPALETTE_LABEL
	},
	when: forwardedPortsViewEnabled
}));
A
Alex Ross 已提交
1034 1035 1036 1037
MenuRegistry.appendMenuItem(MenuId.TunnelTitle, ({
	group: 'navigation',
	order: 0,
	command: {
A
Alex Ross 已提交
1038
		id: ForwardPortAction.INLINE_ID,
A
Alex Ross 已提交
1039
		title: ForwardPortAction.LABEL,
1040
		icon: Codicon.plus
A
Alex Ross 已提交
1041 1042 1043 1044 1045 1046
	}
}));
MenuRegistry.appendMenuItem(MenuId.TunnelContext, ({
	group: '0_manage',
	order: 0,
	command: {
A
Alex Ross 已提交
1047 1048
		id: CopyAddressAction.INLINE_ID,
		title: CopyAddressAction.INLINE_LABEL,
A
Alex Ross 已提交
1049
	},
A
Alex Ross 已提交
1050
	when: ContextKeyExpr.or(TunnelTypeContextKey.isEqualTo(TunnelType.Forwarded), TunnelTypeContextKey.isEqualTo(TunnelType.Detected))
A
Alex Ross 已提交
1051 1052 1053 1054 1055 1056 1057 1058
}));
MenuRegistry.appendMenuItem(MenuId.TunnelContext, ({
	group: '0_manage',
	order: 1,
	command: {
		id: OpenPortInBrowserAction.ID,
		title: OpenPortInBrowserAction.LABEL,
	},
A
Alex Ross 已提交
1059
	when: ContextKeyExpr.or(TunnelTypeContextKey.isEqualTo(TunnelType.Forwarded), TunnelTypeContextKey.isEqualTo(TunnelType.Detected))
A
Alex Ross 已提交
1060 1061 1062 1063 1064
}));
MenuRegistry.appendMenuItem(MenuId.TunnelContext, ({
	group: '0_manage',
	order: 2,
	command: {
1065 1066
		id: LabelTunnelAction.ID,
		title: LabelTunnelAction.LABEL,
A
Alex Ross 已提交
1067 1068 1069
	},
	when: TunnelTypeContextKey.isEqualTo(TunnelType.Forwarded)
}));
1070 1071 1072 1073 1074 1075 1076
MenuRegistry.appendMenuItem(MenuId.TunnelContext, ({
	group: '1_manage',
	order: 0,
	command: {
		id: ChangeLocalPortAction.ID,
		title: ChangeLocalPortAction.LABEL,
	},
1077
	when: ContextKeyExpr.and(TunnelTypeContextKey.isEqualTo(TunnelType.Forwarded), PortChangableContextKey)
1078
}));
A
Alex Ross 已提交
1079 1080 1081 1082
MenuRegistry.appendMenuItem(MenuId.TunnelContext, ({
	group: '0_manage',
	order: 1,
	command: {
1083
		id: ForwardPortAction.INLINE_ID,
1084
		title: ForwardPortAction.TREEITEM_LABEL,
A
Alex Ross 已提交
1085
	},
1086
	when: ContextKeyExpr.or(TunnelTypeContextKey.isEqualTo(TunnelType.Candidate), TunnelTypeContextKey.isEqualTo(TunnelType.Add))
A
Alex Ross 已提交
1087 1088
}));
MenuRegistry.appendMenuItem(MenuId.TunnelContext, ({
1089 1090
	group: '1_manage',
	order: 1,
A
Alex Ross 已提交
1091
	command: {
A
Alex Ross 已提交
1092
		id: ClosePortAction.INLINE_ID,
A
Alex Ross 已提交
1093 1094 1095 1096 1097 1098 1099 1100 1101 1102
		title: ClosePortAction.LABEL,
	},
	when: TunnelCloseableContextKey
}));

MenuRegistry.appendMenuItem(MenuId.TunnelInline, ({
	order: 0,
	command: {
		id: OpenPortInBrowserAction.ID,
		title: OpenPortInBrowserAction.LABEL,
1103
		icon: Codicon.globe
A
Alex Ross 已提交
1104
	},
A
Alex Ross 已提交
1105
	when: ContextKeyExpr.or(TunnelTypeContextKey.isEqualTo(TunnelType.Forwarded), TunnelTypeContextKey.isEqualTo(TunnelType.Detected))
A
Alex Ross 已提交
1106 1107 1108 1109
}));
MenuRegistry.appendMenuItem(MenuId.TunnelInline, ({
	order: 0,
	command: {
1110
		id: ForwardPortAction.INLINE_ID,
1111
		title: ForwardPortAction.TREEITEM_LABEL,
1112
		icon: Codicon.plus
A
Alex Ross 已提交
1113
	},
A
Alex Ross 已提交
1114
	when: TunnelTypeContextKey.isEqualTo(TunnelType.Candidate)
A
Alex Ross 已提交
1115 1116 1117 1118
}));
MenuRegistry.appendMenuItem(MenuId.TunnelInline, ({
	order: 2,
	command: {
A
Alex Ross 已提交
1119
		id: ClosePortAction.INLINE_ID,
A
Alex Ross 已提交
1120
		title: ClosePortAction.LABEL,
1121
		icon: Codicon.x
A
Alex Ross 已提交
1122 1123 1124
	},
	when: TunnelCloseableContextKey
}));