В то время как цикл увеличивается и перезапускается до того, как Promisekit может завершить работу и вернуть правильное значение в Swift

Я пытаюсь использовать цикл while с Promisekit с Alamofire для объединения четырех GET-запросов, возврата значения и повторного запуска четырех запросов с новым параметром. Это текущий код, который я использую:

        var index = 0
        var count = classDictionary["class"]!.count-1

        while index <= count {

        firstly {
            parseBooksXML(index)
            }.then { abrevCode in
                self.parseBooksXML2(index, key: abrevCode)
            }.then { courseNumber in
                self.parseBooksXML3(index, key: courseNumber)
            }.then { instructorCode in
                self.parseBooksXML4(instructorCode)
            }
            index += 1
        }

Каждая из первых трех функций возвращает обещанное строковое значение, которое затем передается следующей функции, пока четвертая и последняя функция не вызовет другую функцию для анализа загруженного HTML-кода с использованием Kanna.

В идеале я хотел бы, чтобы все четыре функции были вызваны и завершены, после чего индекс будет увеличиваться и снова запускать цикл, используя новый индексный номер. Как примечание, индекс передается функциям как способ определить, с каким индексом в массиве следует сравнивать значение.

Для ясности, я включил код для функций parseBooksXML ниже:

func parseBooksXML(index: Int) -> Promise<String> {
    return Promise {fulfill, reject in

    let headers = [
        "Referer": "URL"
    ]


    Alamofire.request(.GET, "URL", headers: headers)
        .responseData { response in

            switch response.result {
            case .Success:
                let xml = SWXMLHash.parse(response.data!)

                do {

                    let range = self.classDictionary["class"]![index].rangeOfString("[a-zA-Z]{2,4}", options: .RegularExpressionSearch)
                    let result = self.classDictionary["class"]![index].substringWithRange(range!)

                    try self.abrevCode = (xml["departments"]["department"].withAttr("abrev", result).element!.attribute(by: "id")!.text)

                }
                catch {
                    print("Error: \(error)")

                }
                fulfill(self.abrevCode)

            case .Failure(let error):
                print(error)
            }
        }
    }
}

func parseBooksXML2(index: Int, key: String) -> Promise<String> {
    return Promise {fulfill, reject in

        let headers = [
        "Referer": "URL"
    ]


    Alamofire.request(.GET, "URL", headers: headers)
        .responseData { response in

            switch response.result {
            case .Success:
                let xml = SWXMLHash.parse(response.data!)
                 do {

                    let range = self.classDictionary["class"]![index].rangeOfString("\\d\\d\\d", options: .RegularExpressionSearch)
                    let result = self.classDictionary["class"]![index].substringWithRange(range!)

                    try self.courseNumber = (xml["courses"]["course"].withAttr("name", result).element?.attribute(by: "id")?.text)!

                 }
                 catch {
                    print("Error: \(error)")

                }
                fulfill(self.courseNumber)

            case .Failure(let error):
                print(error)
            }

        }
    }
}

func parseBooksXML3(index: Int, key: String) -> Promise<String> {
    return Promise {fulfill, reject in

        let headers = [
        "Referer": "URL"
    ]

    Alamofire.request(.GET, "URL", headers: headers)
        .responseData { response in

            switch response.result {
            case .Success:
                let xml = SWXMLHash.parse(response.data!)
                do {

                    let range = self.classDictionary["class"]![index].rangeOfString("[a-zA-Z]{1,3}?\\d?\\d?\\d?$", options: .RegularExpressionSearch)
                    let result = self.classDictionary["class"]![index].substringWithRange(range!)

                    try self.instructorCode = (xml["sections"]["section"].withAttr("instructor", self.classTeacher[index]).element?.attribute(by: "id")?.text)!
                }
                catch {
                    print("Error: \(error)")


                }
                fulfill(self.instructorCode)

            case .Failure(let error):
                print(error)
            }

        }
    }
}

func parseBooksXML4(key: String) -> Void {

        let headers = [
        "Referer": "URL"
    ]

    Alamofire.request(.GET, "URL", headers: headers)
        .responseData { response in

            switch response.result {
            case .Success:
                self.parseISBN(String(data: response.data!, encoding: NSUTF8StringEncoding)!)

            case .Failure(let error):
                print(error)
            }

        }
    }

Любая помощь будет оценена!

2 ответа

Вам нужно использовать when:

let count = classDictionary["class"]!.count-1

let promises = (0..<count).map { index -> Promise<ReplaceMe> in
    return firstly {
        parseBooksXML(index)
    }.then { abrevCode in
        self.parseBooksXML2(index, key: abrevCode)
    }.then { courseNumber in
        self.parseBooksXML3(index, key: courseNumber)
    }.then { instructorCode in
        self.parseBooksXML4(instructorCode)
    }
}

when(fulfilled: promises).then {
    //…
}

Поскольку вызов parseBooksXML4 является асинхронным, вы должны обернуть вызов parseBooksXML4(), чтобы вернуть обещание, и дождаться его завершения, прежде чем увеличивать индекс.

    firstly {
        parseBooksXML(index)
        }.then { abrevCode in
            self.parseBooksXML2(index, key: abrevCode)
        }.then { courseNumber in
            self.parseBooksXML3(index, key: courseNumber)
        }.then { instructorCode in
            self.parseBooksXML4(instructorCode)
        }.then { _ in
            index += 1
        }
    }
Другие вопросы по тегам