|
|
|
@ -667,58 +667,29 @@ MessageReceiver.prototype.extend({
|
|
|
|
|
async decrypt(envelope, ciphertext) {
|
|
|
|
|
let promise;
|
|
|
|
|
|
|
|
|
|
// We don't have source at this point yet (with sealed sender)
|
|
|
|
|
// This needs a massive cleanup!
|
|
|
|
|
const address = new libsignal.SignalProtocolAddress(
|
|
|
|
|
envelope.source,
|
|
|
|
|
envelope.sourceDevice
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
const ourNumber = textsecure.storage.user.getNumber();
|
|
|
|
|
const number = address.toString().split('.')[0];
|
|
|
|
|
const options = {};
|
|
|
|
|
|
|
|
|
|
// No limit on message keys if we're communicating with our other devices
|
|
|
|
|
if (ourNumber === number) {
|
|
|
|
|
options.messageKeysLimit = false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Will become obsolete
|
|
|
|
|
const sessionCipher = new libsignal.SessionCipher(
|
|
|
|
|
textsecure.storage.protocol,
|
|
|
|
|
address,
|
|
|
|
|
options
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
const me = {
|
|
|
|
|
number: ourNumber,
|
|
|
|
|
deviceId: parseInt(textsecure.storage.user.getDeviceId(), 10),
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
// Will become obsolete
|
|
|
|
|
const getCurrentSessionBaseKey = async () => {
|
|
|
|
|
const record = await sessionCipher.getRecord(address.toString());
|
|
|
|
|
if (!record) {
|
|
|
|
|
return null;
|
|
|
|
|
}
|
|
|
|
|
const openSession = record.getOpenSession();
|
|
|
|
|
if (!openSession) {
|
|
|
|
|
return null;
|
|
|
|
|
}
|
|
|
|
|
const { baseKey } = openSession.indexInfo;
|
|
|
|
|
return baseKey;
|
|
|
|
|
};
|
|
|
|
|
// Envelope.source will be null on UNIDENTIFIED_SENDER
|
|
|
|
|
// Don't use it there!
|
|
|
|
|
const address = new libsignal.SignalProtocolAddress(
|
|
|
|
|
envelope.source,
|
|
|
|
|
envelope.sourceDevice
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
// Will become obsolete
|
|
|
|
|
const captureActiveSession = async () => {
|
|
|
|
|
this.activeSessionBaseKey = await getCurrentSessionBaseKey(sessionCipher);
|
|
|
|
|
};
|
|
|
|
|
const lokiSessionCipher = new libloki.crypto.LokiSessionCipher(
|
|
|
|
|
textsecure.storage.protocol,
|
|
|
|
|
address
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
switch (envelope.type) {
|
|
|
|
|
case textsecure.protobuf.Envelope.Type.CIPHERTEXT:
|
|
|
|
|
window.log.info('message from', this.getEnvelopeId(envelope));
|
|
|
|
|
promise = captureActiveSession()
|
|
|
|
|
.then(() => sessionCipher.decryptWhisperMessage(ciphertext))
|
|
|
|
|
promise = lokiSessionCipher
|
|
|
|
|
.decryptWhisperMessage(ciphertext)
|
|
|
|
|
.then(this.unpad);
|
|
|
|
|
break;
|
|
|
|
|
case textsecure.protobuf.Envelope.Type.FRIEND_REQUEST: {
|
|
|
|
@ -735,25 +706,11 @@ MessageReceiver.prototype.extend({
|
|
|
|
|
}
|
|
|
|
|
case textsecure.protobuf.Envelope.Type.PREKEY_BUNDLE:
|
|
|
|
|
window.log.info('prekey message from', this.getEnvelopeId(envelope));
|
|
|
|
|
promise = captureActiveSession(sessionCipher).then(async () => {
|
|
|
|
|
if (!this.activeSessionBaseKey) {
|
|
|
|
|
try {
|
|
|
|
|
const buffer = dcodeIO.ByteBuffer.wrap(ciphertext);
|
|
|
|
|
await window.libloki.storage.verifyFriendRequestAcceptPreKey(
|
|
|
|
|
envelope.source,
|
|
|
|
|
buffer
|
|
|
|
|
);
|
|
|
|
|
} catch (e) {
|
|
|
|
|
await this.removeFromCache(envelope);
|
|
|
|
|
throw e;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return this.decryptPreKeyWhisperMessage(
|
|
|
|
|
ciphertext,
|
|
|
|
|
sessionCipher,
|
|
|
|
|
address
|
|
|
|
|
);
|
|
|
|
|
});
|
|
|
|
|
promise = this.decryptPreKeyWhisperMessage(
|
|
|
|
|
ciphertext,
|
|
|
|
|
lokiSessionCipher,
|
|
|
|
|
address
|
|
|
|
|
);
|
|
|
|
|
break;
|
|
|
|
|
case textsecure.protobuf.Envelope.Type.UNIDENTIFIED_SENDER: {
|
|
|
|
|
window.log.info('received unidentified sender message');
|
|
|
|
@ -856,72 +813,6 @@ MessageReceiver.prototype.extend({
|
|
|
|
|
window.log.info('Error getting conversation: ', envelope.source);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// lint hates anything after // (so /// is no good)
|
|
|
|
|
// *** BEGIN: session reset ***
|
|
|
|
|
|
|
|
|
|
// we have address in scope from parent scope
|
|
|
|
|
// seems to be the same input parameters
|
|
|
|
|
// going to comment out due to lint complaints
|
|
|
|
|
/*
|
|
|
|
|
const address = new libsignal.SignalProtocolAddress(
|
|
|
|
|
envelope.source,
|
|
|
|
|
envelope.sourceDevice
|
|
|
|
|
);
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
const restoreActiveSession = async () => {
|
|
|
|
|
const record = await sessionCipher.getRecord(address.toString());
|
|
|
|
|
if (!record) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
record.archiveCurrentState();
|
|
|
|
|
|
|
|
|
|
// NOTE: activeSessionBaseKey will be undefined here...
|
|
|
|
|
const sessionToRestore = record.sessions[this.activeSessionBaseKey];
|
|
|
|
|
record.promoteState(sessionToRestore);
|
|
|
|
|
record.updateSessionState(sessionToRestore);
|
|
|
|
|
await textsecure.storage.protocol.storeSession(
|
|
|
|
|
address.toString(),
|
|
|
|
|
record.serialize()
|
|
|
|
|
);
|
|
|
|
|
};
|
|
|
|
|
const deleteAllSessionExcept = async sessionBaseKey => {
|
|
|
|
|
const record = await sessionCipher.getRecord(address.toString());
|
|
|
|
|
if (!record) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
const sessionToKeep = record.sessions[sessionBaseKey];
|
|
|
|
|
record.sessions = {};
|
|
|
|
|
record.updateSessionState(sessionToKeep);
|
|
|
|
|
await textsecure.storage.protocol.storeSession(
|
|
|
|
|
address.toString(),
|
|
|
|
|
record.serialize()
|
|
|
|
|
);
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
if (conversation.isSessionResetOngoing()) {
|
|
|
|
|
const currentSessionBaseKey = await getCurrentSessionBaseKey(
|
|
|
|
|
sessionCipher
|
|
|
|
|
);
|
|
|
|
|
if (
|
|
|
|
|
this.activeSessionBaseKey &&
|
|
|
|
|
currentSessionBaseKey !== this.activeSessionBaseKey
|
|
|
|
|
) {
|
|
|
|
|
if (conversation.isSessionResetReceived()) {
|
|
|
|
|
await restoreActiveSession();
|
|
|
|
|
} else {
|
|
|
|
|
await deleteAllSessionExcept(currentSessionBaseKey);
|
|
|
|
|
await conversation.onNewSessionAdopted();
|
|
|
|
|
}
|
|
|
|
|
} else if (conversation.isSessionResetReceived()) {
|
|
|
|
|
await deleteAllSessionExcept(this.activeSessionBaseKey);
|
|
|
|
|
await conversation.onNewSessionAdopted();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// lint hates anything after // (so /// is no good)
|
|
|
|
|
// *** END ***
|
|
|
|
|
|
|
|
|
|
// Type here can actually be UNIDENTIFIED_SENDER even if
|
|
|
|
|
// the underlying message is FRIEND_REQUEST
|
|
|
|
|
if (
|
|
|
|
|