2020-10-30 20:34:04 +00:00
|
|
|
// Copyright 2016-2020 Signal Messenger, LLC
|
|
|
|
// SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
|
2018-11-02 18:02:53 +00:00
|
|
|
/* global libsignal, textsecure */
|
|
|
|
|
2020-11-04 17:42:50 +00:00
|
|
|
describe('Protocol Wrapper', function protocolWrapperDescribe() {
|
2018-07-21 21:51:20 +00:00
|
|
|
const store = textsecure.storage.protocol;
|
2020-11-04 17:42:50 +00:00
|
|
|
const identifier = '+15559999999';
|
2018-11-02 18:02:53 +00:00
|
|
|
|
2018-05-02 16:51:22 +00:00
|
|
|
this.timeout(5000);
|
2018-11-02 18:02:53 +00:00
|
|
|
|
2020-11-04 17:42:50 +00:00
|
|
|
before(async function thisNeeded() {
|
2018-05-02 16:51:22 +00:00
|
|
|
localStorage.clear();
|
2020-11-04 17:42:50 +00:00
|
|
|
this.identityKeyPair = await libsignal.KeyHelper.generateIdentityKeyPair();
|
|
|
|
await textsecure.storage.protocol.saveIdentity(
|
|
|
|
identifier,
|
|
|
|
this.identityKeyPair.pubKey
|
|
|
|
);
|
2018-05-02 16:51:22 +00:00
|
|
|
});
|
2018-11-02 18:02:53 +00:00
|
|
|
|
2018-07-21 21:51:20 +00:00
|
|
|
describe('processPreKey', () => {
|
2020-11-04 17:42:50 +00:00
|
|
|
beforeEach(function thisNeeded() {
|
2018-07-21 21:51:20 +00:00
|
|
|
const address = new libsignal.SignalProtocolAddress(identifier, 1);
|
2020-11-04 17:42:50 +00:00
|
|
|
this.builder = new libsignal.SessionBuilder(store, address);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('can process prekeys', async function thisNeeded() {
|
|
|
|
const signedPreKey = await libsignal.KeyHelper.generateSignedPreKey(
|
|
|
|
this.identityKeyPair,
|
|
|
|
123
|
|
|
|
);
|
|
|
|
|
|
|
|
await this.builder.processPreKey({
|
|
|
|
identityKey: this.identityKeyPair.pubKey,
|
|
|
|
registrationId: 1,
|
|
|
|
preKey: {
|
|
|
|
keyId: 1,
|
|
|
|
publicKey: this.identityKeyPair.pubKey,
|
|
|
|
},
|
|
|
|
signedPreKey: {
|
|
|
|
keyId: 123,
|
|
|
|
publicKey: signedPreKey.keyPair.pubKey,
|
|
|
|
signature: signedPreKey.signature,
|
|
|
|
},
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
it('rejects if the identity key changes', function thisNeeded() {
|
|
|
|
return this.builder
|
2018-05-02 16:51:22 +00:00
|
|
|
.processPreKey({
|
|
|
|
identityKey: textsecure.crypto.getRandomBytes(33),
|
|
|
|
})
|
2018-07-21 21:51:20 +00:00
|
|
|
.then(() => {
|
2018-05-20 03:53:12 +00:00
|
|
|
throw new Error('Allowed to overwrite identity key');
|
2018-05-02 16:51:22 +00:00
|
|
|
})
|
2018-07-21 21:51:20 +00:00
|
|
|
.catch(e => {
|
2018-05-02 16:51:22 +00:00
|
|
|
assert.strictEqual(e.message, 'Identity key changed');
|
2015-07-22 19:48:08 +00:00
|
|
|
});
|
|
|
|
});
|
2020-11-04 17:42:50 +00:00
|
|
|
|
|
|
|
it('rejects with a bad prekey signature', async function thisNeeded() {
|
|
|
|
const signedPreKey = await libsignal.KeyHelper.generateSignedPreKey(
|
|
|
|
this.identityKeyPair,
|
|
|
|
123
|
|
|
|
);
|
|
|
|
const bogusSignature = textsecure.crypto.getRandomBytes(64);
|
|
|
|
|
|
|
|
return this.builder
|
|
|
|
.processPreKey({
|
|
|
|
identityKey: this.identityKeyPair.pubKey,
|
|
|
|
signedPreKey: {
|
|
|
|
keyId: 123,
|
|
|
|
publicKey: signedPreKey.keyPair.pubKey,
|
|
|
|
signature: bogusSignature,
|
|
|
|
},
|
|
|
|
})
|
|
|
|
.then(() => {
|
|
|
|
throw new Error("Didn't reject an invalid signature");
|
|
|
|
})
|
|
|
|
.catch(e => {
|
|
|
|
assert.strictEqual(e.message, 'Signature verification failed');
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
it('rejects with a prekey signature for a different identity', async function thisNeeded() {
|
|
|
|
const bogusSignedPreKey = await libsignal.KeyHelper.generateSignedPreKey(
|
|
|
|
await libsignal.KeyHelper.generateIdentityKeyPair(),
|
|
|
|
123
|
|
|
|
);
|
|
|
|
|
|
|
|
return this.builder
|
|
|
|
.processPreKey({
|
|
|
|
identityKey: this.identityKeyPair.pubKey,
|
|
|
|
signedPreKey: {
|
|
|
|
keyId: 123,
|
|
|
|
publicKey: bogusSignedPreKey.keyPair.pubKey,
|
|
|
|
signature: bogusSignedPreKey.signature,
|
|
|
|
},
|
|
|
|
})
|
|
|
|
.then(() => {
|
|
|
|
throw new Error("Didn't reject an invalid signature");
|
|
|
|
})
|
|
|
|
.catch(e => {
|
|
|
|
assert.strictEqual(e.message, 'Signature verification failed');
|
|
|
|
});
|
|
|
|
});
|
2018-05-02 16:51:22 +00:00
|
|
|
});
|
2021-04-13 15:52:26 +00:00
|
|
|
|
|
|
|
describe('cleanOldMessageKeys', () => {
|
|
|
|
it('should clean old message keys', () => {
|
|
|
|
const messageKeys = {};
|
|
|
|
|
|
|
|
const LIMIT = 2000;
|
|
|
|
|
|
|
|
for (let i = 0; i < 2 * LIMIT; i += 1) {
|
|
|
|
messageKeys[i] = i;
|
|
|
|
}
|
|
|
|
|
|
|
|
libsignal.SessionCipher.cleanOldMessageKeys(messageKeys);
|
|
|
|
|
|
|
|
for (let i = 0; i < LIMIT; i += 1) {
|
|
|
|
assert(
|
|
|
|
!Object.prototype.hasOwnProperty.call(messageKeys, i),
|
|
|
|
`should delete old key ${i}`
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (let i = LIMIT; i < 2 * LIMIT; i += 1) {
|
|
|
|
assert(
|
|
|
|
Object.prototype.hasOwnProperty.call(messageKeys, i),
|
|
|
|
`should have fresh key ${i}`
|
|
|
|
);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
});
|
2015-07-22 19:48:08 +00:00
|
|
|
});
|