fix: Use stanza receival timestamps to guard against stale kex messages
	
		
			
	
		
	
	
		
	
		
			All checks were successful
		
		
	
	
		
			
				
	
				ci/woodpecker/push/woodpecker Pipeline was successful
				
			
		
		
	
	
				
					
				
			
		
			All checks were successful
		
		
	
	ci/woodpecker/push/woodpecker Pipeline was successful
				
			This commit is contained in:
		
							parent
							
								
									0826d043d5
								
							
						
					
					
						commit
						1472624b1d
					
				| @ -119,6 +119,8 @@ void main() async { | |||||||
|     aliceDevice.id, |     aliceDevice.id, | ||||||
|     // The deserialised keys |     // The deserialised keys | ||||||
|     keys, |     keys, | ||||||
|  |     // Since the message was not delayed, we use the current time | ||||||
|  |     DateTime.now().millisecondsSinceEpoch, | ||||||
|   ); |   ); | ||||||
| 
 | 
 | ||||||
|   // All Bob has to do now is replace the OMEMO wrapper element  |   // All Bob has to do now is replace the OMEMO wrapper element  | ||||||
|  | |||||||
| @ -68,6 +68,7 @@ class OmemoDoubleRatchet { | |||||||
|     this.sessionAd, |     this.sessionAd, | ||||||
|     this.mkSkipped, // MKSKIPPED |     this.mkSkipped, // MKSKIPPED | ||||||
|     this.acknowledged, |     this.acknowledged, | ||||||
|  |     this.kexTimestamp, | ||||||
|   ); |   ); | ||||||
|    |    | ||||||
|   factory OmemoDoubleRatchet.fromJson(Map<String, dynamic> data) { |   factory OmemoDoubleRatchet.fromJson(Map<String, dynamic> data) { | ||||||
| @ -85,6 +86,7 @@ class OmemoDoubleRatchet { | |||||||
|       'ik_pub': 'base/64/encoded', |       'ik_pub': 'base/64/encoded', | ||||||
|       'session_ad': 'base/64/encoded', |       'session_ad': 'base/64/encoded', | ||||||
|       'acknowledged': true | false, |       'acknowledged': true | false, | ||||||
|  |       'kex_timestamp': int, | ||||||
|       'mkskipped': [ |       'mkskipped': [ | ||||||
|         { |         { | ||||||
|           'key': 'base/64/encoded', |           'key': 'base/64/encoded', | ||||||
| @ -129,6 +131,7 @@ class OmemoDoubleRatchet { | |||||||
|       base64.decode(data['session_ad']! as String), |       base64.decode(data['session_ad']! as String), | ||||||
|       mkSkipped, |       mkSkipped, | ||||||
|       data['acknowledged']! as bool, |       data['acknowledged']! as bool, | ||||||
|  |       data['kex_timestamp']! as int, | ||||||
|     ); |     ); | ||||||
|   } |   } | ||||||
|    |    | ||||||
| @ -160,14 +163,18 @@ class OmemoDoubleRatchet { | |||||||
| 
 | 
 | ||||||
|   final Map<SkippedKey, List<int>> mkSkipped; |   final Map<SkippedKey, List<int>> mkSkipped; | ||||||
| 
 | 
 | ||||||
|  |   /// The point in time at which we performed the kex exchange to create this ratchet. | ||||||
|  |   /// Precision is milliseconds since epoch. | ||||||
|  |   int kexTimestamp; | ||||||
|  | 
 | ||||||
|   /// Indicates whether we received an empty OMEMO message after building a session with |   /// Indicates whether we received an empty OMEMO message after building a session with | ||||||
|   /// the device. |   /// the device.  | ||||||
|   bool acknowledged; |   bool acknowledged; | ||||||
| 
 | 
 | ||||||
|   /// Create an OMEMO session using the Signed Pre Key [spk], the shared secret [sk] that |   /// Create an OMEMO session using the Signed Pre Key [spk], the shared secret [sk] that | ||||||
|   /// was obtained using a X3DH and the associated data [ad] that was also obtained through |   /// was obtained using a X3DH and the associated data [ad] that was also obtained through | ||||||
|   /// a X3DH. [ik] refers to Bob's (the receiver's) IK public key. |   /// a X3DH. [ik] refers to Bob's (the receiver's) IK public key. | ||||||
|   static Future<OmemoDoubleRatchet> initiateNewSession(OmemoPublicKey spk, OmemoPublicKey ik, List<int> sk, List<int> ad, int pn) async { |   static Future<OmemoDoubleRatchet> initiateNewSession(OmemoPublicKey spk, OmemoPublicKey ik, List<int> sk, List<int> ad, int timestamp) async { | ||||||
|     final dhs = await OmemoKeyPair.generateNewPair(KeyPairType.x25519); |     final dhs = await OmemoKeyPair.generateNewPair(KeyPairType.x25519); | ||||||
|     final dhr = spk; |     final dhr = spk; | ||||||
|     final rk  = await kdfRk(sk, await omemoDH(dhs, dhr, 0)); |     final rk  = await kdfRk(sk, await omemoDH(dhs, dhr, 0)); | ||||||
| @ -181,11 +188,12 @@ class OmemoDoubleRatchet { | |||||||
|       null, |       null, | ||||||
|       0, |       0, | ||||||
|       0, |       0, | ||||||
|       pn, |       0, | ||||||
|       ik, |       ik, | ||||||
|       ad, |       ad, | ||||||
|       {}, |       {}, | ||||||
|       false, |       false, | ||||||
|  |       timestamp, | ||||||
|     ); |     ); | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
| @ -193,7 +201,7 @@ class OmemoDoubleRatchet { | |||||||
|   /// Pre Key keypair [spk], the shared secret [sk] that was obtained through a X3DH and |   /// Pre Key keypair [spk], the shared secret [sk] that was obtained through a X3DH and | ||||||
|   /// the associated data [ad] that was also obtained through a X3DH. [ik] refers to |   /// the associated data [ad] that was also obtained through a X3DH. [ik] refers to | ||||||
|   /// Alice's (the initiator's) IK public key. |   /// Alice's (the initiator's) IK public key. | ||||||
|   static Future<OmemoDoubleRatchet> acceptNewSession(OmemoKeyPair spk, OmemoPublicKey ik, List<int> sk, List<int> ad) async { |   static Future<OmemoDoubleRatchet> acceptNewSession(OmemoKeyPair spk, OmemoPublicKey ik, List<int> sk, List<int> ad, int kexTimestamp) async { | ||||||
|     return OmemoDoubleRatchet( |     return OmemoDoubleRatchet( | ||||||
|       spk, |       spk, | ||||||
|       null, |       null, | ||||||
| @ -207,6 +215,7 @@ class OmemoDoubleRatchet { | |||||||
|       ad, |       ad, | ||||||
|       {}, |       {}, | ||||||
|       false, |       false, | ||||||
|  |       kexTimestamp, | ||||||
|     ); |     ); | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
| @ -233,6 +242,7 @@ class OmemoDoubleRatchet { | |||||||
|       'session_ad': base64.encode(sessionAd), |       'session_ad': base64.encode(sessionAd), | ||||||
|       'mkskipped': mkSkippedSerialised, |       'mkskipped': mkSkippedSerialised, | ||||||
|       'acknowledged': acknowledged, |       'acknowledged': acknowledged, | ||||||
|  |       'kex_timestamp': kexTimestamp, | ||||||
|     }; |     }; | ||||||
|   } |   } | ||||||
|    |    | ||||||
| @ -268,18 +278,18 @@ class OmemoDoubleRatchet { | |||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   Future<void> _dhRatchet(OmemoMessage header) async { |   Future<void> _dhRatchet(OmemoMessage header) async { | ||||||
|     pn = header.n!; |     pn = ns; | ||||||
|     ns = 0; |     ns = 0; | ||||||
|     nr = 0; |     nr = 0; | ||||||
|     dhr = OmemoPublicKey.fromBytes(header.dhPub!, KeyPairType.x25519); |     dhr = OmemoPublicKey.fromBytes(header.dhPub!, KeyPairType.x25519); | ||||||
| 
 | 
 | ||||||
|     final newRk = await kdfRk(rk, await omemoDH(dhs, dhr!, 0)); |     final newRk = await kdfRk(rk, await omemoDH(dhs, dhr!, 0)); | ||||||
|     rk = newRk; |     rk = List.from(newRk); | ||||||
|     ckr = newRk; |     ckr = List.from(newRk); | ||||||
|     dhs = await OmemoKeyPair.generateNewPair(KeyPairType.x25519); |     dhs = await OmemoKeyPair.generateNewPair(KeyPairType.x25519); | ||||||
|     final newNewRk = await kdfRk(rk, await omemoDH(dhs, dhr!, 0)); |     final newNewRk = await kdfRk(rk, await omemoDH(dhs, dhr!, 0)); | ||||||
|     rk = newNewRk; |     rk = List.from(newNewRk); | ||||||
|     cks = newNewRk; |     cks = List.from(newNewRk); | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   /// Encrypt [plaintext] using the Double Ratchet. |   /// Encrypt [plaintext] using the Double Ratchet. | ||||||
| @ -313,8 +323,8 @@ class OmemoDoubleRatchet { | |||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     final dhPubMatches = listsEqual( |     final dhPubMatches = listsEqual( | ||||||
|       header.dhPub ?? <int>[], |       header.dhPub!, | ||||||
|       await dhr?.getBytes() ?? <int>[], |       (await dhr?.getBytes()) ?? <int>[], | ||||||
|     ); |     ); | ||||||
|     if (!dhPubMatches) { |     if (!dhPubMatches) { | ||||||
|       await _skipMessageKeys(header.pn!); |       await _skipMessageKeys(header.pn!); | ||||||
| @ -348,6 +358,7 @@ class OmemoDoubleRatchet { | |||||||
|       sessionAd, |       sessionAd, | ||||||
|       Map<SkippedKey, List<int>>.from(mkSkipped), |       Map<SkippedKey, List<int>>.from(mkSkipped), | ||||||
|       acknowledged, |       acknowledged, | ||||||
|  |       kexTimestamp, | ||||||
|     ); |     ); | ||||||
|   } |   } | ||||||
|    |    | ||||||
| @ -378,6 +389,7 @@ class OmemoDoubleRatchet { | |||||||
|       ns == other.ns && |       ns == other.ns && | ||||||
|       nr == other.nr && |       nr == other.nr && | ||||||
|       pn == other.pn && |       pn == other.pn && | ||||||
|       listsEqual(sessionAd, other.sessionAd); |       listsEqual(sessionAd, other.sessionAd) && | ||||||
|  |       kexTimestamp == other.kexTimestamp; | ||||||
|   } |   } | ||||||
| } | } | ||||||
|  | |||||||
| @ -31,14 +31,11 @@ class UnknownSignedPrekeyException implements Exception { | |||||||
|   String errMsg() => 'Unknown Signed Prekey used.'; |   String errMsg() => 'Unknown Signed Prekey used.'; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /// Triggered by the Session Manager when the received Key Exchange message does not | /// Triggered by the Session Manager when the received Key Exchange message does not meet | ||||||
| /// meet our expectations. This happens when the PN attribute of the message is not equal | /// the requirement that a key exchange, given that the ratchet already exists, must be | ||||||
| /// to our receive number. | /// sent after its creation. | ||||||
| class InvalidKeyExchangeException implements Exception { | class InvalidKeyExchangeException implements Exception { | ||||||
|   const InvalidKeyExchangeException(this.expectedPn, this.actualPn); |   String errMsg() => 'The key exchange was sent before the last kex finished'; | ||||||
|   final int expectedPn; |  | ||||||
|   final int actualPn; |  | ||||||
|   String errMsg() => 'The pn attribute of the key exchange is invalid. Expected $expectedPn, got $actualPn'; |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /// Triggered by the Session Manager when a message's sequence number is smaller than we | /// Triggered by the Session Manager when a message's sequence number is smaller than we | ||||||
|  | |||||||
| @ -73,3 +73,7 @@ OmemoKeyPair? decodeKeyPairIfNotNull(String? pk, String? sk, KeyPairType type) { | |||||||
|     type, |     type, | ||||||
|   ); |   ); | ||||||
| } | } | ||||||
|  | 
 | ||||||
|  | int getTimestamp() { | ||||||
|  |   return DateTime.now().millisecondsSinceEpoch; | ||||||
|  | } | ||||||
|  | |||||||
| @ -143,7 +143,7 @@ class OmemoSessionManager { | |||||||
|   /// Create a ratchet session initiated by Alice to the user with Jid [jid] and the device |   /// Create a ratchet session initiated by Alice to the user with Jid [jid] and the device | ||||||
|   /// [deviceId] from the bundle [bundle]. |   /// [deviceId] from the bundle [bundle]. | ||||||
|   @visibleForTesting |   @visibleForTesting | ||||||
|   Future<OmemoKeyExchange> addSessionFromBundle(String jid, int deviceId, OmemoBundle bundle, int pn) async { |   Future<OmemoKeyExchange> addSessionFromBundle(String jid, int deviceId, OmemoBundle bundle) async { | ||||||
|     final device = await getDevice(); |     final device = await getDevice(); | ||||||
|     final kexResult = await x3dhFromBundle( |     final kexResult = await x3dhFromBundle( | ||||||
|       bundle, |       bundle, | ||||||
| @ -154,7 +154,7 @@ class OmemoSessionManager { | |||||||
|       bundle.ik, |       bundle.ik, | ||||||
|       kexResult.sk, |       kexResult.sk, | ||||||
|       kexResult.ad, |       kexResult.ad, | ||||||
|       pn, |       getTimestamp(), | ||||||
|     ); |     ); | ||||||
| 
 | 
 | ||||||
|     await _trustManager.onNewSession(jid, deviceId); |     await _trustManager.onNewSession(jid, deviceId); | ||||||
| @ -201,6 +201,7 @@ class OmemoSessionManager { | |||||||
|       OmemoPublicKey.fromBytes(kex.ik!, KeyPairType.ed25519), |       OmemoPublicKey.fromBytes(kex.ik!, KeyPairType.ed25519), | ||||||
|       kexResult.sk, |       kexResult.sk, | ||||||
|       kexResult.ad, |       kexResult.ad, | ||||||
|  |       getTimestamp(), | ||||||
|     ); |     ); | ||||||
| 
 | 
 | ||||||
|     await _trustManager.onNewSession(jid, deviceId); |     await _trustManager.onNewSession(jid, deviceId); | ||||||
| @ -241,21 +242,10 @@ class OmemoSessionManager { | |||||||
|     final kex = <int, OmemoKeyExchange>{}; |     final kex = <int, OmemoKeyExchange>{}; | ||||||
|     if (newSessions != null) { |     if (newSessions != null) { | ||||||
|       for (final newSession in newSessions) { |       for (final newSession in newSessions) { | ||||||
|         final session = await _getRatchet( |  | ||||||
|           RatchetMapKey( |  | ||||||
|             newSession.jid, |  | ||||||
|             newSession.id, |  | ||||||
|           ), |  | ||||||
|         ); |  | ||||||
| 
 |  | ||||||
|         final pn = session != null ? |  | ||||||
|           session.ns : |  | ||||||
|           0; |  | ||||||
|         kex[newSession.id] = await addSessionFromBundle( |         kex[newSession.id] = await addSessionFromBundle( | ||||||
|           newSession.jid, |           newSession.jid, | ||||||
|           newSession.id, |           newSession.id, | ||||||
|           newSession, |           newSession, | ||||||
|           pn, |  | ||||||
|         ); |         ); | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
| @ -334,12 +324,15 @@ class OmemoSessionManager { | |||||||
|   /// <keys /> element with a "jid" attribute matching our own. [senderJid] refers to the |   /// <keys /> element with a "jid" attribute matching our own. [senderJid] refers to the | ||||||
|   /// bare Jid of the sender. [senderDeviceId] refers to the "sid" attribute of the |   /// bare Jid of the sender. [senderDeviceId] refers to the "sid" attribute of the | ||||||
|   /// <encrypted /> element. |   /// <encrypted /> element. | ||||||
|  |   /// [timestamp] refers to the time the message was sent. This might be either what the | ||||||
|  |   /// server tells you via "XEP-0203: Delayed Delivery" or the point in time at which | ||||||
|  |   /// you received the stanza, if no Delayed Delivery element was found. | ||||||
|   /// |   /// | ||||||
|   /// If the received message is an empty OMEMO message, i.e. there is no <payload /> |   /// If the received message is an empty OMEMO message, i.e. there is no <payload /> | ||||||
|   /// element, then [ciphertext] must be set to null. In this case, this function |   /// element, then [ciphertext] must be set to null. In this case, this function | ||||||
|   /// will return null as there is no message to be decrypted. This, however, is used |   /// will return null as there is no message to be decrypted. This, however, is used | ||||||
|   /// to set up sessions or advance the ratchets. |   /// to set up sessions or advance the ratchets. | ||||||
|   Future<String?> decryptMessage(List<int>? ciphertext, String senderJid, int senderDeviceId, List<EncryptedKey> keys) async { |   Future<String?> decryptMessage(List<int>? ciphertext, String senderJid, int senderDeviceId, List<EncryptedKey> keys, int timestamp) async { | ||||||
|     // Try to find a session we can decrypt with. |     // Try to find a session we can decrypt with. | ||||||
|     var device = await getDevice(); |     var device = await getDevice(); | ||||||
|     final rawKey = keys.firstWhereOrNull((key) => key.rid == device.id); |     final rawKey = keys.firstWhereOrNull((key) => key.rid == device.id); | ||||||
| @ -363,8 +356,8 @@ class OmemoSessionManager { | |||||||
|       // Guard against old key exchanges |       // Guard against old key exchanges | ||||||
|       if (oldRatchet != null) { |       if (oldRatchet != null) { | ||||||
|         _log.finest('KEX for existent ratchet. ${oldRatchet.pn}'); |         _log.finest('KEX for existent ratchet. ${oldRatchet.pn}'); | ||||||
|         if (message.pn != oldRatchet.nr) { |         if (oldRatchet.kexTimestamp > timestamp) { | ||||||
|           throw InvalidKeyExchangeException(oldRatchet.nr, message.pn!); |           throw InvalidKeyExchangeException(); | ||||||
|         } |         } | ||||||
|       } |       } | ||||||
|        |        | ||||||
| @ -401,12 +394,6 @@ class OmemoSessionManager { | |||||||
|     final ratchet = (await _getRatchet(ratchetKey))!; |     final ratchet = (await _getRatchet(ratchetKey))!; | ||||||
|     oldRatchet ??= ratchet.clone(); |     oldRatchet ??= ratchet.clone(); | ||||||
| 
 | 
 | ||||||
|     if (!rawKey.kex) { |  | ||||||
|       if (message.n! < ratchet.nr - 1) { |  | ||||||
|         throw MessageAlreadyDecryptedException(); |  | ||||||
|       } |  | ||||||
|     } |  | ||||||
|      |  | ||||||
|     try { |     try { | ||||||
|       if (rawKey.kex) { |       if (rawKey.kex) { | ||||||
|         keyAndHmac = await ratchet.ratchetDecrypt(message, authMessage.writeToBuffer()); |         keyAndHmac = await ratchet.ratchetDecrypt(message, authMessage.writeToBuffer()); | ||||||
|  | |||||||
| @ -91,6 +91,7 @@ void main() { | |||||||
|       ikAlice.pk, |       ikAlice.pk, | ||||||
|       resultBob.sk, |       resultBob.sk, | ||||||
|       resultBob.ad, |       resultBob.ad, | ||||||
|  |       0, | ||||||
|     ); |     ); | ||||||
| 
 | 
 | ||||||
|     expect(alicesRatchet.sessionAd, bobsRatchet.sessionAd); |     expect(alicesRatchet.sessionAd, bobsRatchet.sessionAd); | ||||||
|  | |||||||
| @ -90,6 +90,7 @@ void main() { | |||||||
|       aliceJid, |       aliceJid, | ||||||
|       await aliceSession.getDeviceId(), |       await aliceSession.getDeviceId(), | ||||||
|       aliceMessage.encryptedKeys, |       aliceMessage.encryptedKeys, | ||||||
|  |       0, | ||||||
|     ); |     ); | ||||||
|     expect(messagePlaintext, bobMessage); |     expect(messagePlaintext, bobMessage); | ||||||
|     // The ratchet should be modified two times: Once for when the ratchet is created and |     // The ratchet should be modified two times: Once for when the ratchet is created and | ||||||
| @ -121,6 +122,7 @@ void main() { | |||||||
|       bobJid, |       bobJid, | ||||||
|       await bobSession.getDeviceId(), |       await bobSession.getDeviceId(), | ||||||
|       bobResponseMessage.encryptedKeys, |       bobResponseMessage.encryptedKeys, | ||||||
|  |       0, | ||||||
|     ); |     ); | ||||||
|     expect(bobResponseText, aliceReceivedMessage); |     expect(bobResponseText, aliceReceivedMessage); | ||||||
|   }); |   }); | ||||||
| @ -170,6 +172,7 @@ void main() { | |||||||
|       aliceJid, |       aliceJid, | ||||||
|       await aliceSession.getDeviceId(), |       await aliceSession.getDeviceId(), | ||||||
|       aliceMessage.encryptedKeys, |       aliceMessage.encryptedKeys, | ||||||
|  |       0, | ||||||
|     ); |     ); | ||||||
|     expect(messagePlaintext, bobMessage); |     expect(messagePlaintext, bobMessage); | ||||||
| 
 | 
 | ||||||
| @ -189,6 +192,7 @@ void main() { | |||||||
|       bobJid, |       bobJid, | ||||||
|       await bobSession.getDeviceId(), |       await bobSession.getDeviceId(), | ||||||
|       bobResponseMessage.encryptedKeys, |       bobResponseMessage.encryptedKeys, | ||||||
|  |       0, | ||||||
|     ); |     ); | ||||||
|     expect(bobResponseText, aliceReceivedMessage); |     expect(bobResponseText, aliceReceivedMessage); | ||||||
| 
 | 
 | ||||||
| @ -245,6 +249,7 @@ void main() { | |||||||
|       aliceJid, |       aliceJid, | ||||||
|       await aliceSession1.getDeviceId(), |       await aliceSession1.getDeviceId(), | ||||||
|       aliceMessage.encryptedKeys, |       aliceMessage.encryptedKeys, | ||||||
|  |       0, | ||||||
|     ); |     ); | ||||||
|     expect(messagePlaintext, bobMessage); |     expect(messagePlaintext, bobMessage); | ||||||
| 
 | 
 | ||||||
| @ -254,6 +259,7 @@ void main() { | |||||||
|       aliceJid, |       aliceJid, | ||||||
|       await aliceSession1.getDeviceId(), |       await aliceSession1.getDeviceId(), | ||||||
|       aliceMessage.encryptedKeys, |       aliceMessage.encryptedKeys, | ||||||
|  |       0, | ||||||
|     ); |     ); | ||||||
|     expect(messagePlaintext, aliceMessage2); |     expect(messagePlaintext, aliceMessage2); | ||||||
|   }); |   }); | ||||||
| @ -294,6 +300,7 @@ void main() { | |||||||
|       aliceJid, |       aliceJid, | ||||||
|       await aliceSession.getDeviceId(), |       await aliceSession.getDeviceId(), | ||||||
|       aliceMessage.encryptedKeys, |       aliceMessage.encryptedKeys, | ||||||
|  |       0, | ||||||
|     ); |     ); | ||||||
|     expect(bobMessage, null); |     expect(bobMessage, null); | ||||||
| 
 | 
 | ||||||
| @ -371,6 +378,7 @@ void main() { | |||||||
|       aliceJid, |       aliceJid, | ||||||
|       await aliceSession.getDeviceId(), |       await aliceSession.getDeviceId(), | ||||||
|       aliceMessage.encryptedKeys, |       aliceMessage.encryptedKeys, | ||||||
|  |       0, | ||||||
|     ); |     ); | ||||||
|     expect(messagePlaintext, bobMessage); |     expect(messagePlaintext, bobMessage); | ||||||
|   }); |   }); | ||||||
| @ -437,6 +445,7 @@ void main() { | |||||||
|       aliceJid, |       aliceJid, | ||||||
|       await aliceSession.getDeviceId(), |       await aliceSession.getDeviceId(), | ||||||
|       aliceMessage.encryptedKeys, |       aliceMessage.encryptedKeys, | ||||||
|  |       0, | ||||||
|     ); |     ); | ||||||
| 
 | 
 | ||||||
|     for (var i = 0; i < 100; i++) { |     for (var i = 0; i < 100; i++) { | ||||||
| @ -456,6 +465,7 @@ void main() { | |||||||
|         bobJid, |         bobJid, | ||||||
|         await bobSession.getDeviceId(), |         await bobSession.getDeviceId(), | ||||||
|         bobResponseMessage.encryptedKeys, |         bobResponseMessage.encryptedKeys, | ||||||
|  |         0, | ||||||
|       ); |       ); | ||||||
|       expect(messageText, aliceReceivedMessage); |       expect(messageText, aliceReceivedMessage); | ||||||
|     } |     } | ||||||
| @ -610,6 +620,7 @@ void main() { | |||||||
|       aliceJid, |       aliceJid, | ||||||
|       await aliceSession.getDeviceId(), |       await aliceSession.getDeviceId(), | ||||||
|       msg1.encryptedKeys, |       msg1.encryptedKeys, | ||||||
|  |       0, | ||||||
|     ); |     ); | ||||||
|     final aliceRatchet1 = aliceSession.getRatchet( |     final aliceRatchet1 = aliceSession.getRatchet( | ||||||
|       bobJid, |       bobJid, | ||||||
| @ -634,6 +645,7 @@ void main() { | |||||||
|       aliceJid, |       aliceJid, | ||||||
|       await aliceSession.getDeviceId(), |       await aliceSession.getDeviceId(), | ||||||
|       msg2.encryptedKeys, |       msg2.encryptedKeys, | ||||||
|  |       getTimestamp(), | ||||||
|     ); |     ); | ||||||
|     final aliceRatchet2 = aliceSession.getRatchet( |     final aliceRatchet2 = aliceSession.getRatchet( | ||||||
|       bobJid, |       bobJid, | ||||||
| @ -669,6 +681,7 @@ void main() { | |||||||
|     ); |     ); | ||||||
| 
 | 
 | ||||||
|     final bobsReceivedMessages = List<EncryptionResult>.empty(growable: true); |     final bobsReceivedMessages = List<EncryptionResult>.empty(growable: true); | ||||||
|  |     final bobsReceivedMessagesTimestamps = List<int>.empty(growable: true); | ||||||
|      |      | ||||||
|     // Alice sends Bob a message |     // Alice sends Bob a message | ||||||
|     final msg1 = await aliceSession.encryptToJid( |     final msg1 = await aliceSession.encryptToJid( | ||||||
| @ -679,11 +692,15 @@ void main() { | |||||||
|       ], |       ], | ||||||
|     ); |     ); | ||||||
|     bobsReceivedMessages.add(msg1); |     bobsReceivedMessages.add(msg1); | ||||||
|  |     final t1 = getTimestamp(); | ||||||
|  |     bobsReceivedMessagesTimestamps.add(t1); | ||||||
|  | 
 | ||||||
|     await bobSession.decryptMessage( |     await bobSession.decryptMessage( | ||||||
|       msg1.ciphertext, |       msg1.ciphertext, | ||||||
|       aliceJid, |       aliceJid, | ||||||
|       await aliceSession.getDeviceId(), |       await aliceSession.getDeviceId(), | ||||||
|       msg1.encryptedKeys, |       msg1.encryptedKeys, | ||||||
|  |       t1, | ||||||
|     ); |     ); | ||||||
| 
 | 
 | ||||||
|     // Bob responds |     // Bob responds | ||||||
| @ -691,13 +708,15 @@ void main() { | |||||||
|       aliceJid, |       aliceJid, | ||||||
|       'Hello!', |       'Hello!', | ||||||
|     ); |     ); | ||||||
|  | 
 | ||||||
|     await aliceSession.decryptMessage( |     await aliceSession.decryptMessage( | ||||||
|       msg2.ciphertext, |       msg2.ciphertext, | ||||||
|       bobJid, |       bobJid, | ||||||
|       await bobSession.getDeviceId(), |       await bobSession.getDeviceId(), | ||||||
|       msg2.encryptedKeys, |       msg2.encryptedKeys, | ||||||
|  |       getTimestamp(), | ||||||
|     ); |     ); | ||||||
| 
 |      | ||||||
|     // Send some messages between the two |     // Send some messages between the two | ||||||
|     for (var i = 0; i < 100; i++) { |     for (var i = 0; i < 100; i++) { | ||||||
|       final msg = await aliceSession.encryptToJid( |       final msg = await aliceSession.encryptToJid( | ||||||
| @ -705,11 +724,14 @@ void main() { | |||||||
|         'Hello $i', |         'Hello $i', | ||||||
|       ); |       ); | ||||||
|       bobsReceivedMessages.add(msg); |       bobsReceivedMessages.add(msg); | ||||||
|  |       final t = getTimestamp(); | ||||||
|  |       bobsReceivedMessagesTimestamps.add(t); | ||||||
|       final result = await bobSession.decryptMessage( |       final result = await bobSession.decryptMessage( | ||||||
|         msg.ciphertext, |         msg.ciphertext, | ||||||
|         aliceJid, |         aliceJid, | ||||||
|         await aliceSession.getDeviceId(), |         await aliceSession.getDeviceId(), | ||||||
|         msg.encryptedKeys, |         msg.encryptedKeys, | ||||||
|  |         t, | ||||||
|       ); |       ); | ||||||
| 
 | 
 | ||||||
|       expect(result, 'Hello $i'); |       expect(result, 'Hello $i'); | ||||||
| @ -720,20 +742,23 @@ void main() { | |||||||
|     final ratchetPreError = bobSession |     final ratchetPreError = bobSession | ||||||
|       .getRatchet(aliceJid, await aliceSession.getDeviceId()) |       .getRatchet(aliceJid, await aliceSession.getDeviceId()) | ||||||
|       .clone(); |       .clone(); | ||||||
|  |     var invalidKex = 0; | ||||||
|     var errorCounter = 0; |     var errorCounter = 0; | ||||||
|     for (final msg in bobsReceivedMessages) { |     for (var i = 0; i < bobsReceivedMessages.length; i++) { | ||||||
|  |       final msg = bobsReceivedMessages[i]; | ||||||
|       try { |       try { | ||||||
|         await bobSession.decryptMessage( |         await bobSession.decryptMessage( | ||||||
|           msg.ciphertext, |           msg.ciphertext, | ||||||
|           aliceJid, |           aliceJid, | ||||||
|           await aliceSession.getDeviceId(), |           await aliceSession.getDeviceId(), | ||||||
|           msg.encryptedKeys, |           msg.encryptedKeys, | ||||||
|  |           bobsReceivedMessagesTimestamps[i], | ||||||
|         ); |         ); | ||||||
|         expect(true, false); |         expect(true, false); | ||||||
|       } on MessageAlreadyDecryptedException catch (_) { |       } on InvalidMessageHMACException catch (_) { | ||||||
|         errorCounter++; |         errorCounter++; | ||||||
|       } on InvalidKeyExchangeException catch (_) { |       } on InvalidKeyExchangeException catch (_) { | ||||||
|         errorCounter++; |         invalidKex++; | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
|     final ratchetPostError = bobSession |     final ratchetPostError = bobSession | ||||||
| @ -741,7 +766,8 @@ void main() { | |||||||
|       .clone(); |       .clone(); | ||||||
| 
 | 
 | ||||||
|     // The 100 messages including the initial KEX message |     // The 100 messages including the initial KEX message | ||||||
|     expect(errorCounter, 101); |     expect(invalidKex, 1); | ||||||
|  |     expect(errorCounter, 100); | ||||||
|     expect(await ratchetPreError.equals(ratchetPostError), true); |     expect(await ratchetPreError.equals(ratchetPostError), true); | ||||||
| 
 | 
 | ||||||
|      |      | ||||||
| @ -754,6 +780,7 @@ void main() { | |||||||
|       aliceJid, |       aliceJid, | ||||||
|       await aliceSession.getDeviceId(), |       await aliceSession.getDeviceId(), | ||||||
|       msg3.encryptedKeys, |       msg3.encryptedKeys, | ||||||
|  |       104, | ||||||
|     ); |     ); | ||||||
| 
 | 
 | ||||||
|     expect(result, 'Are you okay?'); |     expect(result, 'Are you okay?'); | ||||||
|  | |||||||
| @ -62,6 +62,7 @@ void main() { | |||||||
|       aliceJid, |       aliceJid, | ||||||
|       await aliceSession.getDeviceId(), |       await aliceSession.getDeviceId(), | ||||||
|       aliceMessage.encryptedKeys, |       aliceMessage.encryptedKeys, | ||||||
|  |       getTimestamp(), | ||||||
|     ); |     ); | ||||||
|     final aliceOld = aliceSession.getRatchet(bobJid, await bobSession.getDeviceId()); |     final aliceOld = aliceSession.getRatchet(bobJid, await bobSession.getDeviceId()); | ||||||
|     final aliceSerialised = jsonify(await aliceOld.toJson()); |     final aliceSerialised = jsonify(await aliceOld.toJson()); | ||||||
| @ -86,7 +87,6 @@ void main() { | |||||||
|       'bob@localhost', |       'bob@localhost', | ||||||
|       await bobSession.getDeviceId(), |       await bobSession.getDeviceId(), | ||||||
|       await bobSession.getDeviceBundle(), |       await bobSession.getDeviceBundle(), | ||||||
|       0, |  | ||||||
|     ); |     ); | ||||||
| 
 | 
 | ||||||
|     // Serialise and deserialise |     // Serialise and deserialise | ||||||
|  | |||||||
		Loading…
	
		Reference in New Issue
	
	Block a user