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.
		
		
		
		
		
			
		
			
				
	
	
		
			93 lines
		
	
	
		
			3.6 KiB
		
	
	
	
		
			Swift
		
	
			
		
		
	
	
			93 lines
		
	
	
		
			3.6 KiB
		
	
	
	
		
			Swift
		
	
| // Copyright © 2022 Rangeproof Pty Ltd. All rights reserved.
 | |
| 
 | |
| import Foundation
 | |
| import SessionUtilitiesKit
 | |
| 
 | |
| public class DeleteAllMessagesResponse: SnodeRecursiveResponse<DeleteAllMessagesResponse.SwarmItem> {}
 | |
| 
 | |
| // MARK: - SwarmItem
 | |
| 
 | |
| public extension DeleteAllMessagesResponse {
 | |
|     class SwarmItem: SnodeSwarmItem {
 | |
|         private enum CodingKeys: String, CodingKey {
 | |
|             case deleted
 | |
|         }
 | |
|         
 | |
|         public let deleted: [String]
 | |
|         public let deletedNamespaced: [String: [String]]
 | |
|         
 | |
|         // MARK: - Initialization
 | |
|         
 | |
|         required init(from decoder: Decoder) throws {
 | |
|             let container: KeyedDecodingContainer<CodingKeys> = try decoder.container(keyedBy: CodingKeys.self)
 | |
|             
 | |
|             if let decodedDeletedNamespaced: [String: [String]] = try? container.decode([String: [String]].self, forKey: .deleted) {
 | |
|                 deletedNamespaced = decodedDeletedNamespaced
 | |
|                 
 | |
|                 /// **Note:** When doing a multi-namespace delete the `DELETEDHASH` values are totally
 | |
|                 /// ordered (i.e. among all the hashes deleted regardless of namespace)
 | |
|                 deleted = decodedDeletedNamespaced
 | |
|                     .reduce(into: []) { result, next in result.append(contentsOf: next.value) }
 | |
|                     .sorted()
 | |
|             }
 | |
|             else {
 | |
|                 deleted = ((try? container.decode([String].self, forKey: .deleted)) ?? [])
 | |
|                 deletedNamespaced = [:]
 | |
|             }
 | |
|             
 | |
|             try super.init(from: decoder)
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| // MARK: - ValidatableResponse
 | |
| 
 | |
| extension DeleteAllMessagesResponse: ValidatableResponse {
 | |
|     typealias ValidationData = UInt64
 | |
|     typealias ValidationResponse = Bool
 | |
|     
 | |
|     /// Just one response in the swarm must be valid
 | |
|     internal static var requiredSuccessfulResponses: Int { 1 }
 | |
|     
 | |
|     internal func validResultMap(
 | |
|         swarmPublicKey: String,
 | |
|         validationData: UInt64,
 | |
|         using dependencies: Dependencies
 | |
|     ) 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)
 | |
|             else {
 | |
|                 result[next.key] = false
 | |
|                 
 | |
|                 if let reason: String = next.value.reason, let statusCode: Int = next.value.code {
 | |
|                     SNLog("Couldn't delete data from: \(next.key) due to error: \(reason) (\(statusCode)).")
 | |
|                 }
 | |
|                 else {
 | |
|                     SNLog("Couldn't delete data from: \(next.key).")
 | |
|                 }
 | |
|                 return
 | |
|             }
 | |
|             
 | |
|             /// Signature of `( PUBKEY_HEX || TIMESTAMP || DELETEDHASH[0] || ... || DELETEDHASH[N] )`
 | |
|             /// signed by the node's ed25519 pubkey.  When doing a multi-namespace delete the `DELETEDHASH`
 | |
|             /// values are totally ordered (i.e. among all the hashes deleted regardless of namespace)
 | |
|             let verificationBytes: [UInt8] = swarmPublicKey.bytes
 | |
|                 .appending(contentsOf: "\(validationData)".data(using: .ascii)?.bytes)
 | |
|                 .appending(contentsOf: next.value.deleted.joined().bytes)
 | |
|             
 | |
|             result[next.key] = dependencies.crypto.verify(
 | |
|                 .signature(
 | |
|                     message: verificationBytes,
 | |
|                     publicKey: Data(hex: next.key).bytes,
 | |
|                     signature: encodedSignature.bytes
 | |
|                 )
 | |
|             )
 | |
|         }
 | |
|         
 | |
|         return try Self.validated(map: validationMap)
 | |
|     }
 | |
| }
 |