Vapor 4:如何将急切加载的父关系映射到不同的格式?

问题描述

我在如何返回一个包含父关系的模型,同时将那个急切加载的模型映射到不同的形式上有点挣扎。

让我们考虑以下 2 个模型:CourseUser

final class Course: Model,Content {
  static let schema = "courses"

  @ID(key: .id)
  var id: UUID?

  @Field(key: "name")
  var name: String

  @Parent(key: "teacher_id")
  var teacher: User

  init() { }
}

final class User: Model,Content {
  static let schema = "users"

  @ID(key: .id)
  var id: UUID?

  @OptionalField(key: "avatar")
  var avatar: String?

  @Field(key: "name")
  var name: String

  @Field(key: "private")
  var somePrivateField: String

  init() { }
}

我有一条这样的路线,它返回一系列课程:

func list(req: Request) throws -> EventLoopFuture<[Course]> {
  return Course
    .query(on: req.db)
    .all()
}

生成的 JSON 如下所示:

[
  {
    "id": 1,"name": "Course 1","teacher": {
      "id": 1
    }
]

我想要的是返回教师对象,这很容易通过将 .with(\.$teacher) 添加查询中。 Vapor 4 确实让这一切变得非常简单!

[
  {
    "id": 1,"teacher": {
      "id": 1,"name": "User 1","avatar": "https://www.example.com/avatar.jpg","somePrivateField": "super secret internal info"
    }
]

这是我的问题:返回整个 User 对象,包括所有字段,甚至是我不想公开的字段。

将教师信息转换为不同版本的 User 模型(例如 PublicUser)的最简单方法是什么?这是否意味着我必须为 Course 创建一个 DTO,将我的数组从 [Course] 映射到 [PublicCourse],复制所有属性,并在 Course 模型更改时保持它们同步等?

这似乎是很多样板文件,未来有很多出错的空间。很想知道是否有更好的选择。

解决方法

您可以通过首先对原始模型进行编码,然后将其解码为具有较少字段的结构来实现此目的。因此,对于存储在 Course 中的 course 实例要转换为 PublicCourse,您应该这样做:

struct PublicCourse: Decodable {
    //...
    let teacher: PublicUser
    //...
}

let course:Course = // result of Course.query including `with(\.$teacher)`
let encoder = JSONEncoder()
let decoder = JSONDecoder()
let data = try encoder.encode(course)
let publicCourse = try decoder.decode(PublicCourse.self,from: data)

注意结构中的 PublicUser 字段。如果这是精简版,您可以一次性生成最小的 JSON。

,

好的,这个方法怎么样?创建第二个 Model,称为 Teacher,比方说,它被定义为 User 中您想要在 API/JSON 中公开的字段子集,并且具有相同的架构/表名称如User

final class Teacher: Model,Content {
  static let schema = "users"

  // public fields
}

然后将您在 Course 中的关系更改为:

@Parent(key: "teacher_id")
var teacher: Teacher

您的原始查询将保持不变,但只会返回减少的字段集。如果您使用 Teacher 只读,它肯定会起作用。由于基础表存在,因此无需为 Migrations 创建 Teacher。我找不到避免包含 ID 的方法,但这可能不是问题。

相关问答

Selenium Web驱动程序和Java。元素在(x,y)点处不可单击。其...
Python-如何使用点“。” 访问字典成员?
Java 字符串是不可变的。到底是什么意思?
Java中的“ final”关键字如何工作?(我仍然可以修改对象。...
“loop:”在Java代码中。这是什么,为什么要编译?
java.lang.ClassNotFoundException:sun.jdbc.odbc.JdbcOdbc...