Я пытаюсь реализовать цепочку методов для успешных и неудачных вызовов в моем коде, но у меня, похоже, возникают проблемы с получением фактически вызванных методов onSuccess
.Swift Method Chaining with Success & Failure
- Контроллер вида вызывает функцию
getProduct(_:)
. getProduct(_:)
делает вызов API, а затем вызываетstoreProduct(_:)
с извлеченной JSON
storeProduct(_:)
звонкиfetchProduct(_:)
fetchProduct(_:)
doSuccess(_:)
звонки, но это никогда не возвращается вonSuccess
предыдущих вызовов.
Некоторые фрагменты кода
BSProductChainable.swift
import Foundation
class BSProductChainable<SuccessParams, FailureParams> {
var successClosure: ((SuccessParams) ->())? = nil
var failureClosure: ((FailureParams) ->())? = nil
func onSuccess(closure: (SuccessParams) ->()) -> BSProductChainable {
successClosure = closure
return self
}
func onFailure(closure: (FailureParams) ->()) -> BSProductChainable {
failureClosure = closure
return self
}
func doSuccess(params: SuccessParams) {
if let closure = successClosure {
closure(params)
}
}
func doFailure(params: FailureParams) {
if let closure = failureClosure {
closure(params)
}
}
}
BSProductManagerSwift.swift
class BSProductManagerSwift: NSObject {
typealias productResponseChain = BSProductChainable<Product, NSError?>
typealias productsResponseChain = BSProductChainable<[Product], NSError?>
var serviceClient: BSNetworkingServiceClient!
var objectContext: NSManagedObjectContext!
var productChains: BSProductChainable<Product, NSError?>!
var productsChains: BSProductChainable<[Product], NSError?>!
convenience init(serviceClient: BSNetworkingServiceClient) {
self.init()
self.serviceClient = serviceClient
self.objectContext = managedObjectContext
self.productChains = BSProductChainable<Product, NSError?>()
self.productsChains = BSProductChainable<[Product], NSError?>()
}
func getProduct(ean: String) -> productResponseChain {
let urlString = BSConstants.BarcodeScanner.productEndpoint.stringByAppendingString(ean)
serviceClient.GET(urlString, failure: { (error) in
print("Could not get product")
}) { (response) in
if let json = response {
self.storeProduct(json).onSuccess({ (returedProduct) in
print("Stored product")
})
}
}
return productChains
}
func storeProduct(json: JSON) -> productResponseChain {
fetchProduct(json["ean"].stringValue).onSuccess { (returedProduct) in
self.productChains.doSuccess(returedProduct)
}
return productChains
}
func fetchProduct(ean: String) -> productResponseChain {
let fetchRequest = NSFetchRequest(entityName: "Product")
let predicateEAN = NSPredicate(format: "%K == %@", "ean", ean)
let predicateMarket = NSPredicate(format: "%K == %@", "market", BSCountryManager.sharedInstance().getCurrentCountry().market)
let predicateLocale = NSPredicate(format: "%K == %@", "locale", BSLocalizationManager.sharedManager().currentLocalization.localeIdentifier())
let predicateCurrency = NSPredicate(format: "%K == %@", "currency", BSLocalizationManager.sharedManager().currentLocalization.country.currencyIdentifierDMW)
let compoundPredicate = NSCompoundPredicate(andPredicateWithSubpredicates: [predicateEAN, predicateMarket, predicateLocale, predicateCurrency])
fetchRequest.predicate = compoundPredicate
do {
let matchingProuducts = try objectContext.executeFetchRequest(fetchRequest)
if matchingProuducts.count == 0 {
print("No matching products found")
let entity = NSEntityDescription.entityForName("Product", inManagedObjectContext: objectContext)
productChains.doSuccess(Product(entity: entity!, insertIntoManagedObjectContext: objectContext))
} else {
print("Found matching product")
let d = matchingProuducts.first as! Product
productChains.doSuccess(d)
}
} catch let error as NSError {
print("Could not fetch \(error), \(error.userInfo)")
productChains.doFailure(error)
}
return productChains
}
Я изначально INITIALIS ed chainable для каждой функции, но у этого были свои проблемы, из которых я думал (возможно, неправильно), что я должен только инициализировать classable класс один раз и передать его ссылку.
Некоторые сведения о том, куда я иду не так/что я мог бы попробовать, было бы здорово.
Почему бы не использовать фреймворк вместо того, чтобы создавать свои собственные? Взгляните на перспективу; https://github.com/mxcl/PromiseKit –
Это может быть так, как я иду, поскольку я уже потратил слишком много времени на попытку решить эту проблему. – Hodson