mirror of https://github.com/oxen-io/session-ios
				
				
				
			
			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.
		
		
		
		
		
			
		
			
				
	
	
		
			113 lines
		
	
	
		
			4.7 KiB
		
	
	
	
		
			Swift
		
	
			
		
		
	
	
			113 lines
		
	
	
		
			4.7 KiB
		
	
	
	
		
			Swift
		
	
| // Copyright © 2022 Rangeproof Pty Ltd. All rights reserved.
 | |
| 
 | |
| import Foundation
 | |
| import Sodium
 | |
| import SessionUtilitiesKit
 | |
| 
 | |
| public class UpdateExpiryResponse: SnodeRecursiveResponse<UpdateExpiryResponse.SwarmItem> {}
 | |
| 
 | |
| public struct UpdateExpiryResponseResult {
 | |
|     public let changed: [String: UInt64]
 | |
|     public let unchanged: [String: UInt64]
 | |
|     public let didError: Bool
 | |
| }
 | |
| 
 | |
| // MARK: - SwarmItem
 | |
| 
 | |
| public extension UpdateExpiryResponse {
 | |
|     class SwarmItem: SnodeSwarmItem {
 | |
|         private enum CodingKeys: String, CodingKey {
 | |
|             case updated
 | |
|             case unchanged
 | |
|             case expiry
 | |
|         }
 | |
|         
 | |
|         public let updated: [String]
 | |
|         public let unchanged: [String: UInt64]
 | |
|         public let expiry: UInt64?
 | |
|         
 | |
|         // MARK: - Initialization
 | |
|         
 | |
|         required init(from decoder: Decoder) throws {
 | |
|             let container: KeyedDecodingContainer<CodingKeys> = try decoder.container(keyedBy: CodingKeys.self)
 | |
|             
 | |
|             updated = ((try? container.decode([String].self, forKey: .updated)) ?? [])
 | |
|             unchanged = ((try? container.decode([String: UInt64].self, forKey: .unchanged)) ?? [:])
 | |
|             expiry = try? container.decode(UInt64.self, forKey: .expiry)
 | |
|             
 | |
|             try super.init(from: decoder)
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| // MARK: - ValidatableResponse
 | |
| 
 | |
| extension UpdateExpiryResponse: ValidatableResponse {
 | |
|     typealias ValidationData = [String]
 | |
|     typealias ValidationResponse = UpdateExpiryResponseResult
 | |
|     
 | |
|     /// All responses in the swarm must be valid
 | |
|     internal static var requiredSuccessfulResponses: Int { -1 }
 | |
|     
 | |
|     internal func validResultMap(
 | |
|         sodium: Sodium,
 | |
|         userX25519PublicKey: String,
 | |
|         validationData: [String]
 | |
|     ) throws -> [String: UpdateExpiryResponseResult] {
 | |
|         let validationMap: [String: UpdateExpiryResponseResult] = try swarm.reduce(into: [:]) { result, next in
 | |
|             guard
 | |
|                 !next.value.failed,
 | |
|                 let appliedExpiry: UInt64 = next.value.expiry,
 | |
|                 let signatureBase64: String = next.value.signatureBase64,
 | |
|                 let encodedSignature: Data = Data(base64Encoded: signatureBase64)
 | |
|             else {
 | |
|                 result[next.key] = UpdateExpiryResponseResult(changed: [:], unchanged: [:], didError: true)
 | |
|                 
 | |
|                 if let reason: String = next.value.reason, let statusCode: Int = next.value.code {
 | |
|                     SNLog("Couldn't update expiry from: \(next.key) due to error: \(reason) (\(statusCode)).")
 | |
|                 }
 | |
|                 else {
 | |
|                     SNLog("Couldn't update expiry from: \(next.key).")
 | |
|                 }
 | |
|                 return
 | |
|             }
 | |
|             
 | |
|             /// Signature of
 | |
|             /// `( PUBKEY_HEX || EXPIRY || RMSGs... || UMSGs... || CMSG_EXPs... )`
 | |
|             /// where RMSGs are the requested expiry hashes, UMSGs are the actual updated hashes, and
 | |
|             /// CMSG_EXPs are (HASH || EXPIRY) values, ascii-sorted by hash, for the unchanged message
 | |
|             /// hashes included in the "unchanged" field.  The signature uses the node's ed25519 pubkey.
 | |
|             ///
 | |
|             /// **Note:** If `updated` is empty then the `expiry` value will match the value that was
 | |
|             /// included in the original request
 | |
|             let verificationBytes: [UInt8] = userX25519PublicKey.bytes
 | |
|                 .appending(contentsOf: "\(appliedExpiry)".data(using: .ascii)?.bytes)
 | |
|                 .appending(contentsOf: validationData.joined().bytes)
 | |
|                 .appending(contentsOf: next.value.updated.sorted().joined().bytes)
 | |
|                 .appending(contentsOf: next.value.unchanged
 | |
|                     .sorted(by: { lhs, rhs in lhs.key < rhs.key })
 | |
|                     .reduce(into: [UInt8]()) { result, nextUnchanged in
 | |
|                         result.append(contentsOf: nextUnchanged.key.bytes)
 | |
|                         result.append(contentsOf: "\(nextUnchanged.value)".data(using: .ascii)?.bytes ?? [])
 | |
|                     }
 | |
|                 )
 | |
|             let isValid: Bool = sodium.sign.verify(
 | |
|                 message: verificationBytes,
 | |
|                 publicKey: Data(hex: next.key).bytes,
 | |
|                 signature: encodedSignature.bytes
 | |
|             )
 | |
|             
 | |
|             // If the update signature is invalid then we want to fail here
 | |
|             guard isValid else { throw SnodeAPIError.signatureVerificationFailed }
 | |
|             
 | |
|             result[next.key] = UpdateExpiryResponseResult(
 | |
|                 changed: next.value.updated.reduce(into: [:]) { prev, next in prev[next] = appliedExpiry },
 | |
|                 unchanged: next.value.unchanged,
 | |
|                 didError: false
 | |
|             )
 | |
|         }
 | |
|         
 | |
|         return try Self.validated(map: validationMap, totalResponseCount: swarm.count)
 | |
|     }
 | |
| }
 |