Swift Siesta редактировать извлеченную сущность

Я создаю клиент API с использованием Siesta и Swift 3 на Xcode 8. Я хочу иметь возможность извлекать сущность с использованием ресурса Siesta, затем обновлять некоторые данные и делать patch к API.

Проблема в том, что при наличии сущности, если я сохраняю массивы JSON в полях сущности и не могу отправить их обратно на сервер, я получаю следующую ошибку:

▿ Siesta.RequestError
  - userMessage: "Cannot send request"
  - httpStatusCode: nil
  - entity: nil
  ▿ cause: Optional(Siesta.RequestError.Cause.InvalidJSONObject())
    - some: Siesta.RequestError.Cause.InvalidJSONObject
  - timestamp: 502652734.40489101

Моя сущность это:

import SwiftyJSON
import Foundation
struct Order {
    let id: String?
    let sessionId: String?
    let userId: Int?
    let status: String?
    let comment: String?
    let price: Float?
    let products: Array<JSON>? 

    init(json: JSON) throws {
        id          = json["id"].string
        sessionId      = json["sessionId"].string
        userId      = json["userId"].int
        status        = json["status"].string
        comment        = json["comment"].string
        price       = json["price"].float
        products   = json["products"].arrayValue

    }

    /**
     * Helper method to return data as a Dictionary to be able to modify it and do a patch
     **/
    public func toDictionary() -> Dictionary<String, Any> {
        var dictionary: [String:Any] = [
            "id": id ?? "",
            "sessionId": sessionId ?? "",
            "userId": userId ?? 0,
            "status": status ?? "",
            "comment": comment ?? ""
        ]
        dictionary["products"] = products ?? []

        return dictionary
    }
}

Что я делаю, это:

 MyAPI.sessionOrders(sessionId: sessionId).request(.post, json: ["products": [["product": productId, "amount": 2]], "comment": "get Swifty"]).onSuccess() { response in
    let createdObject : Order? =  response.typedContent()
    expect(createdObject?.sessionId).to(equal(sessionId))
    expect(createdObject?.comment).to(equal("get Swifty"))
    expect(createdObject?.products).to(haveCount(1))
    expect(createdObject?.price).to(equal(product.price! * 2))

    if let createdId = createdObject?.id {
        var data = createdObject?.toDictionary()
        data?["comment"] = "edited Swifty" // can set paid because the user is the business owner
        MyAPI.order(id: createdId).request(.patch, json: data!).onSuccess() { response in
            result = true

        }.onFailure() { response in
                dump(response) //error is here
        }
    }
}

Ресурсы:

func sessionOrders( sessionId: String ) -> Resource {
    return self
        .resource("/sessions")
        .child(sessionId)
        .child("orders")
}
func order( id: String ) -> Resource {
    return self
        .resource("/orders")
        .child(id)
}

Трансформаторы:

    self.configureTransformer("/sessions/*/orders", requestMethods: [.post, .put]) {
        try Order(json: ($0.content as JSON)["data"])
    }

    self.configureTransformer("/orders/*") {
        try Order(json: ($0.content as JSON)["data"])
    }

Мне удалось обвести это, создав словарные структуры, такие как:

let products: Array<Dictionary<String, Any>>?

    products   = json["products"].arrayValue.map({
        ["product": $0.dictionaryValue["product"]!.stringValue, "amount": $0.dictionaryValue["amount"]!.intValue]
    })

Но я живу в адских уныниях, если мне нужно что-то изменить:

var data = createdObject?.toDictionary()
data?["comment"] = "edited Swifty" 
//if I want to modify the products...
var products = data?["products"] as! Array<Dictionary<String, Any>>
products[0]["amount"] = 4
data?["products"] = products

Как я могу отправить эти оригинальные массивы JSON с Siesta? Их действительно легко модифицировать и читать! Я просматривал документы по сиесте и github безуспешно...

1 ответ

Решение

Ваша проблема - несоответствие между SwiftyJSON и JSONSerialization Фонда; Сиеста просто оказывается в середине этого.

InvalidJSONObject Сиеста говорит вам, что Фонд не понимает того, что вы ему дали, что будет значением, возвращаемым вашим toDictionary() метод. Большинство вещей в этом словаре выглядят хорошо: строки, целые числа, число с плавающей запятой. (Осторожно при использовании поплавка для денег, кстати.)

Виновник в том, что products массив: это [JSON], где JSON тип SwiftyJSON, с которым Foundation не знает, что делать Вы должны быть в открытом виде, если вы включите JSON значения возвращаются в простые словари:

dictionary["products"] = (products ?? []).map { $0.dictionaryObject }

Если этого не произойдет, или если вам потребуется диагностировать аналогичную ошибку в будущем, удалите все значения из словаря-нарушителя, а затем добавьте их обратно по одному за раз, чтобы увидеть, какое из них вызывает JSONSerialization.

Другие вопросы по тегам