From 6157e8ff55f6b369b0b28dc44a71d17fa809abb8 Mon Sep 17 00:00:00 2001 From: cameronvoell Date: Thu, 22 Aug 2024 14:44:37 -0700 Subject: [PATCH] Revert "remove the inbox state and revoke installations code" This reverts commit 10cf325641a903fce21da3519989b988ccc2ad8d. --- .../modules/xmtpreactnativesdk/XMTPModule.kt | 23 + .../wrappers/InboxStateWrapper.kt | 23 + example/src/tests/groupTests.ts | 3579 +++++++++-------- ios/Wrappers/InboxStateWrapper.swift | 30 + ios/XMTPModule.swift | 19 + src/index.ts | 12 + src/lib/Client.ts | 45 + src/lib/InboxState.ts | 30 + 8 files changed, 1994 insertions(+), 1767 deletions(-) create mode 100644 android/src/main/java/expo/modules/xmtpreactnativesdk/wrappers/InboxStateWrapper.kt create mode 100644 ios/Wrappers/InboxStateWrapper.swift create mode 100644 src/lib/InboxState.ts diff --git a/android/src/main/java/expo/modules/xmtpreactnativesdk/XMTPModule.kt b/android/src/main/java/expo/modules/xmtpreactnativesdk/XMTPModule.kt index ba5db0330..e81cde291 100644 --- a/android/src/main/java/expo/modules/xmtpreactnativesdk/XMTPModule.kt +++ b/android/src/main/java/expo/modules/xmtpreactnativesdk/XMTPModule.kt @@ -24,6 +24,7 @@ import expo.modules.xmtpreactnativesdk.wrappers.DecodedMessageWrapper import expo.modules.xmtpreactnativesdk.wrappers.DecryptedLocalAttachment import expo.modules.xmtpreactnativesdk.wrappers.EncryptedLocalAttachment import expo.modules.xmtpreactnativesdk.wrappers.GroupWrapper +import expo.modules.xmtpreactnativesdk.wrappers.InboxStateWrapper import expo.modules.xmtpreactnativesdk.wrappers.MemberWrapper import expo.modules.xmtpreactnativesdk.wrappers.PermissionPolicySetWrapper import expo.modules.xmtpreactnativesdk.wrappers.PreparedLocalMessage @@ -65,6 +66,7 @@ import org.xmtp.proto.message.api.v1.MessageApiOuterClass import org.xmtp.proto.message.contents.Invitation.ConsentProofPayload import org.xmtp.proto.message.contents.PrivateKeyOuterClass import uniffi.xmtpv3.org.xmtp.android.library.libxmtp.GroupPermissionPreconfiguration +import uniffi.xmtpv3.org.xmtp.android.library.libxmtp.InboxState import uniffi.xmtpv3.org.xmtp.android.library.libxmtp.PermissionOption import java.io.BufferedReader import java.io.File @@ -274,6 +276,27 @@ class XMTPModule : Module() { } } + AsyncFunction("revokeAllOtherInstallations") Coroutine { inboxId: String -> + withContext(Dispatchers.IO) { + logV("revokeAllOtherInstallations") + val client = clients[inboxId] ?: throw XMTPException("No client") + val reactSigner = + ReactNativeSigner(module = this@XMTPModule, address = client.address) + signer = reactSigner + + client.revokeAllOtherInstallations(reactSigner) + signer = null + } + } + + AsyncFunction("getInboxState") Coroutine { inboxId: String, refreshFromNetwork: Boolean -> + withContext(Dispatchers.IO) { + val client = clients[inboxId] ?: throw XMTPException("No client") + val inboxState = client.inboxState(refreshFromNetwork) + InboxStateWrapper.encode(inboxState) + } + } + // // Auth functions // diff --git a/android/src/main/java/expo/modules/xmtpreactnativesdk/wrappers/InboxStateWrapper.kt b/android/src/main/java/expo/modules/xmtpreactnativesdk/wrappers/InboxStateWrapper.kt new file mode 100644 index 000000000..ceba7eb79 --- /dev/null +++ b/android/src/main/java/expo/modules/xmtpreactnativesdk/wrappers/InboxStateWrapper.kt @@ -0,0 +1,23 @@ +package expo.modules.xmtpreactnativesdk.wrappers + +import com.google.gson.GsonBuilder +import uniffi.xmtpv3.org.xmtp.android.library.libxmtp.InboxState + +class InboxStateWrapper { + companion object { + fun encodeToObj(inboxState: InboxState): Map { + return mapOf( + "inboxId" to inboxState.inboxId, + "addresses" to inboxState.addresses, + "installationIds" to inboxState.installationIds, + "recoveryAddress" to inboxState.recoveryAddress + ) + } + + fun encode(inboxState: InboxState): String { + val gson = GsonBuilder().create() + val obj = encodeToObj(inboxState) + return gson.toJson(obj) + } + } +} \ No newline at end of file diff --git a/example/src/tests/groupTests.ts b/example/src/tests/groupTests.ts index e97531d69..d086fd1f9 100644 --- a/example/src/tests/groupTests.ts +++ b/example/src/tests/groupTests.ts @@ -66,2068 +66,2113 @@ test('can make a MLS V3 client', async () => { return true }) -test('calls preAuthenticateToInboxCallback when supplied', async () => { - let isCallbackCalled = 0 - let isPreAuthCalled = false - const preAuthenticateToInboxCallback = () => { - isCallbackCalled++ - isPreAuthCalled = true - } - const preEnableIdentityCallback = () => { - isCallbackCalled++ - } - const preCreateIdentityCallback = () => { - isCallbackCalled++ - } +test('can revoke all other installations', async () => { const keyBytes = new Uint8Array([ 233, 120, 198, 96, 154, 65, 132, 17, 132, 96, 250, 40, 103, 35, 125, 64, 166, 83, 208, 224, 254, 44, 205, 227, 175, 49, 234, 129, 74, 252, 135, 145, ]) + const alixWallet = Wallet.createRandom() - await Client.createRandom({ - env: 'local', - enableV3: true, - preEnableIdentityCallback, - preCreateIdentityCallback, - preAuthenticateToInboxCallback, - dbEncryptionKey: keyBytes, - }) - - assert( - isCallbackCalled === 3, - `callback should be called 3 times but was ${isCallbackCalled}` - ) - - if (!isPreAuthCalled) { - throw new Error('preAuthenticateToInboxCallback not called') - } - - return true -}) - -test('can delete a local database', async () => { - let [client, anotherClient] = await createClients(2) - - await client.conversations.newGroup([anotherClient.address]) - await client.conversations.syncGroups() - assert( - (await client.conversations.listGroups()).length === 1, - `should have a group size of 1 but was ${ - (await client.conversations.listGroups()).length - }` - ) - - assert( - client.dbPath !== '', - `client dbPath should be set but was ${client.dbPath}` - ) - await client.deleteLocalDatabase() - client = await Client.createRandom({ + const alix = await Client.create(alixWallet, { env: 'local', appVersion: 'Testing/0.0.0', enableV3: true, - dbEncryptionKey: new Uint8Array([ - 233, 120, 198, 96, 154, 65, 132, 17, 132, 96, 250, 40, 103, 35, 125, 64, - 166, 83, 208, 224, 254, 44, 205, 227, 175, 49, 234, 129, 74, 252, 135, - 145, - ]), + dbEncryptionKey: keyBytes, }) - await client.conversations.syncGroups() - assert( - (await client.conversations.listGroups()).length === 0, - `should have a group size of 0 but was ${ - (await client.conversations.listGroups()).length - }` - ) - - return true -}) - -test('can make a MLS V3 client with encryption key and database directory', async () => { - // eslint-disable-next-line @typescript-eslint/no-unused-vars - const dbDirPath = `${RNFS.DocumentDirectoryPath}/xmtp_db` - const directoryExists = await RNFS.exists(dbDirPath) - if (!directoryExists) { - await RNFS.mkdir(dbDirPath) - } - const key = new Uint8Array([ - 233, 120, 198, 96, 154, 65, 132, 17, 132, 96, 250, 40, 103, 35, 125, 64, - 166, 83, 208, 224, 254, 44, 205, 227, 175, 49, 234, 129, 74, 252, 135, 145, - ]) + await alix.deleteLocalDatabase() - const client = await Client.createRandom({ + const alix2 = await Client.create(alixWallet, { env: 'local', appVersion: 'Testing/0.0.0', enableV3: true, - dbEncryptionKey: key, - dbDirectory: dbDirPath, + dbEncryptionKey: keyBytes, }) + await alix2.deleteLocalDatabase() - const anotherClient = await Client.createRandom({ + const alix3 = await Client.create(alixWallet, { env: 'local', appVersion: 'Testing/0.0.0', enableV3: true, - dbEncryptionKey: key, + dbEncryptionKey: keyBytes, }) - - await client.conversations.newGroup([anotherClient.address]) + const inboxState = await alix3.inboxState(true) assert( - (await client.conversations.listGroups()).length === 1, - `should have a group size of 1 but was ${ - (await client.conversations.listGroups()).length - }` + inboxState.installationIds.length === 3, + `installationIds length should be 3 but was ${inboxState.installationIds.length}` ) - const bundle = await client.exportKeyBundle() - const clientFromBundle = await Client.createFromKeyBundle(bundle, { - env: 'local', - appVersion: 'Testing/0.0.0', - enableV3: true, - dbEncryptionKey: key, - dbDirectory: dbDirPath, - }) - - assert( - clientFromBundle.address === client.address, - `clients dont match ${client.address} and ${clientFromBundle.address}` - ) + await alix3.revokeAllOtherInstallations(alixWallet) + const inboxState2 = await alix3.inboxState(true) assert( - (await clientFromBundle.conversations.listGroups()).length === 1, - `should have a group size of 1 but was ${ - (await clientFromBundle.conversations.listGroups()).length - }` + inboxState2.installationIds.length === 1, + `installationIds length should be 1 but was ${inboxState2.installationIds.length}` ) return true }) -test('testing large group listing with metadata performance', async () => { - const [alixClient, boClient] = await createClients(2) +// test('calls preAuthenticateToInboxCallback when supplied', async () => { +// let isCallbackCalled = 0 +// let isPreAuthCalled = false +// const preAuthenticateToInboxCallback = () => { +// isCallbackCalled++ +// isPreAuthCalled = true +// } +// const preEnableIdentityCallback = () => { +// isCallbackCalled++ +// } +// const preCreateIdentityCallback = () => { +// isCallbackCalled++ +// } +// const keyBytes = new Uint8Array([ +// 233, 120, 198, 96, 154, 65, 132, 17, 132, 96, 250, 40, 103, 35, 125, 64, +// 166, 83, 208, 224, 254, 44, 205, 227, 175, 49, 234, 129, 74, 252, 135, 145, +// ]) + +// await Client.createRandom({ +// env: 'local', +// enableV3: true, +// preEnableIdentityCallback, +// preCreateIdentityCallback, +// preAuthenticateToInboxCallback, +// dbEncryptionKey: keyBytes, +// }) + +// assert( +// isCallbackCalled === 3, +// `callback should be called 3 times but was ${isCallbackCalled}` +// ) + +// if (!isPreAuthCalled) { +// throw new Error('preAuthenticateToInboxCallback not called') +// } - await createGroups(alixClient, [boClient], 50, 10) +// return true +// }) - let start = Date.now() - let groups = await alixClient.conversations.listGroups() - let end = Date.now() - console.log(`Alix loaded ${groups.length} groups in ${end - start}ms`) +// test('can delete a local database', async () => { +// let [client, anotherClient] = await createClients(2) + +// await client.conversations.newGroup([anotherClient.address]) +// await client.conversations.syncGroups() +// assert( +// (await client.conversations.listGroups()).length === 1, +// `should have a group size of 1 but was ${ +// (await client.conversations.listGroups()).length +// }` +// ) + +// assert( +// client.dbPath !== '', +// `client dbPath should be set but was ${client.dbPath}` +// ) +// await client.deleteLocalDatabase() +// client = await Client.createRandom({ +// env: 'local', +// appVersion: 'Testing/0.0.0', +// enableV3: true, +// dbEncryptionKey: new Uint8Array([ +// 233, 120, 198, 96, 154, 65, 132, 17, 132, 96, 250, 40, 103, 35, 125, 64, +// 166, 83, 208, 224, 254, 44, 205, 227, 175, 49, 234, 129, 74, 252, 135, +// 145, +// ]), +// }) +// await client.conversations.syncGroups() +// assert( +// (await client.conversations.listGroups()).length === 0, +// `should have a group size of 0 but was ${ +// (await client.conversations.listGroups()).length +// }` +// ) - start = Date.now() - await alixClient.conversations.syncGroups() - end = Date.now() - console.log(`Alix synced ${groups.length} groups in ${end - start}ms`) +// return true +// }) - start = Date.now() - await boClient.conversations.syncGroups() - end = Date.now() - console.log(`Bo synced ${groups.length} groups in ${end - start}ms`) +// test('can make a MLS V3 client with encryption key and database directory', async () => { +// // eslint-disable-next-line @typescript-eslint/no-unused-vars +// const dbDirPath = `${RNFS.DocumentDirectoryPath}/xmtp_db` +// const directoryExists = await RNFS.exists(dbDirPath) +// if (!directoryExists) { +// await RNFS.mkdir(dbDirPath) +// } +// const key = new Uint8Array([ +// 233, 120, 198, 96, 154, 65, 132, 17, 132, 96, 250, 40, 103, 35, 125, 64, +// 166, 83, 208, 224, 254, 44, 205, 227, 175, 49, 234, 129, 74, 252, 135, 145, +// ]) + +// const client = await Client.createRandom({ +// env: 'local', +// appVersion: 'Testing/0.0.0', +// enableV3: true, +// dbEncryptionKey: key, +// dbDirectory: dbDirPath, +// }) + +// const anotherClient = await Client.createRandom({ +// env: 'local', +// appVersion: 'Testing/0.0.0', +// enableV3: true, +// dbEncryptionKey: key, +// }) + +// await client.conversations.newGroup([anotherClient.address]) +// assert( +// (await client.conversations.listGroups()).length === 1, +// `should have a group size of 1 but was ${ +// (await client.conversations.listGroups()).length +// }` +// ) + +// const bundle = await client.exportKeyBundle() +// const clientFromBundle = await Client.createFromKeyBundle(bundle, { +// env: 'local', +// appVersion: 'Testing/0.0.0', +// enableV3: true, +// dbEncryptionKey: key, +// dbDirectory: dbDirPath, +// }) + +// assert( +// clientFromBundle.address === client.address, +// `clients dont match ${client.address} and ${clientFromBundle.address}` +// ) + +// assert( +// (await clientFromBundle.conversations.listGroups()).length === 1, +// `should have a group size of 1 but was ${ +// (await clientFromBundle.conversations.listGroups()).length +// }` +// ) +// return true +// }) - start = Date.now() - groups = await boClient.conversations.listGroups() - end = Date.now() - console.log(`Bo loaded ${groups.length} groups in ${end - start}ms`) +// test('testing large group listing with metadata performance', async () => { +// const [alixClient, boClient] = await createClients(2) - return true -}) +// await createGroups(alixClient, [boClient], 50, 10) -test('can drop a local database', async () => { - const [client, anotherClient] = await createClients(2) +// let start = Date.now() +// let groups = await alixClient.conversations.listGroups() +// let end = Date.now() +// console.log(`Alix loaded ${groups.length} groups in ${end - start}ms`) - const group = await client.conversations.newGroup([anotherClient.address]) - await client.conversations.syncGroups() - assert( - (await client.conversations.listGroups()).length === 1, - `should have a group size of 1 but was ${ - (await client.conversations.listGroups()).length - }` - ) +// start = Date.now() +// await alixClient.conversations.syncGroups() +// end = Date.now() +// console.log(`Alix synced ${groups.length} groups in ${end - start}ms`) - await client.dropLocalDatabaseConnection() +// start = Date.now() +// await boClient.conversations.syncGroups() +// end = Date.now() +// console.log(`Bo synced ${groups.length} groups in ${end - start}ms`) - try { - await group.send('hi') - // eslint-disable-next-line @typescript-eslint/no-unused-vars - } catch (error) { - await client.reconnectLocalDatabase() - await group.send('hi') - return true - } - throw new Error('should throw when local database not connected') -}) +// start = Date.now() +// groups = await boClient.conversations.listGroups() +// end = Date.now() +// console.log(`Bo loaded ${groups.length} groups in ${end - start}ms`) -test('can get a inboxId from an address', async () => { - const [alix, bo] = await createClients(2) +// return true +// }) - const boInboxId = await alix.findInboxIdFromAddress(bo.address) - assert(boInboxId === bo.inboxId, `${boInboxId} should match ${bo.inboxId}`) - return true -}) +// test('can drop a local database', async () => { +// const [client, anotherClient] = await createClients(2) + +// const group = await client.conversations.newGroup([anotherClient.address]) +// await client.conversations.syncGroups() +// assert( +// (await client.conversations.listGroups()).length === 1, +// `should have a group size of 1 but was ${ +// (await client.conversations.listGroups()).length +// }` +// ) + +// await client.dropLocalDatabaseConnection() + +// try { +// await group.send('hi') +// // eslint-disable-next-line @typescript-eslint/no-unused-vars +// } catch (error) { +// await client.reconnectLocalDatabase() +// await group.send('hi') +// return true +// } +// throw new Error('should throw when local database not connected') +// }) -test('can make a MLS V3 client from bundle', async () => { - const key = new Uint8Array([ - 233, 120, 198, 96, 154, 65, 132, 17, 132, 96, 250, 40, 103, 35, 125, 64, - 166, 83, 208, 224, 254, 44, 205, 227, 175, 49, 234, 129, 74, 252, 135, 145, - ]) +// test('can get a inboxId from an address', async () => { +// const [alix, bo] = await createClients(2) - const client = await Client.createRandom({ - env: 'local', - appVersion: 'Testing/0.0.0', - enableV3: true, - dbEncryptionKey: key, - }) +// const boInboxId = await alix.findInboxIdFromAddress(bo.address) +// assert(boInboxId === bo.inboxId, `${boInboxId} should match ${bo.inboxId}`) +// return true +// }) - const anotherClient = await Client.createRandom({ - env: 'local', - appVersion: 'Testing/0.0.0', - enableV3: true, - dbEncryptionKey: key, - }) +// test('can make a MLS V3 client from bundle', async () => { +// const key = new Uint8Array([ +// 233, 120, 198, 96, 154, 65, 132, 17, 132, 96, 250, 40, 103, 35, 125, 64, +// 166, 83, 208, 224, 254, 44, 205, 227, 175, 49, 234, 129, 74, 252, 135, 145, +// ]) + +// const client = await Client.createRandom({ +// env: 'local', +// appVersion: 'Testing/0.0.0', +// enableV3: true, +// dbEncryptionKey: key, +// }) + +// const anotherClient = await Client.createRandom({ +// env: 'local', +// appVersion: 'Testing/0.0.0', +// enableV3: true, +// dbEncryptionKey: key, +// }) + +// const group1 = await client.conversations.newGroup([anotherClient.address]) + +// assert( +// group1.client.address === client.address, +// `clients dont match ${client.address} and ${group1.client.address}` +// ) + +// const bundle = await client.exportKeyBundle() +// const client2 = await Client.createFromKeyBundle(bundle, { +// env: 'local', +// appVersion: 'Testing/0.0.0', +// enableV3: true, +// dbEncryptionKey: key, +// }) + +// assert( +// client.address === client2.address, +// `clients dont match ${client2.address} and ${client.address}` +// ) + +// assert( +// client.inboxId === client2.inboxId, +// `clients dont match ${client2.inboxId} and ${client.inboxId}` +// ) + +// assert( +// client.installationId === client2.installationId, +// `clients dont match ${client2.installationId} and ${client.installationId}` +// ) + +// const randomClient = await Client.createRandom({ +// env: 'local', +// appVersion: 'Testing/0.0.0', +// enableV3: true, +// dbEncryptionKey: key, +// }) + +// const group = await client2.conversations.newGroup([randomClient.address]) + +// assert( +// group.client.address === client2.address, +// `clients dont match ${client2.address} and ${group.client.address}` +// ) - const group1 = await client.conversations.newGroup([anotherClient.address]) +// return true +// }) - assert( - group1.client.address === client.address, - `clients dont match ${client.address} and ${group1.client.address}` - ) +// test('production MLS V3 client creation does not error', async () => { +// const key = new Uint8Array([ +// 233, 120, 198, 96, 154, 65, 132, 17, 132, 96, 250, 40, 103, 35, 125, 64, +// 166, 83, 208, 224, 254, 44, 205, 227, 175, 49, 234, 129, 74, 252, 135, 145, +// ]) + +// try { +// await Client.createRandom({ +// env: 'production', +// appVersion: 'Testing/0.0.0', +// enableV3: true, +// dbEncryptionKey: key, +// }) +// // eslint-disable-next-line @typescript-eslint/no-unused-vars +// } catch (error) { +// throw error +// } +// return true +// }) - const bundle = await client.exportKeyBundle() - const client2 = await Client.createFromKeyBundle(bundle, { - env: 'local', - appVersion: 'Testing/0.0.0', - enableV3: true, - dbEncryptionKey: key, - }) +// test('group message delivery status', async () => { +// const [alixClient, boClient] = await createClients(2) +// const alixGroup = await alixClient.conversations.newGroup([boClient.address]) - assert( - client.address === client2.address, - `clients dont match ${client2.address} and ${client.address}` - ) +// await alixGroup.send('hello, world') - assert( - client.inboxId === client2.inboxId, - `clients dont match ${client2.inboxId} and ${client.inboxId}` - ) +// const alixMessages: DecodedMessage[] = await alixGroup.messages() - assert( - client.installationId === client2.installationId, - `clients dont match ${client2.installationId} and ${client.installationId}` - ) +// assert( +// alixMessages.length === 2, +// `the messages length should be 2 but was ${alixMessages.length}` +// ) - const randomClient = await Client.createRandom({ - env: 'local', - appVersion: 'Testing/0.0.0', - enableV3: true, - dbEncryptionKey: key, - }) +// const alixMessagesFiltered: DecodedMessage[] = await alixGroup.messages({ +// deliveryStatus: MessageDeliveryStatus.PUBLISHED, +// }) - const group = await client2.conversations.newGroup([randomClient.address]) +// assert( +// alixMessagesFiltered.length === 2, +// `the messages length should be 2 but was ${alixMessagesFiltered.length}` +// ) - assert( - group.client.address === client2.address, - `clients dont match ${client2.address} and ${group.client.address}` - ) +// await alixGroup.sync() +// const alixMessages2: DecodedMessage[] = await alixGroup.messages() - return true -}) +// assert( +// alixMessages2.length === 2, +// `the messages length should be 2 but was ${alixMessages.length}` +// ) -test('production MLS V3 client creation does not error', async () => { - const key = new Uint8Array([ - 233, 120, 198, 96, 154, 65, 132, 17, 132, 96, 250, 40, 103, 35, 125, 64, - 166, 83, 208, 224, 254, 44, 205, 227, 175, 49, 234, 129, 74, 252, 135, 145, - ]) +// assert( +// alixMessages2[0].deliveryStatus === 'PUBLISHED', +// `the message should have a delivery status of PUBLISHED but was ${alixMessages2[0].deliveryStatus}` +// ) - try { - await Client.createRandom({ - env: 'production', - appVersion: 'Testing/0.0.0', - enableV3: true, - dbEncryptionKey: key, - }) - // eslint-disable-next-line @typescript-eslint/no-unused-vars - } catch (error) { - throw error - } - return true -}) +// await boClient.conversations.syncGroups() +// const boGroup = (await boClient.conversations.listGroups())[0] +// await boGroup.sync() +// const boMessages: DecodedMessage[] = await boGroup.messages() -test('group message delivery status', async () => { - const [alixClient, boClient] = await createClients(2) - const alixGroup = await alixClient.conversations.newGroup([boClient.address]) +// assert( +// boMessages.length === 1, +// `the messages length should be 1 but was ${boMessages.length}` +// ) - await alixGroup.send('hello, world') +// assert( +// boMessages[0].deliveryStatus === 'PUBLISHED', +// `the message should have a delivery status of PUBLISHED but was ${boMessages[0].deliveryStatus}` +// ) - const alixMessages: DecodedMessage[] = await alixGroup.messages() +// return true +// }) - assert( - alixMessages.length === 2, - `the messages length should be 2 but was ${alixMessages.length}` - ) +// test('can find a group by id', async () => { +// const [alixClient, boClient] = await createClients(2) +// const alixGroup = await alixClient.conversations.newGroup([boClient.address]) - const alixMessagesFiltered: DecodedMessage[] = await alixGroup.messages({ - deliveryStatus: MessageDeliveryStatus.PUBLISHED, - }) +// await boClient.conversations.syncGroups() +// const boGroup = await boClient.conversations.findGroup(alixGroup.id) - assert( - alixMessagesFiltered.length === 2, - `the messages length should be 2 but was ${alixMessagesFiltered.length}` - ) +// assert( +// boGroup?.id === alixGroup.id, +// `bo ${boGroup?.id} does not match alix ${alixGroup.id}` +// ) +// return true +// }) - await alixGroup.sync() - const alixMessages2: DecodedMessage[] = await alixGroup.messages() +// test('can find a message by id', async () => { +// const [alixClient, boClient] = await createClients(2) +// const alixGroup = await alixClient.conversations.newGroup([boClient.address]) +// const alixMessageId = await alixGroup.send('Hello') - assert( - alixMessages2.length === 2, - `the messages length should be 2 but was ${alixMessages.length}` - ) +// await boClient.conversations.syncGroups() +// const boGroup = await boClient.conversations.findGroup(alixGroup.id) +// await boGroup?.sync() +// const boMessage = await boClient.conversations.findV3Message(alixMessageId) - assert( - alixMessages2[0].deliveryStatus === 'PUBLISHED', - `the message should have a delivery status of PUBLISHED but was ${alixMessages2[0].deliveryStatus}` - ) +// assert( +// boMessage?.id === alixMessageId, +// `bo message ${boMessage?.id} does not match ${alixMessageId}` +// ) +// return true +// }) - await boClient.conversations.syncGroups() - const boGroup = (await boClient.conversations.listGroups())[0] - await boGroup.sync() - const boMessages: DecodedMessage[] = await boGroup.messages() +// test('who added me to a group', async () => { +// const [alixClient, boClient] = await createClients(2) +// await alixClient.conversations.newGroup([boClient.address]) - assert( - boMessages.length === 1, - `the messages length should be 1 but was ${boMessages.length}` - ) +// await boClient.conversations.syncGroups() +// const boGroup = (await boClient.conversations.listGroups())[0] +// const addedByInboxId = await boGroup.addedByInboxId - assert( - boMessages[0].deliveryStatus === 'PUBLISHED', - `the message should have a delivery status of PUBLISHED but was ${boMessages[0].deliveryStatus}` - ) +// assert( +// addedByInboxId === alixClient.inboxId, +// `addedByInboxId ${addedByInboxId} does not match ${alixClient.inboxId}` +// ) +// return true +// }) - return true -}) +// test('can get members of a group', async () => { +// const [alixClient, boClient] = await createClients(2) +// const group = await alixClient.conversations.newGroup([boClient.address]) + +// const members = group.members + +// assert(members.length === 2, `Should be 2 members but was ${members.length}`) + +// // We can not be sure of the order that members will be returned in +// for (const member of members) { +// // Alix created the group so they are a super admin +// if ( +// member.addresses[0].toLocaleLowerCase() === +// alixClient.address.toLocaleLowerCase() +// ) { +// assert( +// member.permissionLevel === 'super_admin', +// `Should be super_admin but was ${member.permissionLevel}` +// ) +// } +// // Bo did not create the group so he defaults to permission level "member" +// if ( +// member.addresses[0].toLocaleLowerCase() === +// boClient.address.toLocaleLowerCase() +// ) { +// assert( +// member.permissionLevel === 'member', +// `Should be member but was ${member.permissionLevel}` +// ) +// } +// } +// return true +// }) -test('can find a group by id', async () => { - const [alixClient, boClient] = await createClients(2) - const alixGroup = await alixClient.conversations.newGroup([boClient.address]) +// test('can message in a group', async () => { +// // Create three MLS enabled Clients +// const [alixClient, boClient, caroClient] = await createClients(3) - await boClient.conversations.syncGroups() - const boGroup = await boClient.conversations.findGroup(alixGroup.id) +// // alix's num groups start at 0 +// let alixGroups = await alixClient.conversations.listGroups() +// if (alixGroups.length !== 0) { +// throw new Error('num groups should be 0') +// } - assert( - boGroup?.id === alixGroup.id, - `bo ${boGroup?.id} does not match alix ${alixGroup.id}` - ) - return true -}) +// // alix creates a group +// const alixGroup = await alixClient.conversations.newGroup([ +// boClient.address, +// caroClient.address, +// ]) + +// // alix's num groups == 1 +// await alixClient.conversations.syncGroups() +// alixGroups = await alixClient.conversations.listGroups() +// if (alixGroups.length !== 1) { +// throw new Error('num groups should be 1') +// } -test('can find a message by id', async () => { - const [alixClient, boClient] = await createClients(2) - const alixGroup = await alixClient.conversations.newGroup([boClient.address]) - const alixMessageId = await alixGroup.send('Hello') +// // alix group should match create time from list function +// assert(alixGroups[0].createdAt === alixGroup.createdAt, 'group create time') - await boClient.conversations.syncGroups() - const boGroup = await boClient.conversations.findGroup(alixGroup.id) - await boGroup?.sync() - const boMessage = await boClient.conversations.findV3Message(alixMessageId) +// // alix can confirm memberInboxIds +// await alixGroup.sync() +// const memberInboxIds = await alixGroup.memberInboxIds() +// if (memberInboxIds.length !== 3) { +// throw new Error('num group members should be 3') +// } - assert( - boMessage?.id === alixMessageId, - `bo message ${boMessage?.id} does not match ${alixMessageId}` - ) - return true -}) +// if ( +// !( +// memberInboxIds.includes(alixClient.inboxId) && +// memberInboxIds.includes(boClient.inboxId) && +// memberInboxIds.includes(caroClient.inboxId) +// ) +// ) { +// throw new Error('missing address') +// } -test('who added me to a group', async () => { - const [alixClient, boClient] = await createClients(2) - await alixClient.conversations.newGroup([boClient.address]) +// // alix can send messages +// await alixGroup.send('hello, world') +// await alixGroup.send('gm') + +// // bo's num groups == 1 +// await boClient.conversations.syncGroups() +// const boGroups = await boClient.conversations.listGroups() +// if (boGroups.length !== 1) { +// throw new Error( +// 'num groups for bo should be 1, but it is' + boGroups.length +// ) +// } +// await delayToPropogate() +// // bo can read messages from alix +// await boGroups[0].sync() +// const boMessages: DecodedMessage[] = await boGroups[0].messages() + +// if (boMessages.length !== 2) { +// throw new Error( +// 'num messages for bo should be 2, but it is' + boMessages.length +// ) +// } +// if (boMessages[0].content() !== 'gm') { +// throw new Error("newest message should be 'gm'") +// } +// if (boMessages[1].content() !== 'hello, world') { +// throw new Error("newest message should be 'hello, world'") +// } +// // bo can send a message +// await boGroups[0].send('hey guys!') + +// // caro's num groups == 1 +// await caroClient.conversations.syncGroups() +// const caroGroups = await caroClient.conversations.listGroups() +// if (caroGroups.length !== 1) { +// throw new Error( +// 'num groups for caro should be 1, but it is' + caroGroups.length +// ) +// } - await boClient.conversations.syncGroups() - const boGroup = (await boClient.conversations.listGroups())[0] - const addedByInboxId = await boGroup.addedByInboxId +// // caro can read messages from alix and bo +// await caroGroups[0].sync() +// const caroMessages = await caroGroups[0].messages() - assert( - addedByInboxId === alixClient.inboxId, - `addedByInboxId ${addedByInboxId} does not match ${alixClient.inboxId}` - ) - return true -}) +// if (caroMessages.length !== 3) { +// throw new Error(`length should be 3 but was ${caroMessages.length}`) +// } +// if (caroMessages[0].content() !== 'hey guys!') { +// throw new Error( +// `newest Message should be 'hey guys!' but was ${caroMessages[0].content()}` +// ) +// } +// if (caroMessages[1].content() !== 'gm') { +// throw new Error( +// `second Message should be 'gm' but was ${caroMessages[1].content()}` +// ) +// } -test('can get members of a group', async () => { - const [alixClient, boClient] = await createClients(2) - const group = await alixClient.conversations.newGroup([boClient.address]) - - const members = group.members - - assert(members.length === 2, `Should be 2 members but was ${members.length}`) - - // We can not be sure of the order that members will be returned in - for (const member of members) { - // Alix created the group so they are a super admin - if ( - member.addresses[0].toLocaleLowerCase() === - alixClient.address.toLocaleLowerCase() - ) { - assert( - member.permissionLevel === 'super_admin', - `Should be super_admin but was ${member.permissionLevel}` - ) - } - // Bo did not create the group so he defaults to permission level "member" - if ( - member.addresses[0].toLocaleLowerCase() === - boClient.address.toLocaleLowerCase() - ) { - assert( - member.permissionLevel === 'member', - `Should be member but was ${member.permissionLevel}` - ) - } - } - return true -}) +// return true +// }) -test('can message in a group', async () => { - // Create three MLS enabled Clients - const [alixClient, boClient, caroClient] = await createClients(3) +// test('unpublished messages handling', async () => { +// // Initialize fixture clients +// const [alixClient, boClient] = await createClients(3) - // alix's num groups start at 0 - let alixGroups = await alixClient.conversations.listGroups() - if (alixGroups.length !== 0) { - throw new Error('num groups should be 0') - } +// // Create a new group with Bob and Alice +// const boGroup = await boClient.conversations.newGroup([alixClient.address]) - // alix creates a group - const alixGroup = await alixClient.conversations.newGroup([ - boClient.address, - caroClient.address, - ]) +// // Sync Alice's client to get the new group +// await alixClient.conversations.syncGroups() +// const alixGroup = await alixClient.conversations.findGroup(boGroup.id) +// if (!alixGroup) { +// throw new Error(`Group not found for id: ${boGroup.id}`) +// } - // alix's num groups == 1 - await alixClient.conversations.syncGroups() - alixGroups = await alixClient.conversations.listGroups() - if (alixGroups.length !== 1) { - throw new Error('num groups should be 1') - } +// // Check if the group is allowed initially +// let isGroupAllowed = await alixClient.contacts.isGroupAllowed(boGroup.id) +// if (isGroupAllowed) { +// throw new Error('Group should not be allowed initially') +// } - // alix group should match create time from list function - assert(alixGroups[0].createdAt === alixGroup.createdAt, 'group create time') +// // Prepare a message in the group +// const preparedMessageId = await alixGroup.prepareMessage('Test text') - // alix can confirm memberInboxIds - await alixGroup.sync() - const memberInboxIds = await alixGroup.memberInboxIds() - if (memberInboxIds.length !== 3) { - throw new Error('num group members should be 3') - } +// // Check if the group is allowed after preparing the message +// isGroupAllowed = await alixClient.contacts.isGroupAllowed(boGroup.id) +// if (!isGroupAllowed) { +// throw new Error('Group should be allowed after preparing a message') +// } - if ( - !( - memberInboxIds.includes(alixClient.inboxId) && - memberInboxIds.includes(boClient.inboxId) && - memberInboxIds.includes(caroClient.inboxId) - ) - ) { - throw new Error('missing address') - } +// // Verify the message count in the group +// let messageCount = (await alixGroup.messages()).length +// if (messageCount !== 1) { +// throw new Error(`Message count should be 1, but it is ${messageCount}`) +// } - // alix can send messages - await alixGroup.send('hello, world') - await alixGroup.send('gm') - - // bo's num groups == 1 - await boClient.conversations.syncGroups() - const boGroups = await boClient.conversations.listGroups() - if (boGroups.length !== 1) { - throw new Error( - 'num groups for bo should be 1, but it is' + boGroups.length - ) - } - await delayToPropogate() - // bo can read messages from alix - await boGroups[0].sync() - const boMessages: DecodedMessage[] = await boGroups[0].messages() - - if (boMessages.length !== 2) { - throw new Error( - 'num messages for bo should be 2, but it is' + boMessages.length - ) - } - if (boMessages[0].content() !== 'gm') { - throw new Error("newest message should be 'gm'") - } - if (boMessages[1].content() !== 'hello, world') { - throw new Error("newest message should be 'hello, world'") - } - // bo can send a message - await boGroups[0].send('hey guys!') - - // caro's num groups == 1 - await caroClient.conversations.syncGroups() - const caroGroups = await caroClient.conversations.listGroups() - if (caroGroups.length !== 1) { - throw new Error( - 'num groups for caro should be 1, but it is' + caroGroups.length - ) - } +// // Verify the count of published and unpublished messages +// let messageCountPublished = ( +// await alixGroup.messages({ +// deliveryStatus: MessageDeliveryStatus.PUBLISHED, +// }) +// ).length +// let messageCountUnpublished = ( +// await alixGroup.messages({ +// deliveryStatus: MessageDeliveryStatus.UNPUBLISHED, +// }) +// ).length +// if (messageCountPublished !== 0) { +// throw new Error( +// `Published message count should be 0, but it is ${messageCountPublished}` +// ) +// } +// if (messageCountUnpublished !== 1) { +// throw new Error( +// `Unpublished message count should be 1, but it is ${messageCountUnpublished}` +// ) +// } - // caro can read messages from alix and bo - await caroGroups[0].sync() - const caroMessages = await caroGroups[0].messages() +// // Publish the prepared message +// await alixGroup.publishPreparedMessages() + +// // Sync the group after publishing the message +// await alixGroup.sync() + +// // Verify the message counts again +// messageCountPublished = ( +// await alixGroup.messages({ +// deliveryStatus: MessageDeliveryStatus.PUBLISHED, +// }) +// ).length +// messageCountUnpublished = ( +// await alixGroup.messages({ +// deliveryStatus: MessageDeliveryStatus.UNPUBLISHED, +// }) +// ).length +// messageCount = (await alixGroup.messages()).length +// if (messageCountPublished !== 1) { +// throw new Error( +// `Published message count should be 1, but it is ${messageCountPublished}` +// ) +// } +// if (messageCountUnpublished !== 0) { +// throw new Error( +// `Unpublished message count should be 0, but it is ${messageCountUnpublished}` +// ) +// } +// if (messageCount !== 1) { +// throw new Error(`Message count should be 1, but it is ${messageCount}`) +// } - if (caroMessages.length !== 3) { - throw new Error(`length should be 3 but was ${caroMessages.length}`) - } - if (caroMessages[0].content() !== 'hey guys!') { - throw new Error( - `newest Message should be 'hey guys!' but was ${caroMessages[0].content()}` - ) - } - if (caroMessages[1].content() !== 'gm') { - throw new Error( - `second Message should be 'gm' but was ${caroMessages[1].content()}` - ) - } +// // Retrieve all messages and verify the prepared message ID +// const messages = await alixGroup.messages() +// if (preparedMessageId !== messages[0].id) { +// throw new Error(`Message ID should match the prepared message ID`) +// } - return true -}) +// return true +// }) -test('unpublished messages handling', async () => { - // Initialize fixture clients - const [alixClient, boClient] = await createClients(3) +// test('can add members to a group', async () => { +// // Create three MLS enabled Clients +// const [alixClient, boClient, caroClient] = await createClients(3) - // Create a new group with Bob and Alice - const boGroup = await boClient.conversations.newGroup([alixClient.address]) +// // alix's num groups start at 0 +// let alixGroups = await alixClient.conversations.listGroups() +// if (alixGroups.length !== 0) { +// throw new Error('num groups should be 0') +// } - // Sync Alice's client to get the new group - await alixClient.conversations.syncGroups() - const alixGroup = await alixClient.conversations.findGroup(boGroup.id) - if (!alixGroup) { - throw new Error(`Group not found for id: ${boGroup.id}`) - } +// // bo's num groups start at 0 +// let boGroups = await boClient.conversations.listGroups() +// if (boGroups.length !== 0) { +// throw new Error('num groups should be 0') +// } - // Check if the group is allowed initially - let isGroupAllowed = await alixClient.contacts.isGroupAllowed(boGroup.id) - if (isGroupAllowed) { - throw new Error('Group should not be allowed initially') - } +// // caro's num groups start at 0 +// let caroGroups = await caroClient.conversations.listGroups() +// if (caroGroups.length !== 0) { +// throw new Error('num groups should be 0') +// } - // Prepare a message in the group - const preparedMessageId = await alixGroup.prepareMessage('Test text') +// // alix creates a group +// const alixGroup = await alixClient.conversations.newGroup([boClient.address]) - // Check if the group is allowed after preparing the message - isGroupAllowed = await alixClient.contacts.isGroupAllowed(boGroup.id) - if (!isGroupAllowed) { - throw new Error('Group should be allowed after preparing a message') - } +// // alix's num groups == 1 +// await alixClient.conversations.syncGroups() +// alixGroups = await alixClient.conversations.listGroups() +// if (alixGroups.length !== 1) { +// throw new Error('num groups should be 1') +// } - // Verify the message count in the group - let messageCount = (await alixGroup.messages()).length - if (messageCount !== 1) { - throw new Error(`Message count should be 1, but it is ${messageCount}`) - } +// // alix can confirm memberInboxIds +// await alixGroup.sync() +// const memberInboxIds = await alixGroup.memberInboxIds() +// if (memberInboxIds.length !== 2) { +// throw new Error('num group members should be 2') +// } +// if ( +// !( +// memberInboxIds.includes(alixClient.inboxId) && +// memberInboxIds.includes(boClient.inboxId) +// ) +// ) { +// throw new Error('missing address') +// } - // Verify the count of published and unpublished messages - let messageCountPublished = ( - await alixGroup.messages({ - deliveryStatus: MessageDeliveryStatus.PUBLISHED, - }) - ).length - let messageCountUnpublished = ( - await alixGroup.messages({ - deliveryStatus: MessageDeliveryStatus.UNPUBLISHED, - }) - ).length - if (messageCountPublished !== 0) { - throw new Error( - `Published message count should be 0, but it is ${messageCountPublished}` - ) - } - if (messageCountUnpublished !== 1) { - throw new Error( - `Unpublished message count should be 1, but it is ${messageCountUnpublished}` - ) - } +// // alix can send messages +// await alixGroup.send('hello, world') +// await alixGroup.send('gm') + +// // bo's num groups == 1 +// await boClient.conversations.syncGroups() +// boGroups = await boClient.conversations.listGroups() +// if (boGroups.length !== 1) { +// throw new Error( +// 'num groups for bo should be 1, but it is' + boGroups.length +// ) +// } - // Publish the prepared message - await alixGroup.publishPreparedMessages() +// await alixGroup.addMembers([caroClient.address]) - // Sync the group after publishing the message - await alixGroup.sync() +// // caro's num groups == 1 +// await caroClient.conversations.syncGroups() +// caroGroups = await caroClient.conversations.listGroups() +// if (caroGroups.length !== 1) { +// throw new Error( +// 'num groups for caro should be 1, but it is' + caroGroups.length +// ) +// } +// await caroGroups[0].sync() +// const caroMessages = await caroGroups[0].messages() +// if (caroMessages.length !== 0) { +// throw new Error('num messages for caro should be 0') +// } - // Verify the message counts again - messageCountPublished = ( - await alixGroup.messages({ - deliveryStatus: MessageDeliveryStatus.PUBLISHED, - }) - ).length - messageCountUnpublished = ( - await alixGroup.messages({ - deliveryStatus: MessageDeliveryStatus.UNPUBLISHED, - }) - ).length - messageCount = (await alixGroup.messages()).length - if (messageCountPublished !== 1) { - throw new Error( - `Published message count should be 1, but it is ${messageCountPublished}` - ) - } - if (messageCountUnpublished !== 0) { - throw new Error( - `Unpublished message count should be 0, but it is ${messageCountUnpublished}` - ) - } - if (messageCount !== 1) { - throw new Error(`Message count should be 1, but it is ${messageCount}`) - } +// await boGroups[0].sync() +// const boGroupMembers = await boGroups[0].memberInboxIds() +// if (boGroupMembers.length !== 3) { +// throw new Error('num group members should be 3') +// } - // Retrieve all messages and verify the prepared message ID - const messages = await alixGroup.messages() - if (preparedMessageId !== messages[0].id) { - throw new Error(`Message ID should match the prepared message ID`) - } +// return true +// }) - return true -}) +// test('can remove members from a group', async () => { +// // Create three MLS enabled Clients +// const [alixClient, boClient, caroClient] = await createClients(3) -test('can add members to a group', async () => { - // Create three MLS enabled Clients - const [alixClient, boClient, caroClient] = await createClients(3) +// // alix's num groups start at 0 +// let alixGroups = await alixClient.conversations.listGroups() +// if (alixGroups.length !== 0) { +// throw new Error('num groups should be 0') +// } - // alix's num groups start at 0 - let alixGroups = await alixClient.conversations.listGroups() - if (alixGroups.length !== 0) { - throw new Error('num groups should be 0') - } +// // bo's num groups start at 0 +// let boGroups = await boClient.conversations.listGroups() +// assert(boGroups.length === 0, 'num groups should be 0') - // bo's num groups start at 0 - let boGroups = await boClient.conversations.listGroups() - if (boGroups.length !== 0) { - throw new Error('num groups should be 0') - } +// // caro's num groups start at 0 +// let caroGroups = await caroClient.conversations.listGroups() +// if (caroGroups.length !== 0) { +// throw new Error('num groups should be 0') +// } - // caro's num groups start at 0 - let caroGroups = await caroClient.conversations.listGroups() - if (caroGroups.length !== 0) { - throw new Error('num groups should be 0') - } +// // alix creates a group +// const alixGroup = await alixClient.conversations.newGroup([ +// boClient.address, +// caroClient.address, +// ]) + +// // alix's num groups == 1 +// await alixClient.conversations.syncGroups() +// alixGroups = await alixClient.conversations.listGroups() +// if (alixGroups.length !== 1) { +// throw new Error('num groups should be 1') +// } - // alix creates a group - const alixGroup = await alixClient.conversations.newGroup([boClient.address]) +// // alix can confirm memberInboxIds +// await alixGroup.sync() +// const memberInboxIds = await alixGroup.memberInboxIds() +// if (memberInboxIds.length !== 3) { +// throw new Error('num group members should be 3') +// } +// if ( +// !( +// memberInboxIds.includes(alixClient.inboxId) && +// memberInboxIds.includes(boClient.inboxId) +// ) +// ) { +// throw new Error('missing address') +// } - // alix's num groups == 1 - await alixClient.conversations.syncGroups() - alixGroups = await alixClient.conversations.listGroups() - if (alixGroups.length !== 1) { - throw new Error('num groups should be 1') - } +// // alix can send messages +// await alixGroup.send('hello, world') +// await alixGroup.send('gm') + +// // bo's num groups == 1 +// await boClient.conversations.syncGroups() +// boGroups = await boClient.conversations.listGroups() +// if (boGroups.length !== 1) { +// throw new Error( +// 'num groups for bo should be 1, but it is' + boGroups.length +// ) +// } - // alix can confirm memberInboxIds - await alixGroup.sync() - const memberInboxIds = await alixGroup.memberInboxIds() - if (memberInboxIds.length !== 2) { - throw new Error('num group members should be 2') - } - if ( - !( - memberInboxIds.includes(alixClient.inboxId) && - memberInboxIds.includes(boClient.inboxId) - ) - ) { - throw new Error('missing address') - } +// // caro's num groups == 1 +// await caroClient.conversations.syncGroups() +// caroGroups = await caroClient.conversations.listGroups() +// if (caroGroups.length !== 1) { +// throw new Error( +// 'num groups for caro should be 1, but it is' + caroGroups.length +// ) +// } - // alix can send messages - await alixGroup.send('hello, world') - await alixGroup.send('gm') - - // bo's num groups == 1 - await boClient.conversations.syncGroups() - boGroups = await boClient.conversations.listGroups() - if (boGroups.length !== 1) { - throw new Error( - 'num groups for bo should be 1, but it is' + boGroups.length - ) - } +// await caroGroups[0].sync() +// if (!caroGroups[0].isActive()) { +// throw new Error('caros group should be active') +// } - await alixGroup.addMembers([caroClient.address]) +// await alixGroup.removeMembers([caroClient.address]) +// await alixGroup.sync() +// const alixGroupMembers = await alixGroup.memberInboxIds() +// if (alixGroupMembers.length !== 2) { +// throw new Error( +// 'num group members should be 2 but was' + alixGroupMembers.length +// ) +// } - // caro's num groups == 1 - await caroClient.conversations.syncGroups() - caroGroups = await caroClient.conversations.listGroups() - if (caroGroups.length !== 1) { - throw new Error( - 'num groups for caro should be 1, but it is' + caroGroups.length - ) - } - await caroGroups[0].sync() - const caroMessages = await caroGroups[0].messages() - if (caroMessages.length !== 0) { - throw new Error('num messages for caro should be 0') - } +// await caroGroups[0].sync() +// if (await caroGroups[0].isActive()) { +// throw new Error('caros group should not be active') +// } - await boGroups[0].sync() - const boGroupMembers = await boGroups[0].memberInboxIds() - if (boGroupMembers.length !== 3) { - throw new Error('num group members should be 3') - } +// const caroGroupMembers = await caroGroups[0].memberInboxIds() +// if (caroGroupMembers.length !== 2) { +// throw new Error( +// 'num group members should be 2 but was' + caroGroupMembers.length +// ) +// } - return true -}) +// return true +// }) -test('can remove members from a group', async () => { - // Create three MLS enabled Clients - const [alixClient, boClient, caroClient] = await createClients(3) +// test('can remove and add members from a group by inbox id', async () => { +// // Create three MLS enabled Clients +// const [alixClient, boClient, caroClient] = await createClients(3) + +// // alix creates a group +// const alixGroup = await alixClient.conversations.newGroup([ +// boClient.address, +// caroClient.address, +// ]) + +// // alix can confirm memberInboxIds +// await alixGroup.sync() +// const memberInboxIds = await alixGroup.memberInboxIds() +// if (memberInboxIds.length !== 3) { +// throw new Error('num group members should be 3') +// } - // alix's num groups start at 0 - let alixGroups = await alixClient.conversations.listGroups() - if (alixGroups.length !== 0) { - throw new Error('num groups should be 0') - } +// await alixGroup.removeMembersByInboxId([caroClient.inboxId]) +// await alixGroup.sync() +// const alixGroupMembers = await alixGroup.memberInboxIds() +// if (alixGroupMembers.length !== 2) { +// throw new Error('num group members should be 2') +// } - // bo's num groups start at 0 - let boGroups = await boClient.conversations.listGroups() - assert(boGroups.length === 0, 'num groups should be 0') +// await alixGroup.addMembersByInboxId([caroClient.inboxId]) +// await alixGroup.sync() +// const alixGroupMembers2 = await alixGroup.memberInboxIds() +// if (alixGroupMembers2.length !== 3) { +// throw new Error('num group members should be 3') +// } - // caro's num groups start at 0 - let caroGroups = await caroClient.conversations.listGroups() - if (caroGroups.length !== 0) { - throw new Error('num groups should be 0') - } - - // alix creates a group - const alixGroup = await alixClient.conversations.newGroup([ - boClient.address, - caroClient.address, - ]) - - // alix's num groups == 1 - await alixClient.conversations.syncGroups() - alixGroups = await alixClient.conversations.listGroups() - if (alixGroups.length !== 1) { - throw new Error('num groups should be 1') - } - - // alix can confirm memberInboxIds - await alixGroup.sync() - const memberInboxIds = await alixGroup.memberInboxIds() - if (memberInboxIds.length !== 3) { - throw new Error('num group members should be 3') - } - if ( - !( - memberInboxIds.includes(alixClient.inboxId) && - memberInboxIds.includes(boClient.inboxId) - ) - ) { - throw new Error('missing address') - } - - // alix can send messages - await alixGroup.send('hello, world') - await alixGroup.send('gm') - - // bo's num groups == 1 - await boClient.conversations.syncGroups() - boGroups = await boClient.conversations.listGroups() - if (boGroups.length !== 1) { - throw new Error( - 'num groups for bo should be 1, but it is' + boGroups.length - ) - } - - // caro's num groups == 1 - await caroClient.conversations.syncGroups() - caroGroups = await caroClient.conversations.listGroups() - if (caroGroups.length !== 1) { - throw new Error( - 'num groups for caro should be 1, but it is' + caroGroups.length - ) - } - - await caroGroups[0].sync() - if (!caroGroups[0].isActive()) { - throw new Error('caros group should be active') - } - - await alixGroup.removeMembers([caroClient.address]) - await alixGroup.sync() - const alixGroupMembers = await alixGroup.memberInboxIds() - if (alixGroupMembers.length !== 2) { - throw new Error( - 'num group members should be 2 but was' + alixGroupMembers.length - ) - } - - await caroGroups[0].sync() - if (await caroGroups[0].isActive()) { - throw new Error('caros group should not be active') - } - - const caroGroupMembers = await caroGroups[0].memberInboxIds() - if (caroGroupMembers.length !== 2) { - throw new Error( - 'num group members should be 2 but was' + caroGroupMembers.length - ) - } - - return true -}) - -test('can remove and add members from a group by inbox id', async () => { - // Create three MLS enabled Clients - const [alixClient, boClient, caroClient] = await createClients(3) - - // alix creates a group - const alixGroup = await alixClient.conversations.newGroup([ - boClient.address, - caroClient.address, - ]) - - // alix can confirm memberInboxIds - await alixGroup.sync() - const memberInboxIds = await alixGroup.memberInboxIds() - if (memberInboxIds.length !== 3) { - throw new Error('num group members should be 3') - } - - await alixGroup.removeMembersByInboxId([caroClient.inboxId]) - await alixGroup.sync() - const alixGroupMembers = await alixGroup.memberInboxIds() - if (alixGroupMembers.length !== 2) { - throw new Error('num group members should be 2') - } - - await alixGroup.addMembersByInboxId([caroClient.inboxId]) - await alixGroup.sync() - const alixGroupMembers2 = await alixGroup.memberInboxIds() - if (alixGroupMembers2.length !== 3) { - throw new Error('num group members should be 3') - } - - return true -}) - -test('can stream both groups and messages at same time', async () => { - const [alix, bo] = await createClients(2) - - let groupCallbacks = 0 - let messageCallbacks = 0 - await bo.conversations.streamGroups(async () => { - groupCallbacks++ - }) - - await bo.conversations.streamAllMessages(async () => { - messageCallbacks++ - }, true) - - const group = await alix.conversations.newGroup([bo.address]) - await group.send('hello') - - await delayToPropogate() - // await new Promise((resolve) => setTimeout(resolve, 10000)) - assert( - messageCallbacks === 1, - 'message stream should have received 1 message' - ) - assert(groupCallbacks === 1, 'group stream should have received 1 group') - return true -}) - -test('can stream groups', async () => { - const [alixClient, boClient, caroClient] = await createClients(3) - - // Start streaming groups - const groups: Group[] = [] - const cancelStreamGroups = await alixClient.conversations.streamGroups( - async (group: Group) => { - groups.push(group) - } - ) - - // caro creates a group with alix, so stream callback is fired - // eslint-disable-next-line @typescript-eslint/no-unused-vars - const caroGroup = await caroClient.conversations.newGroup([ - alixClient.address, - ]) - await delayToPropogate() - if ((groups.length as number) !== 1) { - throw Error('Unexpected num groups (should be 1): ' + groups.length) - } - - assert(groups[0].members.length == 2, 'should be 2') - - // bo creates a group with alix so a stream callback is fired - // eslint-disable-next-line @typescript-eslint/no-unused-vars - const boGroup = await boClient.conversations.newGroup([alixClient.address]) - await delayToPropogate() - if ((groups.length as number) !== 2) { - throw Error('Unexpected num groups (should be 2): ' + groups.length) - } - - // * Note alix creating a group does not trigger alix conversations - // group stream. Workaround is to syncGroups after you create and list manually - // See https://github.com/xmtp/libxmtp/issues/504 - - // alix creates a group - // eslint-disable-next-line @typescript-eslint/no-unused-vars - const alixGroup = await alixClient.conversations.newGroup([ - boClient.address, - caroClient.address, - ]) - await delayToPropogate() - if (groups.length !== 3) { - throw Error('Expected group length 3 but it is: ' + groups.length) - } - // Sync groups after creation if you created a group - const listedGroups = await alixClient.conversations.listGroups() - await delayToPropogate() - groups.push(listedGroups[listedGroups.length - 1]) - if ((groups.length as number) !== 4) { - throw Error('Expected group length 4 but it is: ' + groups.length) - } - - cancelStreamGroups() - await delayToPropogate() - - // Creating a group should no longer trigger stream groups - // eslint-disable-next-line @typescript-eslint/no-unused-vars - const caroSecond = await caroClient.conversations.newGroup([ - alixClient.address, - ]) - await delayToPropogate() - if ((groups.length as number) !== 4) { - throw Error('Unexpected num groups (should be 4): ' + groups.length) - } - - return true -}) - -test('can list groups', async () => { - const [alixClient, boClient] = await createClients(2) - - const group1 = await boClient.conversations.newGroup([alixClient.address], { - name: 'group1 name', - imageUrlSquare: 'www.group1image.com', - }) - const group2 = await boClient.conversations.newGroup([alixClient.address], { - name: 'group2 name', - imageUrlSquare: 'www.group2image.com', - }) - - const boGroups = await boClient.conversations.listGroups() - await alixClient.conversations.syncGroups() - const alixGroups = await alixClient.conversations.listGroups() - - assert( - boGroups.length === alixGroups.length, - `group lengths should be the same but bo was ${boGroups.length} and alix was ${alixGroups.length}` - ) - - const boGroup1 = await boClient.conversations.findGroup(group1.id) - const boGroup2 = await boClient.conversations.findGroup(group2.id) - - const alixGroup1 = await alixClient.conversations.findGroup(group1.id) - const alixGroup2 = await alixClient.conversations.findGroup(group2.id) - - assert( - boGroup2?.name === 'group2 name', - `Group 2 name for bo should be group2 name but was ${boGroup2?.name}` - ) - - assert( - boGroup1?.imageUrlSquare === 'www.group1image.com', - `Group 2 url for bo should be www.group1image.com but was ${boGroup1?.imageUrlSquare}` - ) - - assert( - alixGroup1?.name === 'group1 name', - `Group 1 name for alix should be group1 name but was ${alixGroup1?.name}` - ) - - assert( - alixGroup2?.imageUrlSquare === 'www.group2image.com', - `Group 2 url for alix should be www.group2image.com but was ${alixGroup2?.imageUrlSquare}` - ) - - assert(boGroup1?.isGroupActive === true, `Group 1 should be active for bo`) - - return true -}) - -test('can list all groups and conversations', async () => { - const [alixClient, boClient, caroClient] = await createClients(3) - - // Add one group and one conversation - const boGroup = await boClient.conversations.newGroup([alixClient.address]) - const alixConversation = await alixClient.conversations.newConversation( - caroClient.address - ) - - const listedContainers = await alixClient.conversations.listAll() - - // Verify information in listed containers is correct - // BUG - List All returns in Chronological order on iOS - // and reverse Chronological order on Android - const first = 0 - const second = 1 - if ( - listedContainers[first].topic !== boGroup.topic || - listedContainers[first].version !== ConversationVersion.GROUP || - listedContainers[second].version !== ConversationVersion.DIRECT || - listedContainers[second].createdAt !== alixConversation.createdAt - ) { - throw Error('Listed containers should match streamed containers') - } - - return true -}) +// return true +// }) -test('can stream all groups and conversations', async () => { - const [alixClient, boClient, caroClient] = await createClients(3) +// test('can stream both groups and messages at same time', async () => { +// const [alix, bo] = await createClients(2) - // Start streaming groups and conversations - const containers: ConversationContainer[] = [] - const cancelStreamAll = await alixClient.conversations.streamAll( - async (conversationContainer: ConversationContainer) => { - containers.push(conversationContainer) - } - ) +// let groupCallbacks = 0 +// let messageCallbacks = 0 +// await bo.conversations.streamGroups(async () => { +// groupCallbacks++ +// }) - // bo creates a group with alix, so stream callback is fired - // eslint-disable-next-line @typescript-eslint/no-unused-vars - const boGroup = await boClient.conversations.newGroup([alixClient.address]) - await delayToPropogate() - if ((containers.length as number) !== 1) { - throw Error('Unexpected num groups (should be 1): ' + containers.length) - } +// await bo.conversations.streamAllMessages(async () => { +// messageCallbacks++ +// }, true) - // bo creates a v2 Conversation with alix so a stream callback is fired - // eslint-disable-next-line @typescript-eslint/no-unused-vars - const boConversation = await boClient.conversations.newConversation( - alixClient.address - ) - await delayToPropogate() - if ((containers.length as number) !== 2) { - throw Error('Unexpected num groups (should be 2): ' + containers.length) - } +// const group = await alix.conversations.newGroup([bo.address]) +// await group.send('hello') - if ( - containers[1].version === ConversationVersion.DIRECT && - boConversation.conversationID !== - (containers[1] as Conversation).conversationID - ) { - throw Error( - 'Conversation from streamed all should match conversationID with created conversation' - ) - } +// await delayToPropogate() +// // await new Promise((resolve) => setTimeout(resolve, 10000)) +// assert( +// messageCallbacks === 1, +// 'message stream should have received 1 message' +// ) +// assert(groupCallbacks === 1, 'group stream should have received 1 group') +// return true +// }) - // * Note alix creating a v2 Conversation does trigger alix conversations - // stream. +// test('can stream groups', async () => { +// const [alixClient, boClient, caroClient] = await createClients(3) + +// // Start streaming groups +// const groups: Group[] = [] +// const cancelStreamGroups = await alixClient.conversations.streamGroups( +// async (group: Group) => { +// groups.push(group) +// } +// ) + +// // caro creates a group with alix, so stream callback is fired +// // eslint-disable-next-line @typescript-eslint/no-unused-vars +// const caroGroup = await caroClient.conversations.newGroup([ +// alixClient.address, +// ]) +// await delayToPropogate() +// if ((groups.length as number) !== 1) { +// throw Error('Unexpected num groups (should be 1): ' + groups.length) +// } - // alix creates a V2 Conversationgroup - // eslint-disable-next-line @typescript-eslint/no-unused-vars - const alixConversation = await alixClient.conversations.newConversation( - caroClient.address - ) - await delayToPropogate() - if (containers.length !== 3) { - throw Error('Expected group length 3 but it is: ' + containers.length) - } +// assert(groups[0].members.length == 2, 'should be 2') - cancelStreamAll() - await delayToPropogate() +// // bo creates a group with alix so a stream callback is fired +// // eslint-disable-next-line @typescript-eslint/no-unused-vars +// const boGroup = await boClient.conversations.newGroup([alixClient.address]) +// await delayToPropogate() +// if ((groups.length as number) !== 2) { +// throw Error('Unexpected num groups (should be 2): ' + groups.length) +// } - // Creating a group should no longer trigger stream groups - // eslint-disable-next-line @typescript-eslint/no-unused-vars - const caroConversation = await caroClient.conversations.newGroup([ - alixClient.address, - ]) - await delayToPropogate() - if ((containers.length as number) !== 3) { - throw Error('Unexpected num groups (should be 3): ' + containers.length) - } +// // * Note alix creating a group does not trigger alix conversations +// // group stream. Workaround is to syncGroups after you create and list manually +// // See https://github.com/xmtp/libxmtp/issues/504 - return true -}) +// // alix creates a group +// // eslint-disable-next-line @typescript-eslint/no-unused-vars +// const alixGroup = await alixClient.conversations.newGroup([ +// boClient.address, +// caroClient.address, +// ]) +// await delayToPropogate() +// if (groups.length !== 3) { +// throw Error('Expected group length 3 but it is: ' + groups.length) +// } +// // Sync groups after creation if you created a group +// const listedGroups = await alixClient.conversations.listGroups() +// await delayToPropogate() +// groups.push(listedGroups[listedGroups.length - 1]) +// if ((groups.length as number) !== 4) { +// throw Error('Expected group length 4 but it is: ' + groups.length) +// } -test('canMessage', async () => { - const [bo, alix, caro] = await createClients(3) +// cancelStreamGroups() +// await delayToPropogate() - const canMessage = await bo.canMessage(alix.address) - if (!canMessage) { - throw new Error('should be able to message v2 client') - } +// // Creating a group should no longer trigger stream groups +// // eslint-disable-next-line @typescript-eslint/no-unused-vars +// const caroSecond = await caroClient.conversations.newGroup([ +// alixClient.address, +// ]) +// await delayToPropogate() +// if ((groups.length as number) !== 4) { +// throw Error('Unexpected num groups (should be 4): ' + groups.length) +// } - const canMessageV3 = await caro.canGroupMessage([ - caro.address, - alix.address, - '0x0000000000000000000000000000000000000000', - ]) +// return true +// }) - assert( - canMessageV3['0x0000000000000000000000000000000000000000'] === false, - `should not be able to message 0x0000000000000000000000000000000000000000` - ) +// test('can list groups', async () => { +// const [alixClient, boClient] = await createClients(2) - assert( - canMessageV3[caro.address.toLowerCase()] === true, - `should be able to message ${caro.address}` - ) +// const group1 = await boClient.conversations.newGroup([alixClient.address], { +// name: 'group1 name', +// imageUrlSquare: 'www.group1image.com', +// }) +// const group2 = await boClient.conversations.newGroup([alixClient.address], { +// name: 'group2 name', +// imageUrlSquare: 'www.group2image.com', +// }) - assert( - canMessageV3[alix.address.toLowerCase()] === true, - `should be able to message ${alix.address}` - ) +// const boGroups = await boClient.conversations.listGroups() +// await alixClient.conversations.syncGroups() +// const alixGroups = await alixClient.conversations.listGroups() - return true -}) +// assert( +// boGroups.length === alixGroups.length, +// `group lengths should be the same but bo was ${boGroups.length} and alix was ${alixGroups.length}` +// ) -test('can stream group messages', async () => { - // Create three MLS enabled Clients - const [alixClient, boClient, caroClient] = await createClients(3) +// const boGroup1 = await boClient.conversations.findGroup(group1.id) +// const boGroup2 = await boClient.conversations.findGroup(group2.id) - // alix creates a group - const alixGroup = await alixClient.conversations.newGroup([ - boClient.address, - caroClient.address, - ]) +// const alixGroup1 = await alixClient.conversations.findGroup(group1.id) +// const alixGroup2 = await alixClient.conversations.findGroup(group2.id) - // Record message stream for this group - const groupMessages: DecodedMessage[] = [] - const cancelGroupMessageStream = await alixGroup.streamGroupMessages( - async (message) => { - groupMessages.push(message) - } - ) +// assert( +// boGroup2?.name === 'group2 name', +// `Group 2 name for bo should be group2 name but was ${boGroup2?.name}` +// ) - // bo's num groups == 1 - await boClient.conversations.syncGroups() - const boGroup = (await boClient.conversations.listGroups())[0] +// assert( +// boGroup1?.imageUrlSquare === 'www.group1image.com', +// `Group 2 url for bo should be www.group1image.com but was ${boGroup1?.imageUrlSquare}` +// ) - for (let i = 0; i < 5; i++) { - await boGroup.send({ text: `Message ${i}` }) - await delayToPropogate() - } +// assert( +// alixGroup1?.name === 'group1 name', +// `Group 1 name for alix should be group1 name but was ${alixGroup1?.name}` +// ) - if (groupMessages.length !== 5) { - throw Error('Unexpected convo messages count ' + groupMessages.length) - } - for (let i = 0; i < 5; i++) { - if (groupMessages[i].content() !== `Message ${i}`) { - throw Error( - 'Unexpected group message content ' + groupMessages[i].content() - ) - } - } +// assert( +// alixGroup2?.imageUrlSquare === 'www.group2image.com', +// `Group 2 url for alix should be www.group2image.com but was ${alixGroup2?.imageUrlSquare}` +// ) - cancelGroupMessageStream() - for (let i = 0; i < 5; i++) { - await boGroup.send({ text: `Message ${i}` }) - } +// assert(boGroup1?.isGroupActive === true, `Group 1 should be active for bo`) - if (groupMessages.length !== 5) { - throw Error('Unexpected convo messages count ' + groupMessages.length) - } +// return true +// }) - return true -}) +// test('can list all groups and conversations', async () => { +// const [alixClient, boClient, caroClient] = await createClients(3) + +// // Add one group and one conversation +// const boGroup = await boClient.conversations.newGroup([alixClient.address]) +// const alixConversation = await alixClient.conversations.newConversation( +// caroClient.address +// ) + +// const listedContainers = await alixClient.conversations.listAll() + +// // Verify information in listed containers is correct +// // BUG - List All returns in Chronological order on iOS +// // and reverse Chronological order on Android +// const first = 0 +// const second = 1 +// if ( +// listedContainers[first].topic !== boGroup.topic || +// listedContainers[first].version !== ConversationVersion.GROUP || +// listedContainers[second].version !== ConversationVersion.DIRECT || +// listedContainers[second].createdAt !== alixConversation.createdAt +// ) { +// throw Error('Listed containers should match streamed containers') +// } -test('can stream all messages', async () => { - const [alix, bo, caro] = await createClients(3) +// return true +// }) - await delayToPropogate() +// test('can stream all groups and conversations', async () => { +// const [alixClient, boClient, caroClient] = await createClients(3) - // Record message stream across all conversations - const allMessages: DecodedMessage[] = [] - await alix.conversations.streamAllMessages(async (message) => { - allMessages.push(message) - }) +// // Start streaming groups and conversations +// const containers: ConversationContainer[] = [] +// const cancelStreamAll = await alixClient.conversations.streamAll( +// async (conversationContainer: ConversationContainer) => { +// containers.push(conversationContainer) +// } +// ) - // Start bo starts a new conversation. - const boConvo = await bo.conversations.newConversation(alix.address) - await delayToPropogate() +// // bo creates a group with alix, so stream callback is fired +// // eslint-disable-next-line @typescript-eslint/no-unused-vars +// const boGroup = await boClient.conversations.newGroup([alixClient.address]) +// await delayToPropogate() +// if ((containers.length as number) !== 1) { +// throw Error('Unexpected num groups (should be 1): ' + containers.length) +// } - for (let i = 0; i < 5; i++) { - await boConvo.send({ text: `Message ${i}` }) - await delayToPropogate() - } +// // bo creates a v2 Conversation with alix so a stream callback is fired +// // eslint-disable-next-line @typescript-eslint/no-unused-vars +// const boConversation = await boClient.conversations.newConversation( +// alixClient.address +// ) +// await delayToPropogate() +// if ((containers.length as number) !== 2) { +// throw Error('Unexpected num groups (should be 2): ' + containers.length) +// } - const count = allMessages.length - if (count !== 5) { - throw Error('Unexpected all messages count ' + allMessages.length) - } +// if ( +// containers[1].version === ConversationVersion.DIRECT && +// boConversation.conversationID !== +// (containers[1] as Conversation).conversationID +// ) { +// throw Error( +// 'Conversation from streamed all should match conversationID with created conversation' +// ) +// } - const caroConvo = await caro.conversations.newConversation(alix.address) - const caroGroup = await caro.conversations.newGroup([alix.address]) - await delayToPropogate() - for (let i = 0; i < 5; i++) { - await caroConvo.send({ text: `Message ${i}` }) - await caroGroup.send({ text: `Message ${i}` }) - await delayToPropogate() - } +// // * Note alix creating a v2 Conversation does trigger alix conversations +// // stream. - if (allMessages.length !== 10) { - throw Error('Unexpected all messages count ' + allMessages.length) - } +// // alix creates a V2 Conversationgroup +// // eslint-disable-next-line @typescript-eslint/no-unused-vars +// const alixConversation = await alixClient.conversations.newConversation( +// caroClient.address +// ) +// await delayToPropogate() +// if (containers.length !== 3) { +// throw Error('Expected group length 3 but it is: ' + containers.length) +// } - alix.conversations.cancelStreamAllMessages() +// cancelStreamAll() +// await delayToPropogate() - await alix.conversations.streamAllMessages(async (message) => { - allMessages.push(message) - }, true) +// // Creating a group should no longer trigger stream groups +// // eslint-disable-next-line @typescript-eslint/no-unused-vars +// const caroConversation = await caroClient.conversations.newGroup([ +// alixClient.address, +// ]) +// await delayToPropogate() +// if ((containers.length as number) !== 3) { +// throw Error('Unexpected num groups (should be 3): ' + containers.length) +// } - for (let i = 0; i < 5; i++) { - await boConvo.send({ text: `Message ${i}` }) - await caroGroup.send({ text: `Message ${i}` }) - await delayToPropogate() - } - if (allMessages.length <= 15) { - throw Error('Unexpected all messages count ' + allMessages.length) - } +// return true +// }) - return true -}) +// test('canMessage', async () => { +// const [bo, alix, caro] = await createClients(3) -test('can make a group with metadata', async () => { - const [alix, bo] = await createClients(2) - bo.register(new GroupUpdatedCodec()) +// const canMessage = await bo.canMessage(alix.address) +// if (!canMessage) { +// throw new Error('should be able to message v2 client') +// } - const alixGroup = await alix.conversations.newGroup([bo.address], { - name: 'Start Name', - imageUrlSquare: 'starturl.com', - description: 'a fun description', - }) +// const canMessageV3 = await caro.canGroupMessage([ +// caro.address, +// alix.address, +// '0x0000000000000000000000000000000000000000', +// ]) - const groupName1 = await alixGroup.groupName() - const groupImageUrl1 = await alixGroup.groupImageUrlSquare() - const groupDescription1 = await alixGroup.groupDescription() - assert( - groupName1 === 'Start Name', - `the group should start with a name of Start Name not ${groupName1}` - ) +// assert( +// canMessageV3['0x0000000000000000000000000000000000000000'] === false, +// `should not be able to message 0x0000000000000000000000000000000000000000` +// ) - assert( - groupImageUrl1 === 'starturl.com', - `the group should start with a name of starturl.com not ${groupImageUrl1}` - ) +// assert( +// canMessageV3[caro.address.toLowerCase()] === true, +// `should be able to message ${caro.address}` +// ) - assert( - groupDescription1 === 'a fun description', - `the group should start with a name of a fun description not ${groupDescription1}` - ) +// assert( +// canMessageV3[alix.address.toLowerCase()] === true, +// `should be able to message ${alix.address}` +// ) - await alixGroup.updateGroupName('New Name') - await alixGroup.updateGroupImageUrlSquare('newurl.com') - await alixGroup.updateGroupDescription('a new group description') - await alixGroup.sync() - await bo.conversations.syncGroups() - const boGroups = await bo.conversations.listGroups() - const boGroup = boGroups[0] - await boGroup.sync() - - const groupName2 = await alixGroup.groupName() - const groupImageUrl2 = await alixGroup.groupImageUrlSquare() - const groupDescription2 = await alixGroup.groupDescription() - assert( - groupName2 === 'New Name', - `the group should start with a name of New Name not ${groupName2}` - ) +// return true +// }) - assert( - groupImageUrl2 === 'newurl.com', - `the group should start with a name of newurl.com not ${groupImageUrl2}` - ) +// test('can stream group messages', async () => { +// // Create three MLS enabled Clients +// const [alixClient, boClient, caroClient] = await createClients(3) + +// // alix creates a group +// const alixGroup = await alixClient.conversations.newGroup([ +// boClient.address, +// caroClient.address, +// ]) + +// // Record message stream for this group +// const groupMessages: DecodedMessage[] = [] +// const cancelGroupMessageStream = await alixGroup.streamGroupMessages( +// async (message) => { +// groupMessages.push(message) +// } +// ) + +// // bo's num groups == 1 +// await boClient.conversations.syncGroups() +// const boGroup = (await boClient.conversations.listGroups())[0] + +// for (let i = 0; i < 5; i++) { +// await boGroup.send({ text: `Message ${i}` }) +// await delayToPropogate() +// } - assert( - groupDescription2 === 'a new group description', - `the group should start with a name of a new group description not ${groupDescription2}` - ) +// if (groupMessages.length !== 5) { +// throw Error('Unexpected convo messages count ' + groupMessages.length) +// } +// for (let i = 0; i < 5; i++) { +// if (groupMessages[i].content() !== `Message ${i}`) { +// throw Error( +// 'Unexpected group message content ' + groupMessages[i].content() +// ) +// } +// } - const groupName3 = await boGroup.groupName() - const groupImageUrl3 = await boGroup.groupImageUrlSquare() - assert( - groupName3 === 'New Name', - `the group should start with a name of New Name not ${groupName3}` - ) +// cancelGroupMessageStream() +// for (let i = 0; i < 5; i++) { +// await boGroup.send({ text: `Message ${i}` }) +// } - assert( - groupImageUrl3 === 'newurl.com', - `the group should start with a name of newurl.com not ${groupImageUrl3}` - ) +// if (groupMessages.length !== 5) { +// throw Error('Unexpected convo messages count ' + groupMessages.length) +// } - const boMessages = await boGroup.messages() - assert( - boMessages[0].contentTypeId === 'xmtp.org/group_updated:1.0', - 'Unexpected message content ' + JSON.stringify(boMessages[0].contentTypeId) - ) +// return true +// }) - const message = boMessages[1].content() as GroupUpdatedContent - assert( - message.metadataFieldsChanged[0].fieldName === 'group_image_url_square', - `the metadata field changed should be group_image_url_square but was ${message.metadataFieldsChanged[0].fieldName}` - ) - const message2 = boMessages[0].content() as GroupUpdatedContent - assert( - message2.metadataFieldsChanged[0].fieldName === 'description', - `the metadata field changed should be description but was ${message2.metadataFieldsChanged[0].fieldName}` - ) - return true -}) +// test('can stream all messages', async () => { +// const [alix, bo, caro] = await createClients(3) -test('can make a group with admin permissions', async () => { - const [adminClient, anotherClient] = await createClients(2) +// await delayToPropogate() - const group = await adminClient.conversations.newGroup( - [anotherClient.address], - { permissionLevel: 'admin_only' } - ) +// // Record message stream across all conversations +// const allMessages: DecodedMessage[] = [] +// await alix.conversations.streamAllMessages(async (message) => { +// allMessages.push(message) +// }) - if ((await group.permissionPolicySet()).addMemberPolicy !== 'admin') { - throw Error( - `Group permission level should be admin but was ${ - (await group.permissionPolicySet()).addMemberPolicy - }` - ) - } +// // Start bo starts a new conversation. +// const boConvo = await bo.conversations.newConversation(alix.address) +// await delayToPropogate() - const isSuperAdmin = await group.isSuperAdmin(adminClient.inboxId) - if (!isSuperAdmin) { - throw Error(`adminClient should be the super admin`) - } +// for (let i = 0; i < 5; i++) { +// await boConvo.send({ text: `Message ${i}` }) +// await delayToPropogate() +// } - // Creator id not working, see https://github.com/xmtp/libxmtp/issues/788 - // if (group.creatorInboxId !== adminClient.inboxId) { - // throw Error( - // `adminClient should be the creator but was ${group.creatorInboxId}` - // ) - // } +// const count = allMessages.length +// if (count !== 5) { +// throw Error('Unexpected all messages count ' + allMessages.length) +// } - return true -}) +// const caroConvo = await caro.conversations.newConversation(alix.address) +// const caroGroup = await caro.conversations.newGroup([alix.address]) +// await delayToPropogate() +// for (let i = 0; i < 5; i++) { +// await caroConvo.send({ text: `Message ${i}` }) +// await caroGroup.send({ text: `Message ${i}` }) +// await delayToPropogate() +// } -test('can paginate group messages', async () => { - // Create three MLS enabled Clients - const [alixClient, boClient] = await createClients(2) +// if (allMessages.length !== 10) { +// throw Error('Unexpected all messages count ' + allMessages.length) +// } - // alix creates a group - const alixGroup = await alixClient.conversations.newGroup([boClient.address]) +// alix.conversations.cancelStreamAllMessages() - // alix can send messages - await alixGroup.send('hello, world') - await alixGroup.send('gm') +// await alix.conversations.streamAllMessages(async (message) => { +// allMessages.push(message) +// }, true) - await boClient.conversations.syncGroups() - const boGroups = await boClient.conversations.listGroups() - if (boGroups.length !== 1) { - throw new Error( - 'num groups for bo should be 1, but it is' + boGroups.length - ) - } - await delayToPropogate() - // bo can read messages from alix - await boGroups[0].sync() - const boMessages: DecodedMessage[] = await boGroups[0].messages({ - limit: 1, - }) +// for (let i = 0; i < 5; i++) { +// await boConvo.send({ text: `Message ${i}` }) +// await caroGroup.send({ text: `Message ${i}` }) +// await delayToPropogate() +// } +// if (allMessages.length <= 15) { +// throw Error('Unexpected all messages count ' + allMessages.length) +// } - if (boMessages.length !== 1) { - throw Error(`Should limit just 1 message but was ${boMessages.length}`) - } +// return true +// }) - return true -}) +// test('can make a group with metadata', async () => { +// const [alix, bo] = await createClients(2) +// bo.register(new GroupUpdatedCodec()) + +// const alixGroup = await alix.conversations.newGroup([bo.address], { +// name: 'Start Name', +// imageUrlSquare: 'starturl.com', +// description: 'a fun description', +// }) + +// const groupName1 = await alixGroup.groupName() +// const groupImageUrl1 = await alixGroup.groupImageUrlSquare() +// const groupDescription1 = await alixGroup.groupDescription() +// assert( +// groupName1 === 'Start Name', +// `the group should start with a name of Start Name not ${groupName1}` +// ) + +// assert( +// groupImageUrl1 === 'starturl.com', +// `the group should start with a name of starturl.com not ${groupImageUrl1}` +// ) + +// assert( +// groupDescription1 === 'a fun description', +// `the group should start with a name of a fun description not ${groupDescription1}` +// ) + +// await alixGroup.updateGroupName('New Name') +// await alixGroup.updateGroupImageUrlSquare('newurl.com') +// await alixGroup.updateGroupDescription('a new group description') +// await alixGroup.sync() +// await bo.conversations.syncGroups() +// const boGroups = await bo.conversations.listGroups() +// const boGroup = boGroups[0] +// await boGroup.sync() + +// const groupName2 = await alixGroup.groupName() +// const groupImageUrl2 = await alixGroup.groupImageUrlSquare() +// const groupDescription2 = await alixGroup.groupDescription() +// assert( +// groupName2 === 'New Name', +// `the group should start with a name of New Name not ${groupName2}` +// ) + +// assert( +// groupImageUrl2 === 'newurl.com', +// `the group should start with a name of newurl.com not ${groupImageUrl2}` +// ) + +// assert( +// groupDescription2 === 'a new group description', +// `the group should start with a name of a new group description not ${groupDescription2}` +// ) + +// const groupName3 = await boGroup.groupName() +// const groupImageUrl3 = await boGroup.groupImageUrlSquare() +// assert( +// groupName3 === 'New Name', +// `the group should start with a name of New Name not ${groupName3}` +// ) + +// assert( +// groupImageUrl3 === 'newurl.com', +// `the group should start with a name of newurl.com not ${groupImageUrl3}` +// ) + +// const boMessages = await boGroup.messages() +// assert( +// boMessages[0].contentTypeId === 'xmtp.org/group_updated:1.0', +// 'Unexpected message content ' + JSON.stringify(boMessages[0].contentTypeId) +// ) + +// const message = boMessages[1].content() as GroupUpdatedContent +// assert( +// message.metadataFieldsChanged[0].fieldName === 'group_image_url_square', +// `the metadata field changed should be group_image_url_square but was ${message.metadataFieldsChanged[0].fieldName}` +// ) +// const message2 = boMessages[0].content() as GroupUpdatedContent +// assert( +// message2.metadataFieldsChanged[0].fieldName === 'description', +// `the metadata field changed should be description but was ${message2.metadataFieldsChanged[0].fieldName}` +// ) +// return true +// }) -test('can stream all group messages', async () => { - const [alix, bo, caro] = await createClients(3) +// test('can make a group with admin permissions', async () => { +// const [adminClient, anotherClient] = await createClients(2) - await delayToPropogate() +// const group = await adminClient.conversations.newGroup( +// [anotherClient.address], +// { permissionLevel: 'admin_only' } +// ) - // Start bo starts a new group. - const boGroup = await bo.conversations.newGroup([alix.address]) - await delayToPropogate() +// if ((await group.permissionPolicySet()).addMemberPolicy !== 'admin') { +// throw Error( +// `Group permission level should be admin but was ${ +// (await group.permissionPolicySet()).addMemberPolicy +// }` +// ) +// } - // Starts a new conversation. - const caroGroup = await caro.conversations.newGroup([alix.address]) +// const isSuperAdmin = await group.isSuperAdmin(adminClient.inboxId) +// if (!isSuperAdmin) { +// throw Error(`adminClient should be the super admin`) +// } - // Record message stream across all conversations - const allMessages: DecodedMessage[] = [] - // If we don't call syncGroups here, the streamAllGroupMessages will not - // stream the first message. Feels like a bug. - await alix.conversations.syncGroups() - await alix.conversations.streamAllGroupMessages(async (message) => { - allMessages.push(message) - }) +// // Creator id not working, see https://github.com/xmtp/libxmtp/issues/788 +// // if (group.creatorInboxId !== adminClient.inboxId) { +// // throw Error( +// // `adminClient should be the creator but was ${group.creatorInboxId}` +// // ) +// // } - for (let i = 0; i < 5; i++) { - await boGroup.send({ text: `Message ${i}` }) - await delayToPropogate() - } +// return true +// }) - const count = allMessages.length - if (count !== 5) { - throw Error('Unexpected all messages count first' + allMessages.length) - } +// test('can paginate group messages', async () => { +// // Create three MLS enabled Clients +// const [alixClient, boClient] = await createClients(2) - await delayToPropogate() - for (let i = 0; i < 5; i++) { - await caroGroup.send({ text: `Message ${i}` }) - await delayToPropogate() - } +// // alix creates a group +// const alixGroup = await alixClient.conversations.newGroup([boClient.address]) - if (allMessages.length !== 10) { - throw Error('Unexpected all messages count second' + allMessages.length) - } +// // alix can send messages +// await alixGroup.send('hello, world') +// await alixGroup.send('gm') - alix.conversations.cancelStreamAllGroupMessages() - await delayToPropogate() - await alix.conversations.streamAllGroupMessages(async (message) => { - allMessages.push(message) - }) +// await boClient.conversations.syncGroups() +// const boGroups = await boClient.conversations.listGroups() +// if (boGroups.length !== 1) { +// throw new Error( +// 'num groups for bo should be 1, but it is' + boGroups.length +// ) +// } +// await delayToPropogate() +// // bo can read messages from alix +// await boGroups[0].sync() +// const boMessages: DecodedMessage[] = await boGroups[0].messages({ +// limit: 1, +// }) + +// if (boMessages.length !== 1) { +// throw Error(`Should limit just 1 message but was ${boMessages.length}`) +// } - for (let i = 0; i < 5; i++) { - await boGroup.send({ text: `Message ${i}` }) - await delayToPropogate() - } - if (allMessages.length <= 10) { - throw Error('Unexpected all messages count ' + allMessages.length) - } +// return true +// }) - return true -}) +// test('can stream all group messages', async () => { +// const [alix, bo, caro] = await createClients(3) -test('can streamAll from multiple clients', async () => { - const [alix, bo, caro] = await createClients(3) +// await delayToPropogate() - // Setup stream alls - const allBoConversations: any[] = [] - const allAliConversations: any[] = [] +// // Start bo starts a new group. +// const boGroup = await bo.conversations.newGroup([alix.address]) +// await delayToPropogate() - await bo.conversations.streamAll(async (conversation) => { - allBoConversations.push(conversation) - }) - await alix.conversations.streamAll(async (conversation) => { - allAliConversations.push(conversation) - }) +// // Starts a new conversation. +// const caroGroup = await caro.conversations.newGroup([alix.address]) + +// // Record message stream across all conversations +// const allMessages: DecodedMessage[] = [] +// // If we don't call syncGroups here, the streamAllGroupMessages will not +// // stream the first message. Feels like a bug. +// await alix.conversations.syncGroups() +// await alix.conversations.streamAllGroupMessages(async (message) => { +// allMessages.push(message) +// }) + +// for (let i = 0; i < 5; i++) { +// await boGroup.send({ text: `Message ${i}` }) +// await delayToPropogate() +// } - // Start Caro starts a new conversation. - await caro.conversations.newConversation(alix.address) - await delayToPropogate() - if (allBoConversations.length !== 0) { - throw Error( - 'Unexpected all conversations count for Bo ' + - allBoConversations.length + - ' and Alix had ' + - allAliConversations.length - ) - } - if (allAliConversations.length !== 1) { - throw Error( - 'Unexpected all conversations count ' + allAliConversations.length - ) - } - return true -}) +// const count = allMessages.length +// if (count !== 5) { +// throw Error('Unexpected all messages count first' + allMessages.length) +// } -test('can streamAll from multiple clients - swapped orderring', async () => { - const [alix, bo, caro] = await createClients(3) +// await delayToPropogate() +// for (let i = 0; i < 5; i++) { +// await caroGroup.send({ text: `Message ${i}` }) +// await delayToPropogate() +// } - // Setup stream alls - const allBoConversations: any[] = [] - const allAliConversations: any[] = [] +// if (allMessages.length !== 10) { +// throw Error('Unexpected all messages count second' + allMessages.length) +// } - await alix.conversations.streamAll(async (conversation) => { - allAliConversations.push(conversation) - }) +// alix.conversations.cancelStreamAllGroupMessages() +// await delayToPropogate() +// await alix.conversations.streamAllGroupMessages(async (message) => { +// allMessages.push(message) +// }) - await bo.conversations.streamAll(async (conversation) => { - allBoConversations.push(conversation) - }) +// for (let i = 0; i < 5; i++) { +// await boGroup.send({ text: `Message ${i}` }) +// await delayToPropogate() +// } +// if (allMessages.length <= 10) { +// throw Error('Unexpected all messages count ' + allMessages.length) +// } - // Start Caro starts a new conversation. - await caro.conversations.newConversation(alix.address) - await delayToPropogate() - if (allBoConversations.length !== 0) { - throw Error( - 'Unexpected all conversations count for Bo ' + - allBoConversations.length + - ' and Alix had ' + - allAliConversations.length - ) - } - if (allAliConversations.length !== 1) { - throw Error( - 'Unexpected all conversations count ' + allAliConversations.length - ) - } - return true -}) +// return true +// }) -test('can streamAllMessages from multiple clients', async () => { - const [alix, bo, caro] = await createClients(3) +// test('can streamAll from multiple clients', async () => { +// const [alix, bo, caro] = await createClients(3) - // Setup stream - const allBoMessages: any[] = [] - const allAliMessages: any[] = [] +// // Setup stream alls +// const allBoConversations: any[] = [] +// const allAliConversations: any[] = [] - await bo.conversations.streamAllMessages(async (conversation) => { - allBoMessages.push(conversation) - }, true) - await alix.conversations.streamAllMessages(async (conversation) => { - allAliMessages.push(conversation) - }, true) +// await bo.conversations.streamAll(async (conversation) => { +// allBoConversations.push(conversation) +// }) +// await alix.conversations.streamAll(async (conversation) => { +// allAliConversations.push(conversation) +// }) - // Start Caro starts a new conversation. - const caroConversation = await caro.conversations.newConversation( - alix.address - ) - await caroConversation.send({ text: `Message` }) - await delayToPropogate() - if (allBoMessages.length !== 0) { - throw Error('Unexpected all messages count for Bo ' + allBoMessages.length) - } +// // Start Caro starts a new conversation. +// await caro.conversations.newConversation(alix.address) +// await delayToPropogate() +// if (allBoConversations.length !== 0) { +// throw Error( +// 'Unexpected all conversations count for Bo ' + +// allBoConversations.length + +// ' and Alix had ' + +// allAliConversations.length +// ) +// } +// if (allAliConversations.length !== 1) { +// throw Error( +// 'Unexpected all conversations count ' + allAliConversations.length +// ) +// } +// return true +// }) - if (allAliMessages.length !== 1) { - throw Error( - 'Unexpected all conversations count for Ali ' + allAliMessages.length - ) - } +// test('can streamAll from multiple clients - swapped orderring', async () => { +// const [alix, bo, caro] = await createClients(3) - return true -}) +// // Setup stream alls +// const allBoConversations: any[] = [] +// const allAliConversations: any[] = [] -test('can streamAllMessages from multiple clients - swapped', async () => { - const [alix, bo, caro] = await createClients(3) - - // Setup stream - const allBoMessages: any[] = [] - const allAliMessages: any[] = [] - const caroGroup = await caro.conversations.newGroup([alix.address]) - - await alix.conversations.streamAllMessages(async (conversation) => { - allAliMessages.push(conversation) - }, true) - await bo.conversations.streamAllMessages(async (conversation) => { - allBoMessages.push(conversation) - }, true) - - // Start Caro starts a new conversation. - const caroConvo = await caro.conversations.newConversation(alix.address) - await delayToPropogate() - await caroConvo.send({ text: `Message` }) - await caroGroup.send({ text: `Message` }) - await delayToPropogate() - if (allBoMessages.length !== 0) { - throw Error( - 'Unexpected all conversations count for Bo ' + allBoMessages.length - ) - } +// await alix.conversations.streamAll(async (conversation) => { +// allAliConversations.push(conversation) +// }) - if (allAliMessages.length !== 2) { - throw Error( - 'Unexpected all conversations count for Ali ' + allAliMessages.length - ) - } +// await bo.conversations.streamAll(async (conversation) => { +// allBoConversations.push(conversation) +// }) - return true -}) +// // Start Caro starts a new conversation. +// await caro.conversations.newConversation(alix.address) +// await delayToPropogate() +// if (allBoConversations.length !== 0) { +// throw Error( +// 'Unexpected all conversations count for Bo ' + +// allBoConversations.length + +// ' and Alix had ' + +// allAliConversations.length +// ) +// } +// if (allAliConversations.length !== 1) { +// throw Error( +// 'Unexpected all conversations count ' + allAliConversations.length +// ) +// } +// return true +// }) -test('can stream all group Messages from multiple clients', async () => { - const [alix, bo, caro] = await createClients(3) +// test('can streamAllMessages from multiple clients', async () => { +// const [alix, bo, caro] = await createClients(3) - // Setup stream - const allAlixMessages: DecodedMessage[] = [] - const allBoMessages: DecodedMessage[] = [] - const alixGroup = await caro.conversations.newGroup([alix.address]) - const boGroup = await caro.conversations.newGroup([bo.address]) +// // Setup stream +// const allBoMessages: any[] = [] +// const allAliMessages: any[] = [] - await alixGroup.streamGroupMessages(async (message) => { - allAlixMessages.push(message) - }) - await boGroup.streamGroupMessages(async (message) => { - allBoMessages.push(message) - }) +// await bo.conversations.streamAllMessages(async (conversation) => { +// allBoMessages.push(conversation) +// }, true) +// await alix.conversations.streamAllMessages(async (conversation) => { +// allAliMessages.push(conversation) +// }, true) - // Start Caro starts a new conversation. - await delayToPropogate() - await alixGroup.send({ text: `Message` }) - await delayToPropogate() - if (allBoMessages.length !== 0) { - throw Error('Unexpected all messages count for Bo ' + allBoMessages.length) - } +// // Start Caro starts a new conversation. +// const caroConversation = await caro.conversations.newConversation( +// alix.address +// ) +// await caroConversation.send({ text: `Message` }) +// await delayToPropogate() +// if (allBoMessages.length !== 0) { +// throw Error('Unexpected all messages count for Bo ' + allBoMessages.length) +// } - if (allAlixMessages.length !== 1) { - throw Error( - 'Unexpected all messages count for Ali ' + allAlixMessages.length - ) - } +// if (allAliMessages.length !== 1) { +// throw Error( +// 'Unexpected all conversations count for Ali ' + allAliMessages.length +// ) +// } - await alix.conversations.syncGroups() - const alixConv = (await alix.conversations.listGroups())[0] - await alixConv.send({ text: `Message` }) - await delayToPropogate() - if (allBoMessages.length !== 0) { - throw Error('Unexpected all messages count for Bo ' + allBoMessages.length) - } - // @ts-ignore-next-line - if (allAlixMessages.length !== 2) { - throw Error( - 'Unexpected all messages count for Ali ' + allAlixMessages.length - ) - } +// return true +// }) - return true -}) +// test('can streamAllMessages from multiple clients - swapped', async () => { +// const [alix, bo, caro] = await createClients(3) -test('can stream all group Messages from multiple clients - swapped', async () => { - const [alix, bo, caro] = await createClients(3) +// // Setup stream +// const allBoMessages: any[] = [] +// const allAliMessages: any[] = [] +// const caroGroup = await caro.conversations.newGroup([alix.address]) - // Setup stream - const allAlixMessages: DecodedMessage[] = [] - const allBoMessages: DecodedMessage[] = [] - const alixGroup = await caro.conversations.newGroup([alix.address]) - const boGroup = await caro.conversations.newGroup([bo.address]) +// await alix.conversations.streamAllMessages(async (conversation) => { +// allAliMessages.push(conversation) +// }, true) +// await bo.conversations.streamAllMessages(async (conversation) => { +// allBoMessages.push(conversation) +// }, true) - await boGroup.streamGroupMessages(async (message) => { - allBoMessages.push(message) - }) - await alixGroup.streamGroupMessages(async (message) => { - allAlixMessages.push(message) - }) +// // Start Caro starts a new conversation. +// const caroConvo = await caro.conversations.newConversation(alix.address) +// await delayToPropogate() +// await caroConvo.send({ text: `Message` }) +// await caroGroup.send({ text: `Message` }) +// await delayToPropogate() +// if (allBoMessages.length !== 0) { +// throw Error( +// 'Unexpected all conversations count for Bo ' + allBoMessages.length +// ) +// } - // Start Caro starts a new conversation. - await delayToPropogate() - await alixGroup.send({ text: `Message` }) - await delayToPropogate() - if (allBoMessages.length !== 0) { - throw Error('Unexpected all messages count for Bo ' + allBoMessages.length) - } +// if (allAliMessages.length !== 2) { +// throw Error( +// 'Unexpected all conversations count for Ali ' + allAliMessages.length +// ) +// } - if (allAlixMessages.length !== 1) { - throw Error( - 'Unexpected all messages count for Ali ' + allAlixMessages.length - ) - } +// return true +// }) - await alix.conversations.syncGroups() - const alixConv = (await alix.conversations.listGroups())[0] - await alixConv.send({ text: `Message` }) - await delayToPropogate() - if (allBoMessages.length !== 0) { - throw Error('Unexpected all messages count for Bo ' + allBoMessages.length) - } - // @ts-ignore-next-line - if (allAlixMessages.length !== 2) { - throw Error( - 'Unexpected all messages count for Ali ' + allAlixMessages.length - ) - } +// test('can stream all group Messages from multiple clients', async () => { +// const [alix, bo, caro] = await createClients(3) - return true -}) +// // Setup stream +// const allAlixMessages: DecodedMessage[] = [] +// const allBoMessages: DecodedMessage[] = [] +// const alixGroup = await caro.conversations.newGroup([alix.address]) +// const boGroup = await caro.conversations.newGroup([bo.address]) -test('creating a group should allow group', async () => { - const [alix, bo] = await createClients(2) +// await alixGroup.streamGroupMessages(async (message) => { +// allAlixMessages.push(message) +// }) +// await boGroup.streamGroupMessages(async (message) => { +// allBoMessages.push(message) +// }) - const group = await alix.conversations.newGroup([bo.address]) - const consent = await alix.contacts.isGroupAllowed(group.id) - const groupConsent = await group.isAllowed() +// // Start Caro starts a new conversation. +// await delayToPropogate() +// await alixGroup.send({ text: `Message` }) +// await delayToPropogate() +// if (allBoMessages.length !== 0) { +// throw Error('Unexpected all messages count for Bo ' + allBoMessages.length) +// } - if (!consent || !groupConsent) { - throw Error('Group should be allowed') - } +// if (allAlixMessages.length !== 1) { +// throw Error( +// 'Unexpected all messages count for Ali ' + allAlixMessages.length +// ) +// } - const state = await group.consentState() - assert( - state === 'allowed', - `the message should have a consent state of allowed but was ${state}` - ) +// await alix.conversations.syncGroups() +// const alixConv = (await alix.conversations.listGroups())[0] +// await alixConv.send({ text: `Message` }) +// await delayToPropogate() +// if (allBoMessages.length !== 0) { +// throw Error('Unexpected all messages count for Bo ' + allBoMessages.length) +// } +// // @ts-ignore-next-line +// if (allAlixMessages.length !== 2) { +// throw Error( +// 'Unexpected all messages count for Ali ' + allAlixMessages.length +// ) +// } - const consentList = await alix.contacts.consentList() - assert( - consentList[0].permissionType === 'allowed', - `the message should have a consent state of allowed but was ${consentList[0].permissionType}` - ) +// return true +// }) - return true -}) +// test('can stream all group Messages from multiple clients - swapped', async () => { +// const [alix, bo, caro] = await createClients(3) -test('can allow a group', async () => { - const [alix, bo] = await createClients(2) - const alixGroup = await alix.conversations.newGroup([bo.address]) - const startConsent = await bo.contacts.isGroupAllowed(alixGroup.id) - if (startConsent) { - throw Error('Group should not be allowed') - } - await bo.contacts.allowGroups([alixGroup.id]) - const isAllowed = await bo.contacts.isGroupAllowed(alixGroup.id) - if (!isAllowed) { - throw Error('Group should be allowed') - } +// // Setup stream +// const allAlixMessages: DecodedMessage[] = [] +// const allBoMessages: DecodedMessage[] = [] +// const alixGroup = await caro.conversations.newGroup([alix.address]) +// const boGroup = await caro.conversations.newGroup([bo.address]) - return true -}) +// await boGroup.streamGroupMessages(async (message) => { +// allBoMessages.push(message) +// }) +// await alixGroup.streamGroupMessages(async (message) => { +// allAlixMessages.push(message) +// }) -test('can deny a group', async () => { - const [alix, bo] = await createClients(2) - const alixGroup = await alix.conversations.newGroup([bo.address]) - const startConsent = await bo.contacts.isGroupDenied(alixGroup.id) - if (startConsent) { - throw Error('Group should be unknown') - } - await bo.contacts.denyGroups([alixGroup.id]) - await bo.conversations.syncGroups() - const boGroups = await bo.conversations.listGroups() - const isDenied = await bo.contacts.isGroupDenied(alixGroup.id) - const isGroupDenied = await boGroups[0].isDenied() - if (!isDenied || !isGroupDenied) { - throw Error('Group should be denied') - } - await bo.contacts.allowGroups([alixGroup.id]) - const isAllowed = await bo.contacts.isGroupAllowed(alixGroup.id) - if (!isAllowed) { - throw Error('Group should be allowed') - } +// // Start Caro starts a new conversation. +// await delayToPropogate() +// await alixGroup.send({ text: `Message` }) +// await delayToPropogate() +// if (allBoMessages.length !== 0) { +// throw Error('Unexpected all messages count for Bo ' + allBoMessages.length) +// } - return true -}) +// if (allAlixMessages.length !== 1) { +// throw Error( +// 'Unexpected all messages count for Ali ' + allAlixMessages.length +// ) +// } -test('can allow and deny a inbox id', async () => { - const [alix, bo] = await createClients(2) - const startConsent = await bo.contacts.isInboxAllowed(alix.inboxId) - if (startConsent) { - throw Error('inbox id should be unknown') - } - await bo.contacts.denyInboxes([alix.inboxId]) - const isDenied = await bo.contacts.isInboxDenied(alix.inboxId) - if (!isDenied) { - throw Error('inbox id should be denied') - } - await bo.contacts.allowInboxes([alix.inboxId]) - const isAllowed = await bo.contacts.isInboxAllowed(alix.inboxId) - if (!isAllowed) { - throw Error('inbox id should be allowed') - } +// await alix.conversations.syncGroups() +// const alixConv = (await alix.conversations.listGroups())[0] +// await alixConv.send({ text: `Message` }) +// await delayToPropogate() +// if (allBoMessages.length !== 0) { +// throw Error('Unexpected all messages count for Bo ' + allBoMessages.length) +// } +// // @ts-ignore-next-line +// if (allAlixMessages.length !== 2) { +// throw Error( +// 'Unexpected all messages count for Ali ' + allAlixMessages.length +// ) +// } - const consentList = await bo.contacts.consentList() - assert( - consentList[0].entryType === 'inbox_id', - `the message should have a type of inbox_id but was ${consentList[0].entryType}` - ) +// return true +// }) - return true -}) +// test('creating a group should allow group', async () => { +// const [alix, bo] = await createClients(2) -test('can check if group is allowed', async () => { - const [alix, bo] = await createClients(2) - const alixGroup = await alix.conversations.newGroup([bo.address]) - const startConsent = await bo.contacts.isGroupAllowed(alixGroup.id) - if (startConsent) { - throw Error('Group should not be allowed by default') - } - await bo.contacts.allowGroups([alixGroup.id]) - const consent = await bo.contacts.isGroupAllowed(alixGroup.id) - if (!consent) { - throw Error('Group should be allowed') - } +// const group = await alix.conversations.newGroup([bo.address]) +// const consent = await alix.contacts.isGroupAllowed(group.id) +// const groupConsent = await group.isAllowed() - return true -}) +// if (!consent || !groupConsent) { +// throw Error('Group should be allowed') +// } -test('can check if group is denied', async () => { - const [alix, bo] = await createClients(2) - const alixGroup = await alix.conversations.newGroup([bo.address]) - const startConsent = await bo.contacts.isGroupDenied(alixGroup.id) - if (startConsent) { - throw Error('Group should not be denied by default') - } - await bo.contacts.denyGroups([alixGroup.id]) - const consent = await bo.contacts.isGroupDenied(alixGroup.id) - if (!consent) { - throw Error('Group should be denied') - } - return true -}) +// const state = await group.consentState() +// assert( +// state === 'allowed', +// `the message should have a consent state of allowed but was ${state}` +// ) -test('sync function behaves as expected', async () => { - const [alix, bo, caro] = await createClients(3) - const alixGroup = await alix.conversations.newGroup([bo.address]) +// const consentList = await alix.contacts.consentList() +// assert( +// consentList[0].permissionType === 'allowed', +// `the message should have a consent state of allowed but was ${consentList[0].permissionType}` +// ) - await alixGroup.send({ text: 'hello' }) +// return true +// }) - // List groups will return empty until the first sync - let boGroups = await bo.conversations.listGroups() - assert(boGroups.length === 0, 'num groups for bo is 0 until we sync') +// test('can allow a group', async () => { +// const [alix, bo] = await createClients(2) +// const alixGroup = await alix.conversations.newGroup([bo.address]) +// const startConsent = await bo.contacts.isGroupAllowed(alixGroup.id) +// if (startConsent) { +// throw Error('Group should not be allowed') +// } +// await bo.contacts.allowGroups([alixGroup.id]) +// const isAllowed = await bo.contacts.isGroupAllowed(alixGroup.id) +// if (!isAllowed) { +// throw Error('Group should be allowed') +// } - await bo.conversations.syncGroups() +// return true +// }) - boGroups = await bo.conversations.listGroups() - assert(boGroups.length === 1, 'num groups for bo is 1') +// test('can deny a group', async () => { +// const [alix, bo] = await createClients(2) +// const alixGroup = await alix.conversations.newGroup([bo.address]) +// const startConsent = await bo.contacts.isGroupDenied(alixGroup.id) +// if (startConsent) { +// throw Error('Group should be unknown') +// } +// await bo.contacts.denyGroups([alixGroup.id]) +// await bo.conversations.syncGroups() +// const boGroups = await bo.conversations.listGroups() +// const isDenied = await bo.contacts.isGroupDenied(alixGroup.id) +// const isGroupDenied = await boGroups[0].isDenied() +// if (!isDenied || !isGroupDenied) { +// throw Error('Group should be denied') +// } +// await bo.contacts.allowGroups([alixGroup.id]) +// const isAllowed = await bo.contacts.isGroupAllowed(alixGroup.id) +// if (!isAllowed) { +// throw Error('Group should be allowed') +// } - // Num members will include the initial num of members even before sync - let numMembers = (await boGroups[0].memberInboxIds()).length - assert(numMembers === 2, 'num members should be 2') +// return true +// }) - // Num messages for a group will be 0 until we sync the group - let numMessages = (await boGroups[0].messages()).length - assert(numMessages === 0, 'num members should be 1') +// test('can allow and deny a inbox id', async () => { +// const [alix, bo] = await createClients(2) +// const startConsent = await bo.contacts.isInboxAllowed(alix.inboxId) +// if (startConsent) { +// throw Error('inbox id should be unknown') +// } +// await bo.contacts.denyInboxes([alix.inboxId]) +// const isDenied = await bo.contacts.isInboxDenied(alix.inboxId) +// if (!isDenied) { +// throw Error('inbox id should be denied') +// } +// await bo.contacts.allowInboxes([alix.inboxId]) +// const isAllowed = await bo.contacts.isInboxAllowed(alix.inboxId) +// if (!isAllowed) { +// throw Error('inbox id should be allowed') +// } - await bo.conversations.syncGroups() +// const consentList = await bo.contacts.consentList() +// assert( +// consentList[0].entryType === 'inbox_id', +// `the message should have a type of inbox_id but was ${consentList[0].entryType}` +// ) - // Num messages is still 0 because we didnt sync the group itself - numMessages = (await boGroups[0].messages()).length - assert(numMessages === 0, 'num messages should be 0') +// return true +// }) - await boGroups[0].sync() +// test('can check if group is allowed', async () => { +// const [alix, bo] = await createClients(2) +// const alixGroup = await alix.conversations.newGroup([bo.address]) +// const startConsent = await bo.contacts.isGroupAllowed(alixGroup.id) +// if (startConsent) { +// throw Error('Group should not be allowed by default') +// } +// await bo.contacts.allowGroups([alixGroup.id]) +// const consent = await bo.contacts.isGroupAllowed(alixGroup.id) +// if (!consent) { +// throw Error('Group should be allowed') +// } - // after syncing the group we now see the correct number of messages - numMessages = (await boGroups[0].messages()).length - assert(numMessages === 1, 'num members should be 1') +// return true +// }) - await alixGroup.addMembers([caro.address]) +// test('can check if group is denied', async () => { +// const [alix, bo] = await createClients(2) +// const alixGroup = await alix.conversations.newGroup([bo.address]) +// const startConsent = await bo.contacts.isGroupDenied(alixGroup.id) +// if (startConsent) { +// throw Error('Group should not be denied by default') +// } +// await bo.contacts.denyGroups([alixGroup.id]) +// const consent = await bo.contacts.isGroupDenied(alixGroup.id) +// if (!consent) { +// throw Error('Group should be denied') +// } +// return true +// }) - numMembers = (await boGroups[0].memberInboxIds()).length - assert(numMembers === 2, 'num members should be 2') +// test('sync function behaves as expected', async () => { +// const [alix, bo, caro] = await createClients(3) +// const alixGroup = await alix.conversations.newGroup([bo.address]) - await bo.conversations.syncGroups() +// await alixGroup.send({ text: 'hello' }) - // Even though we synced the groups, we need to sync the group itself to see the new member - numMembers = (await boGroups[0].memberInboxIds()).length - assert(numMembers === 2, 'num members should be 2') +// // List groups will return empty until the first sync +// let boGroups = await bo.conversations.listGroups() +// assert(boGroups.length === 0, 'num groups for bo is 0 until we sync') - await boGroups[0].sync() +// await bo.conversations.syncGroups() - numMembers = (await boGroups[0].memberInboxIds()).length - assert(numMembers === 3, 'num members should be 3') +// boGroups = await bo.conversations.listGroups() +// assert(boGroups.length === 1, 'num groups for bo is 1') - // eslint-disable-next-line @typescript-eslint/no-unused-vars - const _alixGroup2 = await alix.conversations.newGroup([ - bo.address, - caro.address, - ]) - await bo.conversations.syncGroups() - boGroups = await bo.conversations.listGroups() - assert(boGroups.length === 2, 'num groups for bo is 2') +// // Num members will include the initial num of members even before sync +// let numMembers = (await boGroups[0].memberInboxIds()).length +// assert(numMembers === 2, 'num members should be 2') - // Even before syncing the group, syncGroups will return the initial number of members - numMembers = (await boGroups[1].memberInboxIds()).length - assert(numMembers === 3, 'num members should be 3') +// // Num messages for a group will be 0 until we sync the group +// let numMessages = (await boGroups[0].messages()).length +// assert(numMessages === 0, 'num members should be 1') - return true -}) +// await bo.conversations.syncGroups() -test('can read and update group name', async () => { - const [alix, bo, caro] = await createClients(3) - const alixGroup = await alix.conversations.newGroup([bo.address]) +// // Num messages is still 0 because we didnt sync the group itself +// numMessages = (await boGroups[0].messages()).length +// assert(numMessages === 0, 'num messages should be 0') - await alixGroup.sync() - let groupName = await alixGroup.groupName() +// await boGroups[0].sync() - assert(groupName === '', 'group name should be empty string') +// // after syncing the group we now see the correct number of messages +// numMessages = (await boGroups[0].messages()).length +// assert(numMessages === 1, 'num members should be 1') - await alixGroup.updateGroupName('Test name update 1') +// await alixGroup.addMembers([caro.address]) - await alixGroup.sync() - groupName = await alixGroup.groupName() +// numMembers = (await boGroups[0].memberInboxIds()).length +// assert(numMembers === 2, 'num members should be 2') - assert( - groupName === 'Test name update 1', - 'group name should be "Test name update 1"' - ) +// await bo.conversations.syncGroups() - await bo.conversations.syncGroups() - const boGroup = (await bo.conversations.listGroups())[0] - groupName = await boGroup.groupName() +// // Even though we synced the groups, we need to sync the group itself to see the new member +// numMembers = (await boGroups[0].memberInboxIds()).length +// assert(numMembers === 2, 'num members should be 2') - assert(groupName === '', 'group name should be empty string') +// await boGroups[0].sync() - await boGroup.sync() +// numMembers = (await boGroups[0].memberInboxIds()).length +// assert(numMembers === 3, 'num members should be 3') - groupName = await boGroup.groupName() +// // eslint-disable-next-line @typescript-eslint/no-unused-vars +// const _alixGroup2 = await alix.conversations.newGroup([ +// bo.address, +// caro.address, +// ]) +// await bo.conversations.syncGroups() +// boGroups = await bo.conversations.listGroups() +// assert(boGroups.length === 2, 'num groups for bo is 2') - assert( - groupName === 'Test name update 1', - 'group name should be "Test name update 1"' - ) +// // Even before syncing the group, syncGroups will return the initial number of members +// numMembers = (await boGroups[1].memberInboxIds()).length +// assert(numMembers === 3, 'num members should be 3') - await alixGroup.addMembers([caro.address]) - await caro.conversations.syncGroups() - const caroGroup = (await caro.conversations.listGroups())[0] +// return true +// }) - await caroGroup.sync() - groupName = await caroGroup.groupName() - assert( - groupName === 'Test name update 1', - 'group name should be "Test name update 1"' - ) - return true -}) +// test('can read and update group name', async () => { +// const [alix, bo, caro] = await createClients(3) +// const alixGroup = await alix.conversations.newGroup([bo.address]) -test('can list groups does not fork', async () => { - const [alix, bo] = await createClients(2) - console.log('created clients') - let groupCallbacks = 0 - //#region Stream groups - await bo.conversations.streamGroups(async () => { - console.log('group received') - groupCallbacks++ - }) - //#region Stream All Messages - await bo.conversations.streamAllMessages(async () => { - console.log('message received') - }, true) - //#endregion - // #region create group - const alixGroup = await alix.conversations.newGroup([bo.address]) - await alixGroup.updateGroupName('hello') - await alixGroup.send('hello1') - console.log('sent group message') - // #endregion - // #region sync groups - await bo.conversations.syncGroups() - // #endregion - const boGroups = await bo.conversations.listGroups() - assert(boGroups.length === 1, 'bo should have 1 group') - const boGroup = boGroups[0] - await boGroup.sync() - - const boMessages1 = await boGroup.messages() - assert( - boMessages1.length === 2, - `should have 2 messages on first load received ${boMessages1.length}` - ) - await boGroup.send('hello2') - await boGroup.send('hello3') - await alixGroup.sync() - const alixMessages = await alixGroup.messages() - for (const message of alixMessages) { - console.log( - 'message', - message.contentTypeId, - message.contentTypeId === 'xmtp.org/text:1.0' - ? message.content() - : 'Group Updated' - ) - } - // alix sees 3 messages - assert( - alixMessages.length === 5, - `should have 5 messages on first load received ${alixMessages.length}` - ) - await alixGroup.send('hello4') - await boGroup.sync() - const boMessages2 = await boGroup.messages() - for (const message of boMessages2) { - console.log( - 'message', - message.contentTypeId, - message.contentTypeId === 'xmtp.org/text:1.0' - ? message.content() - : 'Group Updated' - ) - } - // bo sees 4 messages - assert( - boMessages2.length === 5, - `should have 5 messages on second load received ${boMessages2.length}` - ) +// await alixGroup.sync() +// let groupName = await alixGroup.groupName() - assert(groupCallbacks === 1, 'group stream should have received 1 group') +// assert(groupName === '', 'group name should be empty string') - return true -}) +// await alixGroup.updateGroupName('Test name update 1') -test('can create new installation without breaking group', async () => { - const keyBytes = new Uint8Array([ - 233, 120, 198, 96, 154, 65, 132, 17, 132, 96, 250, 40, 103, 35, 125, 64, - 166, 83, 208, 224, 254, 44, 205, 227, 175, 49, 234, 129, 74, 252, 135, 145, - ]) - const wallet1 = Wallet.createRandom() - const wallet2 = Wallet.createRandom() +// await alixGroup.sync() +// groupName = await alixGroup.groupName() - const client1 = await Client.create(wallet1, { - env: 'local', - appVersion: 'Testing/0.0.0', - enableV3: true, - dbEncryptionKey: keyBytes, - }) - const client2 = await Client.create(wallet2, { - env: 'local', - appVersion: 'Testing/0.0.0', - enableV3: true, - dbEncryptionKey: keyBytes, - }) +// assert( +// groupName === 'Test name update 1', +// 'group name should be "Test name update 1"' +// ) - const group = await client1.conversations.newGroup([wallet2.address]) +// await bo.conversations.syncGroups() +// const boGroup = (await bo.conversations.listGroups())[0] +// groupName = await boGroup.groupName() - await client1.conversations.syncGroups() - await client2.conversations.syncGroups() +// assert(groupName === '', 'group name should be empty string') - const client1Group = await client1.conversations.findGroup(group.id) - const client2Group = await client2.conversations.findGroup(group.id) +// await boGroup.sync() - await client1Group?.sync() - await client2Group?.sync() +// groupName = await boGroup.groupName() - assert(client1Group?.members?.length === 2, `client 1 should see 2 members`) +// assert( +// groupName === 'Test name update 1', +// 'group name should be "Test name update 1"' +// ) - assert( - (await client2Group?.membersList())?.length === 2, - `client 2 should see 2 members` - ) +// await alixGroup.addMembers([caro.address]) +// await caro.conversations.syncGroups() +// const caroGroup = (await caro.conversations.listGroups())[0] - await client2.deleteLocalDatabase() +// await caroGroup.sync() +// groupName = await caroGroup.groupName() +// assert( +// groupName === 'Test name update 1', +// 'group name should be "Test name update 1"' +// ) +// return true +// }) - // Recreating a client with wallet 2 (new installation!) - await Client.create(wallet2, { - env: 'local', - appVersion: 'Testing/0.0.0', - enableV3: true, - dbEncryptionKey: keyBytes, - }) +// test('can list groups does not fork', async () => { +// const [alix, bo] = await createClients(2) +// console.log('created clients') +// let groupCallbacks = 0 +// //#region Stream groups +// await bo.conversations.streamGroups(async () => { +// console.log('group received') +// groupCallbacks++ +// }) +// //#region Stream All Messages +// await bo.conversations.streamAllMessages(async () => { +// console.log('message received') +// }, true) +// //#endregion +// // #region create group +// const alixGroup = await alix.conversations.newGroup([bo.address]) +// await alixGroup.updateGroupName('hello') +// await alixGroup.send('hello1') +// console.log('sent group message') +// // #endregion +// // #region sync groups +// await bo.conversations.syncGroups() +// // #endregion +// const boGroups = await bo.conversations.listGroups() +// assert(boGroups.length === 1, 'bo should have 1 group') +// const boGroup = boGroups[0] +// await boGroup.sync() + +// const boMessages1 = await boGroup.messages() +// assert( +// boMessages1.length === 2, +// `should have 2 messages on first load received ${boMessages1.length}` +// ) +// await boGroup.send('hello2') +// await boGroup.send('hello3') +// await alixGroup.sync() +// const alixMessages = await alixGroup.messages() +// for (const message of alixMessages) { +// console.log( +// 'message', +// message.contentTypeId, +// message.contentTypeId === 'xmtp.org/text:1.0' +// ? message.content() +// : 'Group Updated' +// ) +// } +// // alix sees 3 messages +// assert( +// alixMessages.length === 5, +// `should have 5 messages on first load received ${alixMessages.length}` +// ) +// await alixGroup.send('hello4') +// await boGroup.sync() +// const boMessages2 = await boGroup.messages() +// for (const message of boMessages2) { +// console.log( +// 'message', +// message.contentTypeId, +// message.contentTypeId === 'xmtp.org/text:1.0' +// ? message.content() +// : 'Group Updated' +// ) +// } +// // bo sees 4 messages +// assert( +// boMessages2.length === 5, +// `should have 5 messages on second load received ${boMessages2.length}` +// ) - await client1Group?.send('This message will break the group') - assert( - client1Group?.members?.length === 2, - `client 1 should still see the 2 members` - ) +// assert(groupCallbacks === 1, 'group stream should have received 1 group') - return true -}) +// return true +// }) -test('can list many groups members in parallel', async () => { - const [alix, bo] = await createClients(2) - const groups: Group[] = await createGroups(alix, [bo], 20, 0) +// test('can create new installation without breaking group', async () => { +// const keyBytes = new Uint8Array([ +// 233, 120, 198, 96, 154, 65, 132, 17, 132, 96, 250, 40, 103, 35, 125, 64, +// 166, 83, 208, 224, 254, 44, 205, 227, 175, 49, 234, 129, 74, 252, 135, 145, +// ]) +// const wallet1 = Wallet.createRandom() +// const wallet2 = Wallet.createRandom() + +// const client1 = await Client.create(wallet1, { +// env: 'local', +// appVersion: 'Testing/0.0.0', +// enableV3: true, +// dbEncryptionKey: keyBytes, +// }) +// const client2 = await Client.create(wallet2, { +// env: 'local', +// appVersion: 'Testing/0.0.0', +// enableV3: true, +// dbEncryptionKey: keyBytes, +// }) + +// const group = await client1.conversations.newGroup([wallet2.address]) + +// await client1.conversations.syncGroups() +// await client2.conversations.syncGroups() + +// const client1Group = await client1.conversations.findGroup(group.id) +// const client2Group = await client2.conversations.findGroup(group.id) + +// await client1Group?.sync() +// await client2Group?.sync() + +// assert(client1Group?.members?.length === 2, `client 1 should see 2 members`) + +// assert( +// (await client2Group?.membersList())?.length === 2, +// `client 2 should see 2 members` +// ) + +// await client2.deleteLocalDatabase() + +// // Recreating a client with wallet 2 (new installation!) +// await Client.create(wallet2, { +// env: 'local', +// appVersion: 'Testing/0.0.0', +// enableV3: true, +// dbEncryptionKey: keyBytes, +// }) + +// await client1Group?.send('This message will break the group') +// assert( +// client1Group?.members?.length === 2, +// `client 1 should still see the 2 members` +// ) - try { - await Promise.all(groups.slice(0, 10).map((g) => g.membersList())) - } catch (e) { - throw new Error(`Failed listing 10 groups members with ${e}`) - } +// return true +// }) - try { - await Promise.all(groups.slice(0, 20).map((g) => g.membersList())) - } catch (e) { - throw new Error(`Failed listing 20 groups members with ${e}`) - } +// test('can list many groups members in parallel', async () => { +// const [alix, bo] = await createClients(2) +// const groups: Group[] = await createGroups(alix, [bo], 20, 0) - return true -}) +// try { +// await Promise.all(groups.slice(0, 10).map((g) => g.membersList())) +// } catch (e) { +// throw new Error(`Failed listing 10 groups members with ${e}`) +// } -test('can sync all groups', async () => { - const [alix, bo] = await createClients(2) - const groups: Group[] = await createGroups(alix, [bo], 50, 0) +// try { +// await Promise.all(groups.slice(0, 20).map((g) => g.membersList())) +// } catch (e) { +// throw new Error(`Failed listing 20 groups members with ${e}`) +// } - const alixGroup = groups[0] - await bo.conversations.syncGroups() - const boGroup = await bo.conversations.findGroup(alixGroup.id) - await alixGroup.send('hi') - assert( - (await boGroup?.messages())?.length === 0, - `messages should be empty before sync but was ${boGroup?.messages?.length}` - ) +// return true +// }) - await bo.conversations.syncAllGroups() - assert( - (await boGroup?.messages())?.length === 1, - `messages should be 4 after sync but was ${boGroup?.messages?.length}` - ) - return true -}) +// test('can sync all groups', async () => { +// const [alix, bo] = await createClients(2) +// const groups: Group[] = await createGroups(alix, [bo], 50, 0) + +// const alixGroup = groups[0] +// await bo.conversations.syncGroups() +// const boGroup = await bo.conversations.findGroup(alixGroup.id) +// await alixGroup.send('hi') +// assert( +// (await boGroup?.messages())?.length === 0, +// `messages should be empty before sync but was ${boGroup?.messages?.length}` +// ) + +// await bo.conversations.syncAllGroups() +// assert( +// (await boGroup?.messages())?.length === 1, +// `messages should be 4 after sync but was ${boGroup?.messages?.length}` +// ) +// return true +// }) // Commenting this out so it doesn't block people, but nice to have? // test('can stream messages for a long time', async () => { diff --git a/ios/Wrappers/InboxStateWrapper.swift b/ios/Wrappers/InboxStateWrapper.swift new file mode 100644 index 000000000..62f0f2290 --- /dev/null +++ b/ios/Wrappers/InboxStateWrapper.swift @@ -0,0 +1,30 @@ +// +// InboxStateWrapper.swift +// XMTPReactNative +// +// Created by Naomi Plasterer on 8/21/24. +// + +import Foundation +import XMTP + +// Wrapper around XMTP.InboxState to allow passing these objects back into react native. +struct InboxStateWrapper { + static func encodeToObj(_ inboxState: XMTP.InboxState) throws -> [String: Any] { + return [ + "inboxId": inboxState.inboxId, + "addresses": inboxState.addresses, + "installationIds": inboxState.installationIds, + "recoveryAddress": inboxState.recoveryAddress + ] + } + + static func encode(_ inboxState: XMTP.InboxState) throws -> String { + let obj = try encodeToObj(inboxState) + let data = try JSONSerialization.data(withJSONObject: obj) + guard let result = String(data: data, encoding: .utf8) else { + throw WrapperError.encodeError("could not encode inboxState") + } + return result + } +} diff --git a/ios/XMTPModule.swift b/ios/XMTPModule.swift index ab9b88735..f34cec386 100644 --- a/ios/XMTPModule.swift +++ b/ios/XMTPModule.swift @@ -156,6 +156,25 @@ public class XMTPModule: Module { } try await client.requestMessageHistorySync() } + + AsyncFunction("revokeAllOtherInstallations") { (inboxId: String) in + guard let client = await clientsManager.getClient(key: inboxId) else { + throw Error.noClient + } + let signer = ReactNativeSigner(module: self, address: client.address) + self.signer = signer + + try await client.revokeAllOtherInstallations(signingKey: signer) + self.signer = nil + } + + AsyncFunction("getInboxState") { (inboxId: String, refreshFromNetwork: Bool) -> String in + guard let client = await clientsManager.getClient(key: inboxId) else { + throw Error.noClient + } + let inboxState = try await client.inboxState(refreshFromNetwork: refreshFromNetwork) + return try InboxStateWrapper.encode(inboxState) + } // // Auth functions diff --git a/src/index.ts b/src/index.ts index 303519d4c..d6a5f253b 100644 --- a/src/index.ts +++ b/src/index.ts @@ -70,6 +70,18 @@ export async function requestMessageHistorySync(inboxId: string) { return XMTPModule.requestMessageHistorySync(inboxId) } +export async function getInboxState( + inboxId: string, + refreshFromNetwork: boolean +): Promise { + const inboxState = await XMTPModule.getInboxState(inboxId, refreshFromNetwork) + return InboxState.from(inboxState) +} + +export async function revokeAllOtherInstallations(inboxId: string) { + return XMTPModule.revokeAllOtherInstallations(inboxId) +} + export async function auth( address: string, environment: 'local' | 'dev' | 'production', diff --git a/src/lib/Client.ts b/src/lib/Client.ts index d84e0f69a..85f8dd5d1 100644 --- a/src/lib/Client.ts +++ b/src/lib/Client.ts @@ -455,6 +455,51 @@ export class Client< return await XMTPModule.requestMessageHistorySync(this.inboxId) } + /** + * Revoke all other installations but the current one. + */ + async revokeAllOtherInstallations(wallet: Signer | WalletClient | null) { + const signer = getSigner(wallet) + if (!signer) { + throw new Error('Signer is not configured') + } + XMTPModule.emitter.addListener( + 'sign', + async (message: { id: string; message: string }) => { + const request: { id: string; message: string } = message + try { + const signatureString = await signer.signMessage(request.message) + const eSig = splitSignature(signatureString) + const r = hexToBytes(eSig.r) + const s = hexToBytes(eSig.s) + const sigBytes = new Uint8Array(65) + sigBytes.set(r) + sigBytes.set(s, r.length) + sigBytes[64] = eSig.recoveryParam + + const signature = Buffer.from(sigBytes).toString('base64') + + await XMTPModule.receiveSignature(request.id, signature) + await XMTPModule.revokeAllOtherInstallations(this.inboxId) + } catch (e) { + const errorMessage = + 'ERROR in revoke installations. User rejected signature' + console.info(errorMessage, e) + } + } + ) + } + + /** + * Make a request for a inboxs state. + * + * @param {boolean} refreshFromNetwork - If you want to refresh the current state of in the inbox from the network or not. + * @returns {Promise} A Promise resolving to a InboxState. + */ + async inboxState(refreshFromNetwork: boolean): Promise { + return await XMTPModule.getInboxState(this.inboxId, refreshFromNetwork) + } + /** * Determines whether the current user can send messages to the specified peers over groups. * diff --git a/src/lib/InboxState.ts b/src/lib/InboxState.ts new file mode 100644 index 000000000..89d5b8593 --- /dev/null +++ b/src/lib/InboxState.ts @@ -0,0 +1,30 @@ +import { InboxId } from './Client' + +export class InboxState { + inboxId: InboxId + addresses: string[] + installationIds: string[] + recoveryAddress: string + + constructor( + inboxId: InboxId, + addresses: string[], + installationIds: string[], + recoveryAddress: string + ) { + this.inboxId = inboxId + this.addresses = addresses + this.installationIds = installationIds + this.recoveryAddress = recoveryAddress + } + + static from(json: string): InboxState { + const entry = JSON.parse(json) + return new InboxState( + entry.inboxId, + entry.addresses, + entry.installationIds, + entry.recoveryAddress + ) + } +}