2022-03-23 22:41:12 +08:00
|
|
|
import { safeRun } from "../util";
|
2022-03-20 16:56:28 +08:00
|
|
|
|
|
|
|
// @ts-ignore
|
|
|
|
import sandboxHtml from "bundle-text:./iframe_sandbox.html";
|
2022-03-23 22:41:12 +08:00
|
|
|
import { Sandbox } from "../sandbox";
|
|
|
|
import { WorkerLike } from "./worker";
|
2022-03-25 19:03:06 +08:00
|
|
|
import { Plug } from "../plug";
|
2022-03-20 16:56:28 +08:00
|
|
|
|
|
|
|
class IFrameWrapper implements WorkerLike {
|
|
|
|
private iframe: HTMLIFrameElement;
|
|
|
|
onMessage?: (message: any) => Promise<void>;
|
2022-03-21 22:21:34 +08:00
|
|
|
ready: Promise<void>;
|
2022-03-24 17:48:56 +08:00
|
|
|
private messageListener: (evt: any) => void;
|
2022-03-20 16:56:28 +08:00
|
|
|
|
|
|
|
constructor() {
|
|
|
|
const iframe = document.createElement("iframe", {});
|
|
|
|
this.iframe = iframe;
|
|
|
|
iframe.style.display = "none";
|
|
|
|
// Let's lock this down significantly
|
|
|
|
iframe.setAttribute("sandbox", "allow-scripts");
|
|
|
|
iframe.srcdoc = sandboxHtml;
|
2022-03-24 17:48:56 +08:00
|
|
|
this.messageListener = (evt: any) => {
|
2022-03-20 16:56:28 +08:00
|
|
|
if (evt.source !== iframe.contentWindow) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
let data = evt.data;
|
|
|
|
if (!data) return;
|
|
|
|
safeRun(async () => {
|
|
|
|
await this.onMessage!(data);
|
|
|
|
});
|
2022-03-24 17:48:56 +08:00
|
|
|
};
|
|
|
|
window.addEventListener("message", this.messageListener);
|
2022-03-20 16:56:28 +08:00
|
|
|
document.body.appendChild(iframe);
|
2022-03-21 22:21:34 +08:00
|
|
|
this.ready = new Promise((resolve) => {
|
|
|
|
iframe.onload = () => {
|
|
|
|
resolve();
|
|
|
|
iframe.onload = null;
|
|
|
|
};
|
|
|
|
});
|
2022-03-20 16:56:28 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
postMessage(message: any): void {
|
|
|
|
this.iframe.contentWindow!.postMessage(message, "*");
|
|
|
|
}
|
|
|
|
|
|
|
|
terminate() {
|
2022-03-31 23:25:34 +08:00
|
|
|
console.log("Terminating iframe sandbox");
|
2022-03-24 17:48:56 +08:00
|
|
|
window.removeEventListener("message", this.messageListener);
|
2022-03-20 16:56:28 +08:00
|
|
|
return this.iframe.remove();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-03-25 19:03:06 +08:00
|
|
|
export function createSandbox(plug: Plug<any>) {
|
|
|
|
return new Sandbox(plug, new IFrameWrapper());
|
2022-03-20 16:56:28 +08:00
|
|
|
}
|