From 7ebfec87a051706ea5a905bc6ca5f0d512188dab Mon Sep 17 00:00:00 2001 From: Daniel McCartney Date: Wed, 21 Jun 2023 15:26:08 -0400 Subject: [PATCH] feat: permit creating client from exported key bundle --- .../modules/xmtpreactnativesdk/XMTPModule.kt | 17 + ios/XMTPModule.swift | 614 +++++++++--------- src/index.ts | 11 + src/lib/Client.ts | 16 + 4 files changed, 357 insertions(+), 301 deletions(-) diff --git a/android/src/main/java/expo/modules/xmtpreactnativesdk/XMTPModule.kt b/android/src/main/java/expo/modules/xmtpreactnativesdk/XMTPModule.kt index 0e39441d9..232c90734 100644 --- a/android/src/main/java/expo/modules/xmtpreactnativesdk/XMTPModule.kt +++ b/android/src/main/java/expo/modules/xmtpreactnativesdk/XMTPModule.kt @@ -26,6 +26,7 @@ import org.xmtp.android.library.messages.PrivateKeyBuilder import org.xmtp.android.library.messages.Signature import org.xmtp.android.library.push.XMTPPush import org.xmtp.proto.message.contents.SignatureOuterClass +import org.xmtp.proto.message.contents.PrivateKeyOuterClass import java.util.Date import java.util.UUID import kotlin.coroutines.Continuation @@ -132,6 +133,22 @@ class XMTPModule : Module() { randomClient.address } + AsyncFunction("createFromKeyBundle") { keyBundle: String, environment: String -> + logV("createFromKeyBundle") + val options = + ClientOptions(api = apiEnvironments[environment] ?: apiEnvironments["dev"]!!) + val bundle = PrivateKeyOuterClass.PrivateKeyBundle.parseFrom(Base64.decode(keyBundle, NO_WRAP)) + val client = Client().buildFromBundle(bundle = bundle, options = options) + clients[client.address] = client + client.address + } + + AsyncFunction("exportKeyBundle") { clientAddress: String -> + logV("exportKeyBundle") + val client = clients[clientAddress] ?: throw XMTPException("No client") + Base64.encodeToString(client.privateKeyBundle.toByteArray(), NO_WRAP) + } + // // Client API AsyncFunction("canMessage") { clientAddress: String, peerAddress: String -> diff --git a/ios/XMTPModule.swift b/ios/XMTPModule.swift index 2f422cb63..26e975375 100644 --- a/ios/XMTPModule.swift +++ b/ios/XMTPModule.swift @@ -2,164 +2,181 @@ import ExpoModulesCore import XMTP class ReactNativeSigner: NSObject, XMTP.SigningKey { - enum Error: Swift.Error { - case invalidSignature - } - - var module: XMTPModule - var address: String - var continuations: [String: CheckedContinuation] = [:] - - init(module: XMTPModule, address: String) { - self.module = module - self.address = address - } - - func handle(id: String, signature: String) throws { - guard let continuation = continuations[id] else { - return - } - - guard let signatureData = Data(base64Encoded: Data(signature.utf8)), signatureData.count == 65 else { - continuation.resume(throwing: Error.invalidSignature) - continuations.removeValue(forKey: id) - return - } - - let signature = XMTP.Signature.with { - $0.ecdsaCompact.bytes = signatureData[0 ..< 64] - $0.ecdsaCompact.recovery = UInt32(signatureData[64]) - } - - continuation.resume(returning: signature) - continuations.removeValue(forKey: id) - } - - func sign(_ data: Data) async throws -> XMTP.Signature { - let request = SignatureRequest(message: String(data: data, encoding: .utf8)!) - - module.sendEvent("sign", [ - "id": request.id, - "message": request.message - ]) - - return try await withCheckedThrowingContinuation { continuation in - continuations[request.id] = continuation - } - } - - func sign(message: String) async throws -> XMTP.Signature { - return try await sign(Data(message.utf8)) - } + enum Error: Swift.Error { + case invalidSignature + } + + var module: XMTPModule + var address: String + var continuations: [String: CheckedContinuation] = [:] + + init(module: XMTPModule, address: String) { + self.module = module + self.address = address + } + + func handle(id: String, signature: String) throws { + guard let continuation = continuations[id] else { + return + } + + guard let signatureData = Data(base64Encoded: Data(signature.utf8)), signatureData.count == 65 else { + continuation.resume(throwing: Error.invalidSignature) + continuations.removeValue(forKey: id) + return + } + + let signature = XMTP.Signature.with { + $0.ecdsaCompact.bytes = signatureData[0 ..< 64] + $0.ecdsaCompact.recovery = UInt32(signatureData[64]) + } + + continuation.resume(returning: signature) + continuations.removeValue(forKey: id) + } + + func sign(_ data: Data) async throws -> XMTP.Signature { + let request = SignatureRequest(message: String(data: data, encoding: .utf8)!) + + module.sendEvent("sign", [ + "id": request.id, + "message": request.message + ]) + + return try await withCheckedThrowingContinuation { continuation in + continuations[request.id] = continuation + } + } + + func sign(message: String) async throws -> XMTP.Signature { + return try await sign(Data(message.utf8)) + } } struct SignatureRequest: Codable { - var id = UUID().uuidString - var message: String + var id = UUID().uuidString + var message: String } extension Conversation { - func cacheKey(_ clientAddress: String) -> String { - if let conversationID, conversationID != "" { - return "\(clientAddress):\(topic):\(conversationID)" - } else { - return "\(clientAddress):\(topic)" - } - } + func cacheKey(_ clientAddress: String) -> String { + if let conversationID, conversationID != "" { + return "\(clientAddress):\(topic):\(conversationID)" + } else { + return "\(clientAddress):\(topic)" + } + } } public class XMTPModule: Module { - var apiEnvironments = [ - "local": XMTP.ClientOptions.Api( - env: XMTP.XMTPEnvironment.local, - isSecure: false - ), - "dev": XMTP.ClientOptions.Api( - env: XMTP.XMTPEnvironment.dev, - isSecure: true - ), - "production": XMTP.ClientOptions.Api( - env: XMTP.XMTPEnvironment.production, - isSecure: true - ), - ] - - var clients: [String: XMTP.Client] = [:] - var signer: ReactNativeSigner? - var conversations: [String: Conversation] = [:] - var subscriptions: [String: Task] = [:] - - enum Error: Swift.Error { - case noClient, conversationNotFound(String), noMessage - } - - public func definition() -> ModuleDefinition { + var apiEnvironments = [ + "local": XMTP.ClientOptions.Api( + env: XMTP.XMTPEnvironment.local, + isSecure: false + ), + "dev": XMTP.ClientOptions.Api( + env: XMTP.XMTPEnvironment.dev, + isSecure: true + ), + "production": XMTP.ClientOptions.Api( + env: XMTP.XMTPEnvironment.production, + isSecure: true + ), + ] + + var clients: [String: XMTP.Client] = [:] + var signer: ReactNativeSigner? + var conversations: [String: Conversation] = [:] + var subscriptions: [String: Task] = [:] + + enum Error: Swift.Error { + case noClient, conversationNotFound(String), noMessage + } + + public func definition() -> ModuleDefinition { Name("XMTP") Events("sign", "authed", "conversation", "message") - Function("address") { (clientAddress: String) -> String in - if let client = clients[clientAddress] { - return client.address - } else { - return "No Client." - } - } - - // - // Auth functions - // - AsyncFunction("auth") { (address: String, environment: String) in - let signer = ReactNativeSigner(module: self, address: address) - self.signer = signer - let options = XMTP.ClientOptions(api: apiEnvironments[environment] ?? apiEnvironments["local"]!) - self.clients[address] = try await XMTP.Client.create(account: signer, options: options) - self.signer = nil - sendEvent("authed") - } - - Function("receiveSignature") { (requestID: String, signature: String) in - try signer?.handle(id: requestID, signature: signature) - } - - // Generate a random wallet and set the client to that - AsyncFunction("createRandom") { (environment: String) -> String in - let privateKey = try PrivateKey.generate() - let options = XMTP.ClientOptions(api: apiEnvironments[environment] ?? apiEnvironments["dev"]!) - let client = try await Client.create(account: privateKey, options: options) - - self.clients[client.address] = client - return client.address - } - - // - // Client API - AsyncFunction("canMessage") { (clientAddress: String, peerAddress: String) -> Bool in - guard let client = clients[clientAddress] else { - throw Error.noClient - } - - return try await client.canMessage(peerAddress) - } - - AsyncFunction("listConversations") { (clientAddress: String) -> [String] in - guard let client = clients[clientAddress] else { - throw Error.noClient - } - - let conversations = try await client.conversations.list() - - return try conversations.map { conversation in - self.conversations[conversation.cacheKey(clientAddress)] = conversation - - return try ConversationWrapper.encode(ConversationWithClientAddress(client: client, conversation: conversation)) - } - } - - AsyncFunction("loadMessages") { (clientAddress: String, topics: [String], conversationIDs: [String?], limit: Int?, before: Double?, after: Double?) -> [String] in + Function("address") { (clientAddress: String) -> String in + if let client = clients[clientAddress] { + return client.address + } else { + return "No Client." + } + } + + // + // Auth functions + // + AsyncFunction("auth") { (address: String, environment: String) in + let signer = ReactNativeSigner(module: self, address: address) + self.signer = signer + let options = XMTP.ClientOptions(api: apiEnvironments[environment] ?? apiEnvironments["local"]!) + self.clients[address] = try await XMTP.Client.create(account: signer, options: options) + self.signer = nil + sendEvent("authed") + } + + Function("receiveSignature") { (requestID: String, signature: String) in + try signer?.handle(id: requestID, signature: signature) + } + + // Generate a random wallet and set the client to that + AsyncFunction("createRandom") { (environment: String) -> String in + let privateKey = try PrivateKey.generate() + let options = XMTP.ClientOptions(api: apiEnvironments[environment] ?? apiEnvironments["dev"]!) + let client = try await Client.create(account: privateKey, options: options) + + self.clients[client.address] = client + return client.address + } + + // Create a client using its serialized key bundle. + AsyncFunction("createFromKeyBundle") { (keyBundle: String, environment: String) -> String in + let bundle = try PrivateKeyBundle(serializedData: Data(base64Encoded: Data(keyBundle.utf8))!) + let options = XMTP.ClientOptions(api: apiEnvironments[environment] ?? apiEnvironments["dev"]!) + let client = try await Client.from(bundle: bundle, options: options) + self.clients[client.address] = client + return client.address + } + + // Export the client's serialized key bundle. + AsyncFunction("exportKeyBundle") { (clientAddress: String) -> String in + guard let client = clients[clientAddress] else { + throw Error.noClient + } + let bundle = try client.privateKeyBundle.serializedData().base64EncodedString() + return bundle + } + + // + // Client API + AsyncFunction("canMessage") { (clientAddress: String, peerAddress: String) -> Bool in + guard let client = clients[clientAddress] else { + throw Error.noClient + } + + return try await client.canMessage(peerAddress) + } + + AsyncFunction("listConversations") { (clientAddress: String) -> [String] in + guard let client = clients[clientAddress] else { + throw Error.noClient + } + + let conversations = try await client.conversations.list() + + return try conversations.map { conversation in + self.conversations[conversation.cacheKey(clientAddress)] = conversation + + return try ConversationWrapper.encode(ConversationWithClientAddress(client: client, conversation: conversation)) + } + } + + AsyncFunction("loadMessages") { (clientAddress: String, topics: [String], conversationIDs: [String?], limit: Int?, before: Double?, after: Double?) -> [String] in let beforeDate = before != nil ? Date(timeIntervalSince1970: before!) : nil let afterDate = after != nil ? Date(timeIntervalSince1970: after!) : nil - var messages:[String] = [] guard let client = clients[clientAddress] else { throw Error.noClient } @@ -171,164 +188,159 @@ public class XMTPModule: Module { .map { (msg) in try DecodedMessageWrapper.encode(msg) } } - // TODO: Support content types - AsyncFunction("sendMessage") { (clientAddress: String, conversationTopic: String, conversationID: String?, content: String) -> String in - guard let client = clients[clientAddress] else { - throw Error.noClient - } - - guard let conversation = try await findConversation(clientAddress: clientAddress, topic: conversationTopic, conversationID: conversationID) else { - throw Error.conversationNotFound("no conversation found for \(conversationTopic)") - } - - let preparedMessage = try await conversation.prepareMessage(content: content) - let decodedMessage = try preparedMessage.decodedMessage() - - try await preparedMessage.send() - - return try DecodedMessageWrapper.encode(decodedMessage) - } - - AsyncFunction("createConversation") { (clientAddress: String, peerAddress: String, conversationID: String?) -> String in - guard let client = clients[clientAddress] else { - throw Error.noClient - } - - do { - let conversation = try await client.conversations.newConversation(with: peerAddress, context: .init( - conversationID: conversationID ?? "" - )) - - return try ConversationWrapper.encode(ConversationWithClientAddress(client: client, conversation: conversation)) - } catch { - print("ERRRO!: \(error.localizedDescription)") - throw error - } - } - - Function("subscribeToConversations") { (clientAddress: String) in - subscribeToConversations(clientAddress: clientAddress) - } - - AsyncFunction("subscribeToMessages") { (clientAddress: String, topic: String, conversationID: String?) in - try await subscribeToMessages(clientAddress: clientAddress, topic: topic, conversationID: conversationID) - } - - AsyncFunction("unsubscribeFromMessages") { (clientAddress: String, topic: String, conversationID: String?) in - try await unsubscribeFromMessages(clientAddress: clientAddress, topic: topic, conversationID: conversationID) - } - - AsyncFunction("registerPushToken") { (pushServer: String, token: String) in - XMTPPush.shared.setPushServer(pushServer) - do { - try await XMTPPush.shared.register(token: token) - } catch { - print("Error registering: \(error)") - } - } - - AsyncFunction("subscribePushTopics") { (topics: [String]) in - do { - try await XMTPPush.shared.subscribe(topics: topics) - } catch { - print("Error subscribing: \(error)") - } - } - - AsyncFunction("decodeMessage") { (clientAddress: String, topic: String, encryptedMessage: String, conversationID: String?) -> String in - guard let encryptedMessageData = Data(base64Encoded: Data(encryptedMessage.utf8))else { - throw Error.noMessage - } - - let envelope = XMTP.Envelope.with { envelope in - envelope.message = encryptedMessageData - envelope.contentTopic = topic - } - - guard let conversation = try await findConversation(clientAddress: clientAddress, topic: topic, conversationID: conversationID) else { - throw Error.conversationNotFound("no conversation found for \(topic)") - } - let decodedMessage = try conversation.decode(envelope) - return try DecodedMessageWrapper.encode(decodedMessage) - } + // TODO: Support content types + AsyncFunction("sendMessage") { (clientAddress: String, conversationTopic: String, conversationID: String?, content: String) -> String in + guard let conversation = try await findConversation(clientAddress: clientAddress, topic: conversationTopic, conversationID: conversationID) else { + throw Error.conversationNotFound("no conversation found for \(conversationTopic)") + } + + let preparedMessage = try await conversation.prepareMessage(content: content) + let decodedMessage = try preparedMessage.decodedMessage() + + try await preparedMessage.send() + + return try DecodedMessageWrapper.encode(decodedMessage) + } + + AsyncFunction("createConversation") { (clientAddress: String, peerAddress: String, conversationID: String?) -> String in + guard let client = clients[clientAddress] else { + throw Error.noClient + } + + do { + let conversation = try await client.conversations.newConversation(with: peerAddress, context: .init( + conversationID: conversationID ?? "" + )) + + return try ConversationWrapper.encode(ConversationWithClientAddress(client: client, conversation: conversation)) + } catch { + print("ERRRO!: \(error.localizedDescription)") + throw error + } + } + + Function("subscribeToConversations") { (clientAddress: String) in + subscribeToConversations(clientAddress: clientAddress) + } + + AsyncFunction("subscribeToMessages") { (clientAddress: String, topic: String, conversationID: String?) in + try await subscribeToMessages(clientAddress: clientAddress, topic: topic, conversationID: conversationID) + } + + AsyncFunction("unsubscribeFromMessages") { (clientAddress: String, topic: String, conversationID: String?) in + try await unsubscribeFromMessages(clientAddress: clientAddress, topic: topic, conversationID: conversationID) + } + + AsyncFunction("registerPushToken") { (pushServer: String, token: String) in + XMTPPush.shared.setPushServer(pushServer) + do { + try await XMTPPush.shared.register(token: token) + } catch { + print("Error registering: \(error)") + } + } + + AsyncFunction("subscribePushTopics") { (topics: [String]) in + do { + try await XMTPPush.shared.subscribe(topics: topics) + } catch { + print("Error subscribing: \(error)") + } + } + + AsyncFunction("decodeMessage") { (clientAddress: String, topic: String, encryptedMessage: String, conversationID: String?) -> String in + guard let encryptedMessageData = Data(base64Encoded: Data(encryptedMessage.utf8))else { + throw Error.noMessage + } + + let envelope = XMTP.Envelope.with { envelope in + envelope.message = encryptedMessageData + envelope.contentTopic = topic + } + + guard let conversation = try await findConversation(clientAddress: clientAddress, topic: topic, conversationID: conversationID) else { + throw Error.conversationNotFound("no conversation found for \(topic)") + } + let decodedMessage = try conversation.decode(envelope) + return try DecodedMessageWrapper.encode(decodedMessage) + } } - // - // Helpers - // - - func findConversation(clientAddress: String, topic: String, conversationID: String?) async throws -> Conversation? { - guard let client = clients[clientAddress] else { - fatalError("NO CLIENT FOR ADDRES: \(clientAddress)") - throw Error.noClient - } - - let cacheKey: String - - if let conversationID, conversationID != "" { - cacheKey = "\(clientAddress):\(topic):\(conversationID)" - } else { - cacheKey = "\(clientAddress):\(topic)" - } - - if let conversation = conversations[cacheKey] { - return conversation - } else if let conversation = try await client.conversations.list().first(where: { $0.topic == topic && $0.conversationID == conversationID }) { - conversations[conversation.cacheKey(clientAddress)] = conversation - return conversation - } - - return nil - } - - func subscribeToConversations(clientAddress: String) { - guard let client = clients[clientAddress] else { - return - } - - subscriptions["conversations"] = Task { - do { - for try await conversation in client.conversations.stream() { - sendEvent("conversation", [ - "topic": conversation.topic, - "peerAddress": conversation.peerAddress, - "version": conversation.version == .v1 ? "v1" : "v2", - "conversationID": conversation.conversationID - ]) - } - } catch { - print("Error in conversations subscription: \(error)") - subscriptions["conversations"]?.cancel() - } - } - } - - func subscribeToMessages(clientAddress: String, topic: String, conversationID: String?) async throws { - guard let conversation = try await findConversation(clientAddress: clientAddress, topic: topic, conversationID: conversationID) else { - return - } - - subscriptions[conversation.cacheKey(clientAddress)] = Task { - do { - for try await message in conversation.streamMessages() { - sendEvent("message", [ - "topic": conversation.topic, - "conversationID": conversation.conversationID, - "messageJSON": try DecodedMessageWrapper.encode(message) - ]) - } - } catch { - print("Error in messages subscription: \(error)") - subscriptions[conversation.cacheKey(clientAddress)]?.cancel() - } - } - } - - func unsubscribeFromMessages(clientAddress: String, topic: String, conversationID: String?) async throws { - guard let conversation = try await findConversation(clientAddress: clientAddress, topic: topic, conversationID: conversationID) else { - return - } - - subscriptions[conversation.cacheKey(clientAddress)]?.cancel() - } + // + // Helpers + // + + func findConversation(clientAddress: String, topic: String, conversationID: String?) async throws -> Conversation? { + guard let client = clients[clientAddress] else { + throw Error.noClient + } + + let cacheKey: String + + if let conversationID, conversationID != "" { + cacheKey = "\(clientAddress):\(topic):\(conversationID)" + } else { + cacheKey = "\(clientAddress):\(topic)" + } + + if let conversation = conversations[cacheKey] { + return conversation + } else if let conversation = try await client.conversations.list().first(where: { $0.topic == topic && $0.conversationID == conversationID }) { + conversations[conversation.cacheKey(clientAddress)] = conversation + return conversation + } + + return nil + } + + func subscribeToConversations(clientAddress: String) { + guard let client = clients[clientAddress] else { + return + } + + subscriptions["conversations"] = Task { + do { + for try await conversation in client.conversations.stream() { + sendEvent("conversation", [ + "topic": conversation.topic, + "peerAddress": conversation.peerAddress, + "version": conversation.version == .v1 ? "v1" : "v2", + "conversationID": conversation.conversationID + ]) + } + } catch { + print("Error in conversations subscription: \(error)") + subscriptions["conversations"]?.cancel() + } + } + } + + func subscribeToMessages(clientAddress: String, topic: String, conversationID: String?) async throws { + guard let conversation = try await findConversation(clientAddress: clientAddress, topic: topic, conversationID: conversationID) else { + return + } + + subscriptions[conversation.cacheKey(clientAddress)] = Task { + do { + for try await message in conversation.streamMessages() { + sendEvent("message", [ + "topic": conversation.topic, + "conversationID": conversation.conversationID, + "messageJSON": try DecodedMessageWrapper.encode(message) + ]) + } + } catch { + print("Error in messages subscription: \(error)") + subscriptions[conversation.cacheKey(clientAddress)]?.cancel() + } + } + } + + func unsubscribeFromMessages(clientAddress: String, topic: String, conversationID: String?) async throws { + guard let conversation = try await findConversation(clientAddress: clientAddress, topic: topic, conversationID: conversationID) else { + return + } + + subscriptions[conversation.cacheKey(clientAddress)]?.cancel() + } } diff --git a/src/index.ts b/src/index.ts index b45793942..e7b2bc53b 100644 --- a/src/index.ts +++ b/src/index.ts @@ -25,6 +25,17 @@ export async function createRandom( return await XMTPModule.createRandom(environment); } +export async function createFromKeyBundle( + keyBundle: string, + environment: "local" | "dev" | "production" +): Promise { + return await XMTPModule.createFromKeyBundle(keyBundle, environment); +} + +export async function exportKeyBundle(clientAddress: string): Promise { + return await XMTPModule.exportKeyBundle(clientAddress); +} + export async function canMessage(clientAddress: string, peerAddress: string): Promise { return await XMTPModule.canMessage(clientAddress, peerAddress); } diff --git a/src/lib/Client.ts b/src/lib/Client.ts index b856ef248..729f136e4 100644 --- a/src/lib/Client.ts +++ b/src/lib/Client.ts @@ -48,6 +48,14 @@ export class Client { return new Client(address); } + static async createFromKeyBundle( + keyBundle: string, + environment: "local" | "dev" | "production" + ): Promise { + const address = await XMTPModule.createFromKeyBundle(keyBundle, environment); + return new Client(address); + } + async canMessage( peerAddress: string ): Promise { @@ -59,6 +67,14 @@ export class Client { this.conversations = new Conversations(this); } + async exportKeyBundle(): Promise { + return XMTPModule.exportKeyBundle(this.address); + } + +// TODO: support persisting conversations for quick lookup +// async importConversation(exported: string): Promise { ... } +// async exportConversation(topic: string): Promise { ... } + async listBatchMessages( topics: string[], conversationIDs: (string | undefined)[],