trustedDomains.ts 7.9 KB
Newer Older
P
Pine Wu 已提交
1 2 3 4 5
/*---------------------------------------------------------------------------------------------
 *  Copyright (c) Microsoft Corporation. All rights reserved.
 *  Licensed under the MIT License. See License.txt in the project root for license information.
 *--------------------------------------------------------------------------------------------*/

P
Pine Wu 已提交
6
import { URI } from 'vs/base/common/uri';
P
Pine Wu 已提交
7 8
import { localize } from 'vs/nls';
import { ServicesAccessor } from 'vs/platform/instantiation/common/instantiation';
9
import { IProductService } from 'vs/platform/product/common/productService';
P
Pine Wu 已提交
10 11
import { IQuickInputService, IQuickPickItem } from 'vs/platform/quickinput/common/quickInput';
import { IStorageService, StorageScope } from 'vs/platform/storage/common/storage';
P
Pine Wu 已提交
12
import { IEditorService } from 'vs/workbench/services/editor/common/editorService';
P
Pine Wu 已提交
13
import { ITelemetryService } from 'vs/platform/telemetry/common/telemetry';
14
import { IAuthenticationService } from 'vs/workbench/services/authentication/browser/authenticationService';
15 16 17
import { IFileService } from 'vs/platform/files/common/files';
import { ITextFileService } from 'vs/workbench/services/textfile/common/textfiles';
import { IWorkspaceContextService } from 'vs/platform/workspace/common/workspace';
18 19
import { INotificationService, Severity } from 'vs/platform/notification/common/notification';
import { IClipboardService } from 'vs/platform/clipboard/common/clipboardService';
P
Pine Wu 已提交
20

P
Pine Wu 已提交
21
const TRUSTED_DOMAINS_URI = URI.parse('trustedDomains:/Trusted Domains');
P
Pine Wu 已提交
22

P
Pine Wu 已提交
23 24 25
export const TRUSTED_DOMAINS_STORAGE_KEY = 'http.linkProtectionTrustedDomains';
export const TRUSTED_DOMAINS_CONTENT_STORAGE_KEY = 'http.linkProtectionTrustedDomainsContent';

P
Pine Wu 已提交
26 27
export const manageTrustedDomainSettingsCommand = {
	id: 'workbench.action.manageTrustedDomain',
P
Pine Wu 已提交
28
	description: {
P
Pine Wu 已提交
29
		description: localize('trustedDomain.manageTrustedDomain', 'Manage Trusted Domains'),
P
Pine Wu 已提交
30 31 32
		args: []
	},
	handler: async (accessor: ServicesAccessor) => {
P
Pine Wu 已提交
33 34 35
		const editorService = accessor.get(IEditorService);
		editorService.openEditor({ resource: TRUSTED_DOMAINS_URI, mode: 'jsonc' });
		return;
P
Pine Wu 已提交
36 37 38
	}
};

P
Pine Wu 已提交
39 40 41 42 43 44 45 46
type ConfigureTrustedDomainChoice = 'trustDomain' | 'trustSubdomain' | 'trustAll' | 'manage';
interface ConfigureTrustedDomainsQuickPickItem extends IQuickPickItem {
	id: ConfigureTrustedDomainChoice;
}
type ConfigureTrustedDomainsChoiceClassification = {
	choice: { classification: 'SystemMetaData', purpose: 'FeatureInsight' };
};

P
Pine Wu 已提交
47 48 49
export async function configureOpenerTrustedDomainsHandler(
	trustedDomains: string[],
	domainToConfigure: string,
50
	resource: URI,
P
Pine Wu 已提交
51
	quickInputService: IQuickInputService,
P
Pine Wu 已提交
52
	storageService: IStorageService,
P
Pine Wu 已提交
53
	editorService: IEditorService,
54 55 56
	telemetryService: ITelemetryService,
	notificationService: INotificationService,
	clipboardService: IClipboardService,
P
Pine Wu 已提交
57
) {
P
Pine Wu 已提交
58
	const parsedDomainToConfigure = URI.parse(domainToConfigure);
P
Pine Wu 已提交
59
	const toplevelDomainSegements = parsedDomainToConfigure.authority.split('.');
P
Pine Wu 已提交
60
	const domainEnd = toplevelDomainSegements.slice(toplevelDomainSegements.length - 2).join('.');
P
Pine Wu 已提交
61
	const topLevelDomain = '*.' + domainEnd;
P
Pine Wu 已提交
62

P
Pine Wu 已提交
63
	const trustDomainAndOpenLinkItem: ConfigureTrustedDomainsQuickPickItem = {
P
Pine Wu 已提交
64
		type: 'item',
P
Pine Wu 已提交
65
		label: localize('trustedDomain.trustDomain', 'Trust {0}', domainToConfigure),
P
Pine Wu 已提交
66
		id: 'trustDomain',
P
Pine Wu 已提交
67 68
		picked: true
	};
P
Pine Wu 已提交
69
	const trustSubDomainAndOpenLinkItem: ConfigureTrustedDomainsQuickPickItem = {
P
Pine Wu 已提交
70
		type: 'item',
P
Pine Wu 已提交
71
		label: localize('trustedDomain.trustSubDomain', 'Trust {0} and all its subdomains', domainEnd),
P
Pine Wu 已提交
72
		id: 'trustSubdomain'
P
Pine Wu 已提交
73
	};
P
Pine Wu 已提交
74
	const openAllLinksItem: ConfigureTrustedDomainsQuickPickItem = {
P
Pine Wu 已提交
75
		type: 'item',
P
Pine Wu 已提交
76
		label: localize('trustedDomain.trustAllDomains', 'Trust all domains (disables link protection)'),
P
Pine Wu 已提交
77
		id: 'trustAll'
P
Pine Wu 已提交
78
	};
P
Pine Wu 已提交
79
	const manageTrustedDomainItem: ConfigureTrustedDomainsQuickPickItem = {
P
Pine Wu 已提交
80
		type: 'item',
P
Pine Wu 已提交
81 82
		label: localize('trustedDomain.manageTrustedDomains', 'Manage Trusted Domains'),
		id: 'manage'
P
Pine Wu 已提交
83
	};
P
Pine Wu 已提交
84

P
Pine Wu 已提交
85
	const pickedResult = await quickInputService.pick<ConfigureTrustedDomainsQuickPickItem>(
P
Pine Wu 已提交
86
		[trustDomainAndOpenLinkItem, trustSubDomainAndOpenLinkItem, openAllLinksItem, manageTrustedDomainItem],
P
polish  
Pine Wu 已提交
87 88 89 90
		{
			activeItem: trustDomainAndOpenLinkItem
		}
	);
P
Pine Wu 已提交
91

P
Pine Wu 已提交
92 93 94 95 96 97 98 99
	if (pickedResult && pickedResult.id) {
		telemetryService.publicLog2<{ choice: string }, ConfigureTrustedDomainsChoiceClassification>(
			'trustedDomains.configureTrustedDomainsQuickPickChoice',
			{ choice: pickedResult.id }
		);

		switch (pickedResult.id) {
			case 'manage':
100
				await editorService.openEditor({
P
Pine Wu 已提交
101 102 103
					resource: TRUSTED_DOMAINS_URI,
					mode: 'jsonc'
				});
104 105
				notificationService.prompt(Severity.Info, localize('configuringURL', "Configuring trust for: {0}", resource.toString()),
					[{ label: 'Copy', run: () => clipboardService.writeText(resource.toString()) }]);
P
Pine Wu 已提交
106 107 108 109 110 111 112 113 114
				return trustedDomains;
			case 'trustDomain':
			case 'trustSubdomain':
			case 'trustAll':
				const itemToTrust = pickedResult.id === 'trustDomain'
					? domainToConfigure
					: pickedResult.id === 'trustSubdomain' ? topLevelDomain : '*';

				if (trustedDomains.indexOf(itemToTrust) === -1) {
P
Pine Wu 已提交
115
					storageService.remove(TRUSTED_DOMAINS_CONTENT_STORAGE_KEY, StorageScope.GLOBAL);
P
Pine Wu 已提交
116
					storageService.store(
P
Pine Wu 已提交
117
						TRUSTED_DOMAINS_STORAGE_KEY,
P
Pine Wu 已提交
118 119 120
						JSON.stringify([...trustedDomains, itemToTrust]),
						StorageScope.GLOBAL
					);
P
Pine Wu 已提交
121

122
					return [...trustedDomains, itemToTrust];
P
Pine Wu 已提交
123
				}
P
Pine Wu 已提交
124 125 126 127 128
		}
	}

	return [];
}
P
Pine Wu 已提交
129

J
Jackson Kearl 已提交
130 131 132 133 134 135 136 137 138 139 140 141 142 143 144
// Exported for testing.
export function extractGitHubRemotesFromGitConfig(gitConfig: string): string[] {
	const domains = new Set<string>();
	let match: RegExpExecArray | null;

	const RemoteMatcher = /^\s*url\s*=\s*(?:git@|https:\/\/)github\.com(?::|\/)(\S*)\s*$/mg;
	while (match = RemoteMatcher.exec(gitConfig)) {
		const repo = match[1].replace(/\.git$/, '');
		if (repo) {
			domains.add(`https://github.com/${repo}/`);
		}
	}
	return [...domains];
}

145 146
async function getRemotes(fileService: IFileService, textFileService: ITextFileService, contextService: IWorkspaceContextService): Promise<string[]> {
	const workspaceUris = contextService.getWorkspace().folders.map(folder => folder.uri);
147 148 149 150 151 152 153 154 155 156 157 158
	const domains = await Promise.race([
		new Promise<string[][]>(resolve => setTimeout(() => resolve([]), 250)),
		Promise.all<string[]>(workspaceUris.map(async workspaceUri => {
			const path = workspaceUri.path;
			const uri = workspaceUri.with({ path: `${path !== '/' ? path : ''}/.git/config` });
			const exists = await fileService.exists(uri);
			if (!exists) {
				return [];
			}
			const gitConfig = (await (textFileService.read(uri, { acceptTextOnly: true }).catch(() => ({ value: '' })))).value;
			return extractGitHubRemotesFromGitConfig(gitConfig);
		}))]);
159 160 161 162 163 164 165 166 167 168 169 170 171 172

	const set = domains.reduce((set, list) => list.reduce((set, item) => set.add(item), set), new Set<string>());
	return [...set];
}

export async function readTrustedDomains(accessor: ServicesAccessor) {

	const storageService = accessor.get(IStorageService);
	const productService = accessor.get(IProductService);
	const authenticationService = accessor.get(IAuthenticationService);
	const fileService = accessor.get(IFileService);
	const textFileService = accessor.get(ITextFileService);
	const workspaceContextService = accessor.get(IWorkspaceContextService);

P
Pine Wu 已提交
173
	const defaultTrustedDomains: string[] = productService.linkProtectionTrustedDomains
P
Pine Wu 已提交
174 175 176
		? [...productService.linkProtectionTrustedDomains]
		: [];

P
Pine Wu 已提交
177
	let trustedDomains: string[] = [];
P
Pine Wu 已提交
178
	try {
P
Pine Wu 已提交
179
		const trustedDomainsSrc = storageService.get(TRUSTED_DOMAINS_STORAGE_KEY, StorageScope.GLOBAL);
P
Pine Wu 已提交
180 181 182 183 184
		if (trustedDomainsSrc) {
			trustedDomains = JSON.parse(trustedDomainsSrc);
		}
	} catch (err) { }

J
Jackson Kearl 已提交
185 186 187 188 189 190 191
	const userDomains =
		authenticationService.isAuthenticationProviderRegistered('github')
			? ((await authenticationService.getSessions('github')) ?? [])
				.map(session => session.account.displayName)
				.filter((v, i, a) => a.indexOf(v) === i)
				.map(username => `https://github.com/${username}/`)
			: [];
192

193 194
	const workspaceDomains = await getRemotes(fileService, textFileService, workspaceContextService);

P
Pine Wu 已提交
195 196
	return {
		defaultTrustedDomains,
197
		trustedDomains,
198 199
		userDomains,
		workspaceDomains
P
Pine Wu 已提交
200
	};
P
Pine Wu 已提交
201
}