/* * vim: ts=4:sw=4:expandtab */ 'use strict'; describe("SignalProtocolStore", function() { before(function(done) { storage.put('registrationId', 1337); storage.put('identityKey', identityKey); storage.fetch().then(done, done); }); var store = textsecure.storage.protocol; var identifier = '+5558675309'; var identityKey = { pubKey: libsignal.crypto.getRandomBytes(33), privKey: libsignal.crypto.getRandomBytes(32), }; var testKey = { pubKey: libsignal.crypto.getRandomBytes(33), privKey: libsignal.crypto.getRandomBytes(32), }; 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); }); }); describe('saveIdentity', function() { it('stores identity keys', function(done) { store.saveIdentity(identifier, testKey.pubKey).then(function() { return store.loadIdentityKey(identifier).then(function(key) { assertEqualArrayBuffers(key, testKey.pubKey); }); }).then(done,done); }); it('rejects on key change', function(done) { var newIdentity = libsignal.crypto.getRandomBytes(33); store.saveIdentity(identifier, testKey.pubKey).then(function() { store.saveIdentity(identifier, newIdentity).then(function() { done(new Error('Allowed to overwrite identity key')); }).catch(function(e) { assert(e instanceof Error); done(); }); }); }); }); describe('isTrustedIdentity', function() { it('returns true if a key is trusted', function(done) { store.saveIdentity(identifier, testKey.pubKey).then(function() { store.isTrustedIdentity(identifier, testKey.pubKey).then(function(trusted) { if (trusted) { done(); } else { done(new Error('Allowed to overwrite identity key')); } }).catch(done); }); }); it('returns false if a key is untrusted', function(done) { var newIdentity = libsignal.crypto.getRandomBytes(33); store.saveIdentity(identifier, testKey.pubKey).then(function() { store.isTrustedIdentity(identifier, newIdentity).then(function(trusted) { if (trusted) { done(new Error('Allowed to overwrite identity key')); } else { done(); } }).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); }); }); });