123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406 |
- import {
- ExcalidrawElement,
- NonDeletedExcalidrawElement,
- } from "../element/types";
- import { getDefaultAppState } from "../appState";
- import { AppState } from "../types";
- import { exportToCanvas, exportToSvg } from "../scene/export";
- import { fileSave } from "browser-nativefs";
- import { t } from "../i18n";
- import {
- copyCanvasToClipboardAsPng,
- copyTextToSystemClipboard,
- } from "../clipboard";
- import { serializeAsJSON } from "./json";
- import { ExportType } from "../scene/types";
- import { restore } from "./restore";
- import { ImportedDataState } from "./types";
- import { canvasToBlob } from "./blob";
- import { EVENT_ACTION, trackEvent } from "../analytics";
- export { loadFromBlob } from "./blob";
- export { saveAsJSON, loadFromJSON } from "./json";
- const BACKEND_GET = process.env.REACT_APP_BACKEND_V1_GET_URL;
- const BACKEND_V2_POST = process.env.REACT_APP_BACKEND_V2_POST_URL;
- const BACKEND_V2_GET = process.env.REACT_APP_BACKEND_V2_GET_URL;
- export const SOCKET_SERVER = process.env.REACT_APP_SOCKET_SERVER_URL;
- export type EncryptedData = {
- data: ArrayBuffer;
- iv: Uint8Array;
- };
- export type SocketUpdateDataSource = {
- SCENE_INIT: {
- type: "SCENE_INIT";
- payload: {
- elements: readonly ExcalidrawElement[];
- };
- };
- SCENE_UPDATE: {
- type: "SCENE_UPDATE";
- payload: {
- elements: readonly ExcalidrawElement[];
- };
- };
- MOUSE_LOCATION: {
- type: "MOUSE_LOCATION";
- payload: {
- socketId: string;
- pointer: { x: number; y: number };
- button: "down" | "up";
- selectedElementIds: AppState["selectedElementIds"];
- username: string;
- };
- };
- };
- export type SocketUpdateDataIncoming =
- | SocketUpdateDataSource[keyof SocketUpdateDataSource]
- | {
- type: "INVALID_RESPONSE";
- };
- const byteToHex = (byte: number): string => `0${byte.toString(16)}`.slice(-2);
- const generateRandomID = async () => {
- const arr = new Uint8Array(10);
- window.crypto.getRandomValues(arr);
- return Array.from(arr, byteToHex).join("");
- };
- const generateEncryptionKey = async () => {
- const key = await window.crypto.subtle.generateKey(
- {
- name: "AES-GCM",
- length: 128,
- },
- true, // extractable
- ["encrypt", "decrypt"],
- );
- return (await window.crypto.subtle.exportKey("jwk", key)).k;
- };
- export const createIV = () => {
- const arr = new Uint8Array(12);
- return window.crypto.getRandomValues(arr);
- };
- export const getCollaborationLinkData = (link: string) => {
- if (link.length === 0) {
- return;
- }
- const hash = new URL(link).hash;
- return hash.match(/^#room=([a-zA-Z0-9_-]+),([a-zA-Z0-9_-]+)$/);
- };
- export const generateCollaborationLink = async () => {
- const id = await generateRandomID();
- const key = await generateEncryptionKey();
- return `${window.location.origin}${window.location.pathname}#room=${id},${key}`;
- };
- export const getImportedKey = (key: string, usage: KeyUsage) =>
- window.crypto.subtle.importKey(
- "jwk",
- {
- alg: "A128GCM",
- ext: true,
- k: key,
- key_ops: ["encrypt", "decrypt"],
- kty: "oct",
- },
- {
- name: "AES-GCM",
- length: 128,
- },
- false, // extractable
- [usage],
- );
- export const encryptAESGEM = async (
- data: Uint8Array,
- key: string,
- ): Promise<EncryptedData> => {
- const importedKey = await getImportedKey(key, "encrypt");
- const iv = createIV();
- return {
- data: await window.crypto.subtle.encrypt(
- {
- name: "AES-GCM",
- iv,
- },
- importedKey,
- data,
- ),
- iv,
- };
- };
- export const decryptAESGEM = async (
- data: ArrayBuffer,
- key: string,
- iv: Uint8Array,
- ): Promise<SocketUpdateDataIncoming> => {
- try {
- const importedKey = await getImportedKey(key, "decrypt");
- const decrypted = await window.crypto.subtle.decrypt(
- {
- name: "AES-GCM",
- iv,
- },
- importedKey,
- data,
- );
- const decodedData = new TextDecoder("utf-8").decode(
- new Uint8Array(decrypted) as any,
- );
- return JSON.parse(decodedData);
- } catch (error) {
- window.alert(t("alerts.decryptFailed"));
- console.error(error);
- }
- return {
- type: "INVALID_RESPONSE",
- };
- };
- export const exportToBackend = async (
- elements: readonly ExcalidrawElement[],
- appState: AppState,
- ) => {
- const json = serializeAsJSON(elements, appState);
- const encoded = new TextEncoder().encode(json);
- const key = await window.crypto.subtle.generateKey(
- {
- name: "AES-GCM",
- length: 128,
- },
- true, // extractable
- ["encrypt", "decrypt"],
- );
- // The iv is set to 0. We are never going to reuse the same key so we don't
- // need to have an iv. (I hope that's correct...)
- const iv = new Uint8Array(12);
- // We use symmetric encryption. AES-GCM is the recommended algorithm and
- // includes checks that the ciphertext has not been modified by an attacker.
- const encrypted = await window.crypto.subtle.encrypt(
- {
- name: "AES-GCM",
- iv,
- },
- key,
- encoded,
- );
- // We use jwk encoding to be able to extract just the base64 encoded key.
- // We will hardcode the rest of the attributes when importing back the key.
- const exportedKey = await window.crypto.subtle.exportKey("jwk", key);
- try {
- const response = await fetch(BACKEND_V2_POST, {
- method: "POST",
- body: encrypted,
- });
- const json = await response.json();
- if (json.id) {
- const url = new URL(window.location.href);
- // We need to store the key (and less importantly the id) as hash instead
- // of queryParam in order to never send it to the server
- url.hash = `json=${json.id},${exportedKey.k!}`;
- const urlString = url.toString();
- window.prompt(`🔒${t("alerts.uploadedSecurly")}`, urlString);
- } else if (json.error_class === "RequestTooLargeError") {
- window.alert(t("alerts.couldNotCreateShareableLinkTooBig"));
- } else {
- window.alert(t("alerts.couldNotCreateShareableLink"));
- }
- } catch (error) {
- console.error(error);
- window.alert(t("alerts.couldNotCreateShareableLink"));
- }
- };
- const importFromBackend = async (
- id: string | null,
- privateKey?: string | null,
- ): Promise<ImportedDataState> => {
- try {
- const response = await fetch(
- privateKey ? `${BACKEND_V2_GET}${id}` : `${BACKEND_GET}${id}.json`,
- );
- if (!response.ok) {
- window.alert(t("alerts.importBackendFailed"));
- return {};
- }
- let data: ImportedDataState;
- if (privateKey) {
- const buffer = await response.arrayBuffer();
- const key = await getImportedKey(privateKey, "decrypt");
- const iv = new Uint8Array(12);
- const decrypted = await window.crypto.subtle.decrypt(
- {
- name: "AES-GCM",
- iv,
- },
- key,
- buffer,
- );
- // We need to convert the decrypted array buffer to a string
- const string = new window.TextDecoder("utf-8").decode(
- new Uint8Array(decrypted) as any,
- );
- data = JSON.parse(string);
- } else {
- // Legacy format
- data = await response.json();
- }
- trackEvent(EVENT_ACTION, "import");
- return {
- elements: data.elements || null,
- appState: data.appState || null,
- };
- } catch (error) {
- window.alert(t("alerts.importBackendFailed"));
- console.error(error);
- return {};
- }
- };
- export const exportCanvas = async (
- type: ExportType,
- elements: readonly NonDeletedExcalidrawElement[],
- appState: AppState,
- canvas: HTMLCanvasElement,
- {
- exportBackground,
- exportPadding = 10,
- viewBackgroundColor,
- name,
- scale = 1,
- shouldAddWatermark,
- }: {
- exportBackground: boolean;
- exportPadding?: number;
- viewBackgroundColor: string;
- name: string;
- scale?: number;
- shouldAddWatermark: boolean;
- },
- ) => {
- if (elements.length === 0) {
- return window.alert(t("alerts.cannotExportEmptyCanvas"));
- }
- if (type === "svg" || type === "clipboard-svg") {
- const tempSvg = exportToSvg(elements, {
- exportBackground,
- viewBackgroundColor,
- exportPadding,
- scale,
- shouldAddWatermark,
- metadata:
- appState.exportEmbedScene && type === "svg"
- ? await (
- await import(/* webpackChunkName: "image" */ "./image")
- ).encodeSvgMetadata({
- text: serializeAsJSON(elements, appState),
- })
- : undefined,
- });
- if (type === "svg") {
- await fileSave(new Blob([tempSvg.outerHTML], { type: "image/svg+xml" }), {
- fileName: `${name}.svg`,
- extensions: [".svg"],
- });
- return;
- } else if (type === "clipboard-svg") {
- copyTextToSystemClipboard(tempSvg.outerHTML);
- return;
- }
- }
- const tempCanvas = exportToCanvas(elements, appState, {
- exportBackground,
- viewBackgroundColor,
- exportPadding,
- scale,
- shouldAddWatermark,
- });
- tempCanvas.style.display = "none";
- document.body.appendChild(tempCanvas);
- if (type === "png") {
- const fileName = `${name}.png`;
- let blob = await canvasToBlob(tempCanvas);
- if (appState.exportEmbedScene) {
- blob = await (
- await import(/* webpackChunkName: "image" */ "./image")
- ).encodePngMetadata({
- blob,
- metadata: serializeAsJSON(elements, appState),
- });
- }
- await fileSave(blob, {
- fileName,
- extensions: [".png"],
- });
- } else if (type === "clipboard") {
- try {
- await copyCanvasToClipboardAsPng(tempCanvas);
- } catch (error) {
- if (error.name === "CANVAS_POSSIBLY_TOO_BIG") {
- throw error;
- }
- throw new Error(t("alerts.couldNotCopyToClipboard"));
- }
- } else if (type === "backend") {
- exportToBackend(elements, {
- ...appState,
- viewBackgroundColor: exportBackground
- ? appState.viewBackgroundColor
- : getDefaultAppState().viewBackgroundColor,
- });
- }
- // clean up the DOM
- if (tempCanvas !== canvas) {
- tempCanvas.remove();
- }
- };
- export const loadScene = async (
- id: string | null,
- privateKey: string | null,
- // Supply initialData even if importing from backend to ensure we restore
- // localStorage user settings which we do not persist on server.
- // Non-optional so we don't forget to pass it even if `undefined`.
- initialData: ImportedDataState | undefined | null,
- ) => {
- let data;
- if (id != null) {
- // the private key is used to decrypt the content from the server, take
- // extra care not to leak it
- data = restore(
- await importFromBackend(id, privateKey),
- initialData?.appState,
- );
- } else {
- data = restore(initialData || {}, null);
- }
- return {
- elements: data.elements,
- appState: data.appState,
- commitToHistory: false,
- };
- };
|