a90246cbe5
Co-authored-by: Scott Nonnenberg <scott@signal.org>
160 lines
4.6 KiB
JavaScript
160 lines
4.6 KiB
JavaScript
/* global Whisper, textsecure, QRCode, dcodeIO, libsignal, i18n, _ */
|
|
|
|
/* eslint-disable more/no-then */
|
|
|
|
// eslint-disable-next-line func-names
|
|
(function() {
|
|
'use strict';
|
|
|
|
window.Whisper = window.Whisper || {};
|
|
|
|
Whisper.KeyVerificationPanelView = Whisper.View.extend({
|
|
className: 'key-verification panel',
|
|
templateName: 'key-verification',
|
|
events: {
|
|
'click button.verify': 'toggleVerified',
|
|
},
|
|
initialize(options) {
|
|
this.ourNumber = textsecure.storage.user.getNumber();
|
|
this.ourUuid = textsecure.storage.user.getUuid();
|
|
if (options.newKey) {
|
|
this.theirKey = options.newKey;
|
|
}
|
|
|
|
this.loadTheirKey();
|
|
this.loadOurKey();
|
|
|
|
this.render();
|
|
if (options.onLoad) {
|
|
options.onLoad(this);
|
|
}
|
|
|
|
this.loadKeys().then(() => {
|
|
this.listenTo(this.model, 'change', this.render);
|
|
});
|
|
},
|
|
async loadKeys() {
|
|
await this.generateSecurityNumber();
|
|
this.render();
|
|
},
|
|
makeQRCode() {
|
|
// 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() {
|
|
const item = textsecure.storage.protocol.getIdentityRecord(
|
|
this.model.get('id')
|
|
);
|
|
this.theirKey = item ? item.publicKey : null;
|
|
},
|
|
loadOurKey() {
|
|
const item = textsecure.storage.protocol.getIdentityRecord(
|
|
this.ourUuid || this.ourNumber
|
|
);
|
|
this.ourKey = item ? item.publicKey : null;
|
|
},
|
|
generateSecurityNumber() {
|
|
return new libsignal.FingerprintGenerator(5200)
|
|
.createFor(
|
|
// TODO: we cannot use UUIDs for safety numbers yet
|
|
// this.ourUuid || this.ourNumber,
|
|
this.ourNumber,
|
|
this.ourKey,
|
|
// TODO: we cannot use UUIDs for safety numbers yet
|
|
// this.model.get('uuid') || this.model.get('e164'),
|
|
this.model.get('e164'),
|
|
this.theirKey
|
|
)
|
|
.then(securityNumber => {
|
|
this.securityNumber = securityNumber;
|
|
});
|
|
},
|
|
onSafetyNumberChanged() {
|
|
this.model.getProfiles().then(this.loadKeys.bind(this));
|
|
|
|
const dialog = new Whisper.ConfirmationDialogView({
|
|
message: i18n('changedRightAfterVerify', [
|
|
this.model.getTitle(),
|
|
this.model.getTitle(),
|
|
]),
|
|
hideCancel: true,
|
|
});
|
|
|
|
dialog.$el.insertBefore(this.el);
|
|
dialog.focusCancel();
|
|
},
|
|
toggleVerified() {
|
|
this.$('button.verify').attr('disabled', true);
|
|
this.model
|
|
.toggleVerified()
|
|
.catch(result => {
|
|
if (result instanceof Error) {
|
|
if (result.name === 'OutgoingIdentityKeyError') {
|
|
this.onSafetyNumberChanged();
|
|
} else {
|
|
window.log.error(
|
|
'failed to toggle verified:',
|
|
result && result.stack ? result.stack : result
|
|
);
|
|
}
|
|
} else {
|
|
const keyError = _.some(
|
|
result.errors,
|
|
error => error.name === 'OutgoingIdentityKeyError'
|
|
);
|
|
if (keyError) {
|
|
this.onSafetyNumberChanged();
|
|
} else {
|
|
_.forEach(result.errors, error => {
|
|
window.log.error(
|
|
'failed to toggle verified:',
|
|
error && error.stack ? error.stack : error
|
|
);
|
|
});
|
|
}
|
|
}
|
|
})
|
|
.then(() => {
|
|
this.$('button.verify').removeAttr('disabled');
|
|
});
|
|
},
|
|
render_attributes() {
|
|
const s = this.securityNumber;
|
|
const chunks = [];
|
|
if (s) {
|
|
for (let i = 0; i < s.length; i += 5) {
|
|
chunks.push(s.substring(i, i + 5));
|
|
}
|
|
} else {
|
|
for (let i = 0; i < 12; i += 1) {
|
|
chunks.push('XXXXX');
|
|
}
|
|
}
|
|
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
|
|
? i18n('isVerified', name)
|
|
: i18n('isNotVerified', name);
|
|
|
|
return {
|
|
learnMore: i18n('learnMore'),
|
|
theirKeyUnknown: i18n('theirIdentityUnknown'),
|
|
yourSafetyNumberWith,
|
|
verifyHelp: i18n('verifyHelp', this.model.getTitle()),
|
|
verifyButton,
|
|
hasTheirKey: this.theirKey !== undefined,
|
|
chunks,
|
|
isVerified,
|
|
verifiedStatus,
|
|
};
|
|
},
|
|
});
|
|
})();
|