import 'package:omemo_dart/omemo_dart.dart'; import 'package:test/test.dart'; void main() { test('Test using OMEMO sessions with only one device per user', () async { const aliceJid = 'alice@server.example'; const bobJid = 'bob@other.server.example'; // Alice and Bob generate their sessions var deviceModified = false; final aliceSession = await OmemoSessionManager.generateNewIdentity(opkAmount: 1); final bobSession = await OmemoSessionManager.generateNewIdentity(opkAmount: 1); final bobOpks = (await bobSession.getDevice()).opks.values.toList(); bobSession.eventStream.listen((event) { if (event is DeviceBundleModifiedEvent) { deviceModified = true; } }); // Alice encrypts a message for Bob const messagePlaintext = 'Hello Bob!'; final aliceMessage = await aliceSession.encryptToJid( bobJid, messagePlaintext, newSessions: [ await (await bobSession.getDevice()).toBundle(), ], ); expect(aliceMessage.encryptedKeys.length, 1); // Alice sends the message to Bob // ... // Bob decrypts it final bobMessage = await bobSession.decryptMessage( aliceMessage.ciphertext, aliceJid, (await aliceSession.getDevice()).id, aliceMessage.encryptedKeys, ); expect(messagePlaintext, bobMessage); // The event should be triggered expect(deviceModified, true); // Bob should have replaced his OPK expect( listsEqual(bobOpks, (await bobSession.getDevice()).opks.values.toList()), false, ); // Bob responds to Alice const bobResponseText = 'Oh, hello Alice!'; final bobResponseMessage = await bobSession.encryptToJid( aliceJid, bobResponseText, ); // Bob sends the message to Alice // ... // Alice decrypts it final aliceReceivedMessage = await aliceSession.decryptMessage( bobResponseMessage.ciphertext, bobJid, (await bobSession.getDevice()).id, bobResponseMessage.encryptedKeys, ); expect(bobResponseText, aliceReceivedMessage); }); test('Test using OMEMO sessions with only two devices for the receiver', () async { const aliceJid = 'alice@server.example'; const bobJid = 'bob@other.server.example'; // Alice and Bob generate their sessions final aliceSession = await OmemoSessionManager.generateNewIdentity(opkAmount: 1); final bobSession = await OmemoSessionManager.generateNewIdentity(opkAmount: 1); // Bob's other device final bobSession2 = await OmemoSessionManager.generateNewIdentity(opkAmount: 1); // Alice encrypts a message for Bob const messagePlaintext = 'Hello Bob!'; final aliceMessage = await aliceSession.encryptToJid( bobJid, messagePlaintext, newSessions: [ await (await bobSession.getDevice()).toBundle(), await (await bobSession2.getDevice()).toBundle(), ], ); expect(aliceMessage.encryptedKeys.length, 2); expect(aliceMessage.encryptedKeys[0].kex, true); expect(aliceMessage.encryptedKeys[1].kex, true); // Alice sends the message to Bob // ... // Bob decrypts it final bobMessage = await bobSession.decryptMessage( aliceMessage.ciphertext, aliceJid, (await aliceSession.getDevice()).id, aliceMessage.encryptedKeys, ); expect(messagePlaintext, bobMessage); // Bob responds to Alice const bobResponseText = 'Oh, hello Alice!'; final bobResponseMessage = await bobSession.encryptToJid( aliceJid, bobResponseText, ); // Bob sends the message to Alice // ... // Alice decrypts it final aliceReceivedMessage = await aliceSession.decryptMessage( bobResponseMessage.ciphertext, bobJid, (await bobSession.getDevice()).id, bobResponseMessage.encryptedKeys, ); expect(bobResponseText, aliceReceivedMessage); }); }