UNPKG

matrix-react-sdk

Version:
273 lines (265 loc) 39.9 kB
"use strict"; var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault"); Object.defineProperty(exports, "__esModule", { value: true }); exports.AccessCancelledError = void 0; exports.accessSecretStorage = accessSecretStorage; exports.crossSigningCallbacks = void 0; exports.isSecretStorageBeingAccessed = isSecretStorageBeingAccessed; exports.withSecretStorageKeyCache = withSecretStorageKeyCache; var _cryptoApi = require("matrix-js-sdk/src/crypto-api"); var _logger = require("matrix-js-sdk/src/logger"); var _Modal = _interopRequireDefault(require("./Modal")); var _MatrixClientPeg = require("./MatrixClientPeg"); var _languageHandler = require("./languageHandler"); var _WellKnownUtils = require("./utils/WellKnownUtils"); var _AccessSecretStorageDialog = _interopRequireDefault(require("./components/views/dialogs/security/AccessSecretStorageDialog")); var _ModuleRunner = require("./modules/ModuleRunner"); var _QuestionDialog = _interopRequireDefault(require("./components/views/dialogs/QuestionDialog")); var _InteractiveAuthDialog = _interopRequireDefault(require("./components/views/dialogs/InteractiveAuthDialog")); function _getRequireWildcardCache(e) { if ("function" != typeof WeakMap) return null; var r = new WeakMap(), t = new WeakMap(); return (_getRequireWildcardCache = function (e) { return e ? t : r; })(e); } function _interopRequireWildcard(e, r) { if (!r && e && e.__esModule) return e; if (null === e || "object" != typeof e && "function" != typeof e) return { default: e }; var t = _getRequireWildcardCache(r); if (t && t.has(e)) return t.get(e); var n = { __proto__: null }, a = Object.defineProperty && Object.getOwnPropertyDescriptor; for (var u in e) if ("default" !== u && {}.hasOwnProperty.call(e, u)) { var i = a ? Object.getOwnPropertyDescriptor(e, u) : null; i && (i.get || i.set) ? Object.defineProperty(n, u, i) : n[u] = e[u]; } return n.default = e, t && t.set(e, n), n; } /* Copyright 2024 New Vector Ltd. Copyright 2019, 2020 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. */ // This stores the secret storage private keys in memory for the JS SDK. This is // only meant to act as a cache to avoid prompting the user multiple times // during the same single operation. Use `accessSecretStorage` below to scope a // single secret storage operation, as it will clear the cached keys once the // operation ends. let secretStorageKeys = {}; let secretStorageKeyInfo = {}; let secretStorageBeingAccessed = false; /** * This can be used by other components to check if secret storage access is in * progress, so that we can e.g. avoid intermittently showing toasts during * secret storage setup. * * @returns {bool} */ function isSecretStorageBeingAccessed() { return secretStorageBeingAccessed; } class AccessCancelledError extends Error { constructor() { super("Secret storage access canceled"); } } exports.AccessCancelledError = AccessCancelledError; async function confirmToDismiss() { const [sure] = await _Modal.default.createDialog(_QuestionDialog.default, { title: (0, _languageHandler._t)("encryption|cancel_entering_passphrase_title"), description: (0, _languageHandler._t)("encryption|cancel_entering_passphrase_description"), danger: false, button: (0, _languageHandler._t)("action|go_back"), cancelButton: (0, _languageHandler._t)("action|cancel") }).finished; return !sure; } function makeInputToKey(keyInfo) { return async ({ passphrase, recoveryKey }) => { if (passphrase) { return (0, _cryptoApi.deriveRecoveryKeyFromPassphrase)(passphrase, keyInfo.passphrase.salt, keyInfo.passphrase.iterations); } else if (recoveryKey) { return (0, _cryptoApi.decodeRecoveryKey)(recoveryKey); } throw new Error("Invalid input, passphrase or recoveryKey need to be provided"); }; } async function getSecretStorageKey({ keys: keyInfos }) { const cli = _MatrixClientPeg.MatrixClientPeg.safeGet(); let keyId = await cli.getDefaultSecretStorageKeyId(); let keyInfo; if (keyId) { // use the default SSSS key if set keyInfo = keyInfos[keyId]; if (!keyInfo) { // if the default key is not available, pretend the default key // isn't set keyId = null; } } if (!keyId) { // if no default SSSS key is set, fall back to a heuristic of using the // only available key, if only one key is set const keyInfoEntries = Object.entries(keyInfos); if (keyInfoEntries.length > 1) { throw new Error("Multiple storage key requests not implemented"); } [keyId, keyInfo] = keyInfoEntries[0]; } _logger.logger.debug(`getSecretStorageKey: request for 4S keys [${Object.keys(keyInfos)}]: looking for key ${keyId}`); // Check the in-memory cache if (secretStorageBeingAccessed && secretStorageKeys[keyId]) { _logger.logger.debug(`getSecretStorageKey: returning key ${keyId} from cache`); return [keyId, secretStorageKeys[keyId]]; } const keyFromCustomisations = _ModuleRunner.ModuleRunner.instance.extensions.cryptoSetup.getSecretStorageKey(); if (keyFromCustomisations) { _logger.logger.log("getSecretStorageKey: Using secret storage key from CryptoSetupExtension"); cacheSecretStorageKey(keyId, keyInfo, keyFromCustomisations); return [keyId, keyFromCustomisations]; } _logger.logger.debug("getSecretStorageKey: prompting user for key"); const inputToKey = makeInputToKey(keyInfo); const { finished } = _Modal.default.createDialog(_AccessSecretStorageDialog.default, /* props= */ { keyInfo, checkPrivateKey: async input => { const key = await inputToKey(input); return _MatrixClientPeg.MatrixClientPeg.safeGet().secretStorage.checkKey(key, keyInfo); } }, /* className= */undefined, /* isPriorityModal= */false, /* isStaticModal= */false, /* options= */{ onBeforeClose: async reason => { if (reason === "backgroundClick") { return confirmToDismiss(); } return true; } }); const [keyParams] = await finished; if (!keyParams) { throw new AccessCancelledError(); } _logger.logger.debug("getSecretStorageKey: got key from user"); const key = await inputToKey(keyParams); // Save to cache to avoid future prompts in the current session cacheSecretStorageKey(keyId, keyInfo, key); return [keyId, key]; } function cacheSecretStorageKey(keyId, keyInfo, key) { if (secretStorageBeingAccessed) { secretStorageKeys[keyId] = key; secretStorageKeyInfo[keyId] = keyInfo; } } const crossSigningCallbacks = exports.crossSigningCallbacks = { getSecretStorageKey, cacheSecretStorageKey }; /** * Carry out an operation that may require multiple accesses to secret storage, caching the key. * * Use this helper to wrap an operation that may require multiple accesses to secret storage; the user will be prompted * to enter the 4S key or passphrase on the first access, and the key will be cached for the rest of the operation. * * @param func - The operation to be wrapped. */ async function withSecretStorageKeyCache(func) { _logger.logger.debug("SecurityManager: enabling 4S key cache"); secretStorageBeingAccessed = true; try { return await func(); } finally { // Clear secret storage key cache now that work is complete _logger.logger.debug("SecurityManager: disabling 4S key cache"); secretStorageBeingAccessed = false; secretStorageKeys = {}; secretStorageKeyInfo = {}; } } /** * This helper should be used whenever you need to access secret storage. It * ensures that secret storage (and also cross-signing since they each depend on * each other in a cycle of sorts) have been bootstrapped before running the * provided function. * * Bootstrapping secret storage may take one of these paths: * 1. Create secret storage from a passphrase and store cross-signing keys * in secret storage. * 2. Access existing secret storage by requesting passphrase and accessing * cross-signing keys as needed. * 3. All keys are loaded and there's nothing to do. * * Additionally, the secret storage keys are cached during the scope of this function * to ensure the user is prompted only once for their secret storage * passphrase. The cache is then cleared once the provided function completes. * * @param {Function} [func] An operation to perform once secret storage has been * bootstrapped. Optional. * @param {bool} [forceReset] Reset secret storage even if it's already set up */ async function accessSecretStorage(func = async () => {}, forceReset = false) { await withSecretStorageKeyCache(() => doAccessSecretStorage(func, forceReset)); } /** Helper for {@link #accessSecretStorage} */ async function doAccessSecretStorage(func, forceReset) { try { const cli = _MatrixClientPeg.MatrixClientPeg.safeGet(); const crypto = cli.getCrypto(); if (!crypto) { throw new Error("End-to-end encryption is disabled - unable to access secret storage."); } let createNew = false; if (forceReset) { _logger.logger.debug("accessSecretStorage: resetting 4S"); createNew = true; } else if (!(await cli.secretStorage.hasKey())) { _logger.logger.debug("accessSecretStorage: no 4S key configured, creating a new one"); createNew = true; } if (createNew) { // This dialog calls bootstrap itself after guiding the user through // passphrase creation. const { finished } = _Modal.default.createDialogAsync(Promise.resolve().then(() => _interopRequireWildcard(require("./async-components/views/dialogs/security/CreateSecretStorageDialog"))), { forceReset }, undefined, /* priority = */false, /* static = */true, /* options = */{ onBeforeClose: async reason => { // If Secure Backup is required, you cannot leave the modal. if (reason === "backgroundClick") { return !(0, _WellKnownUtils.isSecureBackupRequired)(cli); } return true; } }); const [confirmed] = await finished; if (!confirmed) { throw new Error("Secret storage creation canceled"); } } else { _logger.logger.debug("accessSecretStorage: bootstrapCrossSigning"); await crypto.bootstrapCrossSigning({ authUploadDeviceSigningKeys: async makeRequest => { _logger.logger.debug("accessSecretStorage: performing UIA to upload cross-signing keys"); const { finished } = _Modal.default.createDialog(_InteractiveAuthDialog.default, { title: (0, _languageHandler._t)("encryption|bootstrap_title"), matrixClient: cli, makeRequest }); const [confirmed] = await finished; if (!confirmed) { throw new Error("Cross-signing key upload auth canceled"); } _logger.logger.debug("accessSecretStorage: Cross-signing key upload successful"); } }); _logger.logger.debug("accessSecretStorage: bootstrapSecretStorage"); await crypto.bootstrapSecretStorage({}); } _logger.logger.debug("accessSecretStorage: 4S now ready"); // `return await` needed here to ensure `finally` block runs after the // inner operation completes. await func(); _logger.logger.debug("accessSecretStorage: operation complete"); } catch (e) { _ModuleRunner.ModuleRunner.instance.extensions.cryptoSetup.catchAccessSecretStorageError(e); _logger.logger.error("accessSecretStorage: error during operation", e); // Re-throw so that higher level logic can abort as needed throw e; } } //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_cryptoApi","require","_logger","_Modal","_interopRequireDefault","_MatrixClientPeg","_languageHandler","_WellKnownUtils","_AccessSecretStorageDialog","_ModuleRunner","_QuestionDialog","_InteractiveAuthDialog","_getRequireWildcardCache","e","WeakMap","r","t","_interopRequireWildcard","__esModule","default","has","get","n","__proto__","a","Object","defineProperty","getOwnPropertyDescriptor","u","hasOwnProperty","call","i","set","secretStorageKeys","secretStorageKeyInfo","secretStorageBeingAccessed","isSecretStorageBeingAccessed","AccessCancelledError","Error","constructor","exports","confirmToDismiss","sure","Modal","createDialog","QuestionDialog","title","_t","description","danger","button","cancelButton","finished","makeInputToKey","keyInfo","passphrase","recoveryKey","deriveRecoveryKeyFromPassphrase","salt","iterations","decodeRecoveryKey","getSecretStorageKey","keys","keyInfos","cli","MatrixClientPeg","safeGet","keyId","getDefaultSecretStorageKeyId","keyInfoEntries","entries","length","logger","debug","keyFromCustomisations","ModuleRunner","instance","extensions","cryptoSetup","log","cacheSecretStorageKey","inputToKey","AccessSecretStorageDialog","checkPrivateKey","input","key","secretStorage","checkKey","undefined","onBeforeClose","reason","keyParams","crossSigningCallbacks","withSecretStorageKeyCache","func","accessSecretStorage","forceReset","doAccessSecretStorage","crypto","getCrypto","createNew","hasKey","createDialogAsync","Promise","resolve","then","isSecureBackupRequired","confirmed","bootstrapCrossSigning","authUploadDeviceSigningKeys","makeRequest","InteractiveAuthDialog","matrixClient","bootstrapSecretStorage","catchAccessSecretStorageError","error"],"sources":["../src/SecurityManager.ts"],"sourcesContent":["/*\nCopyright 2024 New Vector Ltd.\nCopyright 2019, 2020 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 { ICryptoCallbacks, SecretStorage } from \"matrix-js-sdk/src/matrix\";\nimport { deriveRecoveryKeyFromPassphrase, decodeRecoveryKey } from \"matrix-js-sdk/src/crypto-api\";\nimport { logger } from \"matrix-js-sdk/src/logger\";\n\nimport type CreateSecretStorageDialog from \"./async-components/views/dialogs/security/CreateSecretStorageDialog\";\nimport Modal from \"./Modal\";\nimport { MatrixClientPeg } from \"./MatrixClientPeg\";\nimport { _t } from \"./languageHandler\";\nimport { isSecureBackupRequired } from \"./utils/WellKnownUtils\";\nimport AccessSecretStorageDialog, { KeyParams } from \"./components/views/dialogs/security/AccessSecretStorageDialog\";\nimport { ModuleRunner } from \"./modules/ModuleRunner\";\nimport QuestionDialog from \"./components/views/dialogs/QuestionDialog\";\nimport InteractiveAuthDialog from \"./components/views/dialogs/InteractiveAuthDialog\";\n\n// This stores the secret storage private keys in memory for the JS SDK. This is\n// only meant to act as a cache to avoid prompting the user multiple times\n// during the same single operation. Use `accessSecretStorage` below to scope a\n// single secret storage operation, as it will clear the cached keys once the\n// operation ends.\nlet secretStorageKeys: Record<string, Uint8Array> = {};\nlet secretStorageKeyInfo: Record<string, SecretStorage.SecretStorageKeyDescription> = {};\nlet secretStorageBeingAccessed = false;\n\n/**\n * This can be used by other components to check if secret storage access is in\n * progress, so that we can e.g. avoid intermittently showing toasts during\n * secret storage setup.\n *\n * @returns {bool}\n */\nexport function isSecretStorageBeingAccessed(): boolean {\n    return secretStorageBeingAccessed;\n}\n\nexport class AccessCancelledError extends Error {\n    public constructor() {\n        super(\"Secret storage access canceled\");\n    }\n}\n\nasync function confirmToDismiss(): Promise<boolean> {\n    const [sure] = await Modal.createDialog(QuestionDialog, {\n        title: _t(\"encryption|cancel_entering_passphrase_title\"),\n        description: _t(\"encryption|cancel_entering_passphrase_description\"),\n        danger: false,\n        button: _t(\"action|go_back\"),\n        cancelButton: _t(\"action|cancel\"),\n    }).finished;\n    return !sure;\n}\n\nfunction makeInputToKey(\n    keyInfo: SecretStorage.SecretStorageKeyDescription,\n): (keyParams: KeyParams) => Promise<Uint8Array> {\n    return async ({ passphrase, recoveryKey }): Promise<Uint8Array> => {\n        if (passphrase) {\n            return deriveRecoveryKeyFromPassphrase(passphrase, keyInfo.passphrase.salt, keyInfo.passphrase.iterations);\n        } else if (recoveryKey) {\n            return decodeRecoveryKey(recoveryKey);\n        }\n        throw new Error(\"Invalid input, passphrase or recoveryKey need to be provided\");\n    };\n}\n\nasync function getSecretStorageKey({\n    keys: keyInfos,\n}: {\n    keys: Record<string, SecretStorage.SecretStorageKeyDescription>;\n}): Promise<[string, Uint8Array]> {\n    const cli = MatrixClientPeg.safeGet();\n    let keyId = await cli.getDefaultSecretStorageKeyId();\n    let keyInfo!: SecretStorage.SecretStorageKeyDescription;\n    if (keyId) {\n        // use the default SSSS key if set\n        keyInfo = keyInfos[keyId];\n        if (!keyInfo) {\n            // if the default key is not available, pretend the default key\n            // isn't set\n            keyId = null;\n        }\n    }\n    if (!keyId) {\n        // if no default SSSS key is set, fall back to a heuristic of using the\n        // only available key, if only one key is set\n        const keyInfoEntries = Object.entries(keyInfos);\n        if (keyInfoEntries.length > 1) {\n            throw new Error(\"Multiple storage key requests not implemented\");\n        }\n        [keyId, keyInfo] = keyInfoEntries[0];\n    }\n    logger.debug(`getSecretStorageKey: request for 4S keys [${Object.keys(keyInfos)}]: looking for key ${keyId}`);\n\n    // Check the in-memory cache\n    if (secretStorageBeingAccessed && secretStorageKeys[keyId]) {\n        logger.debug(`getSecretStorageKey: returning key ${keyId} from cache`);\n        return [keyId, secretStorageKeys[keyId]];\n    }\n\n    const keyFromCustomisations = ModuleRunner.instance.extensions.cryptoSetup.getSecretStorageKey();\n    if (keyFromCustomisations) {\n        logger.log(\"getSecretStorageKey: Using secret storage key from CryptoSetupExtension\");\n        cacheSecretStorageKey(keyId, keyInfo, keyFromCustomisations);\n        return [keyId, keyFromCustomisations];\n    }\n\n    logger.debug(\"getSecretStorageKey: prompting user for key\");\n    const inputToKey = makeInputToKey(keyInfo);\n    const { finished } = Modal.createDialog(\n        AccessSecretStorageDialog,\n        /* props= */\n        {\n            keyInfo,\n            checkPrivateKey: async (input: KeyParams): Promise<boolean> => {\n                const key = await inputToKey(input);\n                return MatrixClientPeg.safeGet().secretStorage.checkKey(key, keyInfo);\n            },\n        },\n        /* className= */ undefined,\n        /* isPriorityModal= */ false,\n        /* isStaticModal= */ false,\n        /* options= */ {\n            onBeforeClose: async (reason): Promise<boolean> => {\n                if (reason === \"backgroundClick\") {\n                    return confirmToDismiss();\n                }\n                return true;\n            },\n        },\n    );\n    const [keyParams] = await finished;\n    if (!keyParams) {\n        throw new AccessCancelledError();\n    }\n    logger.debug(\"getSecretStorageKey: got key from user\");\n    const key = await inputToKey(keyParams);\n\n    // Save to cache to avoid future prompts in the current session\n    cacheSecretStorageKey(keyId, keyInfo, key);\n\n    return [keyId, key];\n}\n\nfunction cacheSecretStorageKey(\n    keyId: string,\n    keyInfo: SecretStorage.SecretStorageKeyDescription,\n    key: Uint8Array,\n): void {\n    if (secretStorageBeingAccessed) {\n        secretStorageKeys[keyId] = key;\n        secretStorageKeyInfo[keyId] = keyInfo;\n    }\n}\n\nexport const crossSigningCallbacks: ICryptoCallbacks = {\n    getSecretStorageKey,\n    cacheSecretStorageKey,\n};\n\n/**\n * Carry out an operation that may require multiple accesses to secret storage, caching the key.\n *\n * Use this helper to wrap an operation that may require multiple accesses to secret storage; the user will be prompted\n * to enter the 4S key or passphrase on the first access, and the key will be cached for the rest of the operation.\n *\n * @param func - The operation to be wrapped.\n */\nexport async function withSecretStorageKeyCache<T>(func: () => Promise<T>): Promise<T> {\n    logger.debug(\"SecurityManager: enabling 4S key cache\");\n    secretStorageBeingAccessed = true;\n    try {\n        return await func();\n    } finally {\n        // Clear secret storage key cache now that work is complete\n        logger.debug(\"SecurityManager: disabling 4S key cache\");\n        secretStorageBeingAccessed = false;\n        secretStorageKeys = {};\n        secretStorageKeyInfo = {};\n    }\n}\n\n/**\n * This helper should be used whenever you need to access secret storage. It\n * ensures that secret storage (and also cross-signing since they each depend on\n * each other in a cycle of sorts) have been bootstrapped before running the\n * provided function.\n *\n * Bootstrapping secret storage may take one of these paths:\n * 1. Create secret storage from a passphrase and store cross-signing keys\n *    in secret storage.\n * 2. Access existing secret storage by requesting passphrase and accessing\n *    cross-signing keys as needed.\n * 3. All keys are loaded and there's nothing to do.\n *\n * Additionally, the secret storage keys are cached during the scope of this function\n * to ensure the user is prompted only once for their secret storage\n * passphrase. The cache is then cleared once the provided function completes.\n *\n * @param {Function} [func] An operation to perform once secret storage has been\n * bootstrapped. Optional.\n * @param {bool} [forceReset] Reset secret storage even if it's already set up\n */\nexport async function accessSecretStorage(func = async (): Promise<void> => {}, forceReset = false): Promise<void> {\n    await withSecretStorageKeyCache(() => doAccessSecretStorage(func, forceReset));\n}\n\n/** Helper for {@link #accessSecretStorage} */\nasync function doAccessSecretStorage(func: () => Promise<void>, forceReset: boolean): Promise<void> {\n    try {\n        const cli = MatrixClientPeg.safeGet();\n        const crypto = cli.getCrypto();\n        if (!crypto) {\n            throw new Error(\"End-to-end encryption is disabled - unable to access secret storage.\");\n        }\n\n        let createNew = false;\n        if (forceReset) {\n            logger.debug(\"accessSecretStorage: resetting 4S\");\n            createNew = true;\n        } else if (!(await cli.secretStorage.hasKey())) {\n            logger.debug(\"accessSecretStorage: no 4S key configured, creating a new one\");\n            createNew = true;\n        }\n\n        if (createNew) {\n            // This dialog calls bootstrap itself after guiding the user through\n            // passphrase creation.\n            const { finished } = Modal.createDialogAsync(\n                import(\"./async-components/views/dialogs/security/CreateSecretStorageDialog\") as unknown as Promise<\n                    typeof CreateSecretStorageDialog\n                >,\n                {\n                    forceReset,\n                },\n                undefined,\n                /* priority = */ false,\n                /* static = */ true,\n                /* options = */ {\n                    onBeforeClose: async (reason): Promise<boolean> => {\n                        // If Secure Backup is required, you cannot leave the modal.\n                        if (reason === \"backgroundClick\") {\n                            return !isSecureBackupRequired(cli);\n                        }\n                        return true;\n                    },\n                },\n            );\n            const [confirmed] = await finished;\n            if (!confirmed) {\n                throw new Error(\"Secret storage creation canceled\");\n            }\n        } else {\n            logger.debug(\"accessSecretStorage: bootstrapCrossSigning\");\n            await crypto.bootstrapCrossSigning({\n                authUploadDeviceSigningKeys: async (makeRequest): Promise<void> => {\n                    logger.debug(\"accessSecretStorage: performing UIA to upload cross-signing keys\");\n                    const { finished } = Modal.createDialog(InteractiveAuthDialog, {\n                        title: _t(\"encryption|bootstrap_title\"),\n                        matrixClient: cli,\n                        makeRequest,\n                    });\n                    const [confirmed] = await finished;\n                    if (!confirmed) {\n                        throw new Error(\"Cross-signing key upload auth canceled\");\n                    }\n                    logger.debug(\"accessSecretStorage: Cross-signing key upload successful\");\n                },\n            });\n            logger.debug(\"accessSecretStorage: bootstrapSecretStorage\");\n            await crypto.bootstrapSecretStorage({});\n        }\n\n        logger.debug(\"accessSecretStorage: 4S now ready\");\n        // `return await` needed here to ensure `finally` block runs after the\n        // inner operation completes.\n        await func();\n        logger.debug(\"accessSecretStorage: operation complete\");\n    } catch (e) {\n        ModuleRunner.instance.extensions.cryptoSetup.catchAccessSecretStorageError(e as Error);\n        logger.error(\"accessSecretStorage: error during operation\", e);\n        // Re-throw so that higher level logic can abort as needed\n        throw e;\n    }\n}\n"],"mappings":"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","ignoreList":[]}