Files
vscode/src/vs/workbench/services/remote/common/remoteExplorerService.ts
Alex Ross b09701ac1b Fix duplicate port forwarding for statically forwarded ports (#163899)
* Fix duplicate port in view

* Include localhost address when resolving external
2022-10-18 11:08:21 -07:00

1021 lines
40 KiB
TypeScript

/*---------------------------------------------------------------------------------------------
* Copyright (c) Microsoft Corporation. All rights reserved.
* Licensed under the MIT License. See License.txt in the project root for license information.
*--------------------------------------------------------------------------------------------*/
import * as nls from 'vs/nls';
import { Event, Emitter } from 'vs/base/common/event';
import { createDecorator } from 'vs/platform/instantiation/common/instantiation';
import { InstantiationType, registerSingleton } from 'vs/platform/instantiation/common/extensions';
import { IStorageService, StorageScope, StorageTarget } from 'vs/platform/storage/common/storage';
import { ALL_INTERFACES_ADDRESSES, isAllInterfaces, isLocalhost, ITunnelService, LOCALHOST_ADDRESSES, PortAttributesProvider, ProvidedOnAutoForward, ProvidedPortAttributes, RemoteTunnel, TunnelPrivacyId, TunnelProtocol } from 'vs/platform/tunnel/common/tunnel';
import { Disposable, IDisposable } from 'vs/base/common/lifecycle';
import { IEditableData } from 'vs/workbench/common/views';
import { ConfigurationTarget, IConfigurationService } from 'vs/platform/configuration/common/configuration';
import { TunnelInformation, TunnelDescription, IRemoteAuthorityResolverService, TunnelPrivacy } from 'vs/platform/remote/common/remoteAuthorityResolver';
import { IWorkbenchEnvironmentService } from 'vs/workbench/services/environment/common/environmentService';
import { IAddressProvider } from 'vs/platform/remote/common/remoteAgentConnection';
import { isNumber, isObject, isString } from 'vs/base/common/types';
import { IWorkspaceContextService } from 'vs/platform/workspace/common/workspace';
import { hash } from 'vs/base/common/hash';
import { ILogService } from 'vs/platform/log/common/log';
import { CancellationTokenSource } from 'vs/base/common/cancellation';
import { flatten } from 'vs/base/common/arrays';
import Severity from 'vs/base/common/severity';
import { IDialogService } from 'vs/platform/dialogs/common/dialogs';
import { URI } from 'vs/base/common/uri';
import { deepClone } from 'vs/base/common/objects';
export const IRemoteExplorerService = createDecorator<IRemoteExplorerService>('remoteExplorerService');
export const REMOTE_EXPLORER_TYPE_KEY: string = 'remote.explorerType';
const TUNNELS_TO_RESTORE = 'remote.tunnels.toRestore';
export const TUNNEL_VIEW_ID = '~remote.forwardedPorts';
export const TUNNEL_VIEW_CONTAINER_ID = '~remote.forwardedPortsContainer';
export const PORT_AUTO_FORWARD_SETTING = 'remote.autoForwardPorts';
export const PORT_AUTO_SOURCE_SETTING = 'remote.autoForwardPortsSource';
export const PORT_AUTO_SOURCE_SETTING_PROCESS = 'process';
export const PORT_AUTO_SOURCE_SETTING_OUTPUT = 'output';
export enum TunnelType {
Candidate = 'Candidate',
Detected = 'Detected',
Forwarded = 'Forwarded',
Add = 'Add'
}
export interface ITunnelItem {
tunnelType: TunnelType;
remoteHost: string;
remotePort: number;
localAddress?: string;
protocol: TunnelProtocol;
localUri?: URI;
localPort?: number;
name?: string;
closeable?: boolean;
source: {
source: TunnelSource;
description: string;
};
privacy: TunnelPrivacy;
processDescription?: string;
readonly label: string;
}
export enum TunnelEditId {
None = 0,
New = 1,
Label = 2,
LocalPort = 3
}
interface TunnelProperties {
remote: { host: string; port: number };
local?: number;
name?: string;
source?: {
source: TunnelSource;
description: string;
};
elevateIfNeeded?: boolean;
privacy?: string;
}
export enum TunnelSource {
User,
Auto,
Extension
}
export const UserTunnelSource = {
source: TunnelSource.User,
description: nls.localize('tunnel.source.user', "User Forwarded")
};
export const AutoTunnelSource = {
source: TunnelSource.Auto,
description: nls.localize('tunnel.source.auto', "Auto Forwarded")
};
export interface Tunnel {
remoteHost: string;
remotePort: number;
localAddress: string;
localUri: URI;
protocol: TunnelProtocol;
localPort?: number;
name?: string;
closeable?: boolean;
privacy: TunnelPrivacyId | string;
runningProcess: string | undefined;
hasRunningProcess?: boolean;
pid: number | undefined;
source: {
source: TunnelSource;
description: string;
};
}
export function makeAddress(host: string, port: number): string {
return host + ':' + port;
}
export function parseAddress(address: string): { host: string; port: number } | undefined {
const matches = address.match(/^([a-zA-Z0-9-]+(?:\.[a-zA-Z0-9-]+)*:)?([0-9]+)$/);
if (!matches) {
return undefined;
}
return { host: matches[1]?.substring(0, matches[1].length - 1) || 'localhost', port: Number(matches[2]) };
}
export function mapHasAddress<T>(map: Map<string, T>, host: string, port: number): T | undefined {
const initialAddress = map.get(makeAddress(host, port));
if (initialAddress) {
return initialAddress;
}
if (isLocalhost(host)) {
// Do localhost checks
for (const testHost of LOCALHOST_ADDRESSES) {
const testAddress = makeAddress(testHost, port);
if (map.has(testAddress)) {
return map.get(testAddress);
}
}
} else if (isAllInterfaces(host)) {
// Do all interfaces checks
for (const testHost of ALL_INTERFACES_ADDRESSES) {
const testAddress = makeAddress(testHost, port);
if (map.has(testAddress)) {
return map.get(testAddress);
}
}
}
return undefined;
}
export function mapHasAddressLocalhostOrAllInterfaces<T>(map: Map<string, T>, host: string, port: number): T | undefined {
const originalAddress = mapHasAddress(map, host, port);
if (originalAddress) {
return originalAddress;
}
const otherHost = isAllInterfaces(host) ? 'localhost' : (isLocalhost(host) ? '0.0.0.0' : undefined);
if (otherHost) {
return mapHasAddress(map, otherHost, port);
}
return undefined;
}
export enum OnPortForward {
Notify = 'notify',
OpenBrowser = 'openBrowser',
OpenBrowserOnce = 'openBrowserOnce',
OpenPreview = 'openPreview',
Silent = 'silent',
Ignore = 'ignore'
}
export interface Attributes {
label: string | undefined;
onAutoForward: OnPortForward | undefined;
elevateIfNeeded: boolean | undefined;
requireLocalPort: boolean | undefined;
protocol: TunnelProtocol | undefined;
}
interface PortRange { start: number; end: number }
interface HostAndPort { host: string; port: number }
interface PortAttributes extends Attributes {
key: number | PortRange | RegExp | HostAndPort;
}
export class PortsAttributes extends Disposable {
private static SETTING = 'remote.portsAttributes';
private static DEFAULTS = 'remote.otherPortsAttributes';
private static RANGE = /^(\d+)\-(\d+)$/;
private static HOST_AND_PORT = /^([a-z0-9\-]+):(\d{1,5})$/;
private portsAttributes: PortAttributes[] = [];
private defaultPortAttributes: Attributes | undefined;
private _onDidChangeAttributes = new Emitter<void>();
public readonly onDidChangeAttributes = this._onDidChangeAttributes.event;
constructor(private readonly configurationService: IConfigurationService) {
super();
this._register(configurationService.onDidChangeConfiguration(e => {
if (e.affectsConfiguration(PortsAttributes.SETTING) || e.affectsConfiguration(PortsAttributes.DEFAULTS)) {
this.updateAttributes();
}
}));
this.updateAttributes();
}
private updateAttributes() {
this.portsAttributes = this.readSetting();
this._onDidChangeAttributes.fire();
}
getAttributes(port: number, host: string, commandLine?: string): Attributes | undefined {
let index = this.findNextIndex(port, host, commandLine, this.portsAttributes, 0);
const attributes: Attributes = {
label: undefined,
onAutoForward: undefined,
elevateIfNeeded: undefined,
requireLocalPort: undefined,
protocol: undefined
};
while (index >= 0) {
const found = this.portsAttributes[index];
if (found.key === port) {
attributes.onAutoForward = found.onAutoForward ?? attributes.onAutoForward;
attributes.elevateIfNeeded = (found.elevateIfNeeded !== undefined) ? found.elevateIfNeeded : attributes.elevateIfNeeded;
attributes.label = found.label ?? attributes.label;
attributes.requireLocalPort = found.requireLocalPort;
attributes.protocol = found.protocol;
} else {
// It's a range or regex, which means that if the attribute is already set, we keep it
attributes.onAutoForward = attributes.onAutoForward ?? found.onAutoForward;
attributes.elevateIfNeeded = (attributes.elevateIfNeeded !== undefined) ? attributes.elevateIfNeeded : found.elevateIfNeeded;
attributes.label = attributes.label ?? found.label;
attributes.requireLocalPort = (attributes.requireLocalPort !== undefined) ? attributes.requireLocalPort : undefined;
attributes.protocol = attributes.protocol ?? found.protocol;
}
index = this.findNextIndex(port, host, commandLine, this.portsAttributes, index + 1);
}
if (attributes.onAutoForward !== undefined || attributes.elevateIfNeeded !== undefined
|| attributes.label !== undefined || attributes.requireLocalPort !== undefined
|| attributes.protocol !== undefined) {
return attributes;
}
// If we find no matches, then use the other port attributes.
return this.getOtherAttributes();
}
private hasStartEnd(value: number | PortRange | RegExp | HostAndPort): value is PortRange {
return ((<any>value).start !== undefined) && ((<any>value).end !== undefined);
}
private hasHostAndPort(value: number | PortRange | RegExp | HostAndPort): value is HostAndPort {
return ((<any>value).host !== undefined) && ((<any>value).port !== undefined)
&& isString((<any>value).host) && isNumber((<any>value).port);
}
private findNextIndex(port: number, host: string, commandLine: string | undefined, attributes: PortAttributes[], fromIndex: number): number {
if (fromIndex >= attributes.length) {
return -1;
}
const shouldUseHost = !isLocalhost(host) && !isAllInterfaces(host);
const sliced = attributes.slice(fromIndex);
const foundIndex = sliced.findIndex((value) => {
if (isNumber(value.key)) {
return shouldUseHost ? false : value.key === port;
} else if (this.hasStartEnd(value.key)) {
return shouldUseHost ? false : (port >= value.key.start && port <= value.key.end);
} else if (this.hasHostAndPort(value.key)) {
return (port === value.key.port) && (host === value.key.host);
} else {
return commandLine ? value.key.test(commandLine) : false;
}
});
return foundIndex >= 0 ? foundIndex + fromIndex : -1;
}
private readSetting(): PortAttributes[] {
const settingValue = this.configurationService.getValue(PortsAttributes.SETTING);
if (!settingValue || !isObject(settingValue)) {
return [];
}
const attributes: PortAttributes[] = [];
for (const attributesKey in settingValue) {
if (attributesKey === undefined) {
continue;
}
const setting = (<any>settingValue)[attributesKey];
let key: number | PortRange | RegExp | HostAndPort | undefined = undefined;
if (Number(attributesKey)) {
key = Number(attributesKey);
} else if (isString(attributesKey)) {
if (PortsAttributes.RANGE.test(attributesKey)) {
const match = attributesKey.match(PortsAttributes.RANGE);
key = { start: Number(match![1]), end: Number(match![2]) };
} else if (PortsAttributes.HOST_AND_PORT.test(attributesKey)) {
const match = attributesKey.match(PortsAttributes.HOST_AND_PORT);
key = { host: match![1], port: Number(match![2]) };
} else {
let regTest: RegExp | undefined = undefined;
try {
regTest = RegExp(attributesKey);
} catch (e) {
// The user entered an invalid regular expression.
}
if (regTest) {
key = regTest;
}
}
}
if (!key) {
continue;
}
attributes.push({
key: key,
elevateIfNeeded: setting.elevateIfNeeded,
onAutoForward: setting.onAutoForward,
label: setting.label,
requireLocalPort: setting.requireLocalPort,
protocol: setting.protocol
});
}
const defaults = <any>this.configurationService.getValue(PortsAttributes.DEFAULTS);
if (defaults) {
this.defaultPortAttributes = {
elevateIfNeeded: defaults.elevateIfNeeded,
label: defaults.label,
onAutoForward: defaults.onAutoForward,
requireLocalPort: defaults.requireLocalPort,
protocol: defaults.protocol
};
}
return this.sortAttributes(attributes);
}
private sortAttributes(attributes: PortAttributes[]): PortAttributes[] {
function getVal(item: PortAttributes, thisRef: PortsAttributes) {
if (isNumber(item.key)) {
return item.key;
} else if (thisRef.hasStartEnd(item.key)) {
return item.key.start;
} else if (thisRef.hasHostAndPort(item.key)) {
return item.key.port;
} else {
return Number.MAX_VALUE;
}
}
return attributes.sort((a, b) => {
return getVal(a, this) - getVal(b, this);
});
}
private getOtherAttributes() {
return this.defaultPortAttributes;
}
static providedActionToAction(providedAction: ProvidedOnAutoForward | undefined) {
switch (providedAction) {
case ProvidedOnAutoForward.Notify: return OnPortForward.Notify;
case ProvidedOnAutoForward.OpenBrowser: return OnPortForward.OpenBrowser;
case ProvidedOnAutoForward.OpenBrowserOnce: return OnPortForward.OpenBrowserOnce;
case ProvidedOnAutoForward.OpenPreview: return OnPortForward.OpenPreview;
case ProvidedOnAutoForward.Silent: return OnPortForward.Silent;
case ProvidedOnAutoForward.Ignore: return OnPortForward.Ignore;
default: return undefined;
}
}
public async addAttributes(port: number, attributes: Partial<Attributes>, target: ConfigurationTarget) {
const settingValue = this.configurationService.inspect(PortsAttributes.SETTING);
const remoteValue: any = settingValue.userRemoteValue;
let newRemoteValue: any;
if (!remoteValue || !isObject(remoteValue)) {
newRemoteValue = {};
} else {
newRemoteValue = deepClone(remoteValue);
}
if (!newRemoteValue[`${port}`]) {
newRemoteValue[`${port}`] = {};
}
for (const attribute in attributes) {
newRemoteValue[`${port}`][attribute] = (<any>attributes)[attribute];
}
return this.configurationService.updateValue(PortsAttributes.SETTING, newRemoteValue, target);
}
}
const MISMATCH_LOCAL_PORT_COOLDOWN = 10 * 1000; // 10 seconds
export class TunnelModel extends Disposable {
readonly forwarded: Map<string, Tunnel>;
private readonly inProgress: Map<string, true> = new Map();
readonly detected: Map<string, Tunnel>;
private remoteTunnels: Map<string, RemoteTunnel>;
private _onForwardPort: Emitter<Tunnel | void> = new Emitter();
public onForwardPort: Event<Tunnel | void> = this._onForwardPort.event;
private _onClosePort: Emitter<{ host: string; port: number }> = new Emitter();
public onClosePort: Event<{ host: string; port: number }> = this._onClosePort.event;
private _onPortName: Emitter<{ host: string; port: number }> = new Emitter();
public onPortName: Event<{ host: string; port: number }> = this._onPortName.event;
private _candidates: Map<string, CandidatePort> | undefined;
private _onCandidatesChanged: Emitter<Map<string, { host: string; port: number }>> = new Emitter();
// onCandidateChanged returns the removed candidates
public onCandidatesChanged: Event<Map<string, { host: string; port: number }>> = this._onCandidatesChanged.event;
private _candidateFilter: ((candidates: CandidatePort[]) => Promise<CandidatePort[]>) | undefined;
private tunnelRestoreValue: Promise<string | undefined>;
private _onEnvironmentTunnelsSet: Emitter<void> = new Emitter();
public onEnvironmentTunnelsSet: Event<void> = this._onEnvironmentTunnelsSet.event;
private _environmentTunnelsSet: boolean = false;
public readonly configPortsAttributes: PortsAttributes;
private restoreListener: IDisposable | undefined;
private knownPortsRestoreValue: string | undefined;
private portAttributesProviders: PortAttributesProvider[] = [];
constructor(
@ITunnelService private readonly tunnelService: ITunnelService,
@IStorageService private readonly storageService: IStorageService,
@IConfigurationService private readonly configurationService: IConfigurationService,
@IWorkbenchEnvironmentService private readonly environmentService: IWorkbenchEnvironmentService,
@IRemoteAuthorityResolverService private readonly remoteAuthorityResolverService: IRemoteAuthorityResolverService,
@IWorkspaceContextService private readonly workspaceContextService: IWorkspaceContextService,
@ILogService private readonly logService: ILogService,
@IDialogService private readonly dialogService: IDialogService
) {
super();
this.configPortsAttributes = new PortsAttributes(configurationService);
this.tunnelRestoreValue = this.getTunnelRestoreValue();
this._register(this.configPortsAttributes.onDidChangeAttributes(this.updateAttributes, this));
this.forwarded = new Map();
this.remoteTunnels = new Map();
this.tunnelService.tunnels.then(async (tunnels) => {
const attributes = await this.getAttributes(tunnels.map(tunnel => {
return { port: tunnel.tunnelRemotePort, host: tunnel.tunnelRemoteHost };
}));
for (const tunnel of tunnels) {
if (tunnel.localAddress) {
const key = makeAddress(tunnel.tunnelRemoteHost, tunnel.tunnelRemotePort);
const matchingCandidate = mapHasAddressLocalhostOrAllInterfaces(this._candidates ?? new Map(), tunnel.tunnelRemoteHost, tunnel.tunnelRemotePort);
this.forwarded.set(key, {
remotePort: tunnel.tunnelRemotePort,
remoteHost: tunnel.tunnelRemoteHost,
localAddress: tunnel.localAddress,
protocol: attributes?.get(tunnel.tunnelRemotePort)?.protocol ?? TunnelProtocol.Http,
localUri: await this.makeLocalUri(tunnel.localAddress, attributes?.get(tunnel.tunnelRemotePort)),
localPort: tunnel.tunnelLocalPort,
runningProcess: matchingCandidate?.detail,
hasRunningProcess: !!matchingCandidate,
pid: matchingCandidate?.pid,
privacy: tunnel.privacy,
source: UserTunnelSource,
});
this.remoteTunnels.set(key, tunnel);
}
}
});
this.detected = new Map();
this._register(this.tunnelService.onTunnelOpened(async (tunnel) => {
const key = makeAddress(tunnel.tunnelRemoteHost, tunnel.tunnelRemotePort);
if (!mapHasAddressLocalhostOrAllInterfaces(this.forwarded, tunnel.tunnelRemoteHost, tunnel.tunnelRemotePort)
&& !mapHasAddressLocalhostOrAllInterfaces(this.detected, tunnel.tunnelRemoteHost, tunnel.tunnelRemotePort)
&& !mapHasAddressLocalhostOrAllInterfaces(this.inProgress, tunnel.tunnelRemoteHost, tunnel.tunnelRemotePort)
&& tunnel.localAddress) {
const matchingCandidate = mapHasAddressLocalhostOrAllInterfaces(this._candidates ?? new Map(), tunnel.tunnelRemoteHost, tunnel.tunnelRemotePort);
const attributes = (await this.getAttributes([{ port: tunnel.tunnelRemotePort, host: tunnel.tunnelRemoteHost }]))?.get(tunnel.tunnelRemotePort);
this.forwarded.set(key, {
remoteHost: tunnel.tunnelRemoteHost,
remotePort: tunnel.tunnelRemotePort,
localAddress: tunnel.localAddress,
protocol: attributes?.protocol ?? TunnelProtocol.Http,
localUri: await this.makeLocalUri(tunnel.localAddress, attributes),
localPort: tunnel.tunnelLocalPort,
closeable: true,
runningProcess: matchingCandidate?.detail,
hasRunningProcess: !!matchingCandidate,
pid: matchingCandidate?.pid,
privacy: tunnel.privacy,
source: UserTunnelSource,
});
}
await this.storeForwarded();
this.remoteTunnels.set(key, tunnel);
this._onForwardPort.fire(this.forwarded.get(key)!);
}));
this._register(this.tunnelService.onTunnelClosed(address => {
return this.onTunnelClosed(address);
}));
}
private async onTunnelClosed(address: { host: string; port: number }) {
const key = makeAddress(address.host, address.port);
if (this.forwarded.has(key)) {
this.forwarded.delete(key);
await this.storeForwarded();
this._onClosePort.fire(address);
}
}
private makeLocalUri(localAddress: string, attributes?: Attributes) {
if (localAddress.startsWith('http')) {
return URI.parse(localAddress);
}
const protocol = attributes?.protocol ?? 'http';
return URI.parse(`${protocol}://${localAddress}`);
}
private async getStorageKey(): Promise<string> {
const workspace = this.workspaceContextService.getWorkspace();
const workspaceHash = workspace.configuration ? hash(workspace.configuration.path) : (workspace.folders.length > 0 ? hash(workspace.folders[0].uri.path) : undefined);
return `${TUNNELS_TO_RESTORE}.${this.environmentService.remoteAuthority}.${workspaceHash}`;
}
private async getTunnelRestoreValue(): Promise<string | undefined> {
const deprecatedValue = this.storageService.get(TUNNELS_TO_RESTORE, StorageScope.WORKSPACE);
if (deprecatedValue) {
this.storageService.remove(TUNNELS_TO_RESTORE, StorageScope.WORKSPACE);
await this.storeForwarded();
return deprecatedValue;
}
return this.storageService.get(await this.getStorageKey(), StorageScope.PROFILE);
}
async restoreForwarded() {
if (this.configurationService.getValue('remote.restoreForwardedPorts')) {
const tunnelRestoreValue = await this.tunnelRestoreValue;
if (tunnelRestoreValue && (tunnelRestoreValue !== this.knownPortsRestoreValue)) {
const tunnels = <Tunnel[] | undefined>JSON.parse(tunnelRestoreValue) ?? [];
this.logService.trace(`ForwardedPorts: (TunnelModel) restoring ports ${tunnels.map(tunnel => tunnel.remotePort).join(', ')}`);
for (const tunnel of tunnels) {
if (!mapHasAddressLocalhostOrAllInterfaces(this.detected, tunnel.remoteHost, tunnel.remotePort)) {
await this.forward({
remote: { host: tunnel.remoteHost, port: tunnel.remotePort },
local: tunnel.localPort,
name: tunnel.name,
privacy: tunnel.privacy,
elevateIfNeeded: true
});
}
}
}
}
if (!this.restoreListener) {
// It's possible that at restore time the value hasn't synced.
const key = await this.getStorageKey();
this.restoreListener = this._register(this.storageService.onDidChangeValue(async (e) => {
if (e.key === key) {
this.tunnelRestoreValue = Promise.resolve(this.storageService.get(await this.getStorageKey(), StorageScope.PROFILE));
await this.restoreForwarded();
}
}));
}
}
private async storeForwarded() {
if (this.configurationService.getValue('remote.restoreForwardedPorts')) {
const valueToStore = JSON.stringify(Array.from(this.forwarded.values()).filter(value => value.source.source === TunnelSource.User));
if (valueToStore !== this.knownPortsRestoreValue) {
this.knownPortsRestoreValue = valueToStore;
this.storageService.store(await this.getStorageKey(), this.knownPortsRestoreValue, StorageScope.PROFILE, StorageTarget.USER);
}
}
}
private mismatchCooldown = new Date();
private async showPortMismatchModalIfNeeded(tunnel: RemoteTunnel, expectedLocal: number, attributes: Attributes | undefined) {
if (!tunnel.tunnelLocalPort || !attributes?.requireLocalPort) {
return;
}
if (tunnel.tunnelLocalPort === expectedLocal) {
return;
}
const newCooldown = new Date();
if ((this.mismatchCooldown.getTime() + MISMATCH_LOCAL_PORT_COOLDOWN) > newCooldown.getTime()) {
return;
}
this.mismatchCooldown = newCooldown;
const mismatchString = nls.localize('remote.localPortMismatch.single', "Local port {0} could not be used for forwarding to remote port {1}.\n\nThis usually happens when there is already another process using local port {0}.\n\nPort number {2} has been used instead.",
expectedLocal, tunnel.tunnelRemotePort, tunnel.tunnelLocalPort);
return this.dialogService.show(Severity.Info, mismatchString);
}
async forward(tunnelProperties: TunnelProperties, attributes?: Attributes | null): Promise<RemoteTunnel | void> {
const existingTunnel = mapHasAddressLocalhostOrAllInterfaces(this.forwarded, tunnelProperties.remote.host, tunnelProperties.remote.port);
attributes = attributes ??
((attributes !== null)
? (await this.getAttributes([tunnelProperties.remote]))?.get(tunnelProperties.remote.port)
: undefined);
const localPort = (tunnelProperties.local !== undefined) ? tunnelProperties.local : tunnelProperties.remote.port;
if (!existingTunnel) {
const authority = this.environmentService.remoteAuthority;
const addressProvider: IAddressProvider | undefined = authority ? {
getAddress: async () => { return (await this.remoteAuthorityResolverService.resolveAuthority(authority)).authority; }
} : undefined;
const key = makeAddress(tunnelProperties.remote.host, tunnelProperties.remote.port);
this.inProgress.set(key, true);
const tunnel = await this.tunnelService.openTunnel(addressProvider, tunnelProperties.remote.host, tunnelProperties.remote.port, localPort, (!tunnelProperties.elevateIfNeeded) ? attributes?.elevateIfNeeded : tunnelProperties.elevateIfNeeded, tunnelProperties.privacy, attributes?.protocol);
if (tunnel && tunnel.localAddress) {
const matchingCandidate = mapHasAddressLocalhostOrAllInterfaces<CandidatePort>(this._candidates ?? new Map(), tunnelProperties.remote.host, tunnelProperties.remote.port);
const protocol = (tunnel.protocol ?
((tunnel.protocol === TunnelProtocol.Https) ? TunnelProtocol.Https : TunnelProtocol.Http)
: (attributes?.protocol ?? TunnelProtocol.Http));
const newForward: Tunnel = {
remoteHost: tunnel.tunnelRemoteHost,
remotePort: tunnel.tunnelRemotePort,
localPort: tunnel.tunnelLocalPort,
name: attributes?.label ?? tunnelProperties.name,
closeable: true,
localAddress: tunnel.localAddress,
protocol,
localUri: await this.makeLocalUri(tunnel.localAddress, attributes),
runningProcess: matchingCandidate?.detail,
hasRunningProcess: !!matchingCandidate,
pid: matchingCandidate?.pid,
source: tunnelProperties.source ?? UserTunnelSource,
privacy: tunnel.privacy,
};
this.forwarded.set(key, newForward);
this.remoteTunnels.set(key, tunnel);
this.inProgress.delete(key);
await this.storeForwarded();
await this.showPortMismatchModalIfNeeded(tunnel, localPort, attributes);
this._onForwardPort.fire(newForward);
return tunnel;
}
} else {
const newName = attributes?.label ?? tunnelProperties.name;
if (newName !== existingTunnel.name) {
existingTunnel.name = newName;
this._onForwardPort.fire();
}
if ((attributes?.protocol || (existingTunnel.protocol !== TunnelProtocol.Http)) && (attributes?.protocol !== existingTunnel.protocol)) {
await this.close(existingTunnel.remoteHost, existingTunnel.remotePort);
tunnelProperties.source = existingTunnel.source;
await this.forward(tunnelProperties, attributes);
}
return mapHasAddressLocalhostOrAllInterfaces(this.remoteTunnels, tunnelProperties.remote.host, tunnelProperties.remote.port);
}
}
async name(host: string, port: number, name: string) {
const existingForwarded = mapHasAddressLocalhostOrAllInterfaces(this.forwarded, host, port);
const key = makeAddress(host, port);
if (existingForwarded) {
existingForwarded.name = name;
await this.storeForwarded();
this._onPortName.fire({ host, port });
return;
} else if (this.detected.has(key)) {
this.detected.get(key)!.name = name;
this._onPortName.fire({ host, port });
}
}
async close(host: string, port: number): Promise<void> {
await this.tunnelService.closeTunnel(host, port);
return this.onTunnelClosed({ host, port });
}
address(host: string, port: number): string | undefined {
const key = makeAddress(host, port);
return (this.forwarded.get(key) || this.detected.get(key))?.localAddress;
}
public get environmentTunnelsSet(): boolean {
return this._environmentTunnelsSet;
}
addEnvironmentTunnels(tunnels: TunnelDescription[] | undefined): void {
if (tunnels) {
for (const tunnel of tunnels) {
const matchingCandidate = mapHasAddressLocalhostOrAllInterfaces(this._candidates ?? new Map(), tunnel.remoteAddress.host, tunnel.remoteAddress.port);
const localAddress = typeof tunnel.localAddress === 'string' ? tunnel.localAddress : makeAddress(tunnel.localAddress.host, tunnel.localAddress.port);
this.detected.set(makeAddress(tunnel.remoteAddress.host, tunnel.remoteAddress.port), {
remoteHost: tunnel.remoteAddress.host,
remotePort: tunnel.remoteAddress.port,
localAddress: localAddress,
protocol: TunnelProtocol.Http,
localUri: this.makeLocalUri(localAddress),
closeable: false,
runningProcess: matchingCandidate?.detail,
hasRunningProcess: !!matchingCandidate,
pid: matchingCandidate?.pid,
privacy: TunnelPrivacyId.ConstantPrivate,
source: {
source: TunnelSource.Extension,
description: nls.localize('tunnel.staticallyForwarded', "Statically Forwarded")
}
});
this.tunnelService.setEnvironmentTunnel(tunnel.remoteAddress.host, tunnel.remoteAddress.port, localAddress, TunnelPrivacyId.ConstantPrivate, TunnelProtocol.Http);
}
}
this._environmentTunnelsSet = true;
this._onEnvironmentTunnelsSet.fire();
this._onForwardPort.fire();
}
setCandidateFilter(filter: ((candidates: CandidatePort[]) => Promise<CandidatePort[]>) | undefined): void {
this._candidateFilter = filter;
}
async setCandidates(candidates: CandidatePort[]) {
let processedCandidates = candidates;
if (this._candidateFilter) {
// When an extension provides a filter, we do the filtering on the extension host before the candidates are set here.
// However, when the filter doesn't come from an extension we filter here.
processedCandidates = await this._candidateFilter(candidates);
}
const removedCandidates = this.updateInResponseToCandidates(processedCandidates);
this.logService.trace(`ForwardedPorts: (TunnelModel) removed candidates ${Array.from(removedCandidates.values()).map(candidate => candidate.port).join(', ')}`);
this._onCandidatesChanged.fire(removedCandidates);
}
// Returns removed candidates
private updateInResponseToCandidates(candidates: CandidatePort[]): Map<string, { host: string; port: number }> {
const removedCandidates = this._candidates ?? new Map();
const candidatesMap = new Map();
this._candidates = candidatesMap;
candidates.forEach(value => {
const addressKey = makeAddress(value.host, value.port);
candidatesMap.set(addressKey, {
host: value.host,
port: value.port,
detail: value.detail,
pid: value.pid
});
if (removedCandidates.has(addressKey)) {
removedCandidates.delete(addressKey);
}
const forwardedValue = mapHasAddressLocalhostOrAllInterfaces(this.forwarded, value.host, value.port);
if (forwardedValue) {
forwardedValue.runningProcess = value.detail;
forwardedValue.hasRunningProcess = true;
forwardedValue.pid = value.pid;
}
});
removedCandidates.forEach((_value, key) => {
const parsedAddress = parseAddress(key);
if (!parsedAddress) {
return;
}
const forwardedValue = mapHasAddressLocalhostOrAllInterfaces(this.forwarded, parsedAddress.host, parsedAddress.port);
if (forwardedValue) {
forwardedValue.runningProcess = undefined;
forwardedValue.hasRunningProcess = false;
forwardedValue.pid = undefined;
}
const detectedValue = mapHasAddressLocalhostOrAllInterfaces(this.detected, parsedAddress.host, parsedAddress.port);
if (detectedValue) {
detectedValue.runningProcess = undefined;
detectedValue.hasRunningProcess = false;
detectedValue.pid = undefined;
}
});
return removedCandidates;
}
get candidates(): CandidatePort[] {
return this._candidates ? Array.from(this._candidates.values()) : [];
}
get candidatesOrUndefined(): CandidatePort[] | undefined {
return this._candidates ? this.candidates : undefined;
}
private async updateAttributes() {
// If the label changes in the attributes, we should update it.
const tunnels = Array.from(this.forwarded.values());
const allAttributes = await this.getAttributes(tunnels.map(tunnel => {
return { port: tunnel.remotePort, host: tunnel.remoteHost };
}), false);
if (!allAttributes) {
return;
}
for (const forwarded of tunnels) {
const attributes = allAttributes.get(forwarded.remotePort);
if ((attributes?.protocol || (forwarded.protocol !== TunnelProtocol.Http)) && (attributes?.protocol !== forwarded.protocol)) {
await this.forward({
remote: { host: forwarded.remoteHost, port: forwarded.remotePort },
local: forwarded.localPort,
name: forwarded.name,
source: forwarded.source
}, attributes);
}
if (!attributes) {
continue;
}
if (attributes.label && attributes.label !== forwarded.name) {
await this.name(forwarded.remoteHost, forwarded.remotePort, attributes.label);
}
}
}
async getAttributes(forwardedPorts: { host: string; port: number }[], checkProviders: boolean = true): Promise<Map<number, Attributes> | undefined> {
const matchingCandidates: Map<number, CandidatePort> = new Map();
const pidToPortsMapping: Map<number | undefined, number[]> = new Map();
forwardedPorts.forEach(forwardedPort => {
const matchingCandidate = mapHasAddressLocalhostOrAllInterfaces<CandidatePort>(this._candidates ?? new Map(), LOCALHOST_ADDRESSES[0], forwardedPort.port);
if (matchingCandidate) {
matchingCandidates.set(forwardedPort.port, matchingCandidate);
if (!pidToPortsMapping.has(matchingCandidate.pid)) {
pidToPortsMapping.set(matchingCandidate.pid, []);
}
pidToPortsMapping.get(matchingCandidate.pid)?.push(forwardedPort.port);
}
});
const configAttributes: Map<number, Attributes> = new Map();
forwardedPorts.forEach(forwardedPort => {
const attributes = this.configPortsAttributes.getAttributes(forwardedPort.port, forwardedPort.host, matchingCandidates.get(forwardedPort.port)?.detail);
if (attributes) {
configAttributes.set(forwardedPort.port, attributes);
}
});
if ((this.portAttributesProviders.length === 0) || !checkProviders) {
return (configAttributes.size > 0) ? configAttributes : undefined;
}
// Group calls to provide attributes by pid.
const allProviderResults = await Promise.all(flatten(this.portAttributesProviders.map(provider => {
return Array.from(pidToPortsMapping.entries()).map(entry => {
const portGroup = entry[1];
const matchingCandidate = matchingCandidates.get(portGroup[0]);
return provider.providePortAttributes(portGroup,
matchingCandidate?.pid, matchingCandidate?.detail, new CancellationTokenSource().token);
});
})));
const providedAttributes: Map<number, ProvidedPortAttributes> = new Map();
allProviderResults.forEach(attributes => attributes.forEach(attribute => {
if (attribute) {
providedAttributes.set(attribute.port, attribute);
}
}));
if (!configAttributes && !providedAttributes) {
return undefined;
}
// Merge. The config wins.
const mergedAttributes: Map<number, Attributes> = new Map();
forwardedPorts.forEach(forwardedPorts => {
const config = configAttributes.get(forwardedPorts.port);
const provider = providedAttributes.get(forwardedPorts.port);
mergedAttributes.set(forwardedPorts.port, {
elevateIfNeeded: config?.elevateIfNeeded,
label: config?.label,
onAutoForward: config?.onAutoForward ?? PortsAttributes.providedActionToAction(provider?.autoForwardAction),
requireLocalPort: config?.requireLocalPort,
protocol: config?.protocol
});
});
return mergedAttributes;
}
addAttributesProvider(provider: PortAttributesProvider) {
this.portAttributesProviders.push(provider);
}
}
export interface CandidatePort {
host: string;
port: number;
detail?: string;
pid?: number;
}
export interface IRemoteExplorerService {
readonly _serviceBrand: undefined;
onDidChangeTargetType: Event<string[]>;
targetType: string[];
readonly tunnelModel: TunnelModel;
onDidChangeEditable: Event<{ tunnel: ITunnelItem; editId: TunnelEditId } | undefined>;
setEditable(tunnelItem: ITunnelItem | undefined, editId: TunnelEditId, data: IEditableData | null): void;
getEditableData(tunnelItem: ITunnelItem | undefined, editId?: TunnelEditId): IEditableData | undefined;
forward(tunnelProperties: TunnelProperties, attributes?: Attributes | null): Promise<RemoteTunnel | void>;
close(remote: { host: string; port: number }): Promise<void>;
setTunnelInformation(tunnelInformation: TunnelInformation | undefined): void;
setCandidateFilter(filter: ((candidates: CandidatePort[]) => Promise<CandidatePort[]>) | undefined): IDisposable;
onFoundNewCandidates(candidates: CandidatePort[]): void;
restore(): Promise<void>;
enablePortsFeatures(): void;
onEnabledPortsFeatures: Event<void>;
portsFeaturesEnabled: boolean;
readonly namedProcesses: Map<number, string>;
}
class RemoteExplorerService implements IRemoteExplorerService {
public _serviceBrand: undefined;
private _targetType: string[] = [];
private readonly _onDidChangeTargetType: Emitter<string[]> = new Emitter<string[]>();
public readonly onDidChangeTargetType: Event<string[]> = this._onDidChangeTargetType.event;
private _tunnelModel: TunnelModel;
private _editable: { tunnelItem: ITunnelItem | undefined; editId: TunnelEditId; data: IEditableData } | undefined;
private readonly _onDidChangeEditable: Emitter<{ tunnel: ITunnelItem; editId: TunnelEditId } | undefined> = new Emitter();
public readonly onDidChangeEditable: Event<{ tunnel: ITunnelItem; editId: TunnelEditId } | undefined> = this._onDidChangeEditable.event;
private readonly _onEnabledPortsFeatures: Emitter<void> = new Emitter();
public readonly onEnabledPortsFeatures: Event<void> = this._onEnabledPortsFeatures.event;
private _portsFeaturesEnabled: boolean = false;
public readonly namedProcesses = new Map<number, string>();
constructor(
@IStorageService private readonly storageService: IStorageService,
@ITunnelService private readonly tunnelService: ITunnelService,
@IConfigurationService configurationService: IConfigurationService,
@IWorkbenchEnvironmentService environmentService: IWorkbenchEnvironmentService,
@IRemoteAuthorityResolverService remoteAuthorityResolverService: IRemoteAuthorityResolverService,
@IWorkspaceContextService workspaceContextService: IWorkspaceContextService,
@ILogService logService: ILogService,
@IDialogService dialogService: IDialogService
) {
this._tunnelModel = new TunnelModel(tunnelService, storageService, configurationService, environmentService,
remoteAuthorityResolverService, workspaceContextService, logService, dialogService);
}
set targetType(name: string[]) {
// Can just compare the first element of the array since there are no target overlaps
const current: string = this._targetType.length > 0 ? this._targetType[0] : '';
const newName: string = name.length > 0 ? name[0] : '';
if (current !== newName) {
this._targetType = name;
this.storageService.store(REMOTE_EXPLORER_TYPE_KEY, this._targetType.toString(), StorageScope.WORKSPACE, StorageTarget.USER);
this.storageService.store(REMOTE_EXPLORER_TYPE_KEY, this._targetType.toString(), StorageScope.PROFILE, StorageTarget.USER);
this._onDidChangeTargetType.fire(this._targetType);
}
}
get targetType(): string[] {
return this._targetType;
}
get tunnelModel(): TunnelModel {
return this._tunnelModel;
}
forward(tunnelProperties: TunnelProperties, attributes?: Attributes | null): Promise<RemoteTunnel | void> {
return this.tunnelModel.forward(tunnelProperties, attributes);
}
close(remote: { host: string; port: number }): Promise<void> {
return this.tunnelModel.close(remote.host, remote.port);
}
setTunnelInformation(tunnelInformation: TunnelInformation | undefined): void {
if (tunnelInformation?.features) {
this.tunnelService.setTunnelFeatures(tunnelInformation.features);
}
this.tunnelModel.addEnvironmentTunnels(tunnelInformation?.environmentTunnels);
}
setEditable(tunnelItem: ITunnelItem | undefined, editId: TunnelEditId, data: IEditableData | null): void {
if (!data) {
this._editable = undefined;
} else {
this._editable = { tunnelItem, data, editId };
}
this._onDidChangeEditable.fire(tunnelItem ? { tunnel: tunnelItem, editId } : undefined);
}
getEditableData(tunnelItem: ITunnelItem | undefined, editId: TunnelEditId): IEditableData | undefined {
return (this._editable &&
((!tunnelItem && (tunnelItem === this._editable.tunnelItem)) ||
(tunnelItem && (this._editable.tunnelItem?.remotePort === tunnelItem.remotePort) && (this._editable.tunnelItem.remoteHost === tunnelItem.remoteHost)
&& (this._editable.editId === editId)))) ?
this._editable.data : undefined;
}
setCandidateFilter(filter: (candidates: CandidatePort[]) => Promise<CandidatePort[]>): IDisposable {
if (!filter) {
return {
dispose: () => { }
};
}
this.tunnelModel.setCandidateFilter(filter);
return {
dispose: () => {
this.tunnelModel.setCandidateFilter(undefined);
}
};
}
onFoundNewCandidates(candidates: CandidatePort[]): void {
this.tunnelModel.setCandidates(candidates);
}
restore(): Promise<void> {
return this.tunnelModel.restoreForwarded();
}
enablePortsFeatures(): void {
this._portsFeaturesEnabled = true;
this._onEnabledPortsFeatures.fire();
}
get portsFeaturesEnabled(): boolean {
return this._portsFeaturesEnabled;
}
}
registerSingleton(IRemoteExplorerService, RemoteExplorerService, InstantiationType.Delayed);