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.