|
|
- f" z: z% ] o3 ~- a/ h' U/ ` o1 z9 l- Z<h2 id="系列导航">系列导航</h2>
# r# U/ a% z' T( ^, d<p><a href="https://www.cnblogs.com/code4nothing/p/graphql-net6-0.html">使用Hot Chocolate和.NET 6构建GraphQL应用文章索引</a></p>
- S+ A/ g: Z: Q! y6 ?<h2 id="需求">需求</h2>
! ^- p2 h7 ~7 _! f& f5 {+ {<p>在讨论完GraphQL中的查询需求后,这篇文章我们将演示如何实现GraphQL中的数据添加任务。</p>
5 L9 F7 m. F) l3 t! a<h2 id="思路">思路</h2>
D! U' U2 |1 c7 I, H<p>在GraphQL中,对数据进行查询使用<code>query</code>,而对于修改数据则需要使用<code>mutation</code>,包括新增和修改数据。Hot Chocolate在使用Mutation的逻辑上和使用Query的基本一致,但是需要根据需要定义用于创建或者更新的数据对象,所以我们直接进行实现。</p>* b# _: W3 i6 u3 s' a
<h2 id="实现">实现</h2>% j; [' G0 h j. p, E5 W, p# l t o: C
<p>为了保持简单,我们先定义以下两个类型:</p>
* A* g, [! s' c! Y<pre><code class="language-c#">// 定义新增Post的参数
0 N b1 B2 ]4 s# M; X1 tpublic record AddPostInput(string Title, string Author);
* ~8 }$ C6 z8 M2 \ H& V: S6 `) S6 Y' O. h
// 定义新增Post的返回对象5 y+ [ M* X2 w- g% }
public record AddPostPayload(Post Post);$ X; K) q$ S% D
</code></pre>7 v# f/ q. Y3 C! o; \9 b( }
<p>新建<code>Mutation.cs</code>用来定义相关接口:</p>
1 G/ U+ L5 }& k/ i7 K<ul>
9 Z# d* b, b4 W9 \' z( [ ?<li><code>Mutation.cs</code></li>1 w2 o" k% N% a3 A$ D
</ul>9 b {# z! z- b- [) h( A9 P
<pre><code class="language-c#">namespace PostGraphi.Api.GraphQL;
$ v+ s3 b& ]. O+ H m+ e9 g
: w9 N( C/ R6 P# ]3 p) Dpublic class Mutation8 z) U; ~% s: B) G3 f3 {
{
9 C5 R+ T: t; V9 t& a public async Task<AddPostPayload> AddPostAsync(AddPostInput input, [Service] IRepository<Post> repository)
6 O/ [7 I# ~' ?& x {
! [1 `: a/ a7 u1 X7 _ return new AddPostPayload(await repository.AddAsync(new Post
$ m' B. T* P( L {
/ N. ?' C7 [8 R8 b Title = input.Title,' U! d4 m8 x. f0 h
Author = input.Author
& n0 `9 ^. H1 h# c })); _ X- y, x% G N
}* ~0 q" a9 U8 H7 g+ |# P4 w
}
# Z$ C* f% p% A% F, K7 N' j S% d& C
</code></pre>( z% T7 y# J$ f7 p
<p>最后在注入服务的地方进行配置:</p>+ ^- B; K- g: V$ Z( f
<ul>
: f" C' X0 q) \+ Q. q6 a" q' k# m ]<li><code>ProgramExtensions.cs</code></li>& _ U4 D# A; m* p
</ul>$ I3 x* J Z4 C
<pre><code class="language-c#">builder.Services2 h! ` z+ H4 B# k; `/ I: B: d
.AddGraphQLServer()+ `" d# b# T& j, E& h1 K8 f2 E
.SetPagingOptions(new PagingOptions
( B" {* t7 u6 e8 K+ [8 B" T ^ {
8 _- ^) g" L4 `; k! u# l3 H# B, ` MaxPageSize = 50,
. w' ]0 Y: {& p$ u" ^ I IncludeTotalCount = true8 H/ s7 ]% u6 L, `5 T# L
}): z7 w, E* B8 S) L! [$ T0 K2 s* r
.AddFiltering()0 D) r: g/ j0 R$ t
.AddProjections()
. `: x" Z$ ~+ N* r8 M# ?7 t .AddSorting()
. v$ F! d; P: |' B$ j3 g .AddQueryType<Query>()) l, @5 j! V0 K5 `$ u9 M
.AddMutationType<Mutation>()3 g6 e8 |& c( J3 ^- v
.AddMutationConventions(new MutationConventionOptions; q1 x( N0 @+ N- L* ]! K6 w
{
4 F0 U* K& b4 D5 C: C+ _ ApplyToAllMutations = true,+ j% x/ ?, u- B
InputArgumentName = "input",
E' t% |4 q) G1 ~( ~ InputTypeNamePattern = "{MutationName}Input",
4 `4 Y! G; |5 f9 ~. o PayloadTypeNamePattern = "{MutationName}Payload",$ y7 r4 I) p: z0 A" _
PayloadErrorTypeNamePattern = "{MutationName}Error",+ p/ w3 h# P$ I0 C! N
PayloadErrorsFieldName = "errors". b5 ?# U! @5 r
})! U( W: s0 Q" K" x# g: Y7 p; o
.AddType<PostType>();
! v' n6 e, t8 H/ L7 f, S2 a, D Y</code></pre># {7 ^7 w% v5 E5 R7 n
<p>这样就实现了新增Post的需求,下面我们来验证一下。</p>
: n/ e/ h+ H/ j' E/ t" m<h2 id="验证">验证</h2>
; h9 B6 p. h+ a% y<p>启动<code>Api</code>项目,调用接口:</p>
; J: O0 _, }0 O8 o' E" W) b<p><img src="https://img2022.cnblogs.com/blog/2487237/202202/2487237-20220211104544617-1400586374.png" ></p>
$ r7 u7 ^) s: Y1 [- i r<p>终端的日志输出如下:</p>
" W4 T" M( M V+ a1 f4 \9 d d<pre><code class="language-bash">[10:45:15 INF] Executed DbCommand (1ms) [Parameters=[@p0='?' (DbType = Guid), @p1='?', @p2='?' (Size = 13), @p3='?', @p4='?' (DbType = DateTime), @p5='?', @p6='?' (DbType = DateTime), @p7='?', @p8='?', @p9='?' (DbType = DateTime), @p10='?' (Size = 30)], CommandType='Text', CommandTimeout='30']$ P' v0 Z/ v: ^* K
INSERT INTO "Posts" ("Id", "Abstraction", "Author", "Content", "Created", "CreatedBy", "LastModified", "LastModifiedBy", "Link", "PublishedAt", "Title")" B, o% L( ] L8 u2 b
VALUES (@p0, @p1, @p2, @p3, @p4, @p5, @p6, @p7, @p8, @p9, @p10);
: K p1 X# C) ~5 _/ W7 ~. X[10:45:15 INF] Executed endpoint 'Hot Chocolate GraphQL Pipeline'4 l% `4 H9 B! C; {: N
</code></pre>
! J, |& }, v& A g6 k<p>可以看到新建的Post已经存储到数据库中了,我们可以通过查询接口来获取详情:</p>! K' E# r2 m k5 B
<p><img src="https://img2022.cnblogs.com/blog/2487237/202202/2487237-20220211104851825-1533915064.png" ></p>6 w B9 S7 t' S& {; K
<h2 id="总结">总结</h2>% v7 f2 }; {. [. r9 x3 Y5 B8 Q
<p>在本文中我们实现了简单的新增Post操作,这里还有一些涉及到错误处理的内容,我没有在文章中演示,可以参考官方文档 <a href="https://chillicream.com/docs/hotchocolate/defining-a-schema/mutations/#errors">Errors</a>,在自定义异常对象后,有三种方式可以进行错误处理:直接返回异常;使用异常工厂方法;使用构造函数。甚至可以在<code>AggregateExceptions</code>中一次性返回多个异常。基本思路都是通过添加属性<code>[Error(typeof(SomeUserDefinedException))]</code>来实现的。</p>
4 y$ H# A* {: C<p>在下一篇文章中,我们通过<code>Mutation</code>对已有数据进行更新。</p>
. m2 y, |1 i' P3 V: N8 \7 Z
# n4 d$ E, g# K$ J: w |
|