matrix-react-sdk
Version:
SDK for matrix.org using React
187 lines (158 loc) • 23 kB
JavaScript
"use strict";
var _interopRequireWildcard = require("@babel/runtime/helpers/interopRequireWildcard");
var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault");
Object.defineProperty(exports, "__esModule", {
value: true
});
exports.default = void 0;
var _defineProperty2 = _interopRequireDefault(require("@babel/runtime/helpers/defineProperty"));
var _fileSaver = _interopRequireDefault(require("file-saver"));
var _react = _interopRequireWildcard(require("react"));
var _propTypes = _interopRequireDefault(require("prop-types"));
var _languageHandler = require("../../../../languageHandler");
var _client = require("matrix-js-sdk/src/client");
var MegolmExportEncryption = _interopRequireWildcard(require("../../../../utils/MegolmExportEncryption"));
var sdk = _interopRequireWildcard(require("../../../../index"));
/*
Copyright 2017 Vector Creations Ltd
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.
*/
const PHASE_EDIT = 1;
const PHASE_EXPORTING = 2;
class ExportE2eKeysDialog extends _react.default.Component {
constructor(props) {
super(props);
(0, _defineProperty2.default)(this, "_onPassphraseFormSubmit", ev => {
ev.preventDefault();
const passphrase = this._passphrase1.current.value;
if (passphrase !== this._passphrase2.current.value) {
this.setState({
errStr: (0, _languageHandler._t)('Passphrases must match')
});
return false;
}
if (!passphrase) {
this.setState({
errStr: (0, _languageHandler._t)('Passphrase must not be empty')
});
return false;
}
this._startExport(passphrase);
return false;
});
(0, _defineProperty2.default)(this, "_onCancelClick", ev => {
ev.preventDefault();
this.props.onFinished(false);
return false;
});
this._unmounted = false;
this._passphrase1 = /*#__PURE__*/(0, _react.createRef)();
this._passphrase2 = /*#__PURE__*/(0, _react.createRef)();
this.state = {
phase: PHASE_EDIT,
errStr: null
};
}
componentWillUnmount() {
this._unmounted = true;
}
_startExport(passphrase) {
// extra Promise.resolve() to turn synchronous exceptions into
// asynchronous ones.
Promise.resolve().then(() => {
return this.props.matrixClient.exportRoomKeys();
}).then(k => {
return MegolmExportEncryption.encryptMegolmKeyFile(JSON.stringify(k), passphrase);
}).then(f => {
const blob = new Blob([f], {
type: 'text/plain;charset=us-ascii'
});
_fileSaver.default.saveAs(blob, 'element-keys.txt');
this.props.onFinished(true);
}).catch(e => {
console.error("Error exporting e2e keys:", e);
if (this._unmounted) {
return;
}
const msg = e.friendlyText || (0, _languageHandler._t)('Unknown error');
this.setState({
errStr: msg,
phase: PHASE_EDIT
});
});
this.setState({
errStr: null,
phase: PHASE_EXPORTING
});
}
render() {
const BaseDialog = sdk.getComponent('views.dialogs.BaseDialog');
const disableForm = this.state.phase === PHASE_EXPORTING;
return /*#__PURE__*/_react.default.createElement(BaseDialog, {
className: "mx_exportE2eKeysDialog",
onFinished: this.props.onFinished,
title: (0, _languageHandler._t)("Export room keys")
}, /*#__PURE__*/_react.default.createElement("form", {
onSubmit: this._onPassphraseFormSubmit
}, /*#__PURE__*/_react.default.createElement("div", {
className: "mx_Dialog_content"
}, /*#__PURE__*/_react.default.createElement("p", null, (0, _languageHandler._t)('This process allows you to export the keys for messages ' + 'you have received in encrypted rooms to a local file. You ' + 'will then be able to import the file into another Matrix ' + 'client in the future, so that client will also be able to ' + 'decrypt these messages.')), /*#__PURE__*/_react.default.createElement("p", null, (0, _languageHandler._t)('The exported file will allow anyone who can read it to decrypt ' + 'any encrypted messages that you can see, so you should be ' + 'careful to keep it secure. To help with this, you should enter ' + 'a passphrase below, which will be used to encrypt the exported ' + 'data. It will only be possible to import the data by using the ' + 'same passphrase.')), /*#__PURE__*/_react.default.createElement("div", {
className: "error"
}, this.state.errStr), /*#__PURE__*/_react.default.createElement("div", {
className: "mx_E2eKeysDialog_inputTable"
}, /*#__PURE__*/_react.default.createElement("div", {
className: "mx_E2eKeysDialog_inputRow"
}, /*#__PURE__*/_react.default.createElement("div", {
className: "mx_E2eKeysDialog_inputLabel"
}, /*#__PURE__*/_react.default.createElement("label", {
htmlFor: "passphrase1"
}, (0, _languageHandler._t)("Enter passphrase"))), /*#__PURE__*/_react.default.createElement("div", {
className: "mx_E2eKeysDialog_inputCell"
}, /*#__PURE__*/_react.default.createElement("input", {
ref: this._passphrase1,
id: "passphrase1",
autoFocus: true,
size: "64",
type: "password",
disabled: disableForm
}))), /*#__PURE__*/_react.default.createElement("div", {
className: "mx_E2eKeysDialog_inputRow"
}, /*#__PURE__*/_react.default.createElement("div", {
className: "mx_E2eKeysDialog_inputLabel"
}, /*#__PURE__*/_react.default.createElement("label", {
htmlFor: "passphrase2"
}, (0, _languageHandler._t)("Confirm passphrase"))), /*#__PURE__*/_react.default.createElement("div", {
className: "mx_E2eKeysDialog_inputCell"
}, /*#__PURE__*/_react.default.createElement("input", {
ref: this._passphrase2,
id: "passphrase2",
size: "64",
type: "password",
disabled: disableForm
}))))), /*#__PURE__*/_react.default.createElement("div", {
className: "mx_Dialog_buttons"
}, /*#__PURE__*/_react.default.createElement("input", {
className: "mx_Dialog_primary",
type: "submit",
value: (0, _languageHandler._t)('Export'),
disabled: disableForm
}), /*#__PURE__*/_react.default.createElement("button", {
onClick: this._onCancelClick,
disabled: disableForm
}, (0, _languageHandler._t)("Cancel")))));
}
}
exports.default = ExportE2eKeysDialog;
(0, _defineProperty2.default)(ExportE2eKeysDialog, "propTypes", {
matrixClient: _propTypes.default.instanceOf(_client.MatrixClient).isRequired,
onFinished: _propTypes.default.func.isRequired
});
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../../src/async-components/views/dialogs/security/ExportE2eKeysDialog.js"],"names":["PHASE_EDIT","PHASE_EXPORTING","ExportE2eKeysDialog","React","Component","constructor","props","ev","preventDefault","passphrase","_passphrase1","current","value","_passphrase2","setState","errStr","_startExport","onFinished","_unmounted","state","phase","componentWillUnmount","Promise","resolve","then","matrixClient","exportRoomKeys","k","MegolmExportEncryption","encryptMegolmKeyFile","JSON","stringify","f","blob","Blob","type","FileSaver","saveAs","catch","e","console","error","msg","friendlyText","render","BaseDialog","sdk","getComponent","disableForm","_onPassphraseFormSubmit","_onCancelClick","PropTypes","instanceOf","MatrixClient","isRequired","func"],"mappings":"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","sourcesContent":["/*\nCopyright 2017 Vector Creations Ltd\n\nLicensed under the Apache License, Version 2.0 (the \"License\");\nyou may not use this file except in compliance with the License.\nYou may obtain a copy of the License at\n\n    http://www.apache.org/licenses/LICENSE-2.0\n\nUnless required by applicable law or agreed to in writing, software\ndistributed under the License is distributed on an \"AS IS\" BASIS,\nWITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\nSee the License for the specific language governing permissions and\nlimitations under the License.\n*/\n\nimport FileSaver from 'file-saver';\nimport React, {createRef} from 'react';\nimport PropTypes from 'prop-types';\nimport { _t } from '../../../../languageHandler';\n\nimport { MatrixClient } from 'matrix-js-sdk/src/client';\nimport * as MegolmExportEncryption from '../../../../utils/MegolmExportEncryption';\nimport * as sdk from '../../../../index';\n\nconst PHASE_EDIT = 1;\nconst PHASE_EXPORTING = 2;\n\nexport default class ExportE2eKeysDialog extends React.Component {\n    static propTypes = {\n        matrixClient: PropTypes.instanceOf(MatrixClient).isRequired,\n        onFinished: PropTypes.func.isRequired,\n    };\n\n    constructor(props) {\n        super(props);\n\n        this._unmounted = false;\n\n        this._passphrase1 = createRef();\n        this._passphrase2 = createRef();\n\n        this.state = {\n            phase: PHASE_EDIT,\n            errStr: null,\n        };\n    }\n\n    componentWillUnmount() {\n        this._unmounted = true;\n    }\n\n    _onPassphraseFormSubmit = (ev) => {\n        ev.preventDefault();\n\n        const passphrase = this._passphrase1.current.value;\n        if (passphrase !== this._passphrase2.current.value) {\n            this.setState({errStr: _t('Passphrases must match')});\n            return false;\n        }\n        if (!passphrase) {\n            this.setState({errStr: _t('Passphrase must not be empty')});\n            return false;\n        }\n\n        this._startExport(passphrase);\n        return false;\n    };\n\n    _startExport(passphrase) {\n        // extra Promise.resolve() to turn synchronous exceptions into\n        // asynchronous ones.\n        Promise.resolve().then(() => {\n            return this.props.matrixClient.exportRoomKeys();\n        }).then((k) => {\n            return MegolmExportEncryption.encryptMegolmKeyFile(\n                JSON.stringify(k), passphrase,\n            );\n        }).then((f) => {\n            const blob = new Blob([f], {\n                type: 'text/plain;charset=us-ascii',\n            });\n            FileSaver.saveAs(blob, 'element-keys.txt');\n            this.props.onFinished(true);\n        }).catch((e) => {\n            console.error(\"Error exporting e2e keys:\", e);\n            if (this._unmounted) {\n                return;\n            }\n            const msg = e.friendlyText || _t('Unknown error');\n            this.setState({\n                errStr: msg,\n                phase: PHASE_EDIT,\n            });\n        });\n\n        this.setState({\n            errStr: null,\n            phase: PHASE_EXPORTING,\n        });\n    }\n\n    _onCancelClick = (ev) => {\n        ev.preventDefault();\n        this.props.onFinished(false);\n        return false;\n    };\n\n    render() {\n        const BaseDialog = sdk.getComponent('views.dialogs.BaseDialog');\n\n        const disableForm = (this.state.phase === PHASE_EXPORTING);\n\n        return (\n            <BaseDialog className='mx_exportE2eKeysDialog'\n                onFinished={this.props.onFinished}\n                title={_t(\"Export room keys\")}\n            >\n                <form onSubmit={this._onPassphraseFormSubmit}>\n                    <div className=\"mx_Dialog_content\">\n                        <p>\n                            { _t(\n                                'This process allows you to export the keys for messages ' +\n                                'you have received in encrypted rooms to a local file. You ' +\n                                'will then be able to import the file into another Matrix ' +\n                                'client in the future, so that client will also be able to ' +\n                                'decrypt these messages.',\n                            ) }\n                        </p>\n                        <p>\n                            { _t(\n                                'The exported file will allow anyone who can read it to decrypt ' +\n                                'any encrypted messages that you can see, so you should be ' +\n                                'careful to keep it secure. To help with this, you should enter ' +\n                                'a passphrase below, which will be used to encrypt the exported ' +\n                                'data. It will only be possible to import the data by using the ' +\n                                'same passphrase.',\n                            ) }\n                        </p>\n                        <div className='error'>\n                            { this.state.errStr }\n                        </div>\n                        <div className='mx_E2eKeysDialog_inputTable'>\n                            <div className='mx_E2eKeysDialog_inputRow'>\n                                <div className='mx_E2eKeysDialog_inputLabel'>\n                                    <label htmlFor='passphrase1'>\n                                        { _t(\"Enter passphrase\") }\n                                    </label>\n                                </div>\n                                <div className='mx_E2eKeysDialog_inputCell'>\n                                    <input ref={this._passphrase1} id='passphrase1'\n                                        autoFocus={true} size='64' type='password'\n                                        disabled={disableForm}\n                                    />\n                                </div>\n                            </div>\n                            <div className='mx_E2eKeysDialog_inputRow'>\n                                <div className='mx_E2eKeysDialog_inputLabel'>\n                                    <label htmlFor='passphrase2'>\n                                        { _t(\"Confirm passphrase\") }\n                                    </label>\n                                </div>\n                                <div className='mx_E2eKeysDialog_inputCell'>\n                                    <input ref={this._passphrase2} id='passphrase2'\n                                        size='64' type='password'\n                                        disabled={disableForm}\n                                    />\n                                </div>\n                            </div>\n                        </div>\n                    </div>\n                    <div className='mx_Dialog_buttons'>\n                        <input\n                            className='mx_Dialog_primary'\n                            type='submit'\n                            value={_t('Export')}\n                            disabled={disableForm}\n                        />\n                        <button onClick={this._onCancelClick} disabled={disableForm}>\n                            { _t(\"Cancel\") }\n                        </button>\n                    </div>\n                </form>\n            </BaseDialog>\n        );\n    }\n}\n"]}