我有一个模型广告系列,该模型有多个月:
final class Campaign: Content, SQLiteModel {
var id: Int?
var name: String
var months: Children<Campaign, Month> {
return children(\.campaignID)
}
}
当我想以最基本的方式退还广告系列时,它不包含月份,因为据我所知,计算属性不是可编码的。
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”方法是什么?现在,它似乎可以归结为一堆模型的不同版本(用于创建,用于返回,用于实际的完整内部模型),然后在它们之间进行转换,但是我希望这里缺少某些内容吗?因为很容易忘记将一个新添加的模型属性也添加到该特殊的公共模型中。
或者至少,如果确实以推荐的方式在模型之间进行转换,那么是否有一种方法不需要在路由处理程序中使用所有这些嵌套的地图/平面地图?
答案 0 :(得分:3)
您可以使用SwifQL lib进行复杂的查询
我不确定使用SQLite可以对Months
进行子查询,但是使用PostgreSQL
确实很容易,因为它支持JSON
因此对于PostgreSQL,您的查询可能类似于
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)
}
或
func getOneHandler(_ req: Request) throws -> Future<FullCampaignData> {
let monthsSubquery = SwifQL.select(Fn.array_agg(Fn.to_jsonb(Month.table)))
.from(Month.table)
.where(\Month.campaignID == \Campaign.id)
let query = SwifQL.select(Campaign.table.*, |monthsSubquery| => "months")
.from(Campaign.table)
.execute(on: req, as: .psql)
.all(decoding: FullCampaignData.self)
}