UNPKG

matrix-react-sdk

Version:
167 lines (161 loc) 19.8 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); exports.checkConsistency = checkConsistency; exports.setCryptoInitialised = setCryptoInitialised; exports.tryPersistStorage = tryPersistStorage; var _matrix = require("matrix-js-sdk/src/matrix"); var _logger = require("matrix-js-sdk/src/logger"); var _StorageAccess = require("./StorageAccess"); /* Copyright 2024 New Vector Ltd. Copyright 2019-2021 The Matrix.org Foundation C.I.C. SPDX-License-Identifier: AGPL-3.0-only OR GPL-3.0-only Please see LICENSE files in the repository root for full details. */ const localStorage = window.localStorage; // The JS SDK will add a prefix of "matrix-js-sdk:" to the sync store name. const SYNC_STORE_NAME = "riot-web-sync"; const LEGACY_CRYPTO_STORE_NAME = "matrix-js-sdk:crypto"; const RUST_CRYPTO_STORE_NAME = "matrix-js-sdk::matrix-sdk-crypto"; function log(msg) { _logger.logger.log(`StorageManager: ${msg}`); } function error(msg, ...args) { _logger.logger.error(`StorageManager: ${msg}`, ...args); } function tryPersistStorage() { if (navigator.storage && navigator.storage.persist) { navigator.storage.persist().then(persistent => { _logger.logger.log("StorageManager: Persistent?", persistent); }); } else if (document.requestStorageAccess) { // Safari document.requestStorageAccess().then(() => _logger.logger.log("StorageManager: Persistent?", true), () => _logger.logger.log("StorageManager: Persistent?", false)); } else { _logger.logger.log("StorageManager: Persistence unsupported"); } } async function checkConsistency() { log("Checking storage consistency"); log(`Local storage supported? ${!!localStorage}`); log(`IndexedDB supported? ${!!(0, _StorageAccess.getIDBFactory)()}`); let dataInLocalStorage = false; let dataInCryptoStore = false; let cryptoInited = false; let healthy = true; if (localStorage) { dataInLocalStorage = localStorage.length > 0; log(`Local storage contains data? ${dataInLocalStorage}`); cryptoInited = !!localStorage.getItem("mx_crypto_initialised"); log(`Crypto initialised? ${cryptoInited}`); } else { healthy = false; error("Local storage cannot be used on this browser"); } if ((0, _StorageAccess.getIDBFactory)() && localStorage) { const results = await checkSyncStore(); if (!results.healthy) { healthy = false; } } else { healthy = false; error("Sync store cannot be used on this browser"); } if ((0, _StorageAccess.getIDBFactory)()) { const results = await checkCryptoStore(); dataInCryptoStore = results.exists; if (!results.healthy) { healthy = false; } } else { healthy = false; error("Crypto store cannot be used on this browser"); } if (dataInLocalStorage && cryptoInited && !dataInCryptoStore) { healthy = false; error("Data exists in local storage and crypto is marked as initialised " + " but no data found in crypto store. " + "IndexedDB storage has likely been evicted by the browser!"); } if (healthy) { log("Storage consistency checks passed"); } else { error("Storage consistency checks failed"); } return { dataInLocalStorage, dataInCryptoStore, cryptoInited, healthy }; } async function checkSyncStore() { let exists = false; try { exists = await _matrix.IndexedDBStore.exists((0, _StorageAccess.getIDBFactory)(), SYNC_STORE_NAME); log(`Sync store using IndexedDB contains data? ${exists}`); return { exists, healthy: true }; } catch (e) { error("Sync store using IndexedDB inaccessible", e); } log("Sync store using memory only"); return { exists, healthy: false }; } async function checkCryptoStore() { // check first if there is a rust crypto store try { const rustDbExists = await _matrix.IndexedDBCryptoStore.exists((0, _StorageAccess.getIDBFactory)(), RUST_CRYPTO_STORE_NAME); log(`Rust Crypto store using IndexedDB contains data? ${rustDbExists}`); if (rustDbExists) { // There was an existing rust database, so consider it healthy. return { exists: true, healthy: true }; } else { // No rust store, so let's check if there is a legacy store not yet migrated. try { const legacyIdbExists = await _matrix.IndexedDBCryptoStore.existsAndIsNotMigrated((0, _StorageAccess.getIDBFactory)(), LEGACY_CRYPTO_STORE_NAME); log(`Legacy Crypto store using IndexedDB contains non migrated data? ${legacyIdbExists}`); return { exists: legacyIdbExists, healthy: true }; } catch (e) { error("Legacy crypto store using IndexedDB inaccessible", e); } // No need to check local storage or memory as rust stack doesn't support them. // Given that rust stack requires indexeddb, set healthy to false. return { exists: false, healthy: false }; } } catch (e) { error("Rust crypto store using IndexedDB inaccessible", e); return { exists: false, healthy: false }; } } /** * Sets whether crypto has ever been successfully * initialised on this client. * StorageManager uses this to determine whether indexeddb * has been wiped by the browser: this flag is saved to localStorage * and if it is true and not crypto data is found, an error is * presented to the user. * * @param {boolean} cryptoInited True if crypto has been set up */ function setCryptoInitialised(cryptoInited) { localStorage.setItem("mx_crypto_initialised", String(cryptoInited)); } //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_matrix","require","_logger","_StorageAccess","localStorage","window","SYNC_STORE_NAME","LEGACY_CRYPTO_STORE_NAME","RUST_CRYPTO_STORE_NAME","log","msg","logger","error","args","tryPersistStorage","navigator","storage","persist","then","persistent","document","requestStorageAccess","checkConsistency","getIDBFactory","dataInLocalStorage","dataInCryptoStore","cryptoInited","healthy","length","getItem","results","checkSyncStore","checkCryptoStore","exists","IndexedDBStore","e","rustDbExists","IndexedDBCryptoStore","legacyIdbExists","existsAndIsNotMigrated","setCryptoInitialised","setItem","String"],"sources":["../../src/utils/StorageManager.ts"],"sourcesContent":["/*\nCopyright 2024 New Vector Ltd.\nCopyright 2019-2021 The Matrix.org Foundation C.I.C.\n\nSPDX-License-Identifier: AGPL-3.0-only OR GPL-3.0-only\nPlease see LICENSE files in the repository root for full details.\n*/\n\nimport { IndexedDBStore, IndexedDBCryptoStore } from \"matrix-js-sdk/src/matrix\";\nimport { logger } from \"matrix-js-sdk/src/logger\";\n\nimport { getIDBFactory } from \"./StorageAccess\";\n\nconst localStorage = window.localStorage;\n\n// The JS SDK will add a prefix of \"matrix-js-sdk:\" to the sync store name.\nconst SYNC_STORE_NAME = \"riot-web-sync\";\nconst LEGACY_CRYPTO_STORE_NAME = \"matrix-js-sdk:crypto\";\nconst RUST_CRYPTO_STORE_NAME = \"matrix-js-sdk::matrix-sdk-crypto\";\n\nfunction log(msg: string): void {\n    logger.log(`StorageManager: ${msg}`);\n}\n\nfunction error(msg: string, ...args: any[]): void {\n    logger.error(`StorageManager: ${msg}`, ...args);\n}\n\nexport function tryPersistStorage(): void {\n    if (navigator.storage && navigator.storage.persist) {\n        navigator.storage.persist().then((persistent) => {\n            logger.log(\"StorageManager: Persistent?\", persistent);\n        });\n    } else if (document.requestStorageAccess) {\n        // Safari\n        document.requestStorageAccess().then(\n            () => logger.log(\"StorageManager: Persistent?\", true),\n            () => logger.log(\"StorageManager: Persistent?\", false),\n        );\n    } else {\n        logger.log(\"StorageManager: Persistence unsupported\");\n    }\n}\n\nexport async function checkConsistency(): Promise<{\n    healthy: boolean;\n    cryptoInited: boolean;\n    dataInCryptoStore: boolean;\n    dataInLocalStorage: boolean;\n}> {\n    log(\"Checking storage consistency\");\n    log(`Local storage supported? ${!!localStorage}`);\n    log(`IndexedDB supported? ${!!getIDBFactory()}`);\n\n    let dataInLocalStorage = false;\n    let dataInCryptoStore = false;\n    let cryptoInited = false;\n    let healthy = true;\n\n    if (localStorage) {\n        dataInLocalStorage = localStorage.length > 0;\n        log(`Local storage contains data? ${dataInLocalStorage}`);\n\n        cryptoInited = !!localStorage.getItem(\"mx_crypto_initialised\");\n        log(`Crypto initialised? ${cryptoInited}`);\n    } else {\n        healthy = false;\n        error(\"Local storage cannot be used on this browser\");\n    }\n\n    if (getIDBFactory() && localStorage) {\n        const results = await checkSyncStore();\n        if (!results.healthy) {\n            healthy = false;\n        }\n    } else {\n        healthy = false;\n        error(\"Sync store cannot be used on this browser\");\n    }\n\n    if (getIDBFactory()) {\n        const results = await checkCryptoStore();\n        dataInCryptoStore = results.exists;\n        if (!results.healthy) {\n            healthy = false;\n        }\n    } else {\n        healthy = false;\n        error(\"Crypto store cannot be used on this browser\");\n    }\n\n    if (dataInLocalStorage && cryptoInited && !dataInCryptoStore) {\n        healthy = false;\n        error(\n            \"Data exists in local storage and crypto is marked as initialised \" +\n                \" but no data found in crypto store. \" +\n                \"IndexedDB storage has likely been evicted by the browser!\",\n        );\n    }\n\n    if (healthy) {\n        log(\"Storage consistency checks passed\");\n    } else {\n        error(\"Storage consistency checks failed\");\n    }\n\n    return {\n        dataInLocalStorage,\n        dataInCryptoStore,\n        cryptoInited,\n        healthy,\n    };\n}\n\ninterface StoreCheck {\n    exists: boolean;\n    healthy: boolean;\n}\n\nasync function checkSyncStore(): Promise<StoreCheck> {\n    let exists = false;\n    try {\n        exists = await IndexedDBStore.exists(getIDBFactory()!, SYNC_STORE_NAME);\n        log(`Sync store using IndexedDB contains data? ${exists}`);\n        return { exists, healthy: true };\n    } catch (e) {\n        error(\"Sync store using IndexedDB inaccessible\", e);\n    }\n    log(\"Sync store using memory only\");\n    return { exists, healthy: false };\n}\n\nasync function checkCryptoStore(): Promise<StoreCheck> {\n    // check first if there is a rust crypto store\n    try {\n        const rustDbExists = await IndexedDBCryptoStore.exists(getIDBFactory()!, RUST_CRYPTO_STORE_NAME);\n        log(`Rust Crypto store using IndexedDB contains data? ${rustDbExists}`);\n\n        if (rustDbExists) {\n            // There was an existing rust database, so consider it healthy.\n            return { exists: true, healthy: true };\n        } else {\n            // No rust store, so let's check if there is a legacy store not yet migrated.\n            try {\n                const legacyIdbExists = await IndexedDBCryptoStore.existsAndIsNotMigrated(\n                    getIDBFactory()!,\n                    LEGACY_CRYPTO_STORE_NAME,\n                );\n                log(`Legacy Crypto store using IndexedDB contains non migrated data? ${legacyIdbExists}`);\n                return { exists: legacyIdbExists, healthy: true };\n            } catch (e) {\n                error(\"Legacy crypto store using IndexedDB inaccessible\", e);\n            }\n\n            // No need to check local storage or memory as rust stack doesn't support them.\n            // Given that rust stack requires indexeddb, set healthy to false.\n            return { exists: false, healthy: false };\n        }\n    } catch (e) {\n        error(\"Rust crypto store using IndexedDB inaccessible\", e);\n        return { exists: false, healthy: false };\n    }\n}\n\n/**\n * Sets whether crypto has ever been successfully\n * initialised on this client.\n * StorageManager uses this to determine whether indexeddb\n * has been wiped by the browser: this flag is saved to localStorage\n * and if it is true and not crypto data is found, an error is\n * presented to the user.\n *\n * @param {boolean} cryptoInited True if crypto has been set up\n */\nexport function setCryptoInitialised(cryptoInited: boolean): void {\n    localStorage.setItem(\"mx_crypto_initialised\", String(cryptoInited));\n}\n"],"mappings":"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","ignoreList":[]}