我有一个模型广告系列,有多个月:
final class Campaign: Content, SQLiteModel {
var id: Int?
var name: String
var months: Children<Campaign, Month> {
return children(\.campaignID)
}
}
当我想以最基本的方式返回一个Campaign时,它不包括Months,因为我理解计算属性不是Codable。
func getOneHandler(_ req: Request) throws -> Future<Campaign> {
return try req.parameters.next(Campaign.self)
}
所以,我创建了一个新结构来保存我想要返回的完整对象
struct FullCampaignData: Content {
let id: Int
let name: String
var months: [Month]?
}
然后修改我的路由处理程序,如下所示:
func getOneHandler(_ req: Request) throws -> Future<FullCampaignData> {
return try req.parameters.next(Campaign.self).flatMap(to: FullCampaignData.self) { campaign in
return try campaign.months.query(on: req).all().map(to: FullCampaignData.self) { months in
return try FullCampaignData(id: campaign.requireID(), name: campaign.name, months: months)
}
}
}
这确实有效。但是,这样做似乎需要做很多工作和很多样板。通常的Swifty“Vapory”处理子对象或其他计算对象的方法是什么?现在它似乎归结为你的模型的许多不同版本(用于创建,返回,实际的完整内部版本)然后在它们之间进行转换,但我希望我在这里缺少一些东西?因为很容易忘记将新添加的模型属性添加到该特殊公共模型中。
或者至少,如果确实以推荐的方式在模型之间进行转换,有没有办法在路由处理程序中不需要所有这些嵌套的地图/平面图?
您可以使用qazxsw poi库进行复杂查询
我不确定使用SQLite可以子查询SwifQL,但是使用Months
它很容易因为它支持PostgreSQL
所以对于PostgreSQL,您的查询可能看起来像
JSON
要么
func getOneHandler(_ req: Request) throws -> Future<FullCampaignData> {
let monthsSubquery = SwifQL.select(Fn.array_agg(Fn.to_jsonb(Month.table)))
let query = SwifQL.select(Campaign.table.*, |monthsSubquery| => "months")
.from(Campaign.table)
.join(.leftOuter, Month.table, on: \Month.campaignID == \Campaign.id)
.execute(on: req, as: .psql)
.all(decoding: FullCampaignData.self)
}