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.
		
		
		
		
		
			
		
			
				
	
	
		
			92 lines
		
	
	
		
			3.6 KiB
		
	
	
	
		
			Swift
		
	
			
		
		
	
	
			92 lines
		
	
	
		
			3.6 KiB
		
	
	
	
		
			Swift
		
	
// Copyright © 2022 Rangeproof Pty Ltd. All rights reserved.
 | 
						|
 | 
						|
import Foundation
 | 
						|
import Sodium
 | 
						|
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(
 | 
						|
        sodium: Sodium,
 | 
						|
        userX25519PublicKey: String,
 | 
						|
        validationData: UInt64
 | 
						|
    ) 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] = userX25519PublicKey.bytes
 | 
						|
                .appending(contentsOf: "\(validationData)".data(using: .ascii)?.bytes)
 | 
						|
                .appending(contentsOf: next.value.deleted.joined().bytes)
 | 
						|
            
 | 
						|
            result[next.key] = sodium.sign.verify(
 | 
						|
                message: verificationBytes,
 | 
						|
                publicKey: Data(hex: next.key).bytes,
 | 
						|
                signature: encodedSignature.bytes
 | 
						|
            )
 | 
						|
        }
 | 
						|
        
 | 
						|
        return try Self.validated(map: validationMap)
 | 
						|
    }
 | 
						|
}
 |