mirror of
https://github.com/element-hq/element-web.git
synced 2024-12-03 20:36:57 +03:00
Initial implementation of KeyRequestHandler, KeyShareDialog etc
This commit is contained in:
parent
f756d231cf
commit
d5bc24d9f7
3 changed files with 259 additions and 0 deletions
89
src/KeyRequestHandler.js
Normal file
89
src/KeyRequestHandler.js
Normal file
|
@ -0,0 +1,89 @@
|
||||||
|
/*
|
||||||
|
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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import sdk from './index';
|
||||||
|
import Modal from './Modal';
|
||||||
|
|
||||||
|
export default class KeyRequestHandler {
|
||||||
|
constructor(matrixClient) {
|
||||||
|
this._matrixClient = matrixClient;
|
||||||
|
|
||||||
|
this._isDialogOpen = false;
|
||||||
|
|
||||||
|
// userId -> deviceId -> [keyRequest]
|
||||||
|
this._pendingKeyRequests = {};
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
handleKeyRequest(keyRequest) {
|
||||||
|
const userId = keyRequest.userId;
|
||||||
|
const deviceId = keyRequest.deviceId;
|
||||||
|
|
||||||
|
if (!this._pendingKeyRequests[userId]) {
|
||||||
|
this._pendingKeyRequests[userId] = {};
|
||||||
|
}
|
||||||
|
if (!this._pendingKeyRequests[userId][deviceId]) {
|
||||||
|
this._pendingKeyRequests[userId][deviceId] = [];
|
||||||
|
}
|
||||||
|
this._pendingKeyRequests[userId][deviceId].push(keyRequest);
|
||||||
|
|
||||||
|
if (this._isDialogOpen) {
|
||||||
|
// ignore for now
|
||||||
|
console.log("Key request, but we already have a dialog open");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
this._processNextRequest();
|
||||||
|
}
|
||||||
|
|
||||||
|
_processNextRequest() {
|
||||||
|
const userId = Object.keys(this._pendingKeyRequests)[0];
|
||||||
|
if (!userId) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
const deviceId = Object.keys(this._pendingKeyRequests[userId])[0];
|
||||||
|
if (!deviceId) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
console.log(`Starting KeyShareDialog for ${userId}:${deviceId}`);
|
||||||
|
|
||||||
|
const finished = (r) => {
|
||||||
|
this._isDialogOpen = false;
|
||||||
|
|
||||||
|
if (r) {
|
||||||
|
for (const req of this._pendingKeyRequests[userId][deviceId]) {
|
||||||
|
req.share();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
delete this._pendingKeyRequests[userId][deviceId];
|
||||||
|
if (Object.keys(this._pendingKeyRequests[userId]).length === 0) {
|
||||||
|
delete this._pendingKeyRequests[userId];
|
||||||
|
}
|
||||||
|
|
||||||
|
this._processNextRequest();
|
||||||
|
};
|
||||||
|
|
||||||
|
const KeyShareDialog = sdk.getComponent("dialogs.KeyShareDialog");
|
||||||
|
Modal.createDialog(KeyShareDialog, {
|
||||||
|
matrixClient: this._matrixClient,
|
||||||
|
userId: userId,
|
||||||
|
deviceId: deviceId,
|
||||||
|
onFinished: finished,
|
||||||
|
});
|
||||||
|
this._isDialogOpen = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
|
@ -38,6 +38,7 @@ import PageTypes from '../../PageTypes';
|
||||||
|
|
||||||
import createRoom from "../../createRoom";
|
import createRoom from "../../createRoom";
|
||||||
import * as UDEHandler from '../../UnknownDeviceErrorHandler';
|
import * as UDEHandler from '../../UnknownDeviceErrorHandler';
|
||||||
|
import KeyRequestHandler from '../../KeyRequestHandler';
|
||||||
import { _t } from '../../languageHandler';
|
import { _t } from '../../languageHandler';
|
||||||
|
|
||||||
module.exports = React.createClass({
|
module.exports = React.createClass({
|
||||||
|
@ -914,6 +915,11 @@ module.exports = React.createClass({
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
const krh = new KeyRequestHandler(cli);
|
||||||
|
cli.on("crypto.roomKeyRequest", (req) => {
|
||||||
|
krh.handleKeyRequest(req);
|
||||||
|
});
|
||||||
},
|
},
|
||||||
|
|
||||||
showScreen: function(screen, params) {
|
showScreen: function(screen, params) {
|
||||||
|
|
164
src/components/views/dialogs/KeyShareDialog.js
Normal file
164
src/components/views/dialogs/KeyShareDialog.js
Normal file
|
@ -0,0 +1,164 @@
|
||||||
|
/*
|
||||||
|
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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import Modal from '../../../Modal';
|
||||||
|
import React from 'react';
|
||||||
|
import sdk from '../../../index';
|
||||||
|
|
||||||
|
import { _t } from '../../../languageHandler';
|
||||||
|
|
||||||
|
export default React.createClass({
|
||||||
|
propTypes: {
|
||||||
|
matrixClient: React.PropTypes.object.isRequired,
|
||||||
|
userId: React.PropTypes.string.isRequired,
|
||||||
|
deviceId: React.PropTypes.string.isRequired,
|
||||||
|
onFinished: React.PropTypes.func.isRequired,
|
||||||
|
},
|
||||||
|
|
||||||
|
getInitialState: function() {
|
||||||
|
return {
|
||||||
|
deviceInfo: null,
|
||||||
|
wasNewDevice: false,
|
||||||
|
};
|
||||||
|
},
|
||||||
|
|
||||||
|
componentDidMount: function() {
|
||||||
|
this._unmounted = false;
|
||||||
|
const userId = this.props.userId;
|
||||||
|
const deviceId = this.props.deviceId;
|
||||||
|
|
||||||
|
// give the client a chance to refresh the device list
|
||||||
|
this.props.matrixClient.downloadKeys([userId], false).then((r) => {
|
||||||
|
if (this._unmounted) { return; }
|
||||||
|
|
||||||
|
const deviceInfo = r[userId][deviceId];
|
||||||
|
|
||||||
|
if(!deviceInfo) {
|
||||||
|
console.warn(`No details found for device ${userId}:${deviceId}`);
|
||||||
|
|
||||||
|
this.props.onFinished(false);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const wasNewDevice = !deviceInfo.isKnown();
|
||||||
|
|
||||||
|
this.setState({
|
||||||
|
deviceInfo: deviceInfo,
|
||||||
|
wasNewDevice: wasNewDevice,
|
||||||
|
});
|
||||||
|
|
||||||
|
// if the device was new before, it's not any more.
|
||||||
|
if (wasNewDevice) {
|
||||||
|
this.props.matrixClient.setDeviceKnown(
|
||||||
|
userId,
|
||||||
|
deviceId,
|
||||||
|
true,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}).done();
|
||||||
|
},
|
||||||
|
|
||||||
|
componentWillUnmount: function() {
|
||||||
|
this._unmounted = true;
|
||||||
|
},
|
||||||
|
|
||||||
|
|
||||||
|
_onVerifyClicked: function() {
|
||||||
|
const DeviceVerifyDialog = sdk.getComponent('views.dialogs.DeviceVerifyDialog');
|
||||||
|
|
||||||
|
console.log("KeyShareDialog: Starting verify dialog");
|
||||||
|
Modal.createDialog(DeviceVerifyDialog, {
|
||||||
|
userId: this.props.userId,
|
||||||
|
device: this.state.deviceInfo,
|
||||||
|
onFinished: (verified) => {
|
||||||
|
if (verified) {
|
||||||
|
// can automatically share the keys now.
|
||||||
|
this.props.onFinished(true);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
});
|
||||||
|
},
|
||||||
|
|
||||||
|
_onShareClicked: function() {
|
||||||
|
console.log("KeyShareDialog: User clicked 'share'");
|
||||||
|
this.props.onFinished(true);
|
||||||
|
},
|
||||||
|
|
||||||
|
_onIgnoreClicked: function() {
|
||||||
|
console.log("KeyShareDialog: User clicked 'ignore'");
|
||||||
|
this.props.onFinished(false);
|
||||||
|
},
|
||||||
|
|
||||||
|
_renderContent: function() {
|
||||||
|
const displayName = this.state.deviceInfo.getDisplayName() ||
|
||||||
|
this.state.deviceInfo.deviceId;
|
||||||
|
|
||||||
|
let text;
|
||||||
|
if (this.state.wasNewDevice) {
|
||||||
|
text = "You added a new device '%(displayName)s', which is"
|
||||||
|
+ " requesting encryption keys.";
|
||||||
|
} else {
|
||||||
|
text = "Your unverified device '%(displayName)s' is requesting"
|
||||||
|
+ " encryption keys.";
|
||||||
|
}
|
||||||
|
text = _t(text, {displayName: displayName});
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div>
|
||||||
|
<p>{text}</p>
|
||||||
|
|
||||||
|
<div className="mx_Dialog_buttons">
|
||||||
|
<button onClick={this._onVerifyClicked}>
|
||||||
|
{_t('Start verification')}
|
||||||
|
</button>
|
||||||
|
<button onClick={this._onShareClicked}>
|
||||||
|
{_t('Share without verifying')}
|
||||||
|
</button>
|
||||||
|
<button onClick={this._onIgnoreClicked}>
|
||||||
|
{_t('Ignore request')}
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
},
|
||||||
|
|
||||||
|
render: function() {
|
||||||
|
const BaseDialog = sdk.getComponent('views.dialogs.BaseDialog');
|
||||||
|
const Spinner = sdk.getComponent('views.elements.Spinner');
|
||||||
|
|
||||||
|
let content;
|
||||||
|
|
||||||
|
if (this.state.deviceInfo) {
|
||||||
|
content = this._renderContent();
|
||||||
|
} else {
|
||||||
|
content = (
|
||||||
|
<div>
|
||||||
|
<p>{_t('Loading device info...')}</p>
|
||||||
|
<Spinner />
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<BaseDialog className='mx_KeyShareRequestDialog'
|
||||||
|
onFinished={this.props.onFinished}
|
||||||
|
title='Encryption key request'
|
||||||
|
>
|
||||||
|
{content}
|
||||||
|
</BaseDialog>
|
||||||
|
);
|
||||||
|
},
|
||||||
|
});
|
Loading…
Reference in a new issue