/* Copyright 2019, 2020 The Matrix.org Foundation C.I.C. Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with the License. You may obtain a copy of the License at http://www.apache.org/licenses/LICENSE-2.0 Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the specific language governing permissions and limitations under the License. */ import React from 'react'; import {MatrixClientPeg} from '../../../MatrixClientPeg'; import { _t } from '../../../languageHandler'; import * as sdk from '../../../index'; import { accessSecretStorage } from '../../../SecurityManager'; import Modal from '../../../Modal'; export default class CrossSigningPanel extends React.PureComponent { constructor(props) { super(props); this._unmounted = false; this.state = { error: null, crossSigningPublicKeysOnDevice: false, crossSigningPrivateKeysInStorage: false, masterPrivateKeyCached: false, selfSigningPrivateKeyCached: false, userSigningPrivateKeyCached: false, sessionBackupKeyCached: false, secretStorageKeyInAccount: false, }; } componentDidMount() { const cli = MatrixClientPeg.get(); cli.on("accountData", this.onAccountData); cli.on("userTrustStatusChanged", this.onStatusChanged); cli.on("crossSigning.keysChanged", this.onStatusChanged); this._getUpdatedStatus(); } componentWillUnmount() { this._unmounted = true; const cli = MatrixClientPeg.get(); if (!cli) return; cli.removeListener("accountData", this.onAccountData); cli.removeListener("userTrustStatusChanged", this.onStatusChanged); cli.removeListener("crossSigning.keysChanged", this.onStatusChanged); } onAccountData = (event) => { const type = event.getType(); if (type.startsWith("m.cross_signing") || type.startsWith("m.secret_storage")) { this._getUpdatedStatus(); } }; _onBootstrapClick = () => { this._bootstrapSecureSecretStorage(false); }; onStatusChanged = () => { this._getUpdatedStatus(); }; async _getUpdatedStatus() { const cli = MatrixClientPeg.get(); const pkCache = cli.getCrossSigningCacheCallbacks(); const crossSigning = cli._crypto._crossSigningInfo; const secretStorage = cli._crypto._secretStorage; const crossSigningPublicKeysOnDevice = crossSigning.getId(); const crossSigningPrivateKeysInStorage = await crossSigning.isStoredInSecretStorage(secretStorage); const masterPrivateKeyCached = !!(pkCache && await pkCache.getCrossSigningKeyCache("master")); const selfSigningPrivateKeyCached = !!(pkCache && await pkCache.getCrossSigningKeyCache("self_signing")); const userSigningPrivateKeyCached = !!(pkCache && await pkCache.getCrossSigningKeyCache("user_signing")); const sessionBackupKeyFromCache = await cli._crypto.getSessionBackupPrivateKey(); const sessionBackupKeyCached = !!(sessionBackupKeyFromCache); const sessionBackupKeyWellFormed = sessionBackupKeyFromCache instanceof Uint8Array; const secretStorageKeyInAccount = await secretStorage.hasKey(); const homeserverSupportsCrossSigning = await cli.doesServerSupportUnstableFeature("org.matrix.e2e_cross_signing"); const crossSigningReady = await cli.isCrossSigningReady(); const secretStorageReady = await cli.isSecretStorageReady(); this.setState({ crossSigningPublicKeysOnDevice, crossSigningPrivateKeysInStorage, masterPrivateKeyCached, selfSigningPrivateKeyCached, userSigningPrivateKeyCached, sessionBackupKeyCached, sessionBackupKeyWellFormed, secretStorageKeyInAccount, homeserverSupportsCrossSigning, crossSigningReady, secretStorageReady, }); } /** * 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. * @param {bool} [forceReset] Bootstrap again even if keys already present */ _bootstrapSecureSecretStorage = async (forceReset=false) => { this.setState({ error: null }); try { await accessSecretStorage(() => undefined, forceReset); } catch (e) { this.setState({ error: e }); console.error("Error bootstrapping secret storage", e); } if (this._unmounted) return; this._getUpdatedStatus(); } onDestroyStorage = (act) => { if (!act) return; this._bootstrapSecureSecretStorage(true); } _destroySecureSecretStorage = () => { const ConfirmDestroyCrossSigningDialog = sdk.getComponent("dialogs.ConfirmDestroyCrossSigningDialog"); Modal.createDialog(ConfirmDestroyCrossSigningDialog, { onFinished: this.onDestroyStorage, }); } render() { const AccessibleButton = sdk.getComponent("elements.AccessibleButton"); const { error, crossSigningPublicKeysOnDevice, crossSigningPrivateKeysInStorage, masterPrivateKeyCached, selfSigningPrivateKeyCached, userSigningPrivateKeyCached, sessionBackupKeyCached, sessionBackupKeyWellFormed, secretStorageKeyInAccount, homeserverSupportsCrossSigning, crossSigningReady, secretStorageReady, } = this.state; let errorSection; if (error) { errorSection =
{_t( "Your homeserver does not support cross-signing.", )}
; } else if (crossSigningReady && secretStorageReady) { summarisedStatus =✅ {_t( "Cross-signing and secret storage are ready for use.", )}
; } else if (crossSigningReady && !secretStorageReady) { summarisedStatus =✅ {_t( "Cross-signing is ready for use, but secret storage is " + "currently not being used to backup your keys.", )}
; } else if (crossSigningPrivateKeysInStorage) { summarisedStatus ={_t( "Your account has a cross-signing identity in secret storage, " + "but it is not yet trusted by this session.", )}
; } else { summarisedStatus ={_t( "Cross-signing and secret storage are not yet set up.", )}
; } const keysExistAnywhere = ( secretStorageKeyInAccount || crossSigningPrivateKeysInStorage || crossSigningPublicKeysOnDevice ); const keysExistEverywhere = ( secretStorageKeyInAccount && crossSigningPrivateKeysInStorage && crossSigningPublicKeysOnDevice ); let resetButton; if (keysExistAnywhere) { resetButton = ({_t("Cross-signing public keys:")} | {crossSigningPublicKeysOnDevice ? _t("in memory") : _t("not found")} |
{_t("Cross-signing private keys:")} | {crossSigningPrivateKeysInStorage ? _t("in secret storage") : _t("not found")} |
{_t("Master private key:")} | {masterPrivateKeyCached ? _t("cached locally") : _t("not found locally")} |
{_t("Self signing private key:")} | {selfSigningPrivateKeyCached ? _t("cached locally") : _t("not found locally")} |
{_t("User signing private key:")} | {userSigningPrivateKeyCached ? _t("cached locally") : _t("not found locally")} |
{_t("Session backup key:")} | {sessionBackupKeyCached ? _t("cached locally") : _t("not found locally")} {sessionBackupKeyWellFormedText} |
{_t("Secret storage public key:")} | {secretStorageKeyInAccount ? _t("in account data") : _t("not found")} |
{_t("Homeserver feature support:")} | {homeserverSupportsCrossSigning ? _t("exists") : _t("not found")} |