目标是使用 NestJS 使用代码优先方法实现 GraphQL 模式。
假设我的 GraphQL 模式中有一个
Pet
类型,它有两个字段,name
和 age
。如果这两条信息来自不同的真实来源(我并不总是想同时获取两者),我可以实现一个 PetResolver
类,每个字段都有解析器:
@Resolver(() => Pet)
export class PetResolver {
@Query(() => Pet)
pet(): Pet {
return {};
}
@ResolveField()
name(): Promise<string> {
return Promise.resolve('Spot');
}
@ResolveField(() => Int)
age(): Promise<number> {
return Promise.resolve(2);
}
}
可以这样使用:
query GetPet {
pet {
name
}
}
这有效,并确保每个字段的值仅在请求时获取,但是如果我想在我的
pet
类型上有一个 User
字段,我可以像这样查询:
query GetUserWithPet {
currentUser {
email
pet {
name
}
}
}
应用相同的原则,我可以创建一个 UserResolver 类,如下所示:
@Resolver(() => User)
export class UserResolver {
@Query(() => User)
@UseGuards(AuthCognitoGuard)
currentUser(@CurrentUser() user: IdTokenPayload): User {
return {
id: user.sub,
email: user.email,
};
}
@ResolveField()
pet(@Parent() user: User): Promise<Pet> {
return petService.getPetForUserId(user.id);
}
}
但是
PetService
实现如果只想获取相关数据就必须知道请求了哪些字段。
A) 有没有办法在
PetResolver
中使用 UserResolver
来利用单独的字段解析逻辑?
B) 如果不是,使用 NestJS 代码优先约定确定查询中请求了哪些字段的最佳方法是什么?
C) 这是考虑 GraphQL 查询的“错误”方式吗?最佳实践是否要求我保留单独的解析器并使用这样的查询:
query GetUserWithPet {
currentUser {
email
}
pet {
name
}
}
这是与您的问题相关的官方文档的链接 https://docs.nestjs.com/graphql/resolvers#code-first-resolver
附言很抱歉发布了 necropost,但它应该可以帮助其他用户搜索答案)
用户应该包含一些
petIds
[数组]值(内部,数据库存储字段/列)...
... 使解决
pets: [Pet]
道具/关系成为可能 - Pet
列表 ...
...就像
starshipIDs
在https://graphql.org/learn/execution/中解释的那样
注意:宠物服务询问使用宠物id的记录。
...但当然 pet 可以包含一些
ownerId
(仅或明确可见,数据库存储的字段/列)使得解析 owner: User
prop [反向] 关系成为可能 - 这样你就可以:
query PetWithOwner {
pet (id: "someID") {
id
name
owner {
id
email
# more pets?
pets {
id
name
# you can loop it ;)
owner {
id
email
pet.owner
字段解析器只能返回 { id: ownerId }
对象(部分响应)...服务器将尝试使用 email
解析“缺失”(查询需要)User
prop(owner
是 User
类型) 类型解析器,将 id 作为参数传递(check/console.log parent
和 args
解析器参数)。您不必在 pet.owner
字段解析器中“手动”执行此操作。
... [选择集] 可以从
info
对象中读取 - 第 4 个解析器 arg - 阅读文档/教程以获取详细信息