Objectmapper: `Uso ambíguo de mapa` ao usar ImmutableMappable.

Criado em 15 fev. 2017  ·  4Comentários  ·  Fonte: tristanhimmelman/ObjectMapper

@devxoul Oi! Estou tentando usar o protocolo ImmutableMappable , mas não há explicação no README de como usar junto com objetos aninhados.

Eu quero tentar algo simples como isto:

struct Father: ImmutableMappable {
    let name: String
    let children: [Child]

    init(map: Map) throws {
        name = try map.value("name")
        children = try map.value("children", using: ArrayTransform<Child>())
    }
}

struct Child: ImmutableMappable {
    let age: Int

    init(map: Map) throws {
        age = try map.value("age")
    }
}

Portanto, não há Transform trabalhando com matrizes? Então, escrevi o transformador abaixo inspirado em ListTransformer de 'ObjectMapper-Realm'

struct ArrayTransform<T: ImmutableMappable> {}
extension ArrayTransform: TransformType {
    typealias Object = [T]
    typealias JSON = [Any]


    func transformFromJSON(_ value: Any?) -> Object? {
        if let objects = Mapper<T>().mapArray(JSONObject: value) {
            let list = Object()
            list.append(objectsIn: objects)
            return list
        }
        return nil
    }

    public func transformToJSON(_ value: Object?) -> JSON? {
        return value?.flatMap { $0.toJSON() }
    }
}

Resultados em erro de compilação: Ambigious use of 'mapArray(JSONObject:)'

Minha teoria é que é porque o compilador encontra o método nestes dois escopos:
public extension Mapper where N: ImmutableMappable
public final class Mapper<N: BaseMappable> {

Mas como ImmutableMappable herda de BaseMappable , ambas as funções são candidatas válidas.

Como usar ImmutableMappable para objetos aninhados?

E também, eu tenho que escrever um Transformer para isso?

Comentários muito úteis

@Sajjon Btw, AlamofireObjectMapper agora suporta ImmutableMappable na v4.1

Todos 4 comentários

Você não precisa usar ArrayTransform. Basta fazer o seguinte:

init(map: Map) throws {
  // ...
  children = try map.value("children")
}
PS: A razão da falha de construção é porque você não marcou `mapArray ()` com `try`.
 let objects = try Mapper() .mapArray (JSONObject: valor)
 // faça alguma coisa

@devxoul Ei, obrigado pela sua resposta! Hehe sim, isso foi mais fácil do que eu pensava! 😅

Em relação à sua resposta P.S. - Ainda não consigo fazer funcionar! Agora estou tentando implementar uma extensão Alamofire + ObjectMapper analogamente ao AlamofireObjectMapper , mas por ImmutableMappable

Ainda recebo o erro de compilação Ambigious use of map.... , procure os dois comentários "// Compilation Error" no código abaixo.

O que estou fazendo de errado?

import Foundation
import Alamofire
import ObjectMapper

extension DataRequest {

    enum ErrorCode: Int {
        case noData = 1
        case dataSerializationFailed = 2
    }

    internal static func newError(_ code: ErrorCode, failureReason: String) -> NSError {
        let errorDomain = "com.alamofireobjectmapper.error"

        let userInfo = [NSLocalizedFailureReasonErrorKey: failureReason]
        let returnError = NSError(domain: errorDomain, code: code.rawValue, userInfo: userInfo)

        return returnError
    }

    public static func ObjectMapperSerializer<T: ImmutableMappable>(_ keyPath: String?, mapToObject object: T? = nil, context: MapContext? = nil) -> DataResponseSerializer<T> {
        return DataResponseSerializer { request, response, data, error in
            guard error == nil else {
                return .failure(error!)
            }

            guard let _ = data else {
                let failureReason = "Data could not be serialized. Input data was nil."
                let error = newError(.noData, failureReason: failureReason)
                return .failure(error)
            }

            let jsonResponseSerializer = DataRequest.jsonResponseSerializer(options: .allowFragments)
            let result = jsonResponseSerializer.serializeResponse(request, response, data, error)

            let JSONToMap: Any?
            if let keyPath = keyPath , keyPath.isEmpty == false {
                JSONToMap = (result.value as AnyObject?)?.value(forKeyPath: keyPath)
            } else {
                JSONToMap = result.value
            }

            if let object = object {
                _ = Mapper<T>().map(JSONObject: JSONToMap, toObject: object)
                return .success(object)
            } else {
                do {
                    // Compilation Error: "Amigious user of 'map(JSONObject:)'"
                    let parsedObject = try Mapper<T>(context: context).map(JSONObject: JSONToMap)
                    return .success(parsedObject)
                } catch let error {
                    fatalError("Mapping error: \(error)")
                }
            }

            let failureReason = "ObjectMapper failed to serialize response."
            let error = newError(.dataSerializationFailed, failureReason: failureReason)
            return .failure(error)
        }
    }

    <strong i="16">@discardableResult</strong>
    public func responseObject<T: ImmutableMappable>(queue: DispatchQueue? = nil, keyPath: String? = nil, mapToObject object: T? = nil, context: MapContext? = nil, completionHandler: <strong i="17">@escaping</strong> (DataResponse<T>) -> Void) -> Self {
        return response(queue: queue, responseSerializer: DataRequest.ObjectMapperSerializer(keyPath, mapToObject: object, context: context), completionHandler: completionHandler)
    }

    public static func ObjectMapperArraySerializer<T: ImmutableMappable>(_ keyPath: String?, context: MapContext? = nil) -> DataResponseSerializer<[T]> {
        return DataResponseSerializer { request, response, data, error in
            guard error == nil else {
                return .failure(error!)
            }

            guard let _ = data else {
                let failureReason = "Data could not be serialized. Input data was nil."
                let error = newError(.dataSerializationFailed, failureReason: failureReason)
                return .failure(error)
            }

            let jsonResponseSerializer = DataRequest.jsonResponseSerializer(options: .allowFragments)
            let result = jsonResponseSerializer.serializeResponse(request, response, data, error)

            let JSONToMap: Any?
            if let keyPath = keyPath, keyPath.isEmpty == false {
                JSONToMap = (result.value as AnyObject?)?.value(forKeyPath: keyPath)
            } else {
                JSONToMap = result.value
            }

            do {
                // Compilation Error: "Amigious user of 'map(JSONObject:)'"
                let parsedObject = try Mapper<T>(context: context).mapArray(JSONObject: JSONToMap)
                return .success(parsedObject)
            } catch let error {
                fatalError("Failed to map, error: \(error)")
            }

            let failureReason = "ObjectMapper failed to serialize response."
            let error = newError(.dataSerializationFailed, failureReason: failureReason)
            return .failure(error)
        }
    }

    <strong i="18">@discardableResult</strong>
    public func responseArray<T: ImmutableMappable>(queue: DispatchQueue? = nil, keyPath: String? = nil, context: MapContext? = nil, completionHandler: <strong i="19">@escaping</strong> (DataResponse<[T]>) -> Void) -> Self {
        return response(queue: queue, responseSerializer: DataRequest.ObjectMapperArraySerializer(keyPath, context: context), completionHandler: completionHandler)
    }
}

@Sajjon Btw, AlamofireObjectMapper agora suporta ImmutableMappable na v4.1

Fechar isso como a nova versão do AlamofireObjectMapper deve cuidar do problema. Sinta-se à vontade para comentar e posso reabrir o tíquete, se necessário.

Esta página foi útil?
0 / 5 - 0 avaliações

Questões relacionadas

quetool picture quetool  ·  3Comentários

patchthecode picture patchthecode  ·  3Comentários

maksTheAwesome picture maksTheAwesome  ·  4Comentários

nearspears picture nearspears  ·  4Comentários

pcompassion picture pcompassion  ·  3Comentários