Vapor 3:返回模型时,如何轻松返回子对象

问题描述 投票:3回答:1

我有一个模型广告系列,有多个月:

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”处理子对象或其他计算对象的方法是什么?现在它似乎归结为你的模型的许多不同版本(用于创建,返回,实际的完整内部版本)然后在它们之间进行转换,但我希望我在这里缺少一些东西?因为很容易忘记将新添加的模型属性添加到该特殊公共模型中。

或者至少,如果确实以推荐的方式在模型之间进行转换,有没有办法在路由处理程序中不需要所有这些嵌套的地图/平面图?

swift vapor vapor-fluent
1个回答
3
投票

您可以使用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)
}
© www.soinside.com 2019 - 2024. All rights reserved.