signal-desktop/js/views/key_verification_view.js

135 lines
4 KiB
JavaScript
Raw Normal View History

/* global Whisper, textsecure, QRCode, dcodeIO, libsignal, i18n, _ */
/* eslint-disable more/no-then */
// eslint-disable-next-line func-names
2018-04-27 21:25:04 +00:00
(function() {
'use strict';
2018-04-27 21:25:04 +00:00
window.Whisper = window.Whisper || {};
2018-04-27 21:25:04 +00:00
Whisper.KeyVerificationPanelView = Whisper.View.extend({
className: 'key-verification panel',
templateName: 'key-verification',
events: {
'click button.verify': 'toggleVerified',
},
initialize(options) {
2018-04-27 21:25:04 +00:00
this.ourNumber = textsecure.storage.user.getNumber();
if (options.newKey) {
this.theirKey = options.newKey;
}
this.loadKeys().then(() => {
this.listenTo(this.model, 'change', this.render);
});
2018-04-27 21:25:04 +00:00
},
loadKeys() {
2018-04-27 21:25:04 +00:00
return Promise.all([this.loadTheirKey(), this.loadOurKey()])
.then(this.generateSecurityNumber.bind(this))
.then(this.render.bind(this));
// .then(this.makeQRCode.bind(this));
2018-04-27 21:25:04 +00:00
},
makeQRCode() {
2018-04-27 21:25:04 +00:00
// Per Lilia: We can't turn this on until it generates a Latin1 string, as is
// required by the mobile clients.
new QRCode(this.$('.qr')[0]).makeCode(
dcodeIO.ByteBuffer.wrap(this.ourKey).toString('base64')
);
},
loadTheirKey() {
return textsecure.storage.protocol
.loadIdentityKey(this.model.id)
.then(theirKey => {
2018-04-27 21:25:04 +00:00
this.theirKey = theirKey;
});
2018-04-27 21:25:04 +00:00
},
loadOurKey() {
return textsecure.storage.protocol
.loadIdentityKey(this.ourNumber)
.then(ourKey => {
2018-04-27 21:25:04 +00:00
this.ourKey = ourKey;
});
2018-04-27 21:25:04 +00:00
},
generateSecurityNumber() {
2018-04-27 21:25:04 +00:00
return new libsignal.FingerprintGenerator(5200)
.createFor(this.ourNumber, this.ourKey, this.model.id, this.theirKey)
.then(securityNumber => {
this.securityNumber = securityNumber;
});
2018-04-27 21:25:04 +00:00
},
onSafetyNumberChanged() {
2018-04-27 21:25:04 +00:00
this.model.getProfiles().then(this.loadKeys.bind(this));
const dialog = new Whisper.ConfirmationDialogView({
2018-04-27 21:25:04 +00:00
message: i18n('changedRightAfterVerify', [
this.model.getTitle(),
this.model.getTitle(),
]),
hideCancel: true,
});
2018-04-27 21:25:04 +00:00
dialog.$el.insertBefore(this.el);
dialog.focusCancel();
},
toggleVerified() {
2018-04-27 21:25:04 +00:00
this.$('button.verify').attr('disabled', true);
this.model
.toggleVerified()
.catch(result => {
if (result instanceof Error) {
if (result.name === 'OutgoingIdentityKeyError') {
this.onSafetyNumberChanged();
2018-04-27 21:25:04 +00:00
} else {
console.log('failed to toggle verified:', result.stack);
}
} else {
const keyError = _.some(
result.errors,
error => error.name === 'OutgoingIdentityKeyError'
);
if (keyError) {
this.onSafetyNumberChanged();
} else {
_.forEach(result.errors, error => {
console.log('failed to toggle verified:', error.stack);
2018-04-27 21:25:04 +00:00
});
}
}
})
.then(() => {
this.$('button.verify').removeAttr('disabled');
});
2018-04-27 21:25:04 +00:00
},
render_attributes() {
const s = this.securityNumber;
const chunks = [];
for (let i = 0; i < s.length; i += 5) {
2018-04-27 21:25:04 +00:00
chunks.push(s.substring(i, i + 5));
}
const name = this.model.getTitle();
const yourSafetyNumberWith = i18n(
'yourSafetyNumberWith',
this.model.getTitle()
);
const isVerified = this.model.isVerified();
const verifyButton = isVerified ? i18n('unverify') : i18n('verify');
const verifiedStatus = isVerified
2018-04-27 21:25:04 +00:00
? i18n('isVerified', name)
: i18n('isNotVerified', name);
2018-04-27 21:25:04 +00:00
return {
learnMore: i18n('learnMore'),
theirKeyUnknown: i18n('theirIdentityUnknown'),
yourSafetyNumberWith,
2018-04-27 21:25:04 +00:00
verifyHelp: i18n('verifyHelp', this.model.getTitle()),
verifyButton,
2018-04-27 21:25:04 +00:00
hasTheirKey: this.theirKey !== undefined,
chunks,
isVerified,
verifiedStatus,
2018-04-27 21:25:04 +00:00
};
},
});
})();