|
|
O: C: K( W# R<h2 id="系列导航">系列导航</h2>
2 k0 \# E t, W* S. a" L% _<p><a href="https://www.cnblogs.com/code4nothing/p/graphql-net6-0.html">使用Hot Chocolate和.NET 6构建GraphQL应用文章索引</a></p># l3 j; j. T3 L( C0 K+ T+ h. c+ e# F- a
<h2 id="需求">需求</h2>& g& w# \2 W+ B- n
<p>在讨论完GraphQL中的查询需求后,这篇文章我们将演示如何实现GraphQL中的数据添加任务。</p>) |# S" K7 _! ]+ x2 c4 Y
<h2 id="思路">思路</h2>
/ z Y6 q; F4 ~% c& W4 S<p>在GraphQL中,对数据进行查询使用<code>query</code>,而对于修改数据则需要使用<code>mutation</code>,包括新增和修改数据。Hot Chocolate在使用Mutation的逻辑上和使用Query的基本一致,但是需要根据需要定义用于创建或者更新的数据对象,所以我们直接进行实现。</p>
0 X' v7 ]. ^9 D7 ~1 Y<h2 id="实现">实现</h2>% x# b5 n% |( U9 C% `- F
<p>为了保持简单,我们先定义以下两个类型:</p>
1 u" x9 M. x& |+ l. ^ i2 B+ j<pre><code class="language-c#">// 定义新增Post的参数) \7 h, S, _1 J' ^$ @. Q+ S
public record AddPostInput(string Title, string Author);
. e7 N& u, |9 T# e9 Q. ~% D$ c+ `# j/ Z g! a
// 定义新增Post的返回对象
6 k- t9 h, j( ^/ Mpublic record AddPostPayload(Post Post);" c/ V3 K8 F/ H9 W1 Q
</code></pre>' M [4 a; V# |/ a6 e$ _( y
<p>新建<code>Mutation.cs</code>用来定义相关接口:</p>
' E' T; A! X& P( ]. r* x<ul>4 j$ d2 s+ `3 { m7 f7 t
<li><code>Mutation.cs</code></li>3 i9 a8 s( z$ n8 K' K# R6 @7 s# Z/ @
</ul>
9 C) X0 l6 K# {4 U<pre><code class="language-c#">namespace PostGraphi.Api.GraphQL;9 ]" B8 J' O$ c4 Z
$ g# J$ `* f# _) y
public class Mutation
7 ^! ]: Q+ k, J5 U( {& C6 x{
2 _4 K( Q# u9 B: D1 e. o public async Task<AddPostPayload> AddPostAsync(AddPostInput input, [Service] IRepository<Post> repository)8 W- d) T+ ]" [: Q
{& f. F9 O5 V/ u3 w
return new AddPostPayload(await repository.AddAsync(new Post/ K- v, h2 ^; Z1 O' X
{
3 G D; t6 j/ M9 O1 A* A* X6 ~ Title = input.Title,
6 b% e' r% u& x! ?; [7 J8 B Author = input.Author
# M8 q; v1 R7 [3 C }));
- P8 E$ p) n9 P+ X+ P }
: u+ C+ V" i* P8 ?# h- l" T}1 M* ^9 O. G5 p6 N
+ T7 F1 \: U- g5 B( a
</code></pre>) {0 c6 A; X, y' x/ O
<p>最后在注入服务的地方进行配置:</p>
8 X5 m& i, }! c$ V. n<ul>0 C( \8 ~7 n. k. W8 U" @, }
<li><code>ProgramExtensions.cs</code></li>1 J8 E' [ a1 @
</ul>
; ]; s2 s8 b; |0 d, X<pre><code class="language-c#">builder.Services2 @ v! i+ Y& @' w C$ {1 x0 ?$ T
.AddGraphQLServer(); e9 F- f# T4 @# L& t5 d9 C
.SetPagingOptions(new PagingOptions r: F7 [9 G+ J1 U; g
{) ]& f& D, F+ {4 `. w& `6 P
MaxPageSize = 50,1 V" @' b8 b+ s
IncludeTotalCount = true) r( Z! X7 }3 b1 B N
})
$ Q$ t* R# {. V2 z .AddFiltering() ~1 O# m6 G+ P" _7 r+ I& q; u
.AddProjections()
( ]1 K; H+ ^9 m/ f& o, c .AddSorting()+ K& s' a6 F- E6 }
.AddQueryType<Query>()
' }' X7 H6 H% t1 d! t1 ]7 R4 ^ .AddMutationType<Mutation>()1 B0 x3 q; b+ _: \0 e
.AddMutationConventions(new MutationConventionOptions
; \4 x( r5 `& ~1 `3 r8 k# e {
- L1 b$ w4 B' Q2 R: Y ApplyToAllMutations = true,$ R& ~8 d7 t" F% n) Z# c8 z
InputArgumentName = "input",
9 M1 r2 A# t. ]1 _2 W9 N. F InputTypeNamePattern = "{MutationName}Input",
- I9 n4 |9 F# \3 ` PayloadTypeNamePattern = "{MutationName}Payload",
1 @* f, d4 c+ T. S PayloadErrorTypeNamePattern = "{MutationName}Error",
: j/ O }" j* H) x# U$ B PayloadErrorsFieldName = "errors": L' s+ d1 Z' ^
})
2 }( O' h, d* J: P3 S& ]2 r8 { .AddType<PostType>();# E }7 l- A9 b) r( a
</code></pre>
) K R' D( s! e& E/ p( ^2 x<p>这样就实现了新增Post的需求,下面我们来验证一下。</p>
0 m$ [1 b. U) q5 g<h2 id="验证">验证</h2>
2 `$ e- k/ e! M1 d: ~<p>启动<code>Api</code>项目,调用接口:</p># G' w: |' \) @0 {
<p><img src="https://img2022.cnblogs.com/blog/2487237/202202/2487237-20220211104544617-1400586374.png" ></p>8 u0 L8 }& Q* `7 ^. {$ r
<p>终端的日志输出如下:</p>5 p) |5 s* S. t& n# W* F3 h
<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']
3 C, y8 \7 @5 @# e: t+ J i% nINSERT INTO "Posts" ("Id", "Abstraction", "Author", "Content", "Created", "CreatedBy", "LastModified", "LastModifiedBy", "Link", "PublishedAt", "Title")
+ n. `5 ]2 `& u' t* B' P' CVALUES (@p0, @p1, @p2, @p3, @p4, @p5, @p6, @p7, @p8, @p9, @p10);
7 g& w9 e/ W% C[10:45:15 INF] Executed endpoint 'Hot Chocolate GraphQL Pipeline'& @- ^! @1 k6 R) K0 o' K
</code></pre>
! }( n1 ~1 m9 ?/ Y, ^<p>可以看到新建的Post已经存储到数据库中了,我们可以通过查询接口来获取详情:</p>( P/ s- _$ v/ o4 l9 h5 K: Q S; D! f& W
<p><img src="https://img2022.cnblogs.com/blog/2487237/202202/2487237-20220211104851825-1533915064.png" ></p>/ I% }) }3 R+ o) z F X1 q
<h2 id="总结">总结</h2>
! M9 C) R/ W( ^) `5 t<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>$ ]( U2 Y. G) K, Q" P
<p>在下一篇文章中,我们通过<code>Mutation</code>对已有数据进行更新。</p>9 i% Z1 L5 ~. |0 H+ H; E6 x% D" Z
( \: K6 i$ P/ \0 w0 E
|
|