简体   繁体   中英

How to flatten multiple queries in Vapor

I need to do lots of queries to different tables and return the results in one single response. I'm wondering ow could I flatten the extremely nested function?

final class CodeController {
    internal func indexCodes(_ req: Request) throws -> EventLoopFuture<Response> {
        CodeCountry.query(on: req).sort(\.name).all().flatMap { countries -> EventLoopFuture<Response> in
            CodeFloor.query(on: req).sort(\.name).all().flatMap { floors -> EventLoopFuture<Response> in
                CodeRegion.query(on: req).sort(\.name).all().flatMap { regions -> EventLoopFuture<Response> in
                    CodeObjectType.query(on: req).sort(\.name).all().flatMap { objectTypes -> EventLoopFuture<Response> in
                        CodePropertyType.query(on: req).sort(\.name).all().flatMap { propertyTypes -> EventLoopFuture<Response> in
                            CodeRooms.query(on: req).sort(\.name).all().flatMap { rooms -> EventLoopFuture<Response> in
                                let codes = CodesContent(
                                    countries: countries,
                                    floors: floors,
                                    regions: regions,
                                    objectTypes: objectTypes,
                                    propertyTypes: propertyTypes,
                                    rooms: rooms
                                )

                                return codes.encode(status: .ok, for: req)
                            }
                        }
                    }
                }
            }
        }
    }
}

This might not be the best looking code but I think solves the problem. I liked @Nick's approach with introducing a new data model as a collection, so will do similar. You can merge two futures and get one. This comes from SwiftNIO

let futureThingOne: Future<ThingOne> = ...
let futureThingTwo: Future<ThingTwo> = ...
let futureBothThings: Future<(ThingOne, ThingTwo)> = futureThingOne.and(futureThingTwo)

So by merging the futures coming from queries you can construct a flattened structure.

struct Codes: Content {
    let countries: [CodeCountry]
    let floors: [CodeFloor]
    let regions: [CodeRegion]
    let objectTypes: [CodeObjectType]
    let propertyTypes: [CodePropertyType]
    let rooms: [CodeRooms]

    static func codes(req: Request) -> Future<Codes> {
        let futureCountries = CodeCountry.query(on: req).sort(\.name).all()
        let futureFloors = CodeFloor.query(on: req).sort(\.name).all()
        let futureRegions = CodeRegion.query(on: req).sort(\.name).all()
        let futureObjectTypes = CodeObjectType.query(on: req).sort(\.name).all()
        let futurePropertyTypes = CodePropertyType.query(on: req).sort(\.name).all()
        let futureRooms = CodeRooms.query(on: req).sort(\.name).all()

        let combined = futureCountries.and(futureFloors).and(futureRegions).and(futureObjectTypes).and(futurePropertyTypes).and(futureRooms)
       return combined.map {
            Codes.init(
                countries: $0.0.0.0.0.0,
                floors: $0.0.0.0.0.1,
                regions: $0.0.0.0.1,
                objectTypes: $0.0.0.1,
                propertyTypes: $0.0.1,
                rooms: $0.1
            )
        }

    }
}

The technical post webpages of this site follow the CC BY-SA 4.0 protocol. If you need to reprint, please indicate the site URL or the original address.Any question please contact:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM