trustedDomains.ts 7.1 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';
P
Pine Wu 已提交
18

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

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

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

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

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

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

P
Pine Wu 已提交
80
	const pickedResult = await quickInputService.pick<ConfigureTrustedDomainsQuickPickItem>(
P
Pine Wu 已提交
81
		[trustDomainAndOpenLinkItem, trustSubDomainAndOpenLinkItem, openAllLinksItem, manageTrustedDomainItem],
P
polish  
Pine Wu 已提交
82 83 84 85
		{
			activeItem: trustDomainAndOpenLinkItem
		}
	);
P
Pine Wu 已提交
86

P
Pine Wu 已提交
87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107
	if (pickedResult && pickedResult.id) {
		telemetryService.publicLog2<{ choice: string }, ConfigureTrustedDomainsChoiceClassification>(
			'trustedDomains.configureTrustedDomainsQuickPickChoice',
			{ choice: pickedResult.id }
		);

		switch (pickedResult.id) {
			case 'manage':
				editorService.openEditor({
					resource: TRUSTED_DOMAINS_URI,
					mode: 'jsonc'
				});
				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 已提交
108
					storageService.remove(TRUSTED_DOMAINS_CONTENT_STORAGE_KEY, StorageScope.GLOBAL);
P
Pine Wu 已提交
109
					storageService.store(
P
Pine Wu 已提交
110
						TRUSTED_DOMAINS_STORAGE_KEY,
P
Pine Wu 已提交
111 112 113
						JSON.stringify([...trustedDomains, itemToTrust]),
						StorageScope.GLOBAL
					);
P
Pine Wu 已提交
114

P
Pine Wu 已提交
115 116
					return [...trustedDomains, pickedResult.id];
				}
P
Pine Wu 已提交
117 118 119 120 121
		}
	}

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

123 124 125 126 127 128 129 130 131 132 133 134 135
async function getRemotes(fileService: IFileService, textFileService: ITextFileService, contextService: IWorkspaceContextService): Promise<string[]> {
	const workspaceUris = contextService.getWorkspace().folders.map(folder => folder.uri);
	const domains = await 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 content = (await (textFileService.read(uri, { acceptTextOnly: true }).catch(() => ({ value: '' })))).value;
		const domains = new Set<string>();
		let match: RegExpExecArray | null;

136
		const RemoteMatcher = /^\s*url\s*=\s*(?:git@|https:\/\/)github\.com(?::|\/)([^.]*)\.git\s*$/mg;
137 138 139
		while (match = RemoteMatcher.exec(content)) {
			const [domain, repo] = [match[1], match[2]];
			if (domain && repo) {
140
				domains.add(`https://github.com/${repo}/`);
141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158
			}
		}
		return [...domains];
	}));

	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 已提交
159
	const defaultTrustedDomains: string[] = productService.linkProtectionTrustedDomains
P
Pine Wu 已提交
160 161 162
		? [...productService.linkProtectionTrustedDomains]
		: [];

P
Pine Wu 已提交
163
	let trustedDomains: string[] = [];
P
Pine Wu 已提交
164
	try {
P
Pine Wu 已提交
165
		const trustedDomainsSrc = storageService.get(TRUSTED_DOMAINS_STORAGE_KEY, StorageScope.GLOBAL);
P
Pine Wu 已提交
166 167 168 169 170
		if (trustedDomainsSrc) {
			trustedDomains = JSON.parse(trustedDomainsSrc);
		}
	} catch (err) { }

171 172
	const userDomains = ((await authenticationService.getSessions('github')) ?? [])
		.map(session => session.account.displayName)
173
		.filter((v, i, a) => a.indexOf(v) === i)
174 175
		.map(username => `https://github.com/${username}/`);

176 177
	const workspaceDomains = await getRemotes(fileService, textFileService, workspaceContextService);

P
Pine Wu 已提交
178 179
	return {
		defaultTrustedDomains,
180
		trustedDomains,
181 182
		userDomains,
		workspaceDomains
P
Pine Wu 已提交
183
	};
P
Pine Wu 已提交
184
}