|
|
@ -4,6 +4,8 @@ import Sodium
|
|
|
|
|
|
|
|
|
|
|
|
@objc(SNSnodeAPI)
|
|
|
|
@objc(SNSnodeAPI)
|
|
|
|
public final class SnodeAPI : NSObject {
|
|
|
|
public final class SnodeAPI : NSObject {
|
|
|
|
|
|
|
|
private static let sodium = Sodium()
|
|
|
|
|
|
|
|
|
|
|
|
private static var hasLoadedSnodePool = false
|
|
|
|
private static var hasLoadedSnodePool = false
|
|
|
|
private static var loadedSwarms: Set<String> = []
|
|
|
|
private static var loadedSwarms: Set<String> = []
|
|
|
|
private static var getSnodePoolPromise: Promise<Set<Snode>>?
|
|
|
|
private static var getSnodePoolPromise: Promise<Set<Snode>>?
|
|
|
@ -296,7 +298,6 @@ public final class SnodeAPI : NSObject {
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public static func getSessionID(for onsName: String) -> Promise<String> {
|
|
|
|
public static func getSessionID(for onsName: String) -> Promise<String> {
|
|
|
|
let sodium = Sodium()
|
|
|
|
|
|
|
|
let validationCount = 3
|
|
|
|
let validationCount = 3
|
|
|
|
let sessionIDByteCount = 33
|
|
|
|
let sessionIDByteCount = 33
|
|
|
|
// The name must be lowercased
|
|
|
|
// The name must be lowercased
|
|
|
@ -390,26 +391,18 @@ public final class SnodeAPI : NSObject {
|
|
|
|
|
|
|
|
|
|
|
|
public static func getRawMessages(from snode: Snode, associatedWith publicKey: String) -> RawResponsePromise {
|
|
|
|
public static func getRawMessages(from snode: Snode, associatedWith publicKey: String) -> RawResponsePromise {
|
|
|
|
let (promise, seal) = RawResponsePromise.pending()
|
|
|
|
let (promise, seal) = RawResponsePromise.pending()
|
|
|
|
let storage = SNSnodeKitConfiguration.shared.storage
|
|
|
|
|
|
|
|
Threading.workQueue.async {
|
|
|
|
Threading.workQueue.async {
|
|
|
|
storage.pruneLastMessageHashInfoIfExpired(for: snode, associatedWith: publicKey)
|
|
|
|
getMessagesInternal(from: snode, associatedWith: publicKey).done2 { seal.fulfill($0) }.catch2 { seal.reject($0) }
|
|
|
|
let lastHash = storage.getLastMessageHash(for: snode, associatedWith: publicKey) ?? ""
|
|
|
|
|
|
|
|
let parameters = [ "pubKey" : Features.useTestnet ? publicKey.removing05PrefixIfNeeded() : publicKey, "lastHash" : lastHash ]
|
|
|
|
|
|
|
|
invoke(.getMessages, on: snode, associatedWith: publicKey, parameters: parameters).done2 { seal.fulfill($0) }.catch2 { seal.reject($0) }
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return promise
|
|
|
|
return promise
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public static func getMessages(for publicKey: String) -> Promise<Set<MessageListPromise>> {
|
|
|
|
public static func getMessages(for publicKey: String) -> Promise<Set<MessageListPromise>> {
|
|
|
|
let (promise, seal) = Promise<Set<MessageListPromise>>.pending()
|
|
|
|
let (promise, seal) = Promise<Set<MessageListPromise>>.pending()
|
|
|
|
let storage = SNSnodeKitConfiguration.shared.storage
|
|
|
|
|
|
|
|
Threading.workQueue.async {
|
|
|
|
Threading.workQueue.async {
|
|
|
|
attempt(maxRetryCount: maxRetryCount, recoveringOn: Threading.workQueue) {
|
|
|
|
attempt(maxRetryCount: maxRetryCount, recoveringOn: Threading.workQueue) {
|
|
|
|
getTargetSnodes(for: publicKey).mapValues2 { targetSnode in
|
|
|
|
getTargetSnodes(for: publicKey).mapValues2 { targetSnode in
|
|
|
|
storage.pruneLastMessageHashInfoIfExpired(for: targetSnode, associatedWith: publicKey)
|
|
|
|
return getMessagesInternal(from: targetSnode, associatedWith: publicKey).map2 { rawResponse in
|
|
|
|
let lastHash = storage.getLastMessageHash(for: targetSnode, associatedWith: publicKey) ?? ""
|
|
|
|
|
|
|
|
let parameters = [ "pubKey" : Features.useTestnet ? publicKey.removing05PrefixIfNeeded() : publicKey, "lastHash" : lastHash ]
|
|
|
|
|
|
|
|
return invoke(.getMessages, on: targetSnode, associatedWith: publicKey, parameters: parameters).map2 { rawResponse in
|
|
|
|
|
|
|
|
parseRawMessagesResponse(rawResponse, from: targetSnode, associatedWith: publicKey)
|
|
|
|
parseRawMessagesResponse(rawResponse, from: targetSnode, associatedWith: publicKey)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}.map2 { Set($0) }
|
|
|
|
}.map2 { Set($0) }
|
|
|
@ -417,6 +410,28 @@ public final class SnodeAPI : NSObject {
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return promise
|
|
|
|
return promise
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
private static func getMessagesInternal(from snode: Snode, associatedWith publicKey: String) -> RawResponsePromise {
|
|
|
|
|
|
|
|
let storage = SNSnodeKitConfiguration.shared.storage
|
|
|
|
|
|
|
|
guard let userED25519KeyPair = storage.getUserED25519KeyPair() else { return Promise(error: Error.noKeyPair) }
|
|
|
|
|
|
|
|
// Get last message hash
|
|
|
|
|
|
|
|
storage.pruneLastMessageHashInfoIfExpired(for: snode, associatedWith: publicKey)
|
|
|
|
|
|
|
|
let lastHash = storage.getLastMessageHash(for: snode, associatedWith: publicKey) ?? ""
|
|
|
|
|
|
|
|
// Construct signature
|
|
|
|
|
|
|
|
let timestamp = UInt64(Int64(NSDate.millisecondTimestamp()) + SnodeAPI.clockOffset)
|
|
|
|
|
|
|
|
let ed25519PublicKey = userED25519KeyPair.publicKey.toHexString()
|
|
|
|
|
|
|
|
let verificationData = ("retrieve" + String(timestamp)).data(using: String.Encoding.utf8)!
|
|
|
|
|
|
|
|
let signature = sodium.sign.signature(message: Bytes(verificationData), secretKey: userED25519KeyPair.secretKey)!
|
|
|
|
|
|
|
|
// Make the request
|
|
|
|
|
|
|
|
let parameters: JSON = [
|
|
|
|
|
|
|
|
"pubKey" : Features.useTestnet ? publicKey.removing05PrefixIfNeeded() : publicKey,
|
|
|
|
|
|
|
|
"lastHash" : lastHash,
|
|
|
|
|
|
|
|
"timestamp" : timestamp,
|
|
|
|
|
|
|
|
"pubkey_ed25519" : ed25519PublicKey,
|
|
|
|
|
|
|
|
"signature" : signature.toBase64()!
|
|
|
|
|
|
|
|
]
|
|
|
|
|
|
|
|
return invoke(.getMessages, on: snode, associatedWith: publicKey, parameters: parameters)
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public static func sendMessage(_ message: SnodeMessage) -> Promise<Set<RawResponsePromise>> {
|
|
|
|
public static func sendMessage(_ message: SnodeMessage) -> Promise<Set<RawResponsePromise>> {
|
|
|
|
let (promise, seal) = Promise<Set<RawResponsePromise>>.pending()
|
|
|
|
let (promise, seal) = Promise<Set<RawResponsePromise>>.pending()
|
|
|
@ -439,7 +454,6 @@ public final class SnodeAPI : NSObject {
|
|
|
|
let storage = SNSnodeKitConfiguration.shared.storage
|
|
|
|
let storage = SNSnodeKitConfiguration.shared.storage
|
|
|
|
guard let userX25519PublicKey = storage.getUserPublicKey(),
|
|
|
|
guard let userX25519PublicKey = storage.getUserPublicKey(),
|
|
|
|
let userED25519KeyPair = storage.getUserED25519KeyPair() else { return Promise(error: Error.noKeyPair) }
|
|
|
|
let userED25519KeyPair = storage.getUserED25519KeyPair() else { return Promise(error: Error.noKeyPair) }
|
|
|
|
let sodium = Sodium()
|
|
|
|
|
|
|
|
return attempt(maxRetryCount: maxRetryCount, recoveringOn: Threading.workQueue) {
|
|
|
|
return attempt(maxRetryCount: maxRetryCount, recoveringOn: Threading.workQueue) {
|
|
|
|
getSwarm(for: userX25519PublicKey).then2 { swarm -> Promise<[String:Bool]> in
|
|
|
|
getSwarm(for: userX25519PublicKey).then2 { swarm -> Promise<[String:Bool]> in
|
|
|
|
let snode = swarm.randomElement()!
|
|
|
|
let snode = swarm.randomElement()!
|
|
|
@ -532,6 +546,7 @@ public final class SnodeAPI : NSObject {
|
|
|
|
newReceivedMessages.insert(hash)
|
|
|
|
newReceivedMessages.insert(hash)
|
|
|
|
return !isDuplicate
|
|
|
|
return !isDuplicate
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Avoid the sync write transaction if possible
|
|
|
|
if oldReceivedMessages != newReceivedMessages {
|
|
|
|
if oldReceivedMessages != newReceivedMessages {
|
|
|
|
SNSnodeKitConfiguration.shared.storage.writeSync { transaction in
|
|
|
|
SNSnodeKitConfiguration.shared.storage.writeSync { transaction in
|
|
|
|
SNSnodeKitConfiguration.shared.storage.setReceivedMessages(to: newReceivedMessages, for: publicKey, using: transaction)
|
|
|
|
SNSnodeKitConfiguration.shared.storage.setReceivedMessages(to: newReceivedMessages, for: publicKey, using: transaction)
|
|
|
|