matrix-react-sdk
Version:
SDK for matrix.org using React
205 lines (197 loc) • 34.5 kB
JavaScript
;
var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault");
Object.defineProperty(exports, "__esModule", {
value: true
});
exports.default = void 0;
exports.shouldShowLogoutDialog = shouldShowLogoutDialog;
var _defineProperty2 = _interopRequireDefault(require("@babel/runtime/helpers/defineProperty"));
var _react = _interopRequireDefault(require("react"));
var _logger = require("matrix-js-sdk/src/logger");
var _Modal = _interopRequireDefault(require("../../../Modal"));
var _dispatcher = _interopRequireDefault(require("../../../dispatcher/dispatcher"));
var _languageHandler = require("../../../languageHandler");
var _MatrixClientPeg = require("../../../MatrixClientPeg");
var _RestoreKeyBackupDialog = _interopRequireDefault(require("./security/RestoreKeyBackupDialog"));
var _QuestionDialog = _interopRequireDefault(require("./QuestionDialog"));
var _BaseDialog = _interopRequireDefault(require("./BaseDialog"));
var _Spinner = _interopRequireDefault(require("../elements/Spinner"));
var _DialogButtons = _interopRequireDefault(require("../elements/DialogButtons"));
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 2020-2022 The Matrix.org Foundation C.I.C.
Copyright 2018, 2019 New Vector Ltd
SPDX-License-Identifier: AGPL-3.0-only OR GPL-3.0-only
Please see LICENSE files in the repository root for full details.
*/
var BackupStatus = /*#__PURE__*/function (BackupStatus) {
BackupStatus[BackupStatus["LOADING"] = 0] = "LOADING";
BackupStatus[BackupStatus["NO_CRYPTO"] = 1] = "NO_CRYPTO";
BackupStatus[BackupStatus["BACKUP_ACTIVE"] = 2] = "BACKUP_ACTIVE";
BackupStatus[BackupStatus["SERVER_BACKUP_BUT_DISABLED"] = 3] = "SERVER_BACKUP_BUT_DISABLED";
BackupStatus[BackupStatus["NO_BACKUP"] = 4] = "NO_BACKUP";
BackupStatus[BackupStatus["ERROR"] = 5] = "ERROR";
return BackupStatus;
}(BackupStatus || {});
/**
* Checks if the `LogoutDialog` should be shown instead of the simple logout flow.
* The `LogoutDialog` will check the crypto recovery status of the account and
* help the user setup recovery properly if needed.
*/
async function shouldShowLogoutDialog(cli) {
const crypto = cli?.getCrypto();
if (!crypto) return false;
// If any room is encrypted, we need to show the advanced logout flow
const allRooms = cli.getRooms();
for (const room of allRooms) {
const isE2e = await crypto.isEncryptionEnabledInRoom(room.roomId);
if (isE2e) return true;
}
return false;
}
class LogoutDialog extends _react.default.Component {
constructor(props) {
super(props);
(0, _defineProperty2.default)(this, "onExportE2eKeysClicked", () => {
_Modal.default.createDialogAsync(Promise.resolve().then(() => _interopRequireWildcard(require("../../../async-components/views/dialogs/security/ExportE2eKeysDialog"))), {
matrixClient: _MatrixClientPeg.MatrixClientPeg.safeGet()
});
});
(0, _defineProperty2.default)(this, "onFinished", confirmed => {
if (confirmed) {
_dispatcher.default.dispatch({
action: "logout"
});
}
// close dialog
this.props.onFinished(!!confirmed);
});
(0, _defineProperty2.default)(this, "onSetRecoveryMethodClick", () => {
if (this.state.backupStatus === BackupStatus.SERVER_BACKUP_BUT_DISABLED) {
// A key backup exists for this account, but the creating device is not
// verified, so restore the backup which will give us the keys from it and
// allow us to trust it (ie. upload keys to it)
_Modal.default.createDialog(_RestoreKeyBackupDialog.default, undefined, undefined, /* priority = */false, /* static = */true);
} else {
_Modal.default.createDialogAsync(Promise.resolve().then(() => _interopRequireWildcard(require("../../../async-components/views/dialogs/security/CreateKeyBackupDialog"))), undefined, undefined, /* priority = */false, /* static = */true);
}
// close dialog
this.props.onFinished(true);
});
(0, _defineProperty2.default)(this, "onLogoutConfirm", () => {
_dispatcher.default.dispatch({
action: "logout"
});
// close dialog
this.props.onFinished(true);
});
this.state = {
backupStatus: BackupStatus.LOADING
};
// we can't call setState() immediately, so wait a beat
window.setTimeout(() => this.startLoadBackupStatus(), 0);
}
/** kick off the asynchronous calls to populate `state.backupStatus` in the background */
startLoadBackupStatus() {
this.loadBackupStatus().catch(e => {
_logger.logger.log("Unable to fetch key backup status", e);
this.setState({
backupStatus: BackupStatus.ERROR
});
});
}
async loadBackupStatus() {
const client = _MatrixClientPeg.MatrixClientPeg.safeGet();
const crypto = client.getCrypto();
if (!crypto) {
this.setState({
backupStatus: BackupStatus.NO_CRYPTO
});
return;
}
if ((await crypto.getActiveSessionBackupVersion()) !== null) {
this.setState({
backupStatus: BackupStatus.BACKUP_ACTIVE
});
return;
}
// backup is not active. see if there is a backup version on the server we ought to back up to.
const backupInfo = await client.getKeyBackupVersion();
this.setState({
backupStatus: backupInfo ? BackupStatus.SERVER_BACKUP_BUT_DISABLED : BackupStatus.NO_BACKUP
});
}
/**
* Show a dialog prompting the user to set up key backup.
*
* Either there is no backup at all ({@link BackupStatus.NO_BACKUP}), there is a backup on the server but
* we are not connected to it ({@link BackupStatus.SERVER_BACKUP_BUT_DISABLED}), or we were unable to pull the
* backup data ({@link BackupStatus.ERROR}). In all three cases, we should prompt the user to set up key backup.
*/
renderSetupBackupDialog() {
const description = /*#__PURE__*/_react.default.createElement("div", null, /*#__PURE__*/_react.default.createElement("p", null, (0, _languageHandler._t)("auth|logout_dialog|setup_secure_backup_description_1")), /*#__PURE__*/_react.default.createElement("p", null, (0, _languageHandler._t)("auth|logout_dialog|setup_secure_backup_description_2")), /*#__PURE__*/_react.default.createElement("p", null, (0, _languageHandler._t)("encryption|setup_secure_backup|explainer")));
let setupButtonCaption;
if (this.state.backupStatus === BackupStatus.SERVER_BACKUP_BUT_DISABLED) {
setupButtonCaption = (0, _languageHandler._t)("settings|security|key_backup_connect");
} else {
// if there's an error fetching the backup info, we'll just assume there's
// no backup for the purpose of the button caption
setupButtonCaption = (0, _languageHandler._t)("auth|logout_dialog|use_key_backup");
}
const dialogContent = /*#__PURE__*/_react.default.createElement("div", null, /*#__PURE__*/_react.default.createElement("div", {
className: "mx_Dialog_content",
id: "mx_Dialog_content"
}, description), /*#__PURE__*/_react.default.createElement(_DialogButtons.default, {
primaryButton: setupButtonCaption,
hasCancel: false,
onPrimaryButtonClick: this.onSetRecoveryMethodClick,
focus: true
}, /*#__PURE__*/_react.default.createElement("button", {
onClick: this.onLogoutConfirm
}, (0, _languageHandler._t)("auth|logout_dialog|skip_key_backup"))), /*#__PURE__*/_react.default.createElement("details", null, /*#__PURE__*/_react.default.createElement("summary", {
className: "mx_LogoutDialog_ExportKeyAdvanced"
}, (0, _languageHandler._t)("common|advanced")), /*#__PURE__*/_react.default.createElement("p", null, /*#__PURE__*/_react.default.createElement("button", {
onClick: this.onExportE2eKeysClicked
}, (0, _languageHandler._t)("auth|logout_dialog|megolm_export")))));
// Not quite a standard question dialog as the primary button cancels
// the action and does something else instead, whilst non-default button
// confirms the action.
return /*#__PURE__*/_react.default.createElement(_BaseDialog.default, {
title: (0, _languageHandler._t)("auth|logout_dialog|setup_key_backup_title"),
contentId: "mx_Dialog_content",
hasCancel: true,
onFinished: this.onFinished
}, dialogContent);
}
render() {
switch (this.state.backupStatus) {
case BackupStatus.LOADING:
// while we're deciding if we have backups, show a spinner
return /*#__PURE__*/_react.default.createElement(_BaseDialog.default, {
title: (0, _languageHandler._t)("action|sign_out"),
contentId: "mx_Dialog_content",
hasCancel: true,
onFinished: this.onFinished
}, /*#__PURE__*/_react.default.createElement(_Spinner.default, null));
case BackupStatus.NO_CRYPTO:
case BackupStatus.BACKUP_ACTIVE:
return /*#__PURE__*/_react.default.createElement(_QuestionDialog.default, {
hasCancelButton: true,
title: (0, _languageHandler._t)("action|sign_out"),
description: (0, _languageHandler._t)("auth|logout_dialog|description"),
button: (0, _languageHandler._t)("action|sign_out"),
onFinished: this.onFinished
});
case BackupStatus.NO_BACKUP:
case BackupStatus.SERVER_BACKUP_BUT_DISABLED:
case BackupStatus.ERROR:
return this.renderSetupBackupDialog();
}
}
}
exports.default = LogoutDialog;
(0, _defineProperty2.default)(LogoutDialog, "defaultProps", {
onFinished: function () {}
});
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_react","_interopRequireDefault","require","_logger","_Modal","_dispatcher","_languageHandler","_MatrixClientPeg","_RestoreKeyBackupDialog","_QuestionDialog","_BaseDialog","_Spinner","_DialogButtons","_getRequireWildcardCache","e","WeakMap","r","t","_interopRequireWildcard","__esModule","default","has","get","n","__proto__","a","Object","defineProperty","getOwnPropertyDescriptor","u","hasOwnProperty","call","i","set","BackupStatus","shouldShowLogoutDialog","cli","crypto","getCrypto","allRooms","getRooms","room","isE2e","isEncryptionEnabledInRoom","roomId","LogoutDialog","React","Component","constructor","props","_defineProperty2","Modal","createDialogAsync","Promise","resolve","then","matrixClient","MatrixClientPeg","safeGet","confirmed","dis","dispatch","action","onFinished","state","backupStatus","SERVER_BACKUP_BUT_DISABLED","createDialog","RestoreKeyBackupDialog","undefined","LOADING","window","setTimeout","startLoadBackupStatus","loadBackupStatus","catch","logger","log","setState","ERROR","client","NO_CRYPTO","getActiveSessionBackupVersion","BACKUP_ACTIVE","backupInfo","getKeyBackupVersion","NO_BACKUP","renderSetupBackupDialog","description","createElement","_t","setupButtonCaption","dialogContent","className","id","primaryButton","hasCancel","onPrimaryButtonClick","onSetRecoveryMethodClick","focus","onClick","onLogoutConfirm","onExportE2eKeysClicked","title","contentId","render","hasCancelButton","button","exports"],"sources":["../../../../src/components/views/dialogs/LogoutDialog.tsx"],"sourcesContent":["/*\nCopyright 2024 New Vector Ltd.\nCopyright 2020-2022 The Matrix.org Foundation C.I.C.\nCopyright 2018, 2019 New Vector Ltd\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 React from \"react\";\nimport { logger } from \"matrix-js-sdk/src/logger\";\nimport { MatrixClient } from \"matrix-js-sdk/src/matrix\";\n\nimport type CreateKeyBackupDialog from \"../../../async-components/views/dialogs/security/CreateKeyBackupDialog\";\nimport type ExportE2eKeysDialog from \"../../../async-components/views/dialogs/security/ExportE2eKeysDialog\";\nimport Modal from \"../../../Modal\";\nimport dis from \"../../../dispatcher/dispatcher\";\nimport { _t } from \"../../../languageHandler\";\nimport { MatrixClientPeg } from \"../../../MatrixClientPeg\";\nimport RestoreKeyBackupDialog from \"./security/RestoreKeyBackupDialog\";\nimport QuestionDialog from \"./QuestionDialog\";\nimport BaseDialog from \"./BaseDialog\";\nimport Spinner from \"../elements/Spinner\";\nimport DialogButtons from \"../elements/DialogButtons\";\n\ninterface IProps {\n    onFinished: (success: boolean) => void;\n}\n\nenum BackupStatus {\n    /** we're trying to figure out if there is an active backup */\n    LOADING,\n\n    /** crypto is disabled in this client (so no need to back up) */\n    NO_CRYPTO,\n\n    /** Key backup is active and working */\n    BACKUP_ACTIVE,\n\n    /** there is a backup on the server but we are not backing up to it */\n    SERVER_BACKUP_BUT_DISABLED,\n\n    /** backup is not set up locally and there is no backup on the server */\n    NO_BACKUP,\n\n    /** there was an error fetching the state */\n    ERROR,\n}\n\ninterface IState {\n    backupStatus: BackupStatus;\n}\n\n/**\n * Checks if the `LogoutDialog` should be shown instead of the simple logout flow.\n * The `LogoutDialog` will check the crypto recovery status of the account and\n * help the user setup recovery properly if needed.\n */\nexport async function shouldShowLogoutDialog(cli: MatrixClient): Promise<boolean> {\n    const crypto = cli?.getCrypto();\n    if (!crypto) return false;\n\n    // If any room is encrypted, we need to show the advanced logout flow\n    const allRooms = cli!.getRooms();\n    for (const room of allRooms) {\n        const isE2e = await crypto.isEncryptionEnabledInRoom(room.roomId);\n        if (isE2e) return true;\n    }\n\n    return false;\n}\n\nexport default class LogoutDialog extends React.Component<IProps, IState> {\n    public static defaultProps = {\n        onFinished: function () {},\n    };\n\n    public constructor(props: IProps) {\n        super(props);\n\n        this.state = {\n            backupStatus: BackupStatus.LOADING,\n        };\n\n        // we can't call setState() immediately, so wait a beat\n        window.setTimeout(() => this.startLoadBackupStatus(), 0);\n    }\n\n    /** kick off the asynchronous calls to populate `state.backupStatus` in the background */\n    private startLoadBackupStatus(): void {\n        this.loadBackupStatus().catch((e) => {\n            logger.log(\"Unable to fetch key backup status\", e);\n            this.setState({\n                backupStatus: BackupStatus.ERROR,\n            });\n        });\n    }\n\n    private async loadBackupStatus(): Promise<void> {\n        const client = MatrixClientPeg.safeGet();\n        const crypto = client.getCrypto();\n        if (!crypto) {\n            this.setState({ backupStatus: BackupStatus.NO_CRYPTO });\n            return;\n        }\n\n        if ((await crypto.getActiveSessionBackupVersion()) !== null) {\n            this.setState({ backupStatus: BackupStatus.BACKUP_ACTIVE });\n            return;\n        }\n\n        // backup is not active. see if there is a backup version on the server we ought to back up to.\n        const backupInfo = await client.getKeyBackupVersion();\n        this.setState({ backupStatus: backupInfo ? BackupStatus.SERVER_BACKUP_BUT_DISABLED : BackupStatus.NO_BACKUP });\n    }\n\n    private onExportE2eKeysClicked = (): void => {\n        Modal.createDialogAsync(\n            import(\"../../../async-components/views/dialogs/security/ExportE2eKeysDialog\") as unknown as Promise<\n                typeof ExportE2eKeysDialog\n            >,\n            {\n                matrixClient: MatrixClientPeg.safeGet(),\n            },\n        );\n    };\n\n    private onFinished = (confirmed?: boolean): void => {\n        if (confirmed) {\n            dis.dispatch({ action: \"logout\" });\n        }\n        // close dialog\n        this.props.onFinished(!!confirmed);\n    };\n\n    private onSetRecoveryMethodClick = (): void => {\n        if (this.state.backupStatus === BackupStatus.SERVER_BACKUP_BUT_DISABLED) {\n            // A key backup exists for this account, but the creating device is not\n            // verified, so restore the backup which will give us the keys from it and\n            // allow us to trust it (ie. upload keys to it)\n            Modal.createDialog(\n                RestoreKeyBackupDialog,\n                undefined,\n                undefined,\n                /* priority = */ false,\n                /* static = */ true,\n            );\n        } else {\n            Modal.createDialogAsync(\n                import(\"../../../async-components/views/dialogs/security/CreateKeyBackupDialog\") as unknown as Promise<\n                    typeof CreateKeyBackupDialog\n                >,\n                undefined,\n                undefined,\n                /* priority = */ false,\n                /* static = */ true,\n            );\n        }\n\n        // close dialog\n        this.props.onFinished(true);\n    };\n\n    private onLogoutConfirm = (): void => {\n        dis.dispatch({ action: \"logout\" });\n\n        // close dialog\n        this.props.onFinished(true);\n    };\n\n    /**\n     * Show a dialog prompting the user to set up key backup.\n     *\n     * Either there is no backup at all ({@link BackupStatus.NO_BACKUP}), there is a backup on the server but\n     * we are not connected to it ({@link BackupStatus.SERVER_BACKUP_BUT_DISABLED}), or we were unable to pull the\n     * backup data ({@link BackupStatus.ERROR}). In all three cases, we should prompt the user to set up key backup.\n     */\n    private renderSetupBackupDialog(): React.ReactNode {\n        const description = (\n            <div>\n                <p>{_t(\"auth|logout_dialog|setup_secure_backup_description_1\")}</p>\n                <p>{_t(\"auth|logout_dialog|setup_secure_backup_description_2\")}</p>\n                <p>{_t(\"encryption|setup_secure_backup|explainer\")}</p>\n            </div>\n        );\n\n        let setupButtonCaption;\n        if (this.state.backupStatus === BackupStatus.SERVER_BACKUP_BUT_DISABLED) {\n            setupButtonCaption = _t(\"settings|security|key_backup_connect\");\n        } else {\n            // if there's an error fetching the backup info, we'll just assume there's\n            // no backup for the purpose of the button caption\n            setupButtonCaption = _t(\"auth|logout_dialog|use_key_backup\");\n        }\n\n        const dialogContent = (\n            <div>\n                <div className=\"mx_Dialog_content\" id=\"mx_Dialog_content\">\n                    {description}\n                </div>\n                <DialogButtons\n                    primaryButton={setupButtonCaption}\n                    hasCancel={false}\n                    onPrimaryButtonClick={this.onSetRecoveryMethodClick}\n                    focus={true}\n                >\n                    <button onClick={this.onLogoutConfirm}>{_t(\"auth|logout_dialog|skip_key_backup\")}</button>\n                </DialogButtons>\n                <details>\n                    <summary className=\"mx_LogoutDialog_ExportKeyAdvanced\">{_t(\"common|advanced\")}</summary>\n                    <p>\n                        <button onClick={this.onExportE2eKeysClicked}>{_t(\"auth|logout_dialog|megolm_export\")}</button>\n                    </p>\n                </details>\n            </div>\n        );\n        // Not quite a standard question dialog as the primary button cancels\n        // the action and does something else instead, whilst non-default button\n        // confirms the action.\n        return (\n            <BaseDialog\n                title={_t(\"auth|logout_dialog|setup_key_backup_title\")}\n                contentId=\"mx_Dialog_content\"\n                hasCancel={true}\n                onFinished={this.onFinished}\n            >\n                {dialogContent}\n            </BaseDialog>\n        );\n    }\n\n    public render(): React.ReactNode {\n        switch (this.state.backupStatus) {\n            case BackupStatus.LOADING:\n                // while we're deciding if we have backups, show a spinner\n                return (\n                    <BaseDialog\n                        title={_t(\"action|sign_out\")}\n                        contentId=\"mx_Dialog_content\"\n                        hasCancel={true}\n                        onFinished={this.onFinished}\n                    >\n                        <Spinner />\n                    </BaseDialog>\n                );\n\n            case BackupStatus.NO_CRYPTO:\n            case BackupStatus.BACKUP_ACTIVE:\n                return (\n                    <QuestionDialog\n                        hasCancelButton={true}\n                        title={_t(\"action|sign_out\")}\n                        description={_t(\"auth|logout_dialog|description\")}\n                        button={_t(\"action|sign_out\")}\n                        onFinished={this.onFinished}\n                    />\n                );\n\n            case BackupStatus.NO_BACKUP:\n            case BackupStatus.SERVER_BACKUP_BUT_DISABLED:\n            case BackupStatus.ERROR:\n                return this.renderSetupBackupDialog();\n        }\n    }\n}\n"],"mappings":"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","ignoreList":[]}