@zsviczian/excalidraw 0.9.0-obsidian-image-support-1 → 0.9.0-obsidian-image-support-2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
|
@@ -2588,7 +2588,7 @@ eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpac
|
|
|
2588
2588
|
/***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => {
|
|
2589
2589
|
|
|
2590
2590
|
"use strict";
|
|
2591
|
-
eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ \"loadFirebaseStorage\": () => (/* binding */ loadFirebaseStorage),\n/* harmony export */ \"isSavedToFirebase\": () => (/* binding */ isSavedToFirebase),\n/* harmony export */ \"saveFilesToFirebase\": () => (/* binding */ saveFilesToFirebase),\n/* harmony export */ \"saveToFirebase\": () => (/* binding */ saveToFirebase),\n/* harmony export */ \"loadFromFirebase\": () => (/* binding */ loadFromFirebase),\n/* harmony export */ \"loadFilesFromFirebase\": () => (/* binding */ loadFilesFromFirebase)\n/* harmony export */ });\n/* harmony import */ var _data__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./index */ \"../../excalidraw-app/data/index.ts\");\n/* harmony import */ var _element__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ../../element */ \"../../element/index.ts\");\n/* harmony import */ var _data_restore__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ../../data/restore */ \"../../data/restore.ts\");\n/* harmony import */ var _app_constants__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! ../app_constants */ \"../../excalidraw-app/app_constants.ts\");\nvar __awaiter = (undefined && undefined.__awaiter) || function (thisArg, _arguments, P, generator) {\r\n function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }\r\n return new (P || (P = Promise))(function (resolve, reject) {\r\n function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }\r\n function rejected(value) { try { step(generator[\"throw\"](value)); } catch (e) { reject(e); } }\r\n function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }\r\n step((generator = generator.apply(thisArg, _arguments || [])).next());\r\n });\r\n};\r\n\r\n\r\n\r\n\r\n\r\n// private\r\n// -----------------------------------------------------------------------------\r\nconst FIREBASE_CONFIG = JSON.parse(process.env.REACT_APP_FIREBASE_CONFIG);\r\nlet firebasePromise = null;\r\nlet firestorePromise = null;\r\nlet firebseStoragePromise = null;\r\nlet isFirebaseInitialized = false;\r\nconst _loadFirebase = () => __awaiter(void 0, void 0, void 0, function* () {\r\n const firebase = (yield __webpack_require__.e(/*! import() | firebase */ \"vendor\").then(__webpack_require__.bind(__webpack_require__, /*! firebase/app */ \"../../../node_modules/firebase/app/dist/index.esm.js\"))).default;\r\n // due to dev HMR\r\n if (!isFirebaseInitialized) {\r\n isFirebaseInitialized = true;\r\n try {\r\n firebase.initializeApp(FIREBASE_CONFIG);\r\n }\r\n catch (error) {\r\n console.warn(error.name, error.code);\r\n }\r\n }\r\n return firebase;\r\n});\r\nconst _getFirebase = () => __awaiter(void 0, void 0, void 0, function* () {\r\n if (!firebasePromise) {\r\n firebasePromise = _loadFirebase();\r\n }\r\n return firebasePromise;\r\n});\r\n// -----------------------------------------------------------------------------\r\nconst loadFirestore = () => __awaiter(void 0, void 0, void 0, function* () {\r\n const firebase = yield _getFirebase();\r\n if (!firestorePromise) {\r\n firestorePromise = __webpack_require__.e(/*! import() | firestore */ \"vendor\").then(__webpack_require__.bind(__webpack_require__, /*! firebase/firestore */ \"../../../node_modules/firebase/firestore/dist/index.esm.js\"));\r\n }\r\n if (firestorePromise !== true) {\r\n yield firestorePromise;\r\n firestorePromise = true;\r\n }\r\n return firebase;\r\n});\r\nconst loadFirebaseStorage = () => __awaiter(void 0, void 0, void 0, function* () {\r\n const firebase = yield _getFirebase();\r\n if (!firebseStoragePromise) {\r\n firebseStoragePromise = __webpack_require__.e(/*! import() | storage */ \"vendor\").then(__webpack_require__.bind(__webpack_require__, /*! firebase/storage */ \"../../../node_modules/firebase/storage/dist/index.esm.js\"));\r\n }\r\n if (firebseStoragePromise !== true) {\r\n yield firebseStoragePromise;\r\n firebseStoragePromise = true;\r\n }\r\n return firebase;\r\n});\r\nconst encryptElements = (key, elements) => __awaiter(void 0, void 0, void 0, function* () {\r\n const importedKey = yield (0,_data__WEBPACK_IMPORTED_MODULE_0__.getImportedKey)(key, \"encrypt\");\r\n const iv = (0,_data__WEBPACK_IMPORTED_MODULE_0__.createIV)();\r\n const json = JSON.stringify(elements);\r\n const encoded = new TextEncoder().encode(json);\r\n const ciphertext = yield window.crypto.subtle.encrypt({\r\n name: \"AES-GCM\",\r\n iv,\r\n }, importedKey, encoded);\r\n return { ciphertext, iv };\r\n});\r\nconst decryptElements = (key, iv, ciphertext) => __awaiter(void 0, void 0, void 0, function* () {\r\n const importedKey = yield (0,_data__WEBPACK_IMPORTED_MODULE_0__.getImportedKey)(key, \"decrypt\");\r\n const decrypted = yield window.crypto.subtle.decrypt({\r\n name: \"AES-GCM\",\r\n iv,\r\n }, importedKey, ciphertext);\r\n const decodedData = new TextDecoder(\"utf-8\").decode(new Uint8Array(decrypted));\r\n return JSON.parse(decodedData);\r\n});\r\nconst firebaseSceneVersionCache = new WeakMap();\r\nconst isSavedToFirebase = (portal, elements) => {\r\n if (portal.socket && portal.roomId && portal.roomKey) {\r\n const sceneVersion = (0,_element__WEBPACK_IMPORTED_MODULE_1__.getSceneVersion)(elements);\r\n return firebaseSceneVersionCache.get(portal.socket) === sceneVersion;\r\n }\r\n // if no room exists, consider the room saved so that we don't unnecessarily\r\n // prevent unload (there's nothing we could do at that point anyway)\r\n return true;\r\n};\r\nconst saveFilesToFirebase = ({ prefix, decryptionKey, files, allowedTypes, maxBytes, }) => __awaiter(void 0, void 0, void 0, function* () {\r\n const firebase = yield loadFirebaseStorage();\r\n const filesToUpload = [...files].map(([id, dataURL]) => {\r\n const blob = (0,_data__WEBPACK_IMPORTED_MODULE_0__.dataURLToBlob)(dataURL);\r\n if (!allowedTypes.includes(blob.type)) {\r\n throw new Error(\"Disallowed file type.\");\r\n }\r\n if (blob.size > maxBytes) {\r\n throw new Error(`File cannot be larger than ${maxBytes / 1024} kB.`);\r\n }\r\n return { blob, id };\r\n });\r\n const erroredFiles = new Map();\r\n const savedFiles = new Map();\r\n yield Promise.all(filesToUpload.map(({ blob, id }) => __awaiter(void 0, void 0, void 0, function* () {\r\n const encryptedData = yield (0,_data__WEBPACK_IMPORTED_MODULE_0__.encryptData)(decryptionKey, blob);\r\n try {\r\n yield firebase\r\n .storage()\r\n .ref(`${prefix}/${id}`)\r\n .put(new Blob([encryptedData.iv, encryptedData.blob], {\r\n type: blob.type,\r\n }), {\r\n cacheControl: `public, max-age=${_app_constants__WEBPACK_IMPORTED_MODULE_3__.FILE_CACHE_MAX_AGE_SEC}`,\r\n customMetadata: {\r\n data: JSON.stringify({\r\n version: 1,\r\n filename: id,\r\n type: blob.type,\r\n }),\r\n created: Date.now().toString(),\r\n },\r\n });\r\n savedFiles.set(id, true);\r\n }\r\n catch (error) {\r\n erroredFiles.set(id, true);\r\n }\r\n })));\r\n return { savedFiles, erroredFiles };\r\n});\r\nconst decryptData = (iv, encrypted, privateKey) => __awaiter(void 0, void 0, void 0, function* () {\r\n const key = yield (0,_data__WEBPACK_IMPORTED_MODULE_0__.getImportedKey)(privateKey, \"decrypt\");\r\n return window.crypto.subtle.decrypt({\r\n name: \"AES-GCM\",\r\n iv,\r\n }, key, encrypted);\r\n});\r\nconst saveToFirebase = (portal, elements) => __awaiter(void 0, void 0, void 0, function* () {\r\n const { roomId, roomKey, socket } = portal;\r\n if (\r\n // if no room exists, consider the room saved because there's nothing we can\r\n // do at this point\r\n !roomId ||\r\n !roomKey ||\r\n !socket ||\r\n isSavedToFirebase(portal, elements)) {\r\n return true;\r\n }\r\n const firebase = yield loadFirestore();\r\n const sceneVersion = (0,_element__WEBPACK_IMPORTED_MODULE_1__.getSceneVersion)(elements);\r\n const { ciphertext, iv } = yield encryptElements(roomKey, elements);\r\n const nextDocData = {\r\n sceneVersion,\r\n ciphertext: firebase.firestore.Blob.fromUint8Array(new Uint8Array(ciphertext)),\r\n iv: firebase.firestore.Blob.fromUint8Array(iv),\r\n };\r\n const db = firebase.firestore();\r\n const docRef = db.collection(\"scenes\").doc(roomId);\r\n const didUpdate = yield db.runTransaction((transaction) => __awaiter(void 0, void 0, void 0, function* () {\r\n const doc = yield transaction.get(docRef);\r\n if (!doc.exists) {\r\n transaction.set(docRef, nextDocData);\r\n return true;\r\n }\r\n const prevDocData = doc.data();\r\n if (prevDocData.sceneVersion >= nextDocData.sceneVersion) {\r\n return false;\r\n }\r\n transaction.update(docRef, nextDocData);\r\n return true;\r\n }));\r\n if (didUpdate) {\r\n firebaseSceneVersionCache.set(socket, sceneVersion);\r\n }\r\n return didUpdate;\r\n});\r\nconst loadFromFirebase = (roomId, roomKey, socket) => __awaiter(void 0, void 0, void 0, function* () {\r\n const firebase = yield loadFirestore();\r\n const db = firebase.firestore();\r\n const docRef = db.collection(\"scenes\").doc(roomId);\r\n const doc = yield docRef.get();\r\n if (!doc.exists) {\r\n return null;\r\n }\r\n const storedScene = doc.data();\r\n const ciphertext = storedScene.ciphertext.toUint8Array();\r\n const iv = storedScene.iv.toUint8Array();\r\n const elements = yield decryptElements(roomKey, iv, ciphertext);\r\n if (socket) {\r\n firebaseSceneVersionCache.set(socket, (0,_element__WEBPACK_IMPORTED_MODULE_1__.getSceneVersion)(elements));\r\n }\r\n return (0,_data_restore__WEBPACK_IMPORTED_MODULE_2__.restoreElements)(elements, null);\r\n});\r\nconst loadFilesFromFirebase = (prefix, decryptionKey, filesIds) => __awaiter(void 0, void 0, void 0, function* () {\r\n const loadedFiles = [];\r\n const erroredFiles = [];\r\n yield Promise.all([...new Set(filesIds)].map((id) => __awaiter(void 0, void 0, void 0, function* () {\r\n try {\r\n const url = `https://firebasestorage.googleapis.com/v0/b/${FIREBASE_CONFIG.storageBucket}/o/${encodeURIComponent(prefix.replace(/^\\//, \"\"))}%2F${id}`;\r\n const response = yield fetch(`${url}?alt=media`);\r\n if (response.status < 400) {\r\n const contentType = response.headers.get(\"content-type\") || \"image/png\";\r\n const arrayBuffer = yield response.arrayBuffer();\r\n const KEY_LENGTH = 12;\r\n const iv = arrayBuffer.slice(0, KEY_LENGTH);\r\n const encrypted = arrayBuffer.slice(KEY_LENGTH, arrayBuffer.byteLength);\r\n const decrypted = yield decryptData(iv, encrypted, decryptionKey);\r\n const dataURL = yield (0,_data__WEBPACK_IMPORTED_MODULE_0__.arrayBufferToDataURL)(decrypted, contentType);\r\n loadedFiles.push({\r\n type: contentType.includes(\"image/\") ? \"image\" : \"other\",\r\n id,\r\n dataURL,\r\n });\r\n }\r\n }\r\n catch (error) {\r\n erroredFiles.push(id);\r\n console.error(error);\r\n }\r\n })));\r\n return { loadedFiles, erroredFiles };\r\n});\r\n//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"../../excalidraw-app/data/firebase.ts.js","mappings":"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","sources":["webpack:///../../excalidraw-app/data/firebase.ts?913f"],"sourcesContent":["import {\n  arrayBufferToDataURL,\n  dataURLToBlob,\n  encryptData,\n  getImportedKey,\n} from \"../data\";\nimport { createIV } from \"./index\";\nimport { ExcalidrawElement, ImageId } from \"../../element/types\";\nimport { getSceneVersion } from \"../../element\";\nimport Portal from \"../collab/Portal\";\nimport { restoreElements } from \"../../data/restore\";\nimport { BinaryFileData, DataURL } from \"../../types\";\nimport { FILE_CACHE_MAX_AGE_SEC } from \"../app_constants\";\n\n// private\n// -----------------------------------------------------------------------------\n\nconst FIREBASE_CONFIG = JSON.parse(process.env.REACT_APP_FIREBASE_CONFIG);\n\nlet firebasePromise: Promise<\n  typeof import(\"firebase/app\").default\n> | null = null;\nlet firestorePromise: Promise<any> | null | true = null;\nlet firebseStoragePromise: Promise<any> | null | true = null;\n\nlet isFirebaseInitialized = false;\n\nconst _loadFirebase = async () => {\n  const firebase = (\n    await import(/* webpackChunkName: \"firebase\" */ \"firebase/app\")\n  ).default;\n\n  // due to dev HMR\n  if (!isFirebaseInitialized) {\n    isFirebaseInitialized = true;\n    try {\n      firebase.initializeApp(FIREBASE_CONFIG);\n    } catch (error) {\n      console.warn(error.name, error.code);\n    }\n  }\n\n  return firebase;\n};\n\nconst _getFirebase = async (): Promise<\n  typeof import(\"firebase/app\").default\n> => {\n  if (!firebasePromise) {\n    firebasePromise = _loadFirebase();\n  }\n  return firebasePromise;\n};\n\n// -----------------------------------------------------------------------------\n\nconst loadFirestore = async () => {\n  const firebase = await _getFirebase();\n  if (!firestorePromise) {\n    firestorePromise = import(\n      /* webpackChunkName: \"firestore\" */ \"firebase/firestore\"\n    );\n  }\n  if (firestorePromise !== true) {\n    await firestorePromise;\n    firestorePromise = true;\n  }\n  return firebase;\n};\n\nexport const loadFirebaseStorage = async () => {\n  const firebase = await _getFirebase();\n  if (!firebseStoragePromise) {\n    firebseStoragePromise = import(\n      /* webpackChunkName: \"storage\" */ \"firebase/storage\"\n    );\n  }\n  if (firebseStoragePromise !== true) {\n    await firebseStoragePromise;\n    firebseStoragePromise = true;\n  }\n  return firebase;\n};\n\ninterface FirebaseStoredScene {\n  sceneVersion: number;\n  iv: firebase.default.firestore.Blob;\n  ciphertext: firebase.default.firestore.Blob;\n}\n\nconst encryptElements = async (\n  key: string,\n  elements: readonly ExcalidrawElement[],\n): Promise<{ ciphertext: ArrayBuffer; iv: Uint8Array }> => {\n  const importedKey = await getImportedKey(key, \"encrypt\");\n  const iv = createIV();\n  const json = JSON.stringify(elements);\n  const encoded = new TextEncoder().encode(json);\n  const ciphertext = await window.crypto.subtle.encrypt(\n    {\n      name: \"AES-GCM\",\n      iv,\n    },\n    importedKey,\n    encoded,\n  );\n\n  return { ciphertext, iv };\n};\n\nconst decryptElements = async (\n  key: string,\n  iv: Uint8Array,\n  ciphertext: ArrayBuffer,\n): Promise<readonly ExcalidrawElement[]> => {\n  const importedKey = await getImportedKey(key, \"decrypt\");\n  const decrypted = await window.crypto.subtle.decrypt(\n    {\n      name: \"AES-GCM\",\n      iv,\n    },\n    importedKey,\n    ciphertext,\n  );\n\n  const decodedData = new TextDecoder(\"utf-8\").decode(\n    new Uint8Array(decrypted) as any,\n  );\n  return JSON.parse(decodedData);\n};\n\nconst firebaseSceneVersionCache = new WeakMap<SocketIOClient.Socket, number>();\n\nexport const isSavedToFirebase = (\n  portal: Portal,\n  elements: readonly ExcalidrawElement[],\n): boolean => {\n  if (portal.socket && portal.roomId && portal.roomKey) {\n    const sceneVersion = getSceneVersion(elements);\n\n    return firebaseSceneVersionCache.get(portal.socket) === sceneVersion;\n  }\n  // if no room exists, consider the room saved so that we don't unnecessarily\n  // prevent unload (there's nothing we could do at that point anyway)\n  return true;\n};\n\nexport const saveFilesToFirebase = async ({\n  prefix,\n  decryptionKey,\n  files,\n  allowedTypes,\n  maxBytes,\n}: {\n  prefix: string;\n  decryptionKey: string;\n  files: Map<ImageId, DataURL>;\n  allowedTypes: string[];\n  maxBytes: number;\n}) => {\n  const firebase = await loadFirebaseStorage();\n  const filesToUpload = [...files].map(([id, dataURL]) => {\n    const blob = dataURLToBlob(dataURL);\n\n    if (!allowedTypes.includes(blob.type)) {\n      throw new Error(\"Disallowed file type.\");\n    }\n\n    if (blob.size > maxBytes) {\n      throw new Error(`File cannot be larger than ${maxBytes / 1024} kB.`);\n    }\n\n    return { blob, id };\n  });\n\n  const erroredFiles = new Map<ImageId, true>();\n  const savedFiles = new Map<ImageId, true>();\n\n  await Promise.all(\n    filesToUpload.map(async ({ blob, id }) => {\n      const encryptedData = await encryptData(decryptionKey, blob);\n      try {\n        await firebase\n          .storage()\n          .ref(`${prefix}/${id}`)\n          .put(\n            new Blob([encryptedData.iv, encryptedData.blob], {\n              type: blob.type,\n            }),\n            {\n              cacheControl: `public, max-age=${FILE_CACHE_MAX_AGE_SEC}`,\n              customMetadata: {\n                data: JSON.stringify({\n                  version: 1,\n                  filename: id,\n                  type: blob.type,\n                }),\n                created: Date.now().toString(),\n              },\n            },\n          );\n        savedFiles.set(id, true);\n      } catch (error) {\n        erroredFiles.set(id, true);\n      }\n    }),\n  );\n\n  return { savedFiles, erroredFiles };\n};\n\nconst decryptData = async (\n  iv: ArrayBuffer,\n  encrypted: ArrayBuffer,\n  privateKey: string,\n): Promise<ArrayBuffer> => {\n  const key = await getImportedKey(privateKey, \"decrypt\");\n  return window.crypto.subtle.decrypt(\n    {\n      name: \"AES-GCM\",\n      iv,\n    },\n    key,\n    encrypted,\n  );\n};\n\nexport const saveToFirebase = async (\n  portal: Portal,\n  elements: readonly ExcalidrawElement[],\n) => {\n  const { roomId, roomKey, socket } = portal;\n  if (\n    // if no room exists, consider the room saved because there's nothing we can\n    // do at this point\n    !roomId ||\n    !roomKey ||\n    !socket ||\n    isSavedToFirebase(portal, elements)\n  ) {\n    return true;\n  }\n\n  const firebase = await loadFirestore();\n  const sceneVersion = getSceneVersion(elements);\n  const { ciphertext, iv } = await encryptElements(roomKey, elements);\n\n  const nextDocData = {\n    sceneVersion,\n    ciphertext: firebase.firestore.Blob.fromUint8Array(\n      new Uint8Array(ciphertext),\n    ),\n    iv: firebase.firestore.Blob.fromUint8Array(iv),\n  } as FirebaseStoredScene;\n\n  const db = firebase.firestore();\n  const docRef = db.collection(\"scenes\").doc(roomId);\n  const didUpdate = await db.runTransaction(async (transaction) => {\n    const doc = await transaction.get(docRef);\n    if (!doc.exists) {\n      transaction.set(docRef, nextDocData);\n      return true;\n    }\n\n    const prevDocData = doc.data() as FirebaseStoredScene;\n    if (prevDocData.sceneVersion >= nextDocData.sceneVersion) {\n      return false;\n    }\n\n    transaction.update(docRef, nextDocData);\n    return true;\n  });\n\n  if (didUpdate) {\n    firebaseSceneVersionCache.set(socket, sceneVersion);\n  }\n\n  return didUpdate;\n};\n\nexport const loadFromFirebase = async (\n  roomId: string,\n  roomKey: string,\n  socket: SocketIOClient.Socket | null,\n): Promise<readonly ExcalidrawElement[] | null> => {\n  const firebase = await loadFirestore();\n  const db = firebase.firestore();\n\n  const docRef = db.collection(\"scenes\").doc(roomId);\n  const doc = await docRef.get();\n  if (!doc.exists) {\n    return null;\n  }\n  const storedScene = doc.data() as FirebaseStoredScene;\n  const ciphertext = storedScene.ciphertext.toUint8Array();\n  const iv = storedScene.iv.toUint8Array();\n\n  const elements = await decryptElements(roomKey, iv, ciphertext);\n\n  if (socket) {\n    firebaseSceneVersionCache.set(socket, getSceneVersion(elements));\n  }\n\n  return restoreElements(elements, null);\n};\n\nexport const loadFilesFromFirebase = async (\n  prefix: string,\n  decryptionKey: string,\n  filesIds: readonly ImageId[],\n): Promise<{\n  loadedFiles: BinaryFileData[];\n  erroredFiles: ImageId[];\n}> => {\n  const loadedFiles: BinaryFileData[] = [];\n  const erroredFiles: ImageId[] = [];\n\n  await Promise.all(\n    [...new Set(filesIds)].map(async (id) => {\n      try {\n        const url = `https://firebasestorage.googleapis.com/v0/b/${\n          FIREBASE_CONFIG.storageBucket\n        }/o/${encodeURIComponent(prefix.replace(/^\\//, \"\"))}%2F${id}`;\n        const response = await fetch(`${url}?alt=media`);\n        if (response.status < 400) {\n          const contentType =\n            response.headers.get(\"content-type\") || \"image/png\";\n          const arrayBuffer = await response.arrayBuffer();\n\n          const KEY_LENGTH = 12;\n\n          const iv = arrayBuffer.slice(0, KEY_LENGTH);\n          const encrypted = arrayBuffer.slice(\n            KEY_LENGTH,\n            arrayBuffer.byteLength,\n          );\n\n          const decrypted = await decryptData(iv, encrypted, decryptionKey);\n\n          const dataURL = await arrayBufferToDataURL(decrypted, contentType);\n\n          loadedFiles.push({\n            type: contentType.includes(\"image/\") ? \"image\" : \"other\",\n            id,\n            dataURL,\n          });\n        }\n      } catch (error) {\n        erroredFiles.push(id);\n        console.error(error);\n      }\n    }),\n  );\n\n  return { loadedFiles, erroredFiles };\n};\n"],"names":[],"sourceRoot":""}\n//# sourceURL=webpack-internal:///../../excalidraw-app/data/firebase.ts\n");
|
|
2591
|
+
eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ \"loadFirebaseStorage\": () => (/* binding */ loadFirebaseStorage),\n/* harmony export */ \"isSavedToFirebase\": () => (/* binding */ isSavedToFirebase),\n/* harmony export */ \"saveFilesToFirebase\": () => (/* binding */ saveFilesToFirebase),\n/* harmony export */ \"saveToFirebase\": () => (/* binding */ saveToFirebase),\n/* harmony export */ \"loadFromFirebase\": () => (/* binding */ loadFromFirebase),\n/* harmony export */ \"loadFilesFromFirebase\": () => (/* binding */ loadFilesFromFirebase)\n/* harmony export */ });\n/* harmony import */ var _data__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./index */ \"../../excalidraw-app/data/index.ts\");\n/* harmony import */ var _element__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ../../element */ \"../../element/index.ts\");\n/* harmony import */ var _data_restore__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ../../data/restore */ \"../../data/restore.ts\");\n/* harmony import */ var _app_constants__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! ../app_constants */ \"../../excalidraw-app/app_constants.ts\");\nvar __awaiter = (undefined && undefined.__awaiter) || function (thisArg, _arguments, P, generator) {\r\n function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }\r\n return new (P || (P = Promise))(function (resolve, reject) {\r\n function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }\r\n function rejected(value) { try { step(generator[\"throw\"](value)); } catch (e) { reject(e); } }\r\n function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }\r\n step((generator = generator.apply(thisArg, _arguments || [])).next());\r\n });\r\n};\r\nvar _a;\r\n\r\n\r\n\r\n\r\n\r\n// private\r\n// -----------------------------------------------------------------------------\r\nconst FIREBASE_CONFIG = JSON.parse((_a = process.env.REACT_APP_FIREBASE_CONFIG) !== null && _a !== void 0 ? _a : \"{}\");\r\nlet firebasePromise = null;\r\nlet firestorePromise = null;\r\nlet firebseStoragePromise = null;\r\nlet isFirebaseInitialized = false;\r\nconst _loadFirebase = () => __awaiter(void 0, void 0, void 0, function* () {\r\n const firebase = (yield __webpack_require__.e(/*! import() | firebase */ \"vendor\").then(__webpack_require__.bind(__webpack_require__, /*! firebase/app */ \"../../../node_modules/firebase/app/dist/index.esm.js\"))).default;\r\n // due to dev HMR\r\n if (!isFirebaseInitialized) {\r\n isFirebaseInitialized = true;\r\n try {\r\n firebase.initializeApp(FIREBASE_CONFIG);\r\n }\r\n catch (error) {\r\n console.warn(error.name, error.code);\r\n }\r\n }\r\n return firebase;\r\n});\r\nconst _getFirebase = () => __awaiter(void 0, void 0, void 0, function* () {\r\n if (!firebasePromise) {\r\n firebasePromise = _loadFirebase();\r\n }\r\n return firebasePromise;\r\n});\r\n// -----------------------------------------------------------------------------\r\nconst loadFirestore = () => __awaiter(void 0, void 0, void 0, function* () {\r\n const firebase = yield _getFirebase();\r\n if (!firestorePromise) {\r\n firestorePromise = __webpack_require__.e(/*! import() | firestore */ \"vendor\").then(__webpack_require__.bind(__webpack_require__, /*! firebase/firestore */ \"../../../node_modules/firebase/firestore/dist/index.esm.js\"));\r\n }\r\n if (firestorePromise !== true) {\r\n yield firestorePromise;\r\n firestorePromise = true;\r\n }\r\n return firebase;\r\n});\r\nconst loadFirebaseStorage = () => __awaiter(void 0, void 0, void 0, function* () {\r\n const firebase = yield _getFirebase();\r\n if (!firebseStoragePromise) {\r\n firebseStoragePromise = __webpack_require__.e(/*! import() | storage */ \"vendor\").then(__webpack_require__.bind(__webpack_require__, /*! firebase/storage */ \"../../../node_modules/firebase/storage/dist/index.esm.js\"));\r\n }\r\n if (firebseStoragePromise !== true) {\r\n yield firebseStoragePromise;\r\n firebseStoragePromise = true;\r\n }\r\n return firebase;\r\n});\r\nconst encryptElements = (key, elements) => __awaiter(void 0, void 0, void 0, function* () {\r\n const importedKey = yield (0,_data__WEBPACK_IMPORTED_MODULE_0__.getImportedKey)(key, \"encrypt\");\r\n const iv = (0,_data__WEBPACK_IMPORTED_MODULE_0__.createIV)();\r\n const json = JSON.stringify(elements);\r\n const encoded = new TextEncoder().encode(json);\r\n const ciphertext = yield window.crypto.subtle.encrypt({\r\n name: \"AES-GCM\",\r\n iv,\r\n }, importedKey, encoded);\r\n return { ciphertext, iv };\r\n});\r\nconst decryptElements = (key, iv, ciphertext) => __awaiter(void 0, void 0, void 0, function* () {\r\n const importedKey = yield (0,_data__WEBPACK_IMPORTED_MODULE_0__.getImportedKey)(key, \"decrypt\");\r\n const decrypted = yield window.crypto.subtle.decrypt({\r\n name: \"AES-GCM\",\r\n iv,\r\n }, importedKey, ciphertext);\r\n const decodedData = new TextDecoder(\"utf-8\").decode(new Uint8Array(decrypted));\r\n return JSON.parse(decodedData);\r\n});\r\nconst firebaseSceneVersionCache = new WeakMap();\r\nconst isSavedToFirebase = (portal, elements) => {\r\n if (portal.socket && portal.roomId && portal.roomKey) {\r\n const sceneVersion = (0,_element__WEBPACK_IMPORTED_MODULE_1__.getSceneVersion)(elements);\r\n return firebaseSceneVersionCache.get(portal.socket) === sceneVersion;\r\n }\r\n // if no room exists, consider the room saved so that we don't unnecessarily\r\n // prevent unload (there's nothing we could do at that point anyway)\r\n return true;\r\n};\r\nconst saveFilesToFirebase = ({ prefix, decryptionKey, files, allowedTypes, maxBytes, }) => __awaiter(void 0, void 0, void 0, function* () {\r\n const firebase = yield loadFirebaseStorage();\r\n const filesToUpload = [...files].map(([id, dataURL]) => {\r\n const blob = (0,_data__WEBPACK_IMPORTED_MODULE_0__.dataURLToBlob)(dataURL);\r\n if (!allowedTypes.includes(blob.type)) {\r\n throw new Error(\"Disallowed file type.\");\r\n }\r\n if (blob.size > maxBytes) {\r\n throw new Error(`File cannot be larger than ${maxBytes / 1024} kB.`);\r\n }\r\n return { blob, id };\r\n });\r\n const erroredFiles = new Map();\r\n const savedFiles = new Map();\r\n yield Promise.all(filesToUpload.map(({ blob, id }) => __awaiter(void 0, void 0, void 0, function* () {\r\n const encryptedData = yield (0,_data__WEBPACK_IMPORTED_MODULE_0__.encryptData)(decryptionKey, blob);\r\n try {\r\n yield firebase\r\n .storage()\r\n .ref(`${prefix}/${id}`)\r\n .put(new Blob([encryptedData.iv, encryptedData.blob], {\r\n type: blob.type,\r\n }), {\r\n cacheControl: `public, max-age=${_app_constants__WEBPACK_IMPORTED_MODULE_3__.FILE_CACHE_MAX_AGE_SEC}`,\r\n customMetadata: {\r\n data: JSON.stringify({\r\n version: 1,\r\n filename: id,\r\n type: blob.type,\r\n }),\r\n created: Date.now().toString(),\r\n },\r\n });\r\n savedFiles.set(id, true);\r\n }\r\n catch (error) {\r\n erroredFiles.set(id, true);\r\n }\r\n })));\r\n return { savedFiles, erroredFiles };\r\n});\r\nconst decryptData = (iv, encrypted, privateKey) => __awaiter(void 0, void 0, void 0, function* () {\r\n const key = yield (0,_data__WEBPACK_IMPORTED_MODULE_0__.getImportedKey)(privateKey, \"decrypt\");\r\n return window.crypto.subtle.decrypt({\r\n name: \"AES-GCM\",\r\n iv,\r\n }, key, encrypted);\r\n});\r\nconst saveToFirebase = (portal, elements) => __awaiter(void 0, void 0, void 0, function* () {\r\n const { roomId, roomKey, socket } = portal;\r\n if (\r\n // if no room exists, consider the room saved because there's nothing we can\r\n // do at this point\r\n !roomId ||\r\n !roomKey ||\r\n !socket ||\r\n isSavedToFirebase(portal, elements)) {\r\n return true;\r\n }\r\n const firebase = yield loadFirestore();\r\n const sceneVersion = (0,_element__WEBPACK_IMPORTED_MODULE_1__.getSceneVersion)(elements);\r\n const { ciphertext, iv } = yield encryptElements(roomKey, elements);\r\n const nextDocData = {\r\n sceneVersion,\r\n ciphertext: firebase.firestore.Blob.fromUint8Array(new Uint8Array(ciphertext)),\r\n iv: firebase.firestore.Blob.fromUint8Array(iv),\r\n };\r\n const db = firebase.firestore();\r\n const docRef = db.collection(\"scenes\").doc(roomId);\r\n const didUpdate = yield db.runTransaction((transaction) => __awaiter(void 0, void 0, void 0, function* () {\r\n const doc = yield transaction.get(docRef);\r\n if (!doc.exists) {\r\n transaction.set(docRef, nextDocData);\r\n return true;\r\n }\r\n const prevDocData = doc.data();\r\n if (prevDocData.sceneVersion >= nextDocData.sceneVersion) {\r\n return false;\r\n }\r\n transaction.update(docRef, nextDocData);\r\n return true;\r\n }));\r\n if (didUpdate) {\r\n firebaseSceneVersionCache.set(socket, sceneVersion);\r\n }\r\n return didUpdate;\r\n});\r\nconst loadFromFirebase = (roomId, roomKey, socket) => __awaiter(void 0, void 0, void 0, function* () {\r\n const firebase = yield loadFirestore();\r\n const db = firebase.firestore();\r\n const docRef = db.collection(\"scenes\").doc(roomId);\r\n const doc = yield docRef.get();\r\n if (!doc.exists) {\r\n return null;\r\n }\r\n const storedScene = doc.data();\r\n const ciphertext = storedScene.ciphertext.toUint8Array();\r\n const iv = storedScene.iv.toUint8Array();\r\n const elements = yield decryptElements(roomKey, iv, ciphertext);\r\n if (socket) {\r\n firebaseSceneVersionCache.set(socket, (0,_element__WEBPACK_IMPORTED_MODULE_1__.getSceneVersion)(elements));\r\n }\r\n return (0,_data_restore__WEBPACK_IMPORTED_MODULE_2__.restoreElements)(elements, null);\r\n});\r\nconst loadFilesFromFirebase = (prefix, decryptionKey, filesIds) => __awaiter(void 0, void 0, void 0, function* () {\r\n const loadedFiles = [];\r\n const erroredFiles = [];\r\n yield Promise.all([...new Set(filesIds)].map((id) => __awaiter(void 0, void 0, void 0, function* () {\r\n try {\r\n const url = `https://firebasestorage.googleapis.com/v0/b/${FIREBASE_CONFIG.storageBucket}/o/${encodeURIComponent(prefix.replace(/^\\//, \"\"))}%2F${id}`;\r\n const response = yield fetch(`${url}?alt=media`);\r\n if (response.status < 400) {\r\n const contentType = response.headers.get(\"content-type\") || \"image/png\";\r\n const arrayBuffer = yield response.arrayBuffer();\r\n const KEY_LENGTH = 12;\r\n const iv = arrayBuffer.slice(0, KEY_LENGTH);\r\n const encrypted = arrayBuffer.slice(KEY_LENGTH, arrayBuffer.byteLength);\r\n const decrypted = yield decryptData(iv, encrypted, decryptionKey);\r\n const dataURL = yield (0,_data__WEBPACK_IMPORTED_MODULE_0__.arrayBufferToDataURL)(decrypted, contentType);\r\n loadedFiles.push({\r\n type: contentType.includes(\"image/\") ? \"image\" : \"other\",\r\n id,\r\n dataURL,\r\n });\r\n }\r\n }\r\n catch (error) {\r\n erroredFiles.push(id);\r\n console.error(error);\r\n }\r\n })));\r\n return { loadedFiles, erroredFiles };\r\n});\r\n//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"../../excalidraw-app/data/firebase.ts.js","mappings":"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","sources":["webpack:///../../excalidraw-app/data/firebase.ts?913f"],"sourcesContent":["import {\n  arrayBufferToDataURL,\n  dataURLToBlob,\n  encryptData,\n  getImportedKey,\n} from \"../data\";\nimport { createIV } from \"./index\";\nimport { ExcalidrawElement, ImageId } from \"../../element/types\";\nimport { getSceneVersion } from \"../../element\";\nimport Portal from \"../collab/Portal\";\nimport { restoreElements } from \"../../data/restore\";\nimport { BinaryFileData, DataURL } from \"../../types\";\nimport { FILE_CACHE_MAX_AGE_SEC } from \"../app_constants\";\n\n// private\n// -----------------------------------------------------------------------------\n\nconst FIREBASE_CONFIG = JSON.parse(process.env.REACT_APP_FIREBASE_CONFIG??\"{}\");\n\nlet firebasePromise: Promise<\n  typeof import(\"firebase/app\").default\n> | null = null;\nlet firestorePromise: Promise<any> | null | true = null;\nlet firebseStoragePromise: Promise<any> | null | true = null;\n\nlet isFirebaseInitialized = false;\n\nconst _loadFirebase = async () => {\n  const firebase = (\n    await import(/* webpackChunkName: \"firebase\" */ \"firebase/app\")\n  ).default;\n\n  // due to dev HMR\n  if (!isFirebaseInitialized) {\n    isFirebaseInitialized = true;\n    try {\n      firebase.initializeApp(FIREBASE_CONFIG);\n    } catch (error) {\n      console.warn(error.name, error.code);\n    }\n  }\n\n  return firebase;\n};\n\nconst _getFirebase = async (): Promise<\n  typeof import(\"firebase/app\").default\n> => {\n  if (!firebasePromise) {\n    firebasePromise = _loadFirebase();\n  }\n  return firebasePromise;\n};\n\n// -----------------------------------------------------------------------------\n\nconst loadFirestore = async () => {\n  const firebase = await _getFirebase();\n  if (!firestorePromise) {\n    firestorePromise = import(\n      /* webpackChunkName: \"firestore\" */ \"firebase/firestore\"\n    );\n  }\n  if (firestorePromise !== true) {\n    await firestorePromise;\n    firestorePromise = true;\n  }\n  return firebase;\n};\n\nexport const loadFirebaseStorage = async () => {\n  const firebase = await _getFirebase();\n  if (!firebseStoragePromise) {\n    firebseStoragePromise = import(\n      /* webpackChunkName: \"storage\" */ \"firebase/storage\"\n    );\n  }\n  if (firebseStoragePromise !== true) {\n    await firebseStoragePromise;\n    firebseStoragePromise = true;\n  }\n  return firebase;\n};\n\ninterface FirebaseStoredScene {\n  sceneVersion: number;\n  iv: firebase.default.firestore.Blob;\n  ciphertext: firebase.default.firestore.Blob;\n}\n\nconst encryptElements = async (\n  key: string,\n  elements: readonly ExcalidrawElement[],\n): Promise<{ ciphertext: ArrayBuffer; iv: Uint8Array }> => {\n  const importedKey = await getImportedKey(key, \"encrypt\");\n  const iv = createIV();\n  const json = JSON.stringify(elements);\n  const encoded = new TextEncoder().encode(json);\n  const ciphertext = await window.crypto.subtle.encrypt(\n    {\n      name: \"AES-GCM\",\n      iv,\n    },\n    importedKey,\n    encoded,\n  );\n\n  return { ciphertext, iv };\n};\n\nconst decryptElements = async (\n  key: string,\n  iv: Uint8Array,\n  ciphertext: ArrayBuffer,\n): Promise<readonly ExcalidrawElement[]> => {\n  const importedKey = await getImportedKey(key, \"decrypt\");\n  const decrypted = await window.crypto.subtle.decrypt(\n    {\n      name: \"AES-GCM\",\n      iv,\n    },\n    importedKey,\n    ciphertext,\n  );\n\n  const decodedData = new TextDecoder(\"utf-8\").decode(\n    new Uint8Array(decrypted) as any,\n  );\n  return JSON.parse(decodedData);\n};\n\nconst firebaseSceneVersionCache = new WeakMap<SocketIOClient.Socket, number>();\n\nexport const isSavedToFirebase = (\n  portal: Portal,\n  elements: readonly ExcalidrawElement[],\n): boolean => {\n  if (portal.socket && portal.roomId && portal.roomKey) {\n    const sceneVersion = getSceneVersion(elements);\n\n    return firebaseSceneVersionCache.get(portal.socket) === sceneVersion;\n  }\n  // if no room exists, consider the room saved so that we don't unnecessarily\n  // prevent unload (there's nothing we could do at that point anyway)\n  return true;\n};\n\nexport const saveFilesToFirebase = async ({\n  prefix,\n  decryptionKey,\n  files,\n  allowedTypes,\n  maxBytes,\n}: {\n  prefix: string;\n  decryptionKey: string;\n  files: Map<ImageId, DataURL>;\n  allowedTypes: string[];\n  maxBytes: number;\n}) => {\n  const firebase = await loadFirebaseStorage();\n  const filesToUpload = [...files].map(([id, dataURL]) => {\n    const blob = dataURLToBlob(dataURL);\n\n    if (!allowedTypes.includes(blob.type)) {\n      throw new Error(\"Disallowed file type.\");\n    }\n\n    if (blob.size > maxBytes) {\n      throw new Error(`File cannot be larger than ${maxBytes / 1024} kB.`);\n    }\n\n    return { blob, id };\n  });\n\n  const erroredFiles = new Map<ImageId, true>();\n  const savedFiles = new Map<ImageId, true>();\n\n  await Promise.all(\n    filesToUpload.map(async ({ blob, id }) => {\n      const encryptedData = await encryptData(decryptionKey, blob);\n      try {\n        await firebase\n          .storage()\n          .ref(`${prefix}/${id}`)\n          .put(\n            new Blob([encryptedData.iv, encryptedData.blob], {\n              type: blob.type,\n            }),\n            {\n              cacheControl: `public, max-age=${FILE_CACHE_MAX_AGE_SEC}`,\n              customMetadata: {\n                data: JSON.stringify({\n                  version: 1,\n                  filename: id,\n                  type: blob.type,\n                }),\n                created: Date.now().toString(),\n              },\n            },\n          );\n        savedFiles.set(id, true);\n      } catch (error) {\n        erroredFiles.set(id, true);\n      }\n    }),\n  );\n\n  return { savedFiles, erroredFiles };\n};\n\nconst decryptData = async (\n  iv: ArrayBuffer,\n  encrypted: ArrayBuffer,\n  privateKey: string,\n): Promise<ArrayBuffer> => {\n  const key = await getImportedKey(privateKey, \"decrypt\");\n  return window.crypto.subtle.decrypt(\n    {\n      name: \"AES-GCM\",\n      iv,\n    },\n    key,\n    encrypted,\n  );\n};\n\nexport const saveToFirebase = async (\n  portal: Portal,\n  elements: readonly ExcalidrawElement[],\n) => {\n  const { roomId, roomKey, socket } = portal;\n  if (\n    // if no room exists, consider the room saved because there's nothing we can\n    // do at this point\n    !roomId ||\n    !roomKey ||\n    !socket ||\n    isSavedToFirebase(portal, elements)\n  ) {\n    return true;\n  }\n\n  const firebase = await loadFirestore();\n  const sceneVersion = getSceneVersion(elements);\n  const { ciphertext, iv } = await encryptElements(roomKey, elements);\n\n  const nextDocData = {\n    sceneVersion,\n    ciphertext: firebase.firestore.Blob.fromUint8Array(\n      new Uint8Array(ciphertext),\n    ),\n    iv: firebase.firestore.Blob.fromUint8Array(iv),\n  } as FirebaseStoredScene;\n\n  const db = firebase.firestore();\n  const docRef = db.collection(\"scenes\").doc(roomId);\n  const didUpdate = await db.runTransaction(async (transaction) => {\n    const doc = await transaction.get(docRef);\n    if (!doc.exists) {\n      transaction.set(docRef, nextDocData);\n      return true;\n    }\n\n    const prevDocData = doc.data() as FirebaseStoredScene;\n    if (prevDocData.sceneVersion >= nextDocData.sceneVersion) {\n      return false;\n    }\n\n    transaction.update(docRef, nextDocData);\n    return true;\n  });\n\n  if (didUpdate) {\n    firebaseSceneVersionCache.set(socket, sceneVersion);\n  }\n\n  return didUpdate;\n};\n\nexport const loadFromFirebase = async (\n  roomId: string,\n  roomKey: string,\n  socket: SocketIOClient.Socket | null,\n): Promise<readonly ExcalidrawElement[] | null> => {\n  const firebase = await loadFirestore();\n  const db = firebase.firestore();\n\n  const docRef = db.collection(\"scenes\").doc(roomId);\n  const doc = await docRef.get();\n  if (!doc.exists) {\n    return null;\n  }\n  const storedScene = doc.data() as FirebaseStoredScene;\n  const ciphertext = storedScene.ciphertext.toUint8Array();\n  const iv = storedScene.iv.toUint8Array();\n\n  const elements = await decryptElements(roomKey, iv, ciphertext);\n\n  if (socket) {\n    firebaseSceneVersionCache.set(socket, getSceneVersion(elements));\n  }\n\n  return restoreElements(elements, null);\n};\n\nexport const loadFilesFromFirebase = async (\n  prefix: string,\n  decryptionKey: string,\n  filesIds: readonly ImageId[],\n): Promise<{\n  loadedFiles: BinaryFileData[];\n  erroredFiles: ImageId[];\n}> => {\n  const loadedFiles: BinaryFileData[] = [];\n  const erroredFiles: ImageId[] = [];\n\n  await Promise.all(\n    [...new Set(filesIds)].map(async (id) => {\n      try {\n        const url = `https://firebasestorage.googleapis.com/v0/b/${\n          FIREBASE_CONFIG.storageBucket\n        }/o/${encodeURIComponent(prefix.replace(/^\\//, \"\"))}%2F${id}`;\n        const response = await fetch(`${url}?alt=media`);\n        if (response.status < 400) {\n          const contentType =\n            response.headers.get(\"content-type\") || \"image/png\";\n          const arrayBuffer = await response.arrayBuffer();\n\n          const KEY_LENGTH = 12;\n\n          const iv = arrayBuffer.slice(0, KEY_LENGTH);\n          const encrypted = arrayBuffer.slice(\n            KEY_LENGTH,\n            arrayBuffer.byteLength,\n          );\n\n          const decrypted = await decryptData(iv, encrypted, decryptionKey);\n\n          const dataURL = await arrayBufferToDataURL(decrypted, contentType);\n\n          loadedFiles.push({\n            type: contentType.includes(\"image/\") ? \"image\" : \"other\",\n            id,\n            dataURL,\n          });\n        }\n      } catch (error) {\n        erroredFiles.push(id);\n        console.error(error);\n      }\n    }),\n  );\n\n  return { loadedFiles, erroredFiles };\n};\n"],"names":[],"sourceRoot":""}\n//# sourceURL=webpack-internal:///../../excalidraw-app/data/firebase.ts\n");
|
|
2592
2592
|
|
|
2593
2593
|
/***/ }),
|
|
2594
2594
|
|
|
@@ -3105,7 +3105,7 @@ module.exports = JSON.parse('{"ar-SA":100,"bg-BG":68,"ca-ES":84,"cs-CZ":30,"da-D
|
|
|
3105
3105
|
/***/ ((module) => {
|
|
3106
3106
|
|
|
3107
3107
|
"use strict";
|
|
3108
|
-
module.exports = JSON.parse('{"name":"@zsviczian/excalidraw","version":"0.9.0-obsidian-image-support-
|
|
3108
|
+
module.exports = JSON.parse('{"name":"@zsviczian/excalidraw","version":"0.9.0-obsidian-image-support-2","main":"main.js","types":"types/packages/excalidraw/index.d.ts","files":["dist/*","types/*"],"publishConfig":{"access":"public"},"description":"Excalidraw as a React component","repository":"https://github.com/excalidraw/excalidraw","license":"MIT","keywords":["excalidraw","excalidraw-embed","react","npm","npm excalidraw"],"browserslist":{"production":[">0.2%","not dead","not ie <= 11","not op_mini all","not safari < 12","not kaios <= 2.5","not edge < 79","not chrome < 70","not and_uc < 13","not samsung < 10"],"development":["last 1 chrome version","last 1 firefox version","last 1 safari version"]},"peerDependencies":{"react":"^17.0.2","react-dom":"^17.0.2"},"devDependencies":{"@babel/core":"7.14.8","@babel/plugin-transform-arrow-functions":"7.14.5","@babel/plugin-transform-async-to-generator":"7.14.5","@babel/plugin-transform-runtime":"7.14.5","@babel/plugin-transform-typescript":"7.14.6","@babel/preset-env":"7.14.9","@babel/preset-react":"7.14.5","@babel/preset-typescript":"7.14.5","autoprefixer":"10.3.1","babel-loader":"8.2.2","babel-plugin-transform-class-properties":"6.24.1","cross-env":"7.0.3","css-loader":"5.2.6","file-loader":"6.2.0","mini-css-extract-plugin":"1.6.1","postcss-loader":"6.1.1","sass-loader":"12.1.0","terser-webpack-plugin":"5.1.4","ts-loader":"9.2.4","typescript":"4.3.5","webpack":"5.50.0","webpack-bundle-analyzer":"4.4.2","webpack-cli":"4.7.2"},"bugs":"https://github.com/excalidraw/excalidraw/issues","homepage":"https://github.com/excalidraw/excalidraw/tree/master/src/packages/excalidraw","scripts":{"gen:types":"tsc --project ../../../tsconfig-types.json","build:umd":"cross-env NODE_ENV=production webpack --config webpack.prod.config.js && cross-env NODE_ENV=development webpack --config webpack.dev.config.js && yarn gen:types","build:umd:withAnalyzer":"cross-env NODE_ENV=production ANALYZER=true webpack --config webpack.prod.config.js","pack":"yarn build:umd && yarn pack"}}');
|
|
3109
3109
|
|
|
3110
3110
|
/***/ })
|
|
3111
3111
|
|