mirror of
https://github.com/microsoft/vscode.git
synced 2026-04-20 08:38:56 +01:00
235 lines
8.4 KiB
TypeScript
235 lines
8.4 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 { VSBuffer } from 'vs/base/common/buffer';
|
|
import { Emitter, Event } from 'vs/base/common/event';
|
|
import { Schemas } from 'vs/base/common/network';
|
|
import { URI } from 'vs/base/common/uri';
|
|
import { IExtensionDescription } from 'vs/platform/extensions/common/extensions';
|
|
import { normalizeVersion, parseVersion } from 'vs/platform/extensions/common/extensionValidator';
|
|
import { ILogService } from 'vs/platform/log/common/log';
|
|
import { IExtHostApiDeprecationService } from 'vs/workbench/api/common/extHostApiDeprecationService';
|
|
import { serializeWebviewMessage, deserializeWebviewMessage } from 'vs/workbench/api/common/extHostWebviewMessaging';
|
|
import { IExtHostWorkspace } from 'vs/workbench/api/common/extHostWorkspace';
|
|
import { asWebviewUri, webviewGenericCspSource, WebviewRemoteInfo } from 'vs/workbench/common/webview';
|
|
import { SerializableObjectWithBuffers } from 'vs/workbench/services/extensions/common/proxyIdentifier';
|
|
import type * as vscode from 'vscode';
|
|
import * as extHostProtocol from './extHost.protocol';
|
|
|
|
export class ExtHostWebview implements vscode.Webview {
|
|
|
|
readonly #handle: extHostProtocol.WebviewHandle;
|
|
readonly #proxy: extHostProtocol.MainThreadWebviewsShape;
|
|
readonly #deprecationService: IExtHostApiDeprecationService;
|
|
|
|
readonly #remoteInfo: WebviewRemoteInfo;
|
|
readonly #workspace: IExtHostWorkspace | undefined;
|
|
readonly #extension: IExtensionDescription;
|
|
|
|
#html: string = '';
|
|
#options: vscode.WebviewOptions;
|
|
#isDisposed: boolean = false;
|
|
#hasCalledAsWebviewUri = false;
|
|
|
|
#serializeBuffersForPostMessage = false;
|
|
|
|
constructor(
|
|
handle: extHostProtocol.WebviewHandle,
|
|
proxy: extHostProtocol.MainThreadWebviewsShape,
|
|
options: vscode.WebviewOptions,
|
|
remoteInfo: WebviewRemoteInfo,
|
|
workspace: IExtHostWorkspace | undefined,
|
|
extension: IExtensionDescription,
|
|
deprecationService: IExtHostApiDeprecationService,
|
|
) {
|
|
this.#handle = handle;
|
|
this.#proxy = proxy;
|
|
this.#options = options;
|
|
this.#remoteInfo = remoteInfo;
|
|
this.#workspace = workspace;
|
|
this.#extension = extension;
|
|
this.#serializeBuffersForPostMessage = shouldSerializeBuffersForPostMessage(extension);
|
|
this.#deprecationService = deprecationService;
|
|
}
|
|
|
|
/* internal */ readonly _onMessageEmitter = new Emitter<any>();
|
|
public readonly onDidReceiveMessage: Event<any> = this._onMessageEmitter.event;
|
|
|
|
readonly #onDidDisposeEmitter = new Emitter<void>();
|
|
/* internal */ readonly _onDidDispose: Event<void> = this.#onDidDisposeEmitter.event;
|
|
|
|
public dispose() {
|
|
this.#isDisposed = true;
|
|
|
|
this.#onDidDisposeEmitter.fire();
|
|
|
|
this.#onDidDisposeEmitter.dispose();
|
|
this._onMessageEmitter.dispose();
|
|
}
|
|
|
|
public asWebviewUri(resource: vscode.Uri): vscode.Uri {
|
|
this.#hasCalledAsWebviewUri = true;
|
|
return asWebviewUri(resource, this.#remoteInfo);
|
|
}
|
|
|
|
public get cspSource(): string {
|
|
const extensionLocation = this.#extension.extensionLocation;
|
|
if (extensionLocation.scheme === Schemas.https || extensionLocation.scheme === Schemas.http) {
|
|
// The extension is being served up from a CDN.
|
|
// Also include the CDN in the default csp.
|
|
let extensionCspRule = extensionLocation.toString();
|
|
if (!extensionCspRule.endsWith('/')) {
|
|
// Always treat the location as a directory so that we allow all content under it
|
|
extensionCspRule += '/';
|
|
}
|
|
return extensionCspRule + ' ' + webviewGenericCspSource;
|
|
}
|
|
return webviewGenericCspSource;
|
|
}
|
|
|
|
public get html(): string {
|
|
this.assertNotDisposed();
|
|
return this.#html;
|
|
}
|
|
|
|
public set html(value: string) {
|
|
this.assertNotDisposed();
|
|
if (this.#html !== value) {
|
|
this.#html = value;
|
|
if (!this.#hasCalledAsWebviewUri && /(["'])vscode-resource:([^\s'"]+?)(["'])/i.test(value)) {
|
|
this.#hasCalledAsWebviewUri = true;
|
|
this.#deprecationService.report('Webview vscode-resource: uris', this.#extension,
|
|
`Please migrate to use the 'webview.asWebviewUri' api instead: https://aka.ms/vscode-webview-use-aswebviewuri`);
|
|
}
|
|
this.#proxy.$setHtml(this.#handle, value);
|
|
}
|
|
}
|
|
|
|
public get options(): vscode.WebviewOptions {
|
|
this.assertNotDisposed();
|
|
return this.#options;
|
|
}
|
|
|
|
public set options(newOptions: vscode.WebviewOptions) {
|
|
this.assertNotDisposed();
|
|
this.#proxy.$setOptions(this.#handle, serializeWebviewOptions(this.#extension, this.#workspace, newOptions));
|
|
this.#options = newOptions;
|
|
}
|
|
|
|
public async postMessage(message: any): Promise<boolean> {
|
|
if (this.#isDisposed) {
|
|
return false;
|
|
}
|
|
const serialized = serializeWebviewMessage(message, { serializeBuffersForPostMessage: this.#serializeBuffersForPostMessage });
|
|
return this.#proxy.$postMessage(this.#handle, serialized.message, ...serialized.buffers);
|
|
}
|
|
|
|
private assertNotDisposed() {
|
|
if (this.#isDisposed) {
|
|
throw new Error('Webview is disposed');
|
|
}
|
|
}
|
|
}
|
|
|
|
export function shouldSerializeBuffersForPostMessage(extension: IExtensionDescription): boolean {
|
|
try {
|
|
const version = normalizeVersion(parseVersion(extension.engines.vscode));
|
|
return !!version && version.majorBase >= 1 && version.minorBase >= 57;
|
|
} catch {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
export class ExtHostWebviews implements extHostProtocol.ExtHostWebviewsShape {
|
|
|
|
private readonly _webviewProxy: extHostProtocol.MainThreadWebviewsShape;
|
|
|
|
private readonly _webviews = new Map<extHostProtocol.WebviewHandle, ExtHostWebview>();
|
|
|
|
constructor(
|
|
mainContext: extHostProtocol.IMainContext,
|
|
private readonly remoteInfo: WebviewRemoteInfo,
|
|
private readonly workspace: IExtHostWorkspace | undefined,
|
|
private readonly _logService: ILogService,
|
|
private readonly _deprecationService: IExtHostApiDeprecationService,
|
|
) {
|
|
this._webviewProxy = mainContext.getProxy(extHostProtocol.MainContext.MainThreadWebviews);
|
|
}
|
|
|
|
public $onMessage(
|
|
handle: extHostProtocol.WebviewHandle,
|
|
jsonMessage: string,
|
|
buffers: SerializableObjectWithBuffers<VSBuffer[]>
|
|
): void {
|
|
const webview = this.getWebview(handle);
|
|
if (webview) {
|
|
const { message } = deserializeWebviewMessage(jsonMessage, buffers.value);
|
|
webview._onMessageEmitter.fire(message);
|
|
}
|
|
}
|
|
|
|
public $onMissingCsp(
|
|
_handle: extHostProtocol.WebviewHandle,
|
|
extensionId: string
|
|
): void {
|
|
this._logService.warn(`${extensionId} created a webview without a content security policy: https://aka.ms/vscode-webview-missing-csp`);
|
|
}
|
|
|
|
public createNewWebview(handle: string, options: extHostProtocol.IWebviewContentOptions, extension: IExtensionDescription): ExtHostWebview {
|
|
const webview = new ExtHostWebview(handle, this._webviewProxy, reviveOptions(options), this.remoteInfo, this.workspace, extension, this._deprecationService);
|
|
this._webviews.set(handle, webview);
|
|
|
|
webview._onDidDispose(() => { this._webviews.delete(handle); });
|
|
|
|
return webview;
|
|
}
|
|
|
|
public deleteWebview(handle: string) {
|
|
this._webviews.delete(handle);
|
|
}
|
|
|
|
private getWebview(handle: extHostProtocol.WebviewHandle): ExtHostWebview | undefined {
|
|
return this._webviews.get(handle);
|
|
}
|
|
}
|
|
|
|
export function toExtensionData(extension: IExtensionDescription): extHostProtocol.WebviewExtensionDescription {
|
|
return { id: extension.identifier, location: extension.extensionLocation };
|
|
}
|
|
|
|
export function serializeWebviewOptions(
|
|
extension: IExtensionDescription,
|
|
workspace: IExtHostWorkspace | undefined,
|
|
options: vscode.WebviewOptions,
|
|
): extHostProtocol.IWebviewContentOptions {
|
|
return {
|
|
enableCommandUris: options.enableCommandUris,
|
|
enableScripts: options.enableScripts,
|
|
enableForms: options.enableForms,
|
|
portMapping: options.portMapping,
|
|
localResourceRoots: options.localResourceRoots || getDefaultLocalResourceRoots(extension, workspace)
|
|
};
|
|
}
|
|
|
|
export function reviveOptions(options: extHostProtocol.IWebviewContentOptions): vscode.WebviewOptions {
|
|
return {
|
|
enableCommandUris: options.enableCommandUris,
|
|
enableScripts: options.enableScripts,
|
|
enableForms: options.enableForms,
|
|
portMapping: options.portMapping,
|
|
localResourceRoots: options.localResourceRoots?.map(components => URI.from(components)),
|
|
};
|
|
}
|
|
|
|
function getDefaultLocalResourceRoots(
|
|
extension: IExtensionDescription,
|
|
workspace: IExtHostWorkspace | undefined,
|
|
): URI[] {
|
|
return [
|
|
...(workspace?.getWorkspaceFolders() || []).map(x => x.uri),
|
|
extension.extensionLocation,
|
|
];
|
|
}
|