Show warning when we discover key change on verify

FREEBIE
This commit is contained in:
Scott Nonnenberg 2017-07-11 15:54:46 -07:00
parent 91f50c028f
commit 8112cd220a
5 changed files with 79 additions and 42 deletions

View file

@ -95,6 +95,16 @@
} }
} }
}, },
"changedRightAfterVerify": {
"message": "Your verification has been cancelled; please review your new safety number with $name$. Remember, this change could mean that someone is trying to intercept your communication or that $name$ has simply reinstalled Signal.",
"description": "Shown on the safety number screen when the user has selected to verify/unverify a contact's safety number, and we immediately discover a safety number change",
"placeholders": {
"name": {
"content": "$1",
"example": "Bob"
}
}
},
"changedRecentlyMultiple": { "changedRecentlyMultiple": {
"message": "Your safety numbers with multiple group members have changed recently. This could mean that someone is trying to intercept your communication or that they have simply reinstalled Signal.", "message": "Your safety numbers with multiple group members have changed recently. This could mean that someone is trying to intercept your communication or that they have simply reinstalled Signal.",
"description": "Shown on confirmation dialog when user attempts to send a message" "description": "Shown on confirmation dialog when user attempts to send a message"

View file

@ -169,7 +169,9 @@
<div class="content"> <div class="content">
<div class='message'>{{ message }}</div> <div class='message'>{{ message }}</div>
<div class='buttons'> <div class='buttons'>
<button class='cancel' tabindex='2'>{{ cancel }}</button> {{ #showCancel }}
<button class='cancel' tabindex='2'>{{ cancel }}</button>
{{ /showCancel }}
<button class='ok' tabindex='1'>{{ ok }}</button> <button class='ok' tabindex='1'>{{ ok }}</button>
</div> </div>
</div> </div>
@ -346,16 +348,16 @@
</script> </script>
<script type='text/x-tmpl-mustache' id='key-verification'> <script type='text/x-tmpl-mustache' id='key-verification'>
<div class='container'> <div class='container'>
{{ ^has_their_key }} {{ ^hasTheirKey }}
<div class='placeholder'>{{ their_key_unknown }}</div> <div class='placeholder'>{{ theirKeyUnknown }}</div>
{{ /has_their_key }} {{ /hasTheirKey }}
{{ #has_their_key }} {{ #hasTheirKey }}
<label> {{ yourSafetyNumberWith }} </label> <label> {{ yourSafetyNumberWith }} </label>
<!--<div class='qr'></div>--> <!--<div class='qr'></div>-->
<div class='key'> <div class='key'>
{{ #chunks }} <span>{{ . }}</span> {{ /chunks }} {{ #chunks }} <span>{{ . }}</span> {{ /chunks }}
</div> </div>
{{ /has_their_key }} {{ /hasTheirKey }}
{{ verifyHelp }} {{ verifyHelp }}
<p> {{> link_to_support }} </p> <p> {{> link_to_support }} </p>
<div class='summary'> <div class='summary'>

View file

@ -27,6 +27,7 @@
render_attributes: function() { render_attributes: function() {
return { return {
message: this.message, message: this.message,
showCancel: Boolean(this.reject),
cancel: this.cancelText, cancel: this.cancelText,
ok: this.okText ok: this.okText
}; };

View file

@ -12,18 +12,20 @@
'click button.verify': 'toggleVerified', 'click button.verify': 'toggleVerified',
}, },
initialize: function(options) { initialize: function(options) {
this.our_number = textsecure.storage.user.getNumber(); this.ourNumber = textsecure.storage.user.getNumber();
if (options.newKey) { if (options.newKey) {
this.their_key = options.newKey; this.theirKey = options.newKey;
} }
Promise.all([ this.loadKeys().then(function() {
this.listenTo(this.model, 'change', this.render);
}.bind(this));
},
loadKeys: function() {
return Promise.all([
this.loadTheirKey(), this.loadTheirKey(),
this.loadOurKey(), this.loadOurKey(),
]).then(this.generateSecurityNumber.bind(this)) ]).then(this.generateSecurityNumber.bind(this))
.then(function() {
this.listenTo(this.model, 'change', this.render);
}.bind(this))
.then(this.render.bind(this)); .then(this.render.bind(this));
//.then(this.makeQRCode.bind(this)); //.then(this.makeQRCode.bind(this));
}, },
@ -31,40 +33,59 @@
// Per Lilia: We can't turn this on until it generates a Latin1 string, as is // Per Lilia: We can't turn this on until it generates a Latin1 string, as is
// required by the mobile clients. // required by the mobile clients.
new QRCode(this.$('.qr')[0]).makeCode( new QRCode(this.$('.qr')[0]).makeCode(
dcodeIO.ByteBuffer.wrap(this.our_key).toString('base64') dcodeIO.ByteBuffer.wrap(this.ourKey).toString('base64')
); );
}, },
loadTheirKey: function() { loadTheirKey: function() {
if (this.their_key) { return textsecure.storage.protocol.loadIdentityKey(
return Promise.resolve(this.their_key); this.model.id
} else { ).then(function(theirKey) {
return textsecure.storage.protocol.loadIdentityKey( this.theirKey = theirKey;
this.model.id }.bind(this));
).then(function(their_key) {
this.their_key = their_key;
}.bind(this));
}
}, },
loadOurKey: function() { loadOurKey: function() {
if (this.our_key) { return textsecure.storage.protocol.loadIdentityKey(
return Promise.resolve(this.our_key); this.ourNumber
} else { ).then(function(ourKey) {
return textsecure.storage.protocol.loadIdentityKey( this.ourKey = ourKey;
this.our_number }.bind(this));
).then(function(our_key) {
this.our_key = our_key;
}.bind(this));
}
}, },
generateSecurityNumber: function() { generateSecurityNumber: function() {
return new libsignal.FingerprintGenerator(5200).createFor( return new libsignal.FingerprintGenerator(5200).createFor(
this.our_number, this.our_key, this.model.id, this.their_key this.ourNumber, this.ourKey, this.model.id, this.theirKey
).then(function(securityNumber) { ).then(function(securityNumber) {
this.securityNumber = securityNumber; this.securityNumber = securityNumber;
}.bind(this)); }.bind(this));
}, },
onSafetyNumberChanged: function() {
this.model.getProfiles().then(this.loadKeys.bind(this));
var dialog = new Whisper.ConfirmationDialogView({
message: i18n('changedRightAfterVerify', this.model.getTitle(), this.model.getTitle()),
resolve: function() {},
});
dialog.$el.insertBefore(this.el);
dialog.focusCancel();
},
toggleVerified: function() { toggleVerified: function() {
this.model.toggleVerified(); this.$('button.verify').attr('disabled', true);
this.model.toggleVerified().catch(function(result) {
if (result instanceof Error) {
if (result.name === 'OutgoingIdentityKeyError') {
this.onSafetyNumberChanged();
}
} else {
var keyError = _.some(result.errors, function(error) {
return error.name === 'OutgoingIdentityKeyError';
});
if (keyError) {
this.onSafetyNumberChanged();
}
}
}.bind(this)).then(function() {
this.$('button.verify').removeAttr('disabled');
}.bind(this));
}, },
render_attributes: function() { render_attributes: function() {
var s = this.securityNumber; var s = this.securityNumber;
@ -80,11 +101,11 @@
return { return {
learnMore : i18n('learnMore'), learnMore : i18n('learnMore'),
their_key_unknown : i18n('theirIdentityUnknown'), theirKeyUnknown : i18n('theirIdentityUnknown'),
yourSafetyNumberWith : i18n('yourSafetyNumberWith', this.model.getTitle()), yourSafetyNumberWith : i18n('yourSafetyNumberWith', this.model.getTitle()),
verifyHelp : i18n('verifyHelp', this.model.getTitle()), verifyHelp : i18n('verifyHelp', this.model.getTitle()),
verifyButton : verifyButton, verifyButton : verifyButton,
has_their_key : this.their_key !== undefined, hasTheirKey : this.theirKey !== undefined,
chunks : chunks, chunks : chunks,
isVerified : isVerified, isVerified : isVerified,
verifiedStatus : verifiedStatus verifiedStatus : verifiedStatus

View file

@ -173,8 +173,10 @@
<div class="content"> <div class="content">
<div class='message'>{{ message }}</div> <div class='message'>{{ message }}</div>
<div class='buttons'> <div class='buttons'>
<button class='cancel'>{{ cancel }}</button> {{ #showCancel }}
<button class='ok'>{{ ok }}</button> <button class='cancel' tabindex='2'>{{ cancel }}</button>
{{ /showCancel }}
<button class='ok' tabindex='1'>{{ ok }}</button>
</div> </div>
</div> </div>
</script> </script>
@ -337,16 +339,17 @@
</script> </script>
<script type='text/x-tmpl-mustache' id='key-verification'> <script type='text/x-tmpl-mustache' id='key-verification'>
<div class='container'> <div class='container'>
{{ ^has_their_key }} {{ ^hasTheirKey }}
<div class='placeholder'>{{ their_key_unknown }}</div> <div class='placeholder'>{{ theirKeyUnknown }}</div>
{{ /has_their_key }} {{ /hasTheirKey }}
{{ #has_their_key }} {{ #hasTheirKey }}
<label> {{ yourSafetyNumberWith }} </label> <label> {{ yourSafetyNumberWith }} </label>
<!--<div class='qr'></div>--> <!--<div class='qr'></div>-->
<div class='key'> <div class='key'>
{{ #chunks }} <span>{{ . }}</span> {{ /chunks }} {{ #chunks }} <span>{{ . }}</span> {{ /chunks }}
</div> </div>
{{ /has_their_key }} {{ /hasTheirKey }}
{{ verifyHelp }}
<p> {{> link_to_support }} </p> <p> {{> link_to_support }} </p>
<div class='summary'> <div class='summary'>
{{ #isVerified }} {{ #isVerified }}