2018-07-07 00:48:14 +00:00
|
|
|
/* 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-07-07 00:48:14 +00:00
|
|
|
|
2018-04-27 21:25:04 +00:00
|
|
|
window.Whisper = window.Whisper || {};
|
2015-02-27 02:10:04 +00:00
|
|
|
|
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',
|
|
|
|
},
|
2018-07-07 00:48:14 +00:00
|
|
|
initialize(options) {
|
2018-04-27 21:25:04 +00:00
|
|
|
this.ourNumber = textsecure.storage.user.getNumber();
|
|
|
|
if (options.newKey) {
|
|
|
|
this.theirKey = options.newKey;
|
|
|
|
}
|
2017-06-10 19:18:24 +00:00
|
|
|
|
2018-07-07 00:48:14 +00:00
|
|
|
this.loadKeys().then(() => {
|
|
|
|
this.listenTo(this.model, 'change', this.render);
|
|
|
|
});
|
2018-04-27 21:25:04 +00:00
|
|
|
},
|
2018-07-07 00:48:14 +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));
|
2018-07-07 00:48:14 +00:00
|
|
|
// .then(this.makeQRCode.bind(this));
|
2018-04-27 21:25:04 +00:00
|
|
|
},
|
2018-07-07 00:48:14 +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')
|
|
|
|
);
|
|
|
|
},
|
2018-07-07 00:48:14 +00:00
|
|
|
loadTheirKey() {
|
|
|
|
return textsecure.storage.protocol
|
|
|
|
.loadIdentityKey(this.model.id)
|
|
|
|
.then(theirKey => {
|
2018-04-27 21:25:04 +00:00
|
|
|
this.theirKey = theirKey;
|
2018-07-07 00:48:14 +00:00
|
|
|
});
|
2018-04-27 21:25:04 +00:00
|
|
|
},
|
2018-07-07 00:48:14 +00:00
|
|
|
loadOurKey() {
|
|
|
|
return textsecure.storage.protocol
|
|
|
|
.loadIdentityKey(this.ourNumber)
|
|
|
|
.then(ourKey => {
|
2018-04-27 21:25:04 +00:00
|
|
|
this.ourKey = ourKey;
|
2018-07-07 00:48:14 +00:00
|
|
|
});
|
2018-04-27 21:25:04 +00:00
|
|
|
},
|
2018-07-07 00:48:14 +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)
|
2018-07-07 00:48:14 +00:00
|
|
|
.then(securityNumber => {
|
|
|
|
this.securityNumber = securityNumber;
|
|
|
|
});
|
2018-04-27 21:25:04 +00:00
|
|
|
},
|
2018-07-07 00:48:14 +00:00
|
|
|
onSafetyNumberChanged() {
|
2018-04-27 21:25:04 +00:00
|
|
|
this.model.getProfiles().then(this.loadKeys.bind(this));
|
2017-07-11 22:54:46 +00:00
|
|
|
|
2018-07-07 00:48:14 +00:00
|
|
|
const dialog = new Whisper.ConfirmationDialogView({
|
2018-04-27 21:25:04 +00:00
|
|
|
message: i18n('changedRightAfterVerify', [
|
|
|
|
this.model.getTitle(),
|
|
|
|
this.model.getTitle(),
|
|
|
|
]),
|
|
|
|
hideCancel: true,
|
|
|
|
});
|
2017-07-11 22:54:46 +00:00
|
|
|
|
2018-04-27 21:25:04 +00:00
|
|
|
dialog.$el.insertBefore(this.el);
|
|
|
|
dialog.focusCancel();
|
|
|
|
},
|
2018-07-07 00:48:14 +00:00
|
|
|
toggleVerified() {
|
2018-04-27 21:25:04 +00:00
|
|
|
this.$('button.verify').attr('disabled', true);
|
|
|
|
this.model
|
|
|
|
.toggleVerified()
|
2018-07-07 00:48:14 +00:00
|
|
|
.catch(result => {
|
|
|
|
if (result instanceof Error) {
|
|
|
|
if (result.name === 'OutgoingIdentityKeyError') {
|
|
|
|
this.onSafetyNumberChanged();
|
2018-04-27 21:25:04 +00:00
|
|
|
} else {
|
2018-07-21 19:00:08 +00:00
|
|
|
window.log.error(
|
|
|
|
'failed to toggle verified:',
|
|
|
|
result && result.stack ? result.stack : result
|
|
|
|
);
|
2018-07-07 00:48:14 +00:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
const keyError = _.some(
|
|
|
|
result.errors,
|
|
|
|
error => error.name === 'OutgoingIdentityKeyError'
|
|
|
|
);
|
|
|
|
if (keyError) {
|
|
|
|
this.onSafetyNumberChanged();
|
|
|
|
} else {
|
|
|
|
_.forEach(result.errors, error => {
|
2018-07-21 19:00:08 +00:00
|
|
|
window.log.error(
|
|
|
|
'failed to toggle verified:',
|
|
|
|
error && error.stack ? error.stack : error
|
|
|
|
);
|
2018-04-27 21:25:04 +00:00
|
|
|
});
|
2016-09-18 22:16:39 +00:00
|
|
|
}
|
2018-07-07 00:48:14 +00:00
|
|
|
}
|
|
|
|
})
|
|
|
|
.then(() => {
|
|
|
|
this.$('button.verify').removeAttr('disabled');
|
|
|
|
});
|
2018-04-27 21:25:04 +00:00
|
|
|
},
|
2018-07-07 00:48:14 +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));
|
|
|
|
}
|
2018-07-07 00:48:14 +00:00
|
|
|
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);
|
2017-06-10 19:18:24 +00:00
|
|
|
|
2018-04-27 21:25:04 +00:00
|
|
|
return {
|
|
|
|
learnMore: i18n('learnMore'),
|
|
|
|
theirKeyUnknown: i18n('theirIdentityUnknown'),
|
2018-07-07 00:48:14 +00:00
|
|
|
yourSafetyNumberWith,
|
2018-04-27 21:25:04 +00:00
|
|
|
verifyHelp: i18n('verifyHelp', this.model.getTitle()),
|
2018-07-07 00:48:14 +00:00
|
|
|
verifyButton,
|
2018-04-27 21:25:04 +00:00
|
|
|
hasTheirKey: this.theirKey !== undefined,
|
2018-07-07 00:48:14 +00:00
|
|
|
chunks,
|
|
|
|
isVerified,
|
|
|
|
verifiedStatus,
|
2018-04-27 21:25:04 +00:00
|
|
|
};
|
|
|
|
},
|
|
|
|
});
|
2015-02-27 02:10:04 +00:00
|
|
|
})();
|