@ -717,11 +717,72 @@ MessageReceiver.prototype.extend({
deviceId : parseInt ( textsecure . storage . user . getDeviceId ( ) , 10 ) ,
deviceId : parseInt ( textsecure . storage . user . getDeviceId ( ) , 10 ) ,
} ;
} ;
let conversation ;
try {
conversation = await window . ConversationController . getOrCreateAndWait ( envelope . source , 'private' ) ;
} catch ( e ) {
window . log . info ( 'Error getting conversation: ' , envelope . source ) ;
}
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
} ;
const captureActiveSession = async ( ) => {
this . activeSessionBaseKey = await getCurrentSessionBaseKey ( sessionCipher ) ;
} ;
const restoreActiveSession = async ( ) => {
const record = await sessionCipher . getRecord ( address . toString ( ) ) ;
if ( ! record )
return
record . archiveCurrentState ( ) ;
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 ( ) ) ;
} ;
let handleSessionReset ;
if ( conversation . isSessionResetOngoing ( ) ) {
handleSessionReset = async ( result ) => {
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 ( ) ;
}
return result ;
} ;
} else {
handleSessionReset = async ( result ) => result ;
}
switch ( envelope . type ) {
switch ( envelope . type ) {
case textsecure . protobuf . Envelope . Type . CIPHERTEXT :
case textsecure . protobuf . Envelope . Type . CIPHERTEXT :
window . log . info ( 'message from' , this . getEnvelopeId ( envelope ) ) ;
window . log . info ( 'message from' , this . getEnvelopeId ( envelope ) ) ;
promise = sessionCipher . decryptWhisperMessage ( ciphertext )
promise = captureActiveSession ( )
. then ( this . unpad ) ;
. then ( ( ) => sessionCipher . decryptWhisperMessage ( ciphertext ) )
. then ( this . unpad )
. then ( handleSessionReset ) ;
break ;
break ;
case textsecure . protobuf . Envelope . Type . FRIEND _REQUEST : {
case textsecure . protobuf . Envelope . Type . FRIEND _REQUEST : {
window . log . info ( 'friend-request message from ' , envelope . source ) ;
window . log . info ( 'friend-request message from ' , envelope . source ) ;
@ -731,11 +792,13 @@ MessageReceiver.prototype.extend({
}
}
case textsecure . protobuf . Envelope . Type . PREKEY _BUNDLE :
case textsecure . protobuf . Envelope . Type . PREKEY _BUNDLE :
window . log . info ( 'prekey message from' , this . getEnvelopeId ( envelope ) ) ;
window . log . info ( 'prekey message from' , this . getEnvelopeId ( envelope ) ) ;
promise = this . decryptPreKeyWhisperMessage (
promise = captureActiveSession ( sessionCipher )
ciphertext ,
. then ( ( ) => this . decryptPreKeyWhisperMessage (
sessionCipher ,
ciphertext ,
address
sessionCipher ,
) ;
address
) )
. then ( handleSessionReset ) ;
break ;
break ;
case textsecure . protobuf . Envelope . Type . UNIDENTIFIED _SENDER :
case textsecure . protobuf . Envelope . Type . UNIDENTIFIED _SENDER :
window . log . info ( 'received unidentified sender message' ) ;
window . log . info ( 'received unidentified sender message' ) ;
@ -1282,19 +1345,42 @@ MessageReceiver.prototype.extend({
async handleEndSession ( number ) {
async handleEndSession ( number ) {
window . log . info ( 'got end session' ) ;
window . log . info ( 'got end session' ) ;
const deviceIds = await textsecure . storage . protocol . getDeviceIds ( number ) ;
const deviceIds = await textsecure . storage . protocol . getDeviceIds ( number ) ;
const identityKey = StringView . hexToArrayBuffer ( number ) ;
let conversation ;
try {
conversation = window . ConversationController . get ( number ) ;
} catch ( e ) {
window . log . error ( 'Error getting conversation: ' , number ) ;
}
return Promise . all (
// Bail early if a session reset is already ongoing
deviceIds . map ( deviceId => {
if ( conversation . isSessionResetOngoing ( ) ) {
return ;
}
await Promise . all (
deviceIds . map ( async deviceId => {
const address = new libsignal . SignalProtocolAddress ( number , deviceId ) ;
const address = new libsignal . SignalProtocolAddress ( number , deviceId ) ;
const sessionCipher = new libsignal . SessionCipher (
// Instead of deleting the sessions now,
// we process the new prekeys and initiate a new session.
// The old sessions will get deleted once the correspondant
// has switch the the new session.
const [ preKey , signedPreKey ] = await Promise . all ( [
textsecure . storage . protocol . loadContactPreKey ( number ) ,
textsecure . storage . protocol . loadContactSignedPreKey ( number ) ,
] ) ;
if ( preKey === undefined || signedPreKey === undefined ) {
return ;
}
const device = { identityKey , deviceId , preKey , signedPreKey , registrationId : 0 }
const builder = new libsignal . SessionBuilder (
textsecure . storage . protocol ,
textsecure . storage . protocol ,
address
address
) ;
) ;
builder . processPreKey ( device ) ;
window . log . info ( 'deleting sessions for' , address . toString ( ) ) ;
return sessionCipher . deleteAllSessionsForDevice ( ) ;
} )
} )
) ;
) ;
await conversation . onSessionResetReceived ( ) ;
} ,
} ,
processDecrypted ( envelope , decrypted , source ) {
processDecrypted ( envelope , decrypted , source ) {
/* eslint-disable no-bitwise, no-param-reassign */
/* eslint-disable no-bitwise, no-param-reassign */