|
|
9 B9 U* s6 J7 j. k$ C7 s' U9 `<h2 id="系列导航">系列导航</h2>
@& y1 r9 w9 @! d" l$ O" A' n<p><a href="https://www.cnblogs.com/code4nothing/p/graphql-net6-0.html">使用Hot Chocolate和.NET 6构建GraphQL应用文章索引</a></p>
0 T+ W ^! r5 Q. @+ t0 v+ f; e<h2 id="需求">需求</h2>& ]; \& M" p7 t/ k
<p>在讨论完GraphQL中的查询需求后,这篇文章我们将演示如何实现GraphQL中的数据添加任务。</p>. c) D- v) x, t7 L
<h2 id="思路">思路</h2>
9 K5 u R- b2 p( B0 _2 g7 C3 W6 g<p>在GraphQL中,对数据进行查询使用<code>query</code>,而对于修改数据则需要使用<code>mutation</code>,包括新增和修改数据。Hot Chocolate在使用Mutation的逻辑上和使用Query的基本一致,但是需要根据需要定义用于创建或者更新的数据对象,所以我们直接进行实现。</p>2 c$ {/ Q( W$ |8 L! c: M" B
<h2 id="实现">实现</h2>
$ m. W5 Y" z9 X- Q<p>为了保持简单,我们先定义以下两个类型:</p>
* ?& d9 I: o% g& \! w2 D<pre><code class="language-c#">// 定义新增Post的参数
% F6 H* a. h4 S! H6 U5 Xpublic record AddPostInput(string Title, string Author); r" l5 d5 y" ]2 z
( T" M7 s: V% H& V1 X4 m// 定义新增Post的返回对象
$ C8 V8 q9 W$ M3 ]public record AddPostPayload(Post Post);
8 d5 U% Y* K1 o2 H( k6 P$ X</code></pre>
5 E0 O2 I( I9 l! n5 l<p>新建<code>Mutation.cs</code>用来定义相关接口:</p>
- J/ d. {0 J7 S3 p6 V: y<ul>" _& V) ^3 z& R9 r) G
<li><code>Mutation.cs</code></li>' s5 B1 n% w: K2 R7 {% h, K
</ul>' Y7 ~' E& V \. n
<pre><code class="language-c#">namespace PostGraphi.Api.GraphQL;
' A- c/ E' H" O
# o. X% z. m2 Opublic class Mutation
& y0 W [# y" h1 O9 [4 Z{1 A/ I3 \ }3 m( b) q- f5 f" u
public async Task<AddPostPayload> AddPostAsync(AddPostInput input, [Service] IRepository<Post> repository)* @3 W$ K* n4 k! f1 \0 W
{
5 N/ Z' E' x) c U/ U! \ return new AddPostPayload(await repository.AddAsync(new Post s1 d) R5 d/ O1 J& g3 e
{
5 H. i% k {; y) ]6 T Title = input.Title,1 N+ X7 E/ \* P- Q: }9 E
Author = input.Author6 j A( v+ L7 J; q* L4 ]
}));
4 M% n m) M s& _ }5 h. ]5 A" E3 D, t
}. y! G! i- t1 e9 b
2 G: _7 B$ B: y# [! J6 j5 `8 g( I
</code></pre>% a6 ]8 P5 o% H/ I9 Z+ j8 h
<p>最后在注入服务的地方进行配置:</p>
. X1 V$ {2 T8 x: _* i8 o<ul>+ c8 _( C6 w7 D' J/ X5 N8 X4 r. ^
<li><code>ProgramExtensions.cs</code></li>
. v* V L% h# ], K' u8 u4 H$ L6 Z! o' m</ul>! s8 |, b% t$ d, ?3 a
<pre><code class="language-c#">builder.Services9 H# Y2 s- T8 g2 j. b: t# T* o
.AddGraphQLServer()8 P8 { c' Z- b$ p6 C" v
.SetPagingOptions(new PagingOptions
& i6 R8 p) O# o {
8 s* a3 r! h9 `+ `. ]$ f MaxPageSize = 50,
$ X! e# B: C* p1 r: o IncludeTotalCount = true
! E: n1 [7 }& o, s! g( {, P' | })' E4 @0 B& A/ g" C7 ^" W
.AddFiltering()
( O% g0 I# C, v .AddProjections()+ V# y* N8 k' O6 d. e, X
.AddSorting()
V; |; r% t6 Q$ ]7 E .AddQueryType<Query>()6 q% f& q8 R& J9 g% V2 @; b
.AddMutationType<Mutation>()
0 }3 ~7 G# e5 m .AddMutationConventions(new MutationConventionOptions8 a6 I4 J* u# ?4 j: t$ r& f
{
6 W- r% E1 d( Q0 x: F! G: o) J ApplyToAllMutations = true,
$ Z9 \" C. k& b3 }! h, ]6 H, o InputArgumentName = "input",
7 i% ?9 B0 @7 Q InputTypeNamePattern = "{MutationName}Input",
& g) ]1 {- p7 t; o" K2 z( p PayloadTypeNamePattern = "{MutationName}Payload",
% A; w1 T& b5 p% s2 a PayloadErrorTypeNamePattern = "{MutationName}Error",7 ?' C* G8 h: s2 v2 y
PayloadErrorsFieldName = "errors"
7 _' _6 W j: L! t; b$ o })0 B- Y" q- Q0 c1 W4 Z' J
.AddType<PostType>();1 g" }' D6 n4 f* y: X P3 g
</code></pre>
0 C6 z3 ], Z3 r5 A& |<p>这样就实现了新增Post的需求,下面我们来验证一下。</p>
# u* @( e8 y" Z8 S0 z9 A<h2 id="验证">验证</h2>
) _! ~& q2 K* h8 o# t. K* t/ L<p>启动<code>Api</code>项目,调用接口:</p>9 d h& {% c: R3 A3 @, k7 @
<p><img src="https://img2022.cnblogs.com/blog/2487237/202202/2487237-20220211104544617-1400586374.png" ></p>
- R& L9 F4 ]5 I8 u6 [( H<p>终端的日志输出如下:</p>* B" g" L, K0 ?7 g0 j; b$ I
<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']7 W9 P' K6 S& v, j' g P
INSERT INTO "Posts" ("Id", "Abstraction", "Author", "Content", "Created", "CreatedBy", "LastModified", "LastModifiedBy", "Link", "PublishedAt", "Title")7 \2 H" R, h4 _2 x' ^
VALUES (@p0, @p1, @p2, @p3, @p4, @p5, @p6, @p7, @p8, @p9, @p10);7 ^ y+ f5 h0 P- N6 g
[10:45:15 INF] Executed endpoint 'Hot Chocolate GraphQL Pipeline'
' N- y9 s' B, N6 I</code></pre>
9 L; p/ n# _! Z# X) O# G3 j<p>可以看到新建的Post已经存储到数据库中了,我们可以通过查询接口来获取详情:</p>3 w- j/ u2 `9 P" }/ ]9 [0 S2 u
<p><img src="https://img2022.cnblogs.com/blog/2487237/202202/2487237-20220211104851825-1533915064.png" ></p>
* ~$ h( t+ l: e7 z<h2 id="总结">总结</h2>4 |( _* U2 Z Q; D. o3 X5 Q/ I
<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>% n5 O& I0 i/ S- K+ U+ Y
<p>在下一篇文章中,我们通过<code>Mutation</code>对已有数据进行更新。</p>+ O1 X& Y8 ~. \
% A3 m0 D# ~) e: z1 y
|
|