2022-02-08 04:03:25 +08:00
|
|
|
import { Socket } from "phoenix";
|
2022-02-08 21:45:58 +08:00
|
|
|
import { getAttributeOrThrow, parseInteger } from "../lib/attribute";
|
2022-02-02 07:10:17 +08:00
|
|
|
import { randomToken, sha256Base64 } from "../lib/utils";
|
2022-01-06 23:31:26 +08:00
|
|
|
|
2021-12-24 21:18:34 +08:00
|
|
|
/**
|
|
|
|
* A hook used to render JS-enabled cell output.
|
|
|
|
*
|
|
|
|
* The JavaScript is defined by the user, so we sandbox the script
|
|
|
|
* execution inside an iframe.
|
|
|
|
*
|
2022-01-06 23:31:26 +08:00
|
|
|
* The hook connects to a dedicated channel, sending the token and
|
|
|
|
* output ref in an initial message. It expects `init:<ref>` message
|
|
|
|
* with `{ data }` payload, the data is then used in the initial call
|
|
|
|
* to the custom JS module.
|
2021-12-24 21:18:34 +08:00
|
|
|
*
|
2022-01-06 23:31:26 +08:00
|
|
|
* Then, a number of `event:<ref>` with `{ event, payload }` payload
|
|
|
|
* can be sent. The `event` is forwarded to the initialized component.
|
2021-12-24 21:18:34 +08:00
|
|
|
*
|
|
|
|
* Configuration:
|
|
|
|
*
|
2022-01-06 23:31:26 +08:00
|
|
|
* * `data-ref` - a unique identifier used as messages scope
|
2021-12-24 21:18:34 +08:00
|
|
|
*
|
2022-01-20 18:29:45 +08:00
|
|
|
* * `data-assets-base-path` - the path to resolve all relative paths
|
2021-12-24 21:18:34 +08:00
|
|
|
* against in the iframe
|
|
|
|
*
|
|
|
|
* * `data-js-path` - a relative path for the initial output-specific
|
|
|
|
* JS module
|
|
|
|
*
|
2022-01-06 23:31:26 +08:00
|
|
|
* * `data-session-token` - token is sent in the "connect" message to
|
|
|
|
* the channel
|
|
|
|
*
|
2022-02-08 21:45:58 +08:00
|
|
|
* * `data-session-id` - the identifier of the session that this output
|
|
|
|
* belongs go
|
|
|
|
*
|
|
|
|
* * `data-iframe-local-port` - the local port where the iframe is served
|
|
|
|
*
|
2021-12-24 21:18:34 +08:00
|
|
|
*/
|
|
|
|
const JSOutput = {
|
|
|
|
mounted() {
|
|
|
|
this.props = getProps(this);
|
|
|
|
this.state = {
|
2022-01-06 23:31:26 +08:00
|
|
|
childToken: randomToken(),
|
2021-12-24 21:18:34 +08:00
|
|
|
childReadyPromise: null,
|
|
|
|
childReady: false,
|
|
|
|
iframe: null,
|
2022-01-06 23:31:26 +08:00
|
|
|
channelUnsubscribe: null,
|
|
|
|
errorContainer: null,
|
2021-12-24 21:18:34 +08:00
|
|
|
};
|
|
|
|
|
2022-02-08 04:03:25 +08:00
|
|
|
const channel = getChannel(this.props.sessionId);
|
2022-01-06 23:31:26 +08:00
|
|
|
|
2022-02-02 07:10:17 +08:00
|
|
|
// When cells/sections are reordered, morphdom detaches and attaches
|
|
|
|
// the relevant elements in the DOM. Consequently the output element
|
|
|
|
// becomes temporarily detached from the DOM and attaching it back
|
|
|
|
// would cause the iframe to reload. This behaviour is expected, see
|
|
|
|
// https://github.com/whatwg/html/issues/5484 for more details. Reloading
|
|
|
|
// that frequently is inefficient and also clears the iframe state,
|
|
|
|
// which makes is very undesired in our case. To solve this, we insert
|
|
|
|
// the iframe higher in the DOM tree, so that it's never affected by
|
|
|
|
// reordering. Then, we insert a placeholder element in the output to
|
|
|
|
// take up the expected space and we use absolute positioning to place
|
|
|
|
// the iframe exactly over that placeholder. We set up observers to
|
|
|
|
// track the changes in placeholder's position/size and we keep the
|
|
|
|
// absolute iframe in sync.
|
|
|
|
|
2021-12-24 21:18:34 +08:00
|
|
|
const iframePlaceholder = document.createElement("div");
|
2022-02-02 07:10:17 +08:00
|
|
|
this.el.appendChild(iframePlaceholder);
|
|
|
|
|
2021-12-24 21:18:34 +08:00
|
|
|
const iframe = document.createElement("iframe");
|
2022-02-02 07:10:17 +08:00
|
|
|
iframe.className = "w-full h-0 absolute z-[1]";
|
2021-12-24 21:18:34 +08:00
|
|
|
this.state.iframe = iframe;
|
|
|
|
|
2022-02-02 07:10:17 +08:00
|
|
|
this.disconnectObservers = bindIframeSize(iframe, iframePlaceholder);
|
|
|
|
|
|
|
|
// Register message chandler to communicate with the iframe
|
|
|
|
|
2021-12-24 21:18:34 +08:00
|
|
|
function postMessage(message) {
|
|
|
|
iframe.contentWindow.postMessage(message, "*");
|
|
|
|
}
|
|
|
|
|
|
|
|
this.state.childReadyPromise = new Promise((resolve, reject) => {
|
|
|
|
this.handleWindowMessage = (event) => {
|
|
|
|
if (event.source === iframe.contentWindow) {
|
|
|
|
handleChildMessage(event.data);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
window.addEventListener("message", this.handleWindowMessage);
|
|
|
|
|
|
|
|
const handleChildMessage = (message) => {
|
|
|
|
if (message.type === "ready" && !this.state.childReady) {
|
2022-01-20 18:29:45 +08:00
|
|
|
const assetsBaseUrl =
|
|
|
|
window.location.origin + this.props.assetsBasePath;
|
2021-12-24 21:18:34 +08:00
|
|
|
postMessage({
|
|
|
|
type: "readyReply",
|
2022-01-06 23:31:26 +08:00
|
|
|
token: this.state.childToken,
|
2022-01-20 18:29:45 +08:00
|
|
|
baseUrl: assetsBaseUrl,
|
2021-12-24 21:18:34 +08:00
|
|
|
jsPath: this.props.jsPath,
|
|
|
|
});
|
|
|
|
this.state.childReady = true;
|
|
|
|
resolve();
|
|
|
|
} else {
|
|
|
|
// Note: we use a random token to authorize child messages
|
|
|
|
// and do our best to make this token unavailable for the
|
|
|
|
// injected script on the child side. In the worst case scenario,
|
|
|
|
// the script manages to extract the token and can then send
|
|
|
|
// any of those messages, so we can treat this as a possible
|
|
|
|
// surface for attacks. In this case the most "critical" actions
|
|
|
|
// are shortcuts, neither of which is particularly dangerous.
|
2022-01-06 23:31:26 +08:00
|
|
|
if (message.token !== this.state.childToken) {
|
2021-12-24 21:18:34 +08:00
|
|
|
throw new Error("Token mismatch");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (message.type === "resize") {
|
|
|
|
iframePlaceholder.style.height = `${message.height}px`;
|
|
|
|
iframe.style.height = `${message.height}px`;
|
|
|
|
} else if (message.type === "domEvent") {
|
|
|
|
// Replicate the child events on the current element,
|
|
|
|
// so that they are detected upstream in the session hook
|
|
|
|
const event = replicateDomEvent(message.event);
|
|
|
|
this.el.dispatchEvent(event);
|
|
|
|
} else if (message.type === "event") {
|
|
|
|
const { event, payload } = message;
|
2022-02-08 04:03:25 +08:00
|
|
|
const raw = transportEncode([event, this.props.ref], payload);
|
|
|
|
channel.push("event", raw);
|
2021-12-24 21:18:34 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
const replicateDomEvent = (event) => {
|
|
|
|
if (event.type === "focus") {
|
|
|
|
return new FocusEvent("focus");
|
|
|
|
} else if (event.type === "mousedown") {
|
|
|
|
return new MouseEvent("mousedown", { bubbles: true });
|
|
|
|
} else if (event.type === "keydown") {
|
|
|
|
return new KeyboardEvent(event.type, event.props);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
});
|
|
|
|
|
2022-02-02 07:10:17 +08:00
|
|
|
// Load the iframe content
|
|
|
|
const iframesEl = document.querySelector(`[data-element="output-iframes"]`);
|
2022-02-08 21:45:58 +08:00
|
|
|
initializeIframeSource(iframe, this.props.iframePort).then(() => {
|
2022-02-02 07:10:17 +08:00
|
|
|
iframesEl.appendChild(iframe);
|
|
|
|
});
|
2021-12-24 21:18:34 +08:00
|
|
|
|
|
|
|
// Event handlers
|
|
|
|
|
2022-01-06 23:31:26 +08:00
|
|
|
channel.push("connect", {
|
|
|
|
session_token: this.props.sessionToken,
|
|
|
|
ref: this.props.ref,
|
|
|
|
});
|
|
|
|
|
2022-02-08 04:03:25 +08:00
|
|
|
const initRef = channel.on(`init:${this.props.ref}`, (raw) => {
|
|
|
|
const [, payload] = transportDecode(raw);
|
|
|
|
|
2021-12-24 21:18:34 +08:00
|
|
|
this.state.childReadyPromise.then(() => {
|
2022-02-08 04:03:25 +08:00
|
|
|
postMessage({ type: "init", data: payload });
|
2021-12-24 21:18:34 +08:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2022-02-08 04:03:25 +08:00
|
|
|
const eventRef = channel.on(`event:${this.props.ref}`, (raw) => {
|
|
|
|
const [[event], payload] = transportDecode(raw);
|
2022-02-08 21:45:58 +08:00
|
|
|
|
|
|
|
this.state.childReadyPromise.then(() => {
|
|
|
|
postMessage({ type: "event", event, payload });
|
|
|
|
});
|
2022-02-08 04:03:25 +08:00
|
|
|
});
|
2022-01-06 23:31:26 +08:00
|
|
|
|
|
|
|
const errorRef = channel.on(`error:${this.props.ref}`, ({ message }) => {
|
|
|
|
if (!this.state.errorContainer) {
|
|
|
|
this.state.errorContainer = document.createElement("div");
|
|
|
|
this.state.errorContainer.classList.add("error-box", "mb-4");
|
|
|
|
this.el.prepend(this.state.errorContainer);
|
|
|
|
}
|
|
|
|
|
|
|
|
this.state.errorContainer.textContent = message;
|
|
|
|
});
|
|
|
|
|
|
|
|
this.state.channelUnsubscribe = () => {
|
|
|
|
channel.off(`init:${this.props.ref}`, initRef);
|
|
|
|
channel.off(`event:${this.props.ref}`, eventRef);
|
|
|
|
channel.off(`error:${this.props.ref}`, errorRef);
|
|
|
|
};
|
2021-12-24 21:18:34 +08:00
|
|
|
},
|
|
|
|
|
|
|
|
updated() {
|
|
|
|
this.props = getProps(this);
|
|
|
|
},
|
|
|
|
|
|
|
|
destroyed() {
|
|
|
|
window.removeEventListener("message", this.handleWindowMessage);
|
2022-02-02 07:10:17 +08:00
|
|
|
this.disconnectObservers();
|
2021-12-24 21:18:34 +08:00
|
|
|
this.state.iframe.remove();
|
2022-01-06 23:31:26 +08:00
|
|
|
|
2022-02-08 04:03:25 +08:00
|
|
|
const channel = getChannel(this.props.sessionId, { create: false });
|
2022-01-06 23:31:26 +08:00
|
|
|
|
|
|
|
if (channel) {
|
|
|
|
this.state.channelUnsubscribe();
|
|
|
|
channel.push("disconnect", { ref: this.props.ref });
|
|
|
|
}
|
2021-12-24 21:18:34 +08:00
|
|
|
},
|
|
|
|
};
|
|
|
|
|
|
|
|
function getProps(hook) {
|
|
|
|
return {
|
2022-01-06 23:31:26 +08:00
|
|
|
ref: getAttributeOrThrow(hook.el, "data-ref"),
|
2022-01-20 18:29:45 +08:00
|
|
|
assetsBasePath: getAttributeOrThrow(hook.el, "data-assets-base-path"),
|
2021-12-24 21:18:34 +08:00
|
|
|
jsPath: getAttributeOrThrow(hook.el, "data-js-path"),
|
2022-01-06 23:31:26 +08:00
|
|
|
sessionToken: getAttributeOrThrow(hook.el, "data-session-token"),
|
2022-01-11 01:38:08 +08:00
|
|
|
sessionId: getAttributeOrThrow(hook.el, "data-session-id"),
|
2022-02-08 21:45:58 +08:00
|
|
|
iframePort: getAttributeOrThrow(
|
|
|
|
hook.el,
|
|
|
|
"data-iframe-local-port",
|
|
|
|
parseInteger
|
|
|
|
),
|
2021-12-24 21:18:34 +08:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2022-02-08 04:03:25 +08:00
|
|
|
const csrfToken = document
|
|
|
|
.querySelector("meta[name='csrf-token']")
|
|
|
|
.getAttribute("content");
|
|
|
|
const socket = new Socket("/socket", { params: { _csrf_token: csrfToken } });
|
|
|
|
|
2022-02-02 07:10:17 +08:00
|
|
|
let channel = null;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns channel used for all JS outputs in the current session.
|
|
|
|
*/
|
2022-02-08 04:03:25 +08:00
|
|
|
function getChannel(sessionId, { create = true } = {}) {
|
2022-02-02 07:10:17 +08:00
|
|
|
if (!channel && create) {
|
2022-02-08 04:03:25 +08:00
|
|
|
socket.connect();
|
2022-02-02 07:10:17 +08:00
|
|
|
channel = socket.channel("js_output", { session_id: sessionId });
|
|
|
|
channel.join();
|
|
|
|
}
|
|
|
|
|
|
|
|
return channel;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Leaves the JS outputs channel tied to the current session.
|
|
|
|
*/
|
|
|
|
export function leaveChannel() {
|
2022-02-08 04:03:25 +08:00
|
|
|
socket.disconnect();
|
|
|
|
channel = null;
|
2022-02-02 07:10:17 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Sets up observers to resize and reposition the iframe
|
|
|
|
* whenever the placeholder moves.
|
|
|
|
*/
|
|
|
|
function bindIframeSize(iframe, iframePlaceholder) {
|
|
|
|
const notebookEl = document.querySelector(`[data-element="notebook"]`);
|
|
|
|
const notebookContentEl = notebookEl.querySelector(
|
|
|
|
`[data-element="notebook-content"]`
|
|
|
|
);
|
|
|
|
|
|
|
|
function repositionIframe() {
|
|
|
|
const notebookBox = notebookEl.getBoundingClientRect();
|
|
|
|
const placeholderBox = iframePlaceholder.getBoundingClientRect();
|
|
|
|
const top = placeholderBox.top - notebookBox.top + notebookEl.scrollTop;
|
|
|
|
iframe.style.top = `${top}px`;
|
|
|
|
const left = placeholderBox.left - notebookBox.left + notebookEl.scrollLeft;
|
|
|
|
iframe.style.left = `${left}px`;
|
|
|
|
iframe.style.height = `${placeholderBox.height}px`;
|
|
|
|
iframe.style.width = `${placeholderBox.width}px`;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Most output position changes are accompanied by changes to the
|
|
|
|
// notebook content element (adding cells, inserting newlines in
|
|
|
|
// the editor, etc)
|
|
|
|
const resizeObserver = new ResizeObserver((entries) => repositionIframe());
|
|
|
|
resizeObserver.observe(notebookContentEl);
|
|
|
|
|
|
|
|
// On lower level cell/section reordering is applied as element
|
|
|
|
// removal followed by insert, consequently the intersection
|
|
|
|
// between the output and notebook content changes (becomes none
|
|
|
|
// for a brief moment)
|
|
|
|
const intersectionObserver = new IntersectionObserver(
|
|
|
|
(entries) => repositionIframe(),
|
|
|
|
{ root: notebookContentEl }
|
|
|
|
);
|
|
|
|
intersectionObserver.observe(iframePlaceholder);
|
|
|
|
|
|
|
|
return () => {
|
|
|
|
resizeObserver.disconnect();
|
|
|
|
intersectionObserver.disconnect();
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
// Loading iframe using `srcdoc` disables cookies and browser APIs,
|
|
|
|
// such as camera and microphone (1), the same applies to `src` with
|
|
|
|
// data URL, so we need to load the iframe through a regular request.
|
|
|
|
// Since the iframe is sandboxed we also need `allow-same-origin`.
|
|
|
|
// Additionally, we cannot load the iframe from the same origin as
|
|
|
|
// the app, because using `allow-same-origin` together with `allow-scripts`
|
|
|
|
// would be insecure (2). Consequently, we need to load the iframe
|
|
|
|
// from a different origin.
|
|
|
|
//
|
2022-02-08 21:45:58 +08:00
|
|
|
// When running Livebook on https:// we load the iframe from another
|
|
|
|
// https:// origin. On the other hand, when running on http:// we want
|
|
|
|
// to load the iframe from http:// as well, otherwise the browser could
|
|
|
|
// block asset requests from the https:// iframe to http:// Livebook.
|
|
|
|
// However, external http:// content is not considered a secure context (3),
|
|
|
|
// which implies no access to user media. Therefore, instead of using
|
|
|
|
// http://livebook.space we use another localhost endpoint. Note that
|
|
|
|
// this endpoint has a different port than the Livebook web app, that's
|
|
|
|
// because we need separate origins, as outlined above.
|
|
|
|
//
|
2022-02-02 07:10:17 +08:00
|
|
|
// To ensure integrity of the loaded content we manually verify the
|
|
|
|
// checksum against the expected value.
|
|
|
|
//
|
|
|
|
// (1): https://developer.mozilla.org/en-US/docs/Web/API/MediaDevices/getUserMedia#document_source_security
|
|
|
|
// (2): https://developer.mozilla.org/en-US/docs/Web/HTML/Element/iframe#attr-sandbox
|
2022-02-08 21:45:58 +08:00
|
|
|
// (3): https://developer.mozilla.org/en-US/docs/Web/Security/Secure_Contexts
|
2022-02-02 07:10:17 +08:00
|
|
|
|
2022-02-04 01:51:33 +08:00
|
|
|
const IFRAME_SHA256 = "+uJyGu0Ey7uVV7WwRwg7GyjwCkMNRBnyNc25iGFpYXc=";
|
2022-02-02 07:10:17 +08:00
|
|
|
|
2022-02-08 21:45:58 +08:00
|
|
|
function getIframeUrl(iframePort) {
|
|
|
|
return window.location.protocol === "https:"
|
|
|
|
? "https://livebook.space/iframe/v2.html"
|
|
|
|
: `http://${window.location.hostname}:${iframePort}/iframe/v2.html`;
|
|
|
|
}
|
|
|
|
|
|
|
|
function initializeIframeSource(iframe, iframePort) {
|
|
|
|
const iframeUrl = getIframeUrl(iframePort);
|
|
|
|
|
|
|
|
return verifyIframeSource(iframeUrl).then(() => {
|
2022-02-02 07:10:17 +08:00
|
|
|
iframe.sandbox =
|
|
|
|
"allow-scripts allow-same-origin allow-downloads allow-modals";
|
|
|
|
iframe.allow =
|
|
|
|
"accelerometer; ambient-light-sensor; camera; display-capture; encrypted-media; geolocation; gyroscope; microphone; midi; usb; xr-spatial-tracking";
|
2022-02-08 21:45:58 +08:00
|
|
|
iframe.src = iframeUrl;
|
2022-02-02 07:10:17 +08:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
let iframeVerificationPromise = null;
|
|
|
|
|
2022-02-08 21:45:58 +08:00
|
|
|
function verifyIframeSource(iframeUrl) {
|
2022-02-02 07:10:17 +08:00
|
|
|
if (!iframeVerificationPromise) {
|
2022-02-08 21:45:58 +08:00
|
|
|
iframeVerificationPromise = fetch(iframeUrl)
|
2022-02-02 07:10:17 +08:00
|
|
|
.then((response) => response.text())
|
|
|
|
.then((html) => {
|
|
|
|
if (sha256Base64(html) !== IFRAME_SHA256) {
|
|
|
|
throw new Error(
|
|
|
|
"The loaded iframe content doesn't have the expected checksum"
|
|
|
|
);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
return iframeVerificationPromise;
|
|
|
|
}
|
|
|
|
|
2022-02-08 21:45:58 +08:00
|
|
|
// Encoding/decoding of channel payloads
|
|
|
|
|
2022-02-08 04:03:25 +08:00
|
|
|
function transportEncode(meta, payload) {
|
|
|
|
if (
|
|
|
|
Array.isArray(payload) &&
|
|
|
|
payload[1] &&
|
|
|
|
payload[1].constructor === ArrayBuffer
|
|
|
|
) {
|
|
|
|
const [info, buffer] = payload;
|
|
|
|
return encode([meta, info], buffer);
|
|
|
|
} else {
|
|
|
|
return { root: [meta, payload] };
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function transportDecode(raw) {
|
|
|
|
if (raw.constructor === ArrayBuffer) {
|
|
|
|
const [[meta, info], buffer] = decode(raw);
|
|
|
|
return [meta, [info, buffer]];
|
|
|
|
} else {
|
|
|
|
const {
|
|
|
|
root: [meta, payload],
|
|
|
|
} = raw;
|
|
|
|
return [meta, payload];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const HEADER_LENGTH = 4;
|
|
|
|
|
|
|
|
function encode(meta, buffer) {
|
|
|
|
const encoder = new TextEncoder();
|
|
|
|
const metaArray = encoder.encode(JSON.stringify(meta));
|
|
|
|
|
|
|
|
const raw = new ArrayBuffer(
|
|
|
|
HEADER_LENGTH + metaArray.byteLength + buffer.byteLength
|
|
|
|
);
|
|
|
|
const view = new DataView(raw);
|
|
|
|
|
|
|
|
view.setUint32(0, metaArray.byteLength);
|
|
|
|
new Uint8Array(raw, HEADER_LENGTH, metaArray.byteLength).set(metaArray);
|
|
|
|
new Uint8Array(raw, HEADER_LENGTH + metaArray.byteLength).set(
|
|
|
|
new Uint8Array(buffer)
|
|
|
|
);
|
|
|
|
|
|
|
|
return raw;
|
|
|
|
}
|
|
|
|
|
|
|
|
function decode(raw) {
|
|
|
|
const view = new DataView(raw);
|
|
|
|
const metaArrayLength = view.getUint32(0);
|
|
|
|
|
|
|
|
const metaArray = new Uint8Array(raw, HEADER_LENGTH, metaArrayLength);
|
|
|
|
const buffer = raw.slice(HEADER_LENGTH + metaArrayLength);
|
|
|
|
|
|
|
|
const decoder = new TextDecoder();
|
|
|
|
const meta = JSON.parse(decoder.decode(metaArray));
|
|
|
|
|
|
|
|
return [meta, buffer];
|
|
|
|
}
|
|
|
|
|
2021-12-24 21:18:34 +08:00
|
|
|
export default JSOutput;
|