|
|
' i3 o( y i. s/ l K* W1 k<h2 id="系列导航">系列导航</h2>
+ q% W, s2 m; P- |4 A& ?% H<p><a href="https://www.cnblogs.com/code4nothing/p/graphql-net6-0.html">使用Hot Chocolate和.NET 6构建GraphQL应用文章索引</a></p>
7 c8 v- o1 Z1 j/ @0 O. `" H<h2 id="需求">需求</h2>
- Q8 l$ w) Y: |2 g<p>在讨论完GraphQL中的查询需求后,这篇文章我们将演示如何实现GraphQL中的数据添加任务。</p>
6 }2 p( g7 X [2 }3 P7 a7 E7 Y4 b<h2 id="思路">思路</h2>
% a4 I, I6 L/ O- @* N<p>在GraphQL中,对数据进行查询使用<code>query</code>,而对于修改数据则需要使用<code>mutation</code>,包括新增和修改数据。Hot Chocolate在使用Mutation的逻辑上和使用Query的基本一致,但是需要根据需要定义用于创建或者更新的数据对象,所以我们直接进行实现。</p>
e/ U- L8 a8 N0 Y8 H9 y<h2 id="实现">实现</h2>7 z2 B; `' k: f( R9 k
<p>为了保持简单,我们先定义以下两个类型:</p>& z! X; e& Z/ X# D" t+ P
<pre><code class="language-c#">// 定义新增Post的参数) D s) [, {* Z+ L/ K) m
public record AddPostInput(string Title, string Author);
! ^7 U! b4 M& k* ]; {0 g5 E% O2 r H
3 X' Z: g7 k2 h& ~6 _// 定义新增Post的返回对象+ E! p' ]3 _0 q5 }+ }
public record AddPostPayload(Post Post);8 h& M; } \& r% ^( r0 ?
</code></pre> p Q( ?1 j- G- p* U: f
<p>新建<code>Mutation.cs</code>用来定义相关接口:</p>1 l1 t6 B; v' I0 ~5 K
<ul>! g! l5 h5 a! M' x
<li><code>Mutation.cs</code></li>
' Q# J* x7 o9 x# C' w0 t1 }, A1 L</ul>& j, H9 R; v4 Z
<pre><code class="language-c#">namespace PostGraphi.Api.GraphQL;
6 Q+ ^5 J) w* A9 s1 m' O f' X
# d, ]+ @9 d; E6 y8 d' @, xpublic class Mutation( f! F+ ^. J/ c# e9 m
{/ x$ u3 P/ G6 Y `
public async Task<AddPostPayload> AddPostAsync(AddPostInput input, [Service] IRepository<Post> repository)
0 x+ z7 v$ l9 o" p" U {
% E; M# ^' ?6 s6 w1 X! f return new AddPostPayload(await repository.AddAsync(new Post" J) F, Q4 w. f2 {$ ^3 o- N
{& W& [! O2 T% V/ y, ]
Title = input.Title,/ ~2 b5 b) `# w
Author = input.Author8 ~. H& l9 Y5 t# X% `1 e
}));) a2 n4 B0 Q$ V7 n d8 |
}" ~" i/ g) n7 y! J' ]( h3 c
}
2 l7 l: D1 S: C5 Q& I
/ c2 H5 _" H8 e</code></pre>
$ X* l- n/ J. {: W2 b<p>最后在注入服务的地方进行配置:</p>/ z* d3 \- ] K6 D( l9 w7 \1 y
<ul>
: a7 u$ H8 S! b7 F! ` M& N. ]) _( c<li><code>ProgramExtensions.cs</code></li>4 u0 |' V2 g+ W( }* j/ H: q
</ul>
# S$ ]6 I! u' j$ ^! |<pre><code class="language-c#">builder.Services6 q/ Q8 D7 O. [
.AddGraphQLServer()8 B6 O3 }0 W8 v; r4 d9 r0 Y* T
.SetPagingOptions(new PagingOptions
9 F- ?' l. {/ P; d4 }$ f {7 g+ F# a) z/ M5 k
MaxPageSize = 50,! N! C% x' O' R4 L% j4 f0 a& X8 x% O. p
IncludeTotalCount = true5 Q/ j8 X# F5 g' z& B) ?
})
) N1 M7 O( `& i4 E( ?) v .AddFiltering()
$ w2 v5 G# y7 A: _ x8 T- E- t* q .AddProjections()/ F h8 j+ L9 B/ ?8 F# }
.AddSorting()
8 a' p O# M( ^! ] .AddQueryType<Query>()+ U) U |; j7 \3 f3 r# d; o" z/ M
.AddMutationType<Mutation>()% B- @# y- p; ~8 |3 J8 j* o8 X* b
.AddMutationConventions(new MutationConventionOptions) q$ q$ T [& U) k* R- g
{7 H& b$ g: |1 O9 Q7 r
ApplyToAllMutations = true,8 r* y% F+ G/ B2 O% h @
InputArgumentName = "input",
^" {! C8 l: d6 T* S5 q InputTypeNamePattern = "{MutationName}Input",
, M9 ]" \" v4 w PayloadTypeNamePattern = "{MutationName}Payload",
( ?. H0 d# T! X0 q" | PayloadErrorTypeNamePattern = "{MutationName}Error",. D. ` H$ J; `# |
PayloadErrorsFieldName = "errors"" O9 Z' U! t+ W
})
% \0 L7 p5 N! c: v$ [' W/ T .AddType<PostType>();
, c& m$ G S! X- S</code></pre>( V/ B/ N2 d# j" c9 _) O1 v
<p>这样就实现了新增Post的需求,下面我们来验证一下。</p>* N: L: l. c& S9 g/ V& E& ^
<h2 id="验证">验证</h2>- k2 S9 w& E8 j: {
<p>启动<code>Api</code>项目,调用接口:</p>3 }2 |2 a7 p- K* T. D
<p><img src="https://img2022.cnblogs.com/blog/2487237/202202/2487237-20220211104544617-1400586374.png" ></p>! R9 l4 [) Z% _) u4 a" y; s
<p>终端的日志输出如下:</p>; F I: V) J/ c, v8 i+ L$ s
<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']
E7 c1 y' C, V/ F# f6 qINSERT INTO "Posts" ("Id", "Abstraction", "Author", "Content", "Created", "CreatedBy", "LastModified", "LastModifiedBy", "Link", "PublishedAt", "Title")
" ~4 v% R& P' Y- CVALUES (@p0, @p1, @p2, @p3, @p4, @p5, @p6, @p7, @p8, @p9, @p10);
( m: V1 b/ |: q[10:45:15 INF] Executed endpoint 'Hot Chocolate GraphQL Pipeline', T X2 L7 v0 ~2 F
</code></pre>% s1 I; ]3 X6 A8 ]) U) F9 T: b
<p>可以看到新建的Post已经存储到数据库中了,我们可以通过查询接口来获取详情:</p>
0 \2 E$ d: W& t( N" b% U* @<p><img src="https://img2022.cnblogs.com/blog/2487237/202202/2487237-20220211104851825-1533915064.png" ></p>
; c- q A: Q {# E2 ]8 J<h2 id="总结">总结</h2>1 B, S- N: S! L' e# S6 z2 }
<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>
. l- `& }8 Z: [) |<p>在下一篇文章中,我们通过<code>Mutation</code>对已有数据进行更新。</p>4 j# f( a6 |) z7 m0 v
2 s' I# X1 I) A& j6 U- \+ D& m
|
|