You cannot select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
	
	
		
			990 lines
		
	
	
		
			37 KiB
		
	
	
	
		
			JavaScript
		
	
			
		
		
	
	
			990 lines
		
	
	
		
			37 KiB
		
	
	
	
		
			JavaScript
		
	
| 'use strict';
 | |
| 
 | |
| describe("SignalProtocolStore", function() {
 | |
|   var identifier = '+5558675309';
 | |
|   var store;
 | |
|   var identityKey;
 | |
|   var testKey;
 | |
| 
 | |
|   function wrapDeferred(deferred) {
 | |
|     return new Promise(function(resolve, reject) {
 | |
|       return deferred.then(resolve, reject);
 | |
|     });
 | |
|   }
 | |
| 
 | |
|   before(function(done) {
 | |
|     store = textsecure.storage.protocol;
 | |
|     identityKey = {
 | |
|       pubKey: libsignal.crypto.getRandomBytes(33),
 | |
|       privKey: libsignal.crypto.getRandomBytes(32),
 | |
|     };
 | |
|     testKey = {
 | |
|       pubKey: libsignal.crypto.getRandomBytes(33),
 | |
|       privKey: libsignal.crypto.getRandomBytes(32),
 | |
|     };
 | |
| 
 | |
|     storage.put('registrationId', 1337);
 | |
|     storage.put('identityKey', identityKey);
 | |
|     storage.fetch().then(done, done);
 | |
|   });
 | |
| 
 | |
|   describe('getLocalRegistrationId', function() {
 | |
|     it('retrieves my registration id', function(done) {
 | |
|       store.getLocalRegistrationId().then(function(reg) {
 | |
|         assert.strictEqual(reg, 1337);
 | |
|       }).then(done, done);
 | |
|     });
 | |
|   });
 | |
|   describe('getIdentityKeyPair', function() {
 | |
|     it('retrieves my identity key', function(done) {
 | |
|       store.getIdentityKeyPair().then(function(key) {
 | |
|         assertEqualArrayBuffers(key.pubKey, identityKey.pubKey);
 | |
|         assertEqualArrayBuffers(key.privKey, identityKey.privKey);
 | |
|       }).then(done,done);
 | |
|     });
 | |
|   });
 | |
| 
 | |
| 
 | |
| 
 | |
|   var IdentityKeyRecord = Backbone.Model.extend({
 | |
|     database: Whisper.Database,
 | |
|     storeName: 'identityKeys'
 | |
|   });
 | |
|   describe('saveIdentity', function() {
 | |
|     var record = new IdentityKeyRecord({id: identifier});
 | |
|     var address = new libsignal.SignalProtocolAddress(identifier, 1);
 | |
| 
 | |
|     it('stores identity keys', function(done) {
 | |
|       store.saveIdentity(address.toString(), testKey.pubKey).then(function() {
 | |
|         return store.loadIdentityKey(identifier).then(function(key) {
 | |
|           assertEqualArrayBuffers(key, testKey.pubKey);
 | |
|         });
 | |
|       }).then(done,done);
 | |
|     });
 | |
|     it('allows key changes', function(done) {
 | |
|       var newIdentity = libsignal.crypto.getRandomBytes(33);
 | |
|       store.saveIdentity(address.toString(), testKey.pubKey).then(function() {
 | |
|         store.saveIdentity(address.toString(), newIdentity).then(function() {
 | |
|           done();
 | |
|         });
 | |
|       }).catch(done);
 | |
|     });
 | |
| 
 | |
|     describe('When there is no existing key (first use)', function() {
 | |
|       before(function(done) {
 | |
|         store.removeIdentityKey(identifier).then(function() {
 | |
|           store.saveIdentity(address.toString(), testKey.pubKey).then(function() {
 | |
|             record.fetch().then(function() { done(); });
 | |
|           });
 | |
|         });
 | |
|       });
 | |
|       it('marks the key firstUse', function() {
 | |
|         assert(record.get('firstUse'));
 | |
|       });
 | |
|       it('sets the timestamp', function() {
 | |
|         assert(record.get('timestamp'));
 | |
|       });
 | |
|       it('sets the verified status to DEFAULT', function() {
 | |
|         assert.strictEqual(record.get('verified'), store.VerifiedStatus.DEFAULT);
 | |
|       });
 | |
|     });
 | |
|     describe('When there is a different existing key (non first use)', function() {
 | |
|       var newIdentity = libsignal.crypto.getRandomBytes(33);
 | |
|       var oldTimestamp = Date.now();
 | |
|       before(function(done) {
 | |
|         record.save({
 | |
|           publicKey           : testKey.pubKey,
 | |
|           firstUse            : true,
 | |
|           timestamp           : oldTimestamp,
 | |
|           nonblockingApproval : false,
 | |
|           verified            : store.VerifiedStatus.DEFAULT
 | |
|         }).then(function() {
 | |
|           store.saveIdentity(address.toString(), newIdentity).then(function() {
 | |
|             record.fetch().then(function() { done(); });
 | |
|           });
 | |
|         });
 | |
|       });
 | |
|       it('marks the key not firstUse', function() {
 | |
|         assert(!record.get('firstUse'));
 | |
|       });
 | |
|       it('updates the timestamp', function() {
 | |
|         assert.notEqual(record.get('timestamp'), oldTimestamp);
 | |
|       });
 | |
| 
 | |
|       describe('The previous verified status was DEFAULT', function() {
 | |
|         before(function(done) {
 | |
|           record.save({
 | |
|             publicKey           : testKey.pubKey,
 | |
|             firstUse            : true,
 | |
|             timestamp           : oldTimestamp,
 | |
|             nonblockingApproval : false,
 | |
|             verified            : store.VerifiedStatus.DEFAULT
 | |
|           }).then(function() {
 | |
|             store.saveIdentity(address.toString(), newIdentity).then(function() {
 | |
|               record.fetch().then(function() { done(); });
 | |
|             });
 | |
|           });
 | |
|         });
 | |
|         it('sets the new key to unverified', function() {
 | |
|           assert.strictEqual(record.get('verified'), store.VerifiedStatus.DEFAULT);
 | |
|         });
 | |
|       });
 | |
|       describe('The previous verified status was VERIFIED', function() {
 | |
|         before(function(done) {
 | |
|           record.save({
 | |
|             publicKey           : testKey.pubKey,
 | |
|             firstUse            : true,
 | |
|             timestamp           : oldTimestamp,
 | |
|             nonblockingApproval : false,
 | |
|             verified            : store.VerifiedStatus.VERIFIED
 | |
|           }).then(function() {
 | |
|             store.saveIdentity(address.toString(), newIdentity).then(function() {
 | |
|               record.fetch().then(function() { done(); });
 | |
|             });
 | |
|           });
 | |
|         });
 | |
|         it('sets the new key to unverified', function() {
 | |
|           assert.strictEqual(record.get('verified'), store.VerifiedStatus.UNVERIFIED);
 | |
|         });
 | |
|       });
 | |
|       describe('The previous verified status was UNVERIFIED', function() {
 | |
|         before(function(done) {
 | |
|           record.save({
 | |
|             publicKey           : testKey.pubKey,
 | |
|             firstUse            : true,
 | |
|             timestamp           : oldTimestamp,
 | |
|             nonblockingApproval : false,
 | |
|             verified            : store.VerifiedStatus.UNVERIFIED
 | |
|           }).then(function() {
 | |
|             store.saveIdentity(address.toString(), newIdentity).then(function() {
 | |
|               record.fetch().then(function() { done(); });
 | |
|             });
 | |
|           });
 | |
|         });
 | |
|         it('sets the new key to unverified', function() {
 | |
|           assert.strictEqual(record.get('verified'), store.VerifiedStatus.UNVERIFIED);
 | |
|         });
 | |
|       });
 | |
|     });
 | |
|     describe('When the key has not changed', function() {
 | |
|       var oldTimestamp = Date.now();
 | |
|       before(function(done) {
 | |
|         record.save({
 | |
|           publicKey           : testKey.pubKey,
 | |
|           timestamp           : oldTimestamp,
 | |
|           nonblockingApproval : false,
 | |
|           verified            : store.VerifiedStatus.DEFAULT
 | |
|         }).then(function() { done(); });
 | |
|       });
 | |
|       describe('If it is marked firstUse', function() {
 | |
|         before(function(done) {
 | |
|           record.save({ firstUse: true }).then(function() { done(); });
 | |
|         });
 | |
|         it('nothing changes', function(done) {
 | |
|           store.saveIdentity(address.toString(), testKey.pubKey, true).then(function() {
 | |
|             record.fetch().then(function() {
 | |
|               assert(!record.get('nonblockingApproval'));
 | |
|               assert.strictEqual(record.get('timestamp'), oldTimestamp);
 | |
|               done();
 | |
|             });
 | |
|           });
 | |
|         });
 | |
|       });
 | |
|       describe('If it is not marked firstUse', function() {
 | |
|         before(function(done) {
 | |
|           record.save({ firstUse: false }).then(function() { done(); });
 | |
|         });
 | |
|         describe('If nonblocking approval is required', function() {
 | |
|           var now;
 | |
|           before(function(done) {
 | |
|             now = Date.now();
 | |
|             record.save({ timestamp: now }).then(function() { done(); });
 | |
|           });
 | |
|           it('sets non-blocking approval', function(done) {
 | |
|             store.saveIdentity(address.toString(), testKey.pubKey, true).then(function() {
 | |
|               record.fetch().then(function() {
 | |
|                 assert.strictEqual(record.get('nonblockingApproval'), true);
 | |
|                 assert.strictEqual(record.get('timestamp'), now);
 | |
|                 assert.strictEqual(record.get('firstUse'), false);
 | |
|                 done();
 | |
|               });
 | |
|             });
 | |
|           });
 | |
|         });
 | |
|       });
 | |
|     });
 | |
|   });
 | |
|   describe('saveIdentityWithAttributes', function() {
 | |
|     var now;
 | |
|     var record;
 | |
|     var validAttributes;
 | |
| 
 | |
|     before(function(done) {
 | |
|       now = Date.now();
 | |
|       record = new IdentityKeyRecord({id: identifier});
 | |
|       validAttributes = {
 | |
|         publicKey           : testKey.pubKey,
 | |
|         firstUse            : true,
 | |
|         timestamp           : now,
 | |
|         verified            : store.VerifiedStatus.VERIFIED,
 | |
|         nonblockingApproval : false
 | |
|       };
 | |
| 
 | |
|       store.removeIdentityKey(identifier).then(function() { done(); });
 | |
|     });
 | |
|     describe('with valid attributes', function() {
 | |
|       before(function(done) {
 | |
|         store.saveIdentityWithAttributes(identifier, validAttributes).then(function() {
 | |
|           return new Promise(function(resolve) {
 | |
|             record.fetch().then(resolve);
 | |
|           });
 | |
|         }).then(done, done);
 | |
|       });
 | |
| 
 | |
|       it('publicKey is saved', function() {
 | |
|           assertEqualArrayBuffers(record.get('publicKey'), testKey.pubKey);
 | |
|       });
 | |
|       it('firstUse is saved', function() {
 | |
|           assert.strictEqual(record.get('firstUse'), true);
 | |
|       });
 | |
|       it('timestamp is saved', function() {
 | |
|           assert.strictEqual(record.get('timestamp'), now);
 | |
|       });
 | |
|       it('verified is saved', function() {
 | |
|           assert.strictEqual(record.get('verified'), store.VerifiedStatus.VERIFIED);
 | |
|       });
 | |
|       it('nonblockingApproval is saved', function() {
 | |
|           assert.strictEqual(record.get('nonblockingApproval'), false);
 | |
|       });
 | |
|     });
 | |
|     describe('with invalid attributes', function() {
 | |
|       var attributes;
 | |
|       beforeEach(function() {
 | |
|         attributes = _.clone(validAttributes);
 | |
|       });
 | |
| 
 | |
|       function testInvalidAttributes(done) {
 | |
|         store.saveIdentityWithAttributes(identifier, attributes).then(function() {
 | |
|           done(new Error("saveIdentityWithAttributes should have failed"));
 | |
|         }, function() {
 | |
|           done(); // good. we expect to fail with invalid attributes.
 | |
|         });
 | |
|       }
 | |
| 
 | |
|       it('rejects an invalid publicKey', function(done) {
 | |
|         attributes.publicKey = 'a string';
 | |
|         testInvalidAttributes(done);
 | |
|       });
 | |
|       it('rejects invalid firstUse', function(done) {
 | |
|         attributes.firstUse = 0;
 | |
|         testInvalidAttributes(done);
 | |
|       });
 | |
|       it('rejects invalid timestamp', function(done) {
 | |
|         attributes.timestamp = NaN;
 | |
|         testInvalidAttributes(done);
 | |
|       });
 | |
|       it('rejects invalid verified', function(done) {
 | |
|         attributes.verified = null;
 | |
|         testInvalidAttributes(done);
 | |
|       });
 | |
|       it('rejects invalid nonblockingApproval', function(done) {
 | |
|         attributes.nonblockingApproval = 0;
 | |
|         testInvalidAttributes(done);
 | |
|       });
 | |
|     });
 | |
|   });
 | |
|   describe('setApproval', function() {
 | |
|     var record = new IdentityKeyRecord({id: identifier});
 | |
|     function fetchRecord() {
 | |
|       return new Promise(function(resolve) {
 | |
|         record.fetch().then(resolve);
 | |
|       });
 | |
|     }
 | |
|     it ('sets nonblockingApproval', function(done) {
 | |
|       store.setApproval(identifier, true).then(fetchRecord).then(function() {
 | |
|         assert.strictEqual(record.get('nonblockingApproval'), true);
 | |
|       }).then(done, done);
 | |
|     });
 | |
|   });
 | |
|   describe('setVerified', function() {
 | |
|     var record;
 | |
|     function saveRecordDefault() {
 | |
|       record = new IdentityKeyRecord({
 | |
|         id                  : identifier,
 | |
|         publicKey           : testKey.pubKey,
 | |
|         firstUse            : true,
 | |
|         timestamp           : Date.now(),
 | |
|         verified            : store.VerifiedStatus.DEFAULT,
 | |
|         nonblockingApproval : false
 | |
|       });
 | |
|       return new Promise(function(resolve, reject) {
 | |
|         record.save().then(resolve, reject);
 | |
|       });
 | |
|     }
 | |
|     function fetchRecord() {
 | |
|       return new Promise(function(resolve, reject) {
 | |
|         record.fetch().then(resolve, reject);
 | |
|       });
 | |
|     }
 | |
|     describe('with no public key argument', function() {
 | |
|       before(saveRecordDefault);
 | |
|       it ('updates the verified status', function() {
 | |
|         return store.setVerified(
 | |
|           identifier, store.VerifiedStatus.VERIFIED
 | |
|         ).then(fetchRecord).then(function() {
 | |
|           assert.strictEqual(record.get('verified'), store.VerifiedStatus.VERIFIED);
 | |
|           assertEqualArrayBuffers(record.get('publicKey'), testKey.pubKey);
 | |
|         });
 | |
|       });
 | |
|     });
 | |
|     describe('with the current public key', function() {
 | |
|       before(saveRecordDefault);
 | |
|       it ('updates the verified status', function() {
 | |
|         return store.setVerified(
 | |
|           identifier, store.VerifiedStatus.VERIFIED, testKey.pubKey
 | |
|         ).then(fetchRecord).then(function() {
 | |
|           assert.strictEqual(record.get('verified'), store.VerifiedStatus.VERIFIED);
 | |
|           assertEqualArrayBuffers(record.get('publicKey'), testKey.pubKey);
 | |
|         });
 | |
|       });
 | |
|     });
 | |
|     describe('with a mismatching public key', function() {
 | |
|       var newIdentity = libsignal.crypto.getRandomBytes(33);
 | |
|       before(saveRecordDefault);
 | |
|       it ('does not change the record.', function() {
 | |
|         return store.setVerified(
 | |
|           identifier, store.VerifiedStatus.VERIFIED, newIdentity
 | |
|         ).then(fetchRecord).then(function() {
 | |
|           assert.strictEqual(record.get('verified'), store.VerifiedStatus.DEFAULT);
 | |
|           assertEqualArrayBuffers(record.get('publicKey'), testKey.pubKey);
 | |
|         });
 | |
|       });
 | |
|     });
 | |
|   });
 | |
|   describe('processContactSyncVerificationState', function() {
 | |
|     var record;
 | |
|     var newIdentity = libsignal.crypto.getRandomBytes(33);
 | |
|     var keychangeTriggered;
 | |
| 
 | |
|     function fetchRecord() {
 | |
|       return wrapDeferred(record.fetch());
 | |
|     }
 | |
| 
 | |
|     beforeEach(function() {
 | |
|       keychangeTriggered = 0;
 | |
|       store.bind('keychange', function() {
 | |
|         keychangeTriggered++;
 | |
|       });
 | |
|     });
 | |
|     afterEach(function() {
 | |
|       store.unbind('keychange');
 | |
|     });
 | |
| 
 | |
|     describe('when the new verified status is DEFAULT', function() {
 | |
|       describe('when there is no existing record', function() {
 | |
|         before(function() {
 | |
|           record = new IdentityKeyRecord({ id: identifier });
 | |
|           return wrapDeferred(record.destroy());
 | |
|         });
 | |
| 
 | |
|         it ('does nothing', function() {
 | |
|           return store.processContactSyncVerificationState(
 | |
|             identifier, store.VerifiedStatus.DEFAULT, newIdentity
 | |
|           ).then(fetchRecord).then(function() {
 | |
|             // fetchRecord resolved so there is a record.
 | |
|             // Bad.
 | |
|             throw new Error("processContactSyncVerificationState should not save new records");
 | |
|           }, function() {
 | |
|             assert.strictEqual(keychangeTriggered, 0);
 | |
|           });
 | |
|         });
 | |
|       });
 | |
|       describe('when the record exists', function() {
 | |
|         describe('when the existing key is different', function() {
 | |
|           before(function() {
 | |
|             record = new IdentityKeyRecord({
 | |
|               id                  : identifier,
 | |
|               publicKey           : testKey.pubKey,
 | |
|               firstUse            : true,
 | |
|               timestamp           : Date.now(),
 | |
|               verified            : store.VerifiedStatus.VERIFIED,
 | |
|               nonblockingApproval : false
 | |
|             });
 | |
|             return wrapDeferred(record.save());
 | |
|           });
 | |
| 
 | |
|           it ('does not save the new identity (because this is a less secure state)', function() {
 | |
|             return store.processContactSyncVerificationState(
 | |
|               identifier, store.VerifiedStatus.DEFAULT, newIdentity
 | |
|             ).then(fetchRecord).then(function() {
 | |
|               assert.strictEqual(record.get('verified'), store.VerifiedStatus.VERIFIED);
 | |
|               assertEqualArrayBuffers(record.get('publicKey'), testKey.pubKey);
 | |
|               assert.strictEqual(keychangeTriggered, 0);
 | |
|             });
 | |
|           });
 | |
|         });
 | |
|         describe('when the existing key is the same but VERIFIED', function() {
 | |
|           before(function() {
 | |
|             record = new IdentityKeyRecord({
 | |
|               id                  : identifier,
 | |
|               publicKey           : testKey.pubKey,
 | |
|               firstUse            : true,
 | |
|               timestamp           : Date.now(),
 | |
|               verified            : store.VerifiedStatus.VERIFIED,
 | |
|               nonblockingApproval : false
 | |
|             });
 | |
|             return wrapDeferred(record.save());
 | |
|           });
 | |
| 
 | |
|           it ('updates the verified status', function() {
 | |
|             return store.processContactSyncVerificationState(
 | |
|               identifier, store.VerifiedStatus.DEFAULT, testKey.pubKey
 | |
|             ).then(fetchRecord).then(function() {
 | |
|               assert.strictEqual(record.get('verified'), store.VerifiedStatus.DEFAULT);
 | |
|               assertEqualArrayBuffers(record.get('publicKey'), testKey.pubKey);
 | |
|               assert.strictEqual(keychangeTriggered, 0);
 | |
|             });
 | |
|           });
 | |
|         });
 | |
|         describe('when the existing key is the same and already DEFAULT', function() {
 | |
|           before(function() {
 | |
|             record = new IdentityKeyRecord({
 | |
|               id                  : identifier,
 | |
|               publicKey           : testKey.pubKey,
 | |
|               firstUse            : true,
 | |
|               timestamp           : Date.now(),
 | |
|               verified            : store.VerifiedStatus.DEFAULT,
 | |
|               nonblockingApproval : false
 | |
|             });
 | |
|             return wrapDeferred(record.save());
 | |
|           });
 | |
| 
 | |
|           it ('does not hang', function() {
 | |
|             return store.processContactSyncVerificationState(
 | |
|               identifier, store.VerifiedStatus.DEFAULT, testKey.pubKey
 | |
|             ).then(fetchRecord).then(function() {
 | |
|               assert.strictEqual(keychangeTriggered, 0);
 | |
|             });
 | |
|           });
 | |
|         });
 | |
|       });
 | |
|     });
 | |
|     describe('when the new verified status is UNVERIFIED', function() {
 | |
|       describe('when there is no existing record', function() {
 | |
|         before(function() {
 | |
|           record = new IdentityKeyRecord({ id: identifier });
 | |
|           return wrapDeferred(record.destroy());
 | |
|         });
 | |
| 
 | |
|         it ('saves the new identity and marks it verified', function() {
 | |
|           return store.processContactSyncVerificationState(
 | |
|             identifier, store.VerifiedStatus.UNVERIFIED, newIdentity
 | |
|           ).then(fetchRecord).then(function() {
 | |
|             assert.strictEqual(record.get('verified'), store.VerifiedStatus.UNVERIFIED);
 | |
|             assertEqualArrayBuffers(record.get('publicKey'), newIdentity);
 | |
|             assert.strictEqual(keychangeTriggered, 0);
 | |
|           });
 | |
|         });
 | |
|       });
 | |
|       describe('when the record exists', function() {
 | |
|         describe('when the existing key is different', function() {
 | |
|           before(function() {
 | |
|             record = new IdentityKeyRecord({
 | |
|               id                  : identifier,
 | |
|               publicKey           : testKey.pubKey,
 | |
|               firstUse            : true,
 | |
|               timestamp           : Date.now(),
 | |
|               verified            : store.VerifiedStatus.VERIFIED,
 | |
|               nonblockingApproval : false
 | |
|             });
 | |
|             return wrapDeferred(record.save());
 | |
|           });
 | |
| 
 | |
|           it ('saves the new identity and marks it UNVERIFIED', function() {
 | |
|             return store.processContactSyncVerificationState(
 | |
|               identifier, store.VerifiedStatus.UNVERIFIED, newIdentity
 | |
|             ).then(fetchRecord).then(function() {
 | |
|               assert.strictEqual(record.get('verified'), store.VerifiedStatus.UNVERIFIED);
 | |
|               assertEqualArrayBuffers(record.get('publicKey'), newIdentity);
 | |
|               assert.strictEqual(keychangeTriggered, 1);
 | |
|             });
 | |
|           });
 | |
|         });
 | |
|         describe('when the key exists and is DEFAULT', function() {
 | |
|           before(function() {
 | |
|             record = new IdentityKeyRecord({
 | |
|               id                  : identifier,
 | |
|               publicKey           : testKey.pubKey,
 | |
|               firstUse            : true,
 | |
|               timestamp           : Date.now(),
 | |
|               verified            : store.VerifiedStatus.DEFAULT,
 | |
|               nonblockingApproval : false
 | |
|             });
 | |
|             return wrapDeferred(record.save());
 | |
|           });
 | |
| 
 | |
|           it ('updates the verified status', function() {
 | |
|             return store.processContactSyncVerificationState(
 | |
|               identifier, store.VerifiedStatus.UNVERIFIED, testKey.pubKey
 | |
|             ).then(fetchRecord).then(function() {
 | |
|               assert.strictEqual(record.get('verified'), store.VerifiedStatus.UNVERIFIED);
 | |
|               assertEqualArrayBuffers(record.get('publicKey'), testKey.pubKey);
 | |
|               assert.strictEqual(keychangeTriggered, 0);
 | |
|             });
 | |
|           });
 | |
|         });
 | |
|         describe('when the key exists and is already UNVERIFIED', function() {
 | |
|           before(function() {
 | |
|             record = new IdentityKeyRecord({
 | |
|               id                  : identifier,
 | |
|               publicKey           : testKey.pubKey,
 | |
|               firstUse            : true,
 | |
|               timestamp           : Date.now(),
 | |
|               verified            : store.VerifiedStatus.UNVERIFIED,
 | |
|               nonblockingApproval : false
 | |
|             });
 | |
|             return wrapDeferred(record.save());
 | |
|           });
 | |
| 
 | |
|           it ('does not hang', function() {
 | |
|             return store.processContactSyncVerificationState(
 | |
|               identifier, store.VerifiedStatus.UNVERIFIED, testKey.pubKey
 | |
|             ).then(fetchRecord).then(function() {
 | |
|               assert.strictEqual(keychangeTriggered, 0);
 | |
|             });
 | |
|           });
 | |
|         });
 | |
|       });
 | |
|     });
 | |
|     describe('when the new verified status is VERIFIED', function() {
 | |
|       describe('when there is no existing record', function() {
 | |
|         before(function() {
 | |
|           record = new IdentityKeyRecord({ id: identifier });
 | |
|           return new Promise(function(resolve, reject) {
 | |
|             record.destroy().then(resolve, reject);
 | |
|           });
 | |
|         });
 | |
| 
 | |
|         it ('saves the new identity and marks it verified', function() {
 | |
|           return store.processContactSyncVerificationState(
 | |
|             identifier, store.VerifiedStatus.VERIFIED, newIdentity
 | |
|           ).then(fetchRecord).then(function() {
 | |
|             assert.strictEqual(record.get('verified'), store.VerifiedStatus.VERIFIED);
 | |
|             assertEqualArrayBuffers(record.get('publicKey'), newIdentity);
 | |
|             assert.strictEqual(keychangeTriggered, 0);
 | |
|           });
 | |
|         });
 | |
|       });
 | |
|       describe('when the record exists', function() {
 | |
|         describe('when the existing key is different', function() {
 | |
|           before(function() {
 | |
|             record = new IdentityKeyRecord({
 | |
|               id                  : identifier,
 | |
|               publicKey           : testKey.pubKey,
 | |
|               firstUse            : true,
 | |
|               timestamp           : Date.now(),
 | |
|               verified            : store.VerifiedStatus.VERIFIED,
 | |
|               nonblockingApproval : false
 | |
|             });
 | |
|             return wrapDeferred(record.save());
 | |
|           });
 | |
| 
 | |
|           it ('saves the new identity and marks it VERIFIED', function() {
 | |
|             return store.processContactSyncVerificationState(
 | |
|               identifier, store.VerifiedStatus.VERIFIED, newIdentity
 | |
|             ).then(fetchRecord).then(function() {
 | |
|               assert.strictEqual(record.get('verified'), store.VerifiedStatus.VERIFIED);
 | |
|               assertEqualArrayBuffers(record.get('publicKey'), newIdentity);
 | |
|               assert.strictEqual(keychangeTriggered, 1);
 | |
|             });
 | |
|           });
 | |
|         });
 | |
|         describe('when the existing key is the same but UNVERIFIED', function() {
 | |
|           before(function() {
 | |
|             record = new IdentityKeyRecord({
 | |
|               id                  : identifier,
 | |
|               publicKey           : testKey.pubKey,
 | |
|               firstUse            : true,
 | |
|               timestamp           : Date.now(),
 | |
|               verified            : store.VerifiedStatus.UNVERIFIED,
 | |
|               nonblockingApproval : false
 | |
|             });
 | |
|             return wrapDeferred(record.save());
 | |
|           });
 | |
| 
 | |
|           it ('saves the identity and marks it verified', function() {
 | |
|             return store.processContactSyncVerificationState(
 | |
|               identifier, store.VerifiedStatus.VERIFIED, testKey.pubKey
 | |
|             ).then(fetchRecord).then(function() {
 | |
|               assert.strictEqual(record.get('verified'), store.VerifiedStatus.VERIFIED);
 | |
|               assertEqualArrayBuffers(record.get('publicKey'), testKey.pubKey);
 | |
|               assert.strictEqual(keychangeTriggered, 0);
 | |
|             });
 | |
|           });
 | |
|         });
 | |
|         describe('when the existing key is the same and already VERIFIED', function() {
 | |
|           before(function() {
 | |
|             record = new IdentityKeyRecord({
 | |
|               id                  : identifier,
 | |
|               publicKey           : testKey.pubKey,
 | |
|               firstUse            : true,
 | |
|               timestamp           : Date.now(),
 | |
|               verified            : store.VerifiedStatus.VERIFIED,
 | |
|               nonblockingApproval : false
 | |
|             });
 | |
|             return wrapDeferred(record.save());
 | |
|           });
 | |
| 
 | |
|           it ('does not hang', function() {
 | |
|             return store.processContactSyncVerificationState(
 | |
|               identifier, store.VerifiedStatus.VERIFIED, testKey.pubKey
 | |
|             ).then(fetchRecord).then(function() {
 | |
|               assert.strictEqual(keychangeTriggered, 0);
 | |
|             });
 | |
|           });
 | |
|         });
 | |
|       });
 | |
|     });
 | |
| 
 | |
|   });
 | |
| 
 | |
|   describe('isUntrusted', function() {
 | |
|     it('returns false if identity key old enough', function() {
 | |
|       var record = new IdentityKeyRecord({
 | |
|         id                  : identifier,
 | |
|         publicKey           : testKey.pubKey,
 | |
|         timestamp           : Date.now() - 10 * 1000 * 60,
 | |
|         verified            : store.VerifiedStatus.DEFAULT,
 | |
|         firstUse            : false,
 | |
|         nonblockingApproval : false
 | |
|       });
 | |
|       return wrapDeferred(record.save()).then(function() {
 | |
|         return store.isUntrusted(identifier);
 | |
|       }).then(function(untrusted) {
 | |
|         assert.strictEqual(untrusted, false);
 | |
|       });
 | |
|     });
 | |
| 
 | |
|     it('returns false if new but nonblockingApproval is true', function() {
 | |
|       var record = new IdentityKeyRecord({
 | |
|         id                  : identifier,
 | |
|         publicKey           : testKey.pubKey,
 | |
|         timestamp           : Date.now(),
 | |
|         verified            : store.VerifiedStatus.DEFAULT,
 | |
|         firstUse            : false,
 | |
|         nonblockingApproval : true
 | |
|       });
 | |
|       return wrapDeferred(record.save()).then(function() {
 | |
|         return store.isUntrusted(identifier);
 | |
|       }).then(function(untrusted) {
 | |
|         assert.strictEqual(untrusted, false);
 | |
|       });
 | |
|     });
 | |
| 
 | |
|     it('returns false if new but firstUse is true', function() {
 | |
|       var record = new IdentityKeyRecord({
 | |
|         id                  : identifier,
 | |
|         publicKey           : testKey.pubKey,
 | |
|         timestamp           : Date.now(),
 | |
|         verified            : store.VerifiedStatus.DEFAULT,
 | |
|         firstUse            : true,
 | |
|         nonblockingApproval : false
 | |
|       });
 | |
|       return wrapDeferred(record.save()).then(function() {
 | |
|         return store.isUntrusted(identifier);
 | |
|       }).then(function(untrusted) {
 | |
|         assert.strictEqual(untrusted, false);
 | |
|       });
 | |
|     });
 | |
| 
 | |
|     it('returns true if new, and no flags are set', function() {
 | |
|       var record = new IdentityKeyRecord({
 | |
|         id                  : identifier,
 | |
|         publicKey           : testKey.pubKey,
 | |
|         timestamp           : Date.now(),
 | |
|         verified            : store.VerifiedStatus.DEFAULT,
 | |
|         firstUse            : false,
 | |
|         nonblockingApproval : false
 | |
|       });
 | |
|       return wrapDeferred(record.save()).then(function() {
 | |
|         return store.isUntrusted(identifier);
 | |
|       }).then(function(untrusted) {
 | |
|         assert.strictEqual(untrusted, true);
 | |
|       });
 | |
|     });
 | |
|   });
 | |
| 
 | |
|   describe('getVerified', function() {
 | |
|     before(function(done) {
 | |
|       store.setVerified(identifier, store.VerifiedStatus.VERIFIED).then(done, done);
 | |
|     });
 | |
|     it ('resolves to the verified status', function(done) {
 | |
|       store.getVerified(identifier).then(function(result) {
 | |
|         assert.strictEqual(result, store.VerifiedStatus.VERIFIED);
 | |
|       }).then(done, done);
 | |
|     });
 | |
|   });
 | |
|   describe('isTrustedIdentity', function() {
 | |
|     var address = new libsignal.SignalProtocolAddress(identifier, 1);
 | |
|     describe('When invalid direction is given', function(done) {
 | |
|       it('should fail', function(done) {
 | |
|         store.isTrustedIdentity(identifier, testKey.pubKey).then(function() {
 | |
|           done(new Error('isTrustedIdentity should have failed'));
 | |
|         }).catch(function(e) {
 | |
|           done();
 | |
|         });
 | |
|       });
 | |
|     });
 | |
|     describe('When direction is RECEIVING', function() {
 | |
|       it('always returns true', function(done) {
 | |
|         var newIdentity = libsignal.crypto.getRandomBytes(33);
 | |
|         store.saveIdentity(address.toString(), testKey.pubKey).then(function() {
 | |
|           store.isTrustedIdentity(identifier, newIdentity, store.Direction.RECEIVING).then(function(trusted) {
 | |
|             if (trusted) {
 | |
|               done();
 | |
|             } else {
 | |
|               done(new Error('isTrusted returned false when receiving'));
 | |
|             }
 | |
|           }).catch(done);
 | |
|         });
 | |
|       });
 | |
|     });
 | |
|     describe('When direction is SENDING', function() {
 | |
|       describe('When there is no existing key (first use)', function() {
 | |
|         before(function(done) {
 | |
|           store.removeIdentityKey(identifier).then(function() {
 | |
|             done();
 | |
|           });
 | |
|         });
 | |
|         it('returns true', function(done) {
 | |
|           var newIdentity = libsignal.crypto.getRandomBytes(33);
 | |
|           store.isTrustedIdentity(identifier, newIdentity, store.Direction.SENDING).then(function(trusted) {
 | |
|             if (trusted) {
 | |
|               done();
 | |
|             } else {
 | |
|               done(new Error('isTrusted returned false on first use'));
 | |
|             }
 | |
|           }).catch(done);
 | |
|         });
 | |
|       });
 | |
|       describe('When there is an existing key', function() {
 | |
|         before(function(done) {
 | |
|           store.saveIdentity(address.toString(), testKey.pubKey).then(function() {
 | |
|             done();
 | |
|           });
 | |
|         });
 | |
|         describe('When the existing key is different', function() {
 | |
|           it('returns false', function(done) {
 | |
|             var newIdentity = libsignal.crypto.getRandomBytes(33);
 | |
|             store.isTrustedIdentity(identifier, newIdentity, store.Direction.SENDING).then(function(trusted) {
 | |
|               if (trusted) {
 | |
|                 done(new Error('isTrusted returned true on untrusted key'));
 | |
|               } else {
 | |
|                 done();
 | |
|               }
 | |
|             }).catch(done);
 | |
|           });
 | |
|         });
 | |
|         describe('When the existing key matches the new key', function() {
 | |
|           var newIdentity = libsignal.crypto.getRandomBytes(33);
 | |
|           before(function(done) {
 | |
|             store.saveIdentity(address.toString(), newIdentity).then(function() {
 | |
|               done();
 | |
|             });
 | |
|           });
 | |
|           it('returns false if keys match but we just received this new identiy', function(done) {
 | |
|             store.isTrustedIdentity(identifier, newIdentity, store.Direction.SENDING).then(function(trusted) {
 | |
|               if (trusted) {
 | |
|                 done(new Error('isTrusted returned true on untrusted key'));
 | |
|               } else {
 | |
|                 done();
 | |
|               }
 | |
|             }).catch(done);
 | |
|           });
 | |
|           it('returns true if we have already approved identity', function(done) {
 | |
|             store.saveIdentity(address.toString(), newIdentity, true).then(function() {
 | |
|               store.isTrustedIdentity(identifier, newIdentity, store.Direction.SENDING).then(function(trusted) {
 | |
|                 if (trusted) {
 | |
|                   done();
 | |
|                 } else {
 | |
|                   done(new Error('isTrusted returned false on an approved key'));
 | |
|                 }
 | |
|               }).catch(done);
 | |
|             });
 | |
|           });
 | |
|         });
 | |
|       });
 | |
|     });
 | |
|   });
 | |
|   describe('storePreKey', function() {
 | |
|     it('stores prekeys', function(done) {
 | |
|       store.storePreKey(1, testKey).then(function() {
 | |
|         return store.loadPreKey(1).then(function(key) {
 | |
|           assertEqualArrayBuffers(key.pubKey, testKey.pubKey);
 | |
|           assertEqualArrayBuffers(key.privKey, testKey.privKey);
 | |
|         });
 | |
|       }).then(done,done);
 | |
|     });
 | |
|   });
 | |
|   describe('removePreKey', function() {
 | |
|     before(function(done) {
 | |
|       store.storePreKey(2, testKey).then(done);
 | |
|     });
 | |
|     it('deletes prekeys', function(done) {
 | |
|       store.removePreKey(2, testKey).then(function() {
 | |
|         return store.loadPreKey(2).then(function(key) {
 | |
|           assert.isUndefined(key);
 | |
|         });
 | |
|       }).then(done,done);
 | |
|     });
 | |
|   });
 | |
|   describe('storeSignedPreKey', function() {
 | |
|     it('stores signed prekeys', function(done) {
 | |
|       store.storeSignedPreKey(3, testKey).then(function() {
 | |
|         return store.loadSignedPreKey(3).then(function(key) {
 | |
|           assertEqualArrayBuffers(key.pubKey, testKey.pubKey);
 | |
|           assertEqualArrayBuffers(key.privKey, testKey.privKey);
 | |
|         });
 | |
|       }).then(done,done);
 | |
|     });
 | |
|   });
 | |
|   describe('removeSignedPreKey', function() {
 | |
|     before(function(done) {
 | |
|       store.storeSignedPreKey(4, testKey).then(done);
 | |
|     });
 | |
|     it('deletes signed prekeys', function(done) {
 | |
|       store.removeSignedPreKey(4, testKey).then(function() {
 | |
|         return store.loadSignedPreKey(4).then(function(key) {
 | |
|           assert.isUndefined(key);
 | |
|         });
 | |
|       }).then(done,done);
 | |
|     });
 | |
|   });
 | |
|   describe('storeSession', function() {
 | |
|     it('stores sessions', function(done) {
 | |
|       var testRecord = "an opaque string";
 | |
|       store.storeSession(identifier + '.1', testRecord).then(function() {
 | |
|         return store.loadSession(identifier + '.1').then(function(record) {
 | |
|           assert.deepEqual(record, testRecord);
 | |
|         });
 | |
|       }).then(done,done);
 | |
|     });
 | |
|   });
 | |
|   describe('removeAllSessions', function() {
 | |
|     it('removes all sessions for a number', function(done) {
 | |
|       var testRecord = "an opaque string";
 | |
|       var devices = [1, 2, 3].map(function(deviceId) {
 | |
|         return [identifier, deviceId].join('.');
 | |
|       });
 | |
|       var promise = Promise.resolve();
 | |
|       devices.forEach(function(encodedNumber) {
 | |
|         promise = promise.then(function() {
 | |
|           return store.storeSession(encodedNumber, testRecord + encodedNumber);
 | |
|         });
 | |
|       });
 | |
|       promise.then(function() {
 | |
|         return store.removeAllSessions(identifier).then(function(record) {
 | |
|           return Promise.all(devices.map(store.loadSession.bind(store))).then(function(records) {
 | |
|             for (var i in records) {
 | |
|               assert.isUndefined(records[i]);
 | |
|             };
 | |
|           });
 | |
|         });
 | |
|       }).then(done,done);
 | |
|     });
 | |
|   });
 | |
|   describe('clearSessionStore', function() {
 | |
|     it ('clears the session store', function(done) {
 | |
|       var testRecord = "an opaque string";
 | |
|       store.storeSession(identifier + '.1', testRecord).then(function() {
 | |
|         return store.clearSessionStore().then(function() {
 | |
|           return store.loadSession(identifier + '.1').then(function(record) {
 | |
|             assert.isUndefined(record);
 | |
|           });
 | |
|         });
 | |
|       }).then(done,done);
 | |
| 
 | |
|     });
 | |
|   });
 | |
|   describe('getDeviceIds', function() {
 | |
|     it('returns deviceIds for a number', function(done) {
 | |
|       var testRecord = "an opaque string";
 | |
|       var devices = [1, 2, 3].map(function(deviceId) {
 | |
|         return [identifier, deviceId].join('.');
 | |
|       });
 | |
|       var promise = Promise.resolve();
 | |
|       devices.forEach(function(encodedNumber) {
 | |
|         promise = promise.then(function() {
 | |
|           return store.storeSession(encodedNumber, testRecord + encodedNumber);
 | |
|         });
 | |
|       });
 | |
|       promise.then(function() {
 | |
|         return store.getDeviceIds(identifier).then(function(deviceIds) {
 | |
|           assert.sameMembers(deviceIds, [1, 2, 3]);
 | |
|         });
 | |
|       }).then(done,done);
 | |
|     });
 | |
|     it('returns empty array for a number with no device ids', function(done) {
 | |
|       return store.getDeviceIds('foo').then(function(deviceIds) {
 | |
|         assert.sameMembers(deviceIds,[]);
 | |
|       }).then(done,done);
 | |
|     });
 | |
|   });
 | |
| 
 | |
|   describe('Not yet processed messages', function() {
 | |
|     beforeEach(function() {
 | |
|       return store.getAllUnprocessed().then(function(items) {
 | |
|         return Promise.all(_.map(items, function(item) {
 | |
|           return store.removeUnprocessed(item.id);
 | |
|         }));
 | |
|       }).then(function() {
 | |
|         return store.getAllUnprocessed();
 | |
|       }).then(function(items) {
 | |
|         assert.strictEqual(items.length, 0);
 | |
|       });
 | |
|     });
 | |
| 
 | |
|     it('adds two and gets them back', function() {
 | |
|       return Promise.all([
 | |
|         store.addUnprocessed({id: 2, name: 'second', timestamp: 2}),
 | |
|         store.addUnprocessed({id: 3, name: 'third', timestamp: 3}),
 | |
|         store.addUnprocessed({id: 1, name: 'first', timestamp: 1})
 | |
|       ]).then(function() {
 | |
|         return store.getAllUnprocessed();
 | |
|       }).then(function(items) {
 | |
|         assert.strictEqual(items.length, 3);
 | |
| 
 | |
|         // they are in the proper order because the collection comparator is 'timestamp'
 | |
|         assert.strictEqual(items[0].name, 'first');
 | |
|         assert.strictEqual(items[1].name, 'second');
 | |
|         assert.strictEqual(items[2].name, 'third');
 | |
|       });
 | |
|     });
 | |
| 
 | |
|     it('updateUnprocessed successfully updates only part of itme', function() {
 | |
|       var id = 1;
 | |
|       return store.addUnprocessed({id: id, name: 'first', timestamp: 1}).then(function() {
 | |
|         return store.updateUnprocessed(id, {name: 'updated'});
 | |
|       }).then(function() {
 | |
|         return store.getAllUnprocessed();
 | |
|       }).then(function(items) {
 | |
|         assert.strictEqual(items.length, 1);
 | |
|         assert.strictEqual(items[0].name, 'updated');
 | |
|         assert.strictEqual(items[0].timestamp, 1);
 | |
|       });
 | |
|     });
 | |
| 
 | |
|     it('removeUnprocessed successfully deletes item', function() {
 | |
|       var id = 1;
 | |
|       return store.addUnprocessed({id: id, name: 'first', timestamp: 1}).then(function() {
 | |
|         return store.removeUnprocessed(id);
 | |
|       }).then(function() {
 | |
|         return store.getAllUnprocessed();
 | |
|       }).then(function(items) {
 | |
|         assert.strictEqual(items.length, 0);
 | |
|       });
 | |
|     });
 | |
|   });
 | |
| 
 | |
| });
 |