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.
		
		
		
		
		
			
		
			
				
	
	
		
			150 lines
		
	
	
		
			5.5 KiB
		
	
	
	
		
			Swift
		
	
			
		
		
	
	
			150 lines
		
	
	
		
			5.5 KiB
		
	
	
	
		
			Swift
		
	
| // Copyright © 2022 Rangeproof Pty Ltd. All rights reserved.
 | |
| 
 | |
| import Foundation
 | |
| import Combine
 | |
| 
 | |
| public extension HTTP {
 | |
|     typealias BatchResponseTypes = [Codable.Type]
 | |
|     
 | |
|     // MARK: - BatchResponse
 | |
|     
 | |
|     struct BatchResponse {
 | |
|         public let info: ResponseInfoType
 | |
|         public let responses: [Codable]
 | |
|     }
 | |
|     
 | |
|     // MARK: - BatchSubResponse<T>
 | |
|     
 | |
|     struct BatchSubResponse<T: Codable>: BatchSubResponseType {
 | |
|         /// The numeric http response code (e.g. 200 for success)
 | |
|         public let code: Int
 | |
|         
 | |
|         /// Any headers returned by the request
 | |
|         public let headers: [String: String]
 | |
|         
 | |
|         /// The body of the request; will be plain json if content-type is `application/json`, otherwise it will be base64 encoded data
 | |
|         public let body: T?
 | |
|         
 | |
|         /// A flag to indicate that there was a body but it failed to parse
 | |
|         public let failedToParseBody: Bool
 | |
|         
 | |
|         public init(
 | |
|             code: Int,
 | |
|             headers: [String: String] = [:],
 | |
|             body: T? = nil,
 | |
|             failedToParseBody: Bool = false
 | |
|         ) {
 | |
|             self.code = code
 | |
|             self.headers = headers
 | |
|             self.body = body
 | |
|             self.failedToParseBody = failedToParseBody
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| public protocol BatchSubResponseType: Codable {
 | |
|     var code: Int { get }
 | |
|     var headers: [String: String] { get }
 | |
|     var failedToParseBody: Bool { get }
 | |
| }
 | |
| 
 | |
| extension BatchSubResponseType {
 | |
|     public var responseInfo: ResponseInfoType { HTTP.ResponseInfo(code: code, headers: headers) }
 | |
| }
 | |
| 
 | |
| public extension HTTP.BatchSubResponse {
 | |
|     init(from decoder: Decoder) throws {
 | |
|         let container: KeyedDecodingContainer<CodingKeys> = try decoder.container(keyedBy: CodingKeys.self)
 | |
|         let body: T? = try? container.decode(T.self, forKey: .body)
 | |
|         
 | |
|         self = HTTP.BatchSubResponse(
 | |
|             code: try container.decode(Int.self, forKey: .code),
 | |
|             headers: ((try? container.decode([String: String].self, forKey: .headers)) ?? [:]),
 | |
|             body: body,
 | |
|             failedToParseBody: (
 | |
|                 body == nil &&
 | |
|                 T.self != NoResponse.self &&
 | |
|                 !(T.self is ExpressibleByNilLiteral.Type)
 | |
|             )
 | |
|         )
 | |
|     }
 | |
| }
 | |
| 
 | |
| // MARK: - Convenience
 | |
| 
 | |
| public extension Decodable {
 | |
|     static func decoded(from data: Data, using dependencies: Dependencies = Dependencies()) throws -> Self {
 | |
|         return try data.decoded(as: Self.self, using: dependencies)
 | |
|     }
 | |
| }
 | |
| 
 | |
| public extension AnyPublisher where Output == (ResponseInfoType, Data?), Failure == Error {
 | |
|     func decoded(
 | |
|         as types: HTTP.BatchResponseTypes,
 | |
|         requireAllResults: Bool = true,
 | |
|         using dependencies: Dependencies = Dependencies()
 | |
|     ) -> AnyPublisher<HTTP.BatchResponse, Error> {
 | |
|         self
 | |
|             .flatMap { responseInfo, maybeData -> AnyPublisher<HTTP.BatchResponse, Error> in
 | |
|                 // Need to split the data into an array of data so each item can be Decoded correctly
 | |
|                 guard let data: Data = maybeData else {
 | |
|                     return Fail(error: HTTPError.parsingFailed)
 | |
|                         .eraseToAnyPublisher()
 | |
|                 }
 | |
|                 guard let jsonObject: Any = try? JSONSerialization.jsonObject(with: data, options: [.fragmentsAllowed]) else {
 | |
|                     return Fail(error: HTTPError.parsingFailed)
 | |
|                         .eraseToAnyPublisher()
 | |
|                 }
 | |
|                 
 | |
|                 let dataArray: [Data]
 | |
|                 
 | |
|                 switch jsonObject {
 | |
|                     case let anyArray as [Any]:
 | |
|                         dataArray = anyArray.compactMap { try? JSONSerialization.data(withJSONObject: $0) }
 | |
|                         
 | |
|                         guard !requireAllResults || dataArray.count == types.count else {
 | |
|                             return Fail(error: HTTPError.parsingFailed)
 | |
|                                 .eraseToAnyPublisher()
 | |
|                         }
 | |
|                         
 | |
|                     case let anyDict as [String: Any]:
 | |
|                         guard
 | |
|                             let resultsArray: [Data] = (anyDict["results"] as? [Any])?
 | |
|                                 .compactMap({ try? JSONSerialization.data(withJSONObject: $0) }),
 | |
|                             (
 | |
|                                 !requireAllResults ||
 | |
|                                 resultsArray.count == types.count
 | |
|                             )
 | |
|                         else {
 | |
|                             return Fail(error: HTTPError.parsingFailed)
 | |
|                                 .eraseToAnyPublisher()
 | |
|                         }
 | |
|                         
 | |
|                         dataArray = resultsArray
 | |
|                         
 | |
|                     default:
 | |
|                         return Fail(error: HTTPError.parsingFailed)
 | |
|                             .eraseToAnyPublisher()
 | |
|                 }
 | |
|                 
 | |
|                 do {
 | |
|                     // TODO: Remove the 'Swift.'
 | |
|                     return Just(
 | |
|                         HTTP.BatchResponse(
 | |
|                             info: responseInfo,
 | |
|                             responses: try Swift.zip(dataArray, types)
 | |
|                                 .map { data, type in try type.decoded(from: data, using: dependencies) }
 | |
|                         )
 | |
|                     )
 | |
|                     .setFailureType(to: Error.self)
 | |
|                     .eraseToAnyPublisher()
 | |
|                 }
 | |
|                 catch {
 | |
|                     return Fail(error: HTTPError.parsingFailed)
 | |
|                         .eraseToAnyPublisher()
 | |
|                 }
 | |
|             }
 | |
|             .eraseToAnyPublisher()
 | |
|     }
 | |
| }
 |