You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
session-ios/SessionSnodeKit/Models/SendMessageResponse.swift

100 lines
3.2 KiB
Swift

// Copyright © 2022 Rangeproof Pty Ltd. All rights reserved.
import Foundation
import Sodium
import SessionUtilitiesKit
public class SendMessagesResponse: SnodeRecursiveResponse<SendMessagesResponse.SwarmItem> {
private enum CodingKeys: String, CodingKey {
case hash
case swarm
}
public let hash: String
// MARK: - Initialization
required init(from decoder: Decoder) throws {
let container: KeyedDecodingContainer<CodingKeys> = try decoder.container(keyedBy: CodingKeys.self)
hash = try container.decode(String.self, forKey: .hash)
try super.init(from: decoder)
}
}
// MARK: - SwarmItem
public extension SendMessagesResponse {
class SwarmItem: SnodeSwarmItem {
private enum CodingKeys: String, CodingKey {
case hash
case already
}
public let hash: String?
/// `true` if a message with this hash was already stored
///
/// **Note:** The `hash` is still included and signed even if this occurs
public let already: Bool
// MARK: - Initialization
required init(from decoder: Decoder) throws {
let container: KeyedDecodingContainer<CodingKeys> = try decoder.container(keyedBy: CodingKeys.self)
hash = try? container.decode(String.self, forKey: .hash)
already = ((try? container.decode(Bool.self, forKey: .already)) ?? false)
try super.init(from: decoder)
}
}
}
// MARK: - ValidatableResponse
extension SendMessagesResponse: ValidatableResponse {
typealias ValidationData = Void
typealias ValidationResponse = Bool
/// Half of the responses in the swarm must be valid
internal static var requiredSuccessfulResponses: Int { -2 }
internal func validResultMap(
sodium: Sodium,
userX25519PublicKey: String,
validationData: Void
) throws -> [String: Bool] {
let validationMap: [String: Bool] = swarm.reduce(into: [:]) { result, next in
guard
!next.value.failed,
let signatureBase64: String = next.value.signatureBase64,
let encodedSignature: Data = Data(base64Encoded: signatureBase64),
let hash: String = next.value.hash
else {
result[next.key] = false
if let reason: String = next.value.reason, let statusCode: Int = next.value.code {
SNLog("Couldn't store message on: \(next.key) due to error: \(reason) (\(statusCode)).")
}
else {
SNLog("Couldn't store message on: \(next.key).")
}
return
}
/// Signature of `hash` signed by the node's ed25519 pubkey
let verificationBytes: [UInt8] = hash.bytes
result[next.key] = sodium.sign.verify(
message: verificationBytes,
publicKey: Data(hex: next.key).bytes,
signature: encodedSignature.bytes
)
}
return try Self.validated(map: validationMap)
}
}