|
0 O: c4 \ @6 `2 q/ p( p7 b2 o3 S
<h2 id="系列导航">系列导航</h2>5 u7 D- W x# q( ?8 t5 h* m
<p><a href="https://www.cnblogs.com/code4nothing/p/graphql-net6-0.html">使用Hot Chocolate和.NET 6构建GraphQL应用文章索引</a></p>
# i! M7 @" J% t H; V6 _# b" N<h2 id="需求">需求</h2>$ J2 O; I4 {. U0 f1 _' p
<p>在讨论完GraphQL中的查询需求后,这篇文章我们将演示如何实现GraphQL中的数据添加任务。</p>7 w4 o4 _+ i' }8 K' {2 Z l
<h2 id="思路">思路</h2>! t$ Q4 O, C* F$ d5 X. H( E
<p>在GraphQL中,对数据进行查询使用<code>query</code>,而对于修改数据则需要使用<code>mutation</code>,包括新增和修改数据。Hot Chocolate在使用Mutation的逻辑上和使用Query的基本一致,但是需要根据需要定义用于创建或者更新的数据对象,所以我们直接进行实现。</p>7 ^. i; Q+ q3 _- t. a
<h2 id="实现">实现</h2>
, v0 b) w- s. [; A6 r<p>为了保持简单,我们先定义以下两个类型:</p>
% }* ^. J' k5 r; D, F<pre><code class="language-c#">// 定义新增Post的参数
' _ b" y! E/ _0 L- Q. bpublic record AddPostInput(string Title, string Author);
# d9 P+ G2 W! o# ~2 ]2 W9 d; s# Q {* u
// 定义新增Post的返回对象
+ D- {$ I" k, Tpublic record AddPostPayload(Post Post);
7 M( {* D! b. d/ w</code></pre>
- ^! K) h& Q2 F) K<p>新建<code>Mutation.cs</code>用来定义相关接口:</p>
5 _6 k+ a% Y: v% F w<ul>' [& Z U0 u/ L4 K$ i
<li><code>Mutation.cs</code></li>
$ T+ _" S6 c% t! k</ul>
8 q, M9 d( K& E1 _$ F" B<pre><code class="language-c#">namespace PostGraphi.Api.GraphQL;; M, {8 V6 G/ r( O3 {( l
+ ^/ K: H* V( {: O/ t" j
public class Mutation
1 ~7 ]2 E# J3 U% c ?3 A1 u) K/ a{
% M- ]/ [& B) L6 S2 q public async Task<AddPostPayload> AddPostAsync(AddPostInput input, [Service] IRepository<Post> repository)4 G1 y: n: U( d6 L" T/ e
{
2 j+ b; a5 F+ I' ?& b) U+ U return new AddPostPayload(await repository.AddAsync(new Post
$ A1 c$ K. O2 s% E5 v {
! ]$ W: O2 O' G$ ]: ` Title = input.Title,
3 U4 U5 l0 f( L7 V8 ]; s Author = input.Author( Z# {# W+ Y- l( \5 \
}));7 D/ }6 l, I3 _
}& E6 c9 s5 x$ ~6 [' r
}! m8 C" e' N( f5 e% C" {7 Q
# C8 W5 d1 ^7 D) t3 V7 A</code></pre>
% ^% q3 y- ^. i% x: d |1 v2 @<p>最后在注入服务的地方进行配置:</p>
) F* n+ b1 `7 T* h- r<ul>
% V0 c) ~" c" L<li><code>ProgramExtensions.cs</code></li>
6 e9 [; S) _7 H2 P6 U. K</ul>+ K2 ^+ t8 p/ T# K E; y
<pre><code class="language-c#">builder.Services
0 V: D/ V) E& o* r .AddGraphQLServer()
7 f+ b4 h1 O1 A7 E) @6 u* k9 g7 G .SetPagingOptions(new PagingOptions9 K1 g; M( a4 n, _, Z8 N9 Y
{
0 y9 d# \. n; m7 | MaxPageSize = 50,
) y( g$ Y. F3 E4 s+ X7 q IncludeTotalCount = true
+ @& ~, o" `+ S) C( }2 } })
3 S q P- c, F& e# Q6 @( L .AddFiltering()
* p6 D b9 |* ^$ v4 P& Q9 j4 U. G .AddProjections()
4 i3 g1 y: C+ _2 m# e .AddSorting()% i, S/ p) X: w$ q3 Y2 P
.AddQueryType<Query>()
9 D9 Z) k& U, q+ A .AddMutationType<Mutation>()
; S( G9 [1 ~* H' W; M' @ .AddMutationConventions(new MutationConventionOptions4 u$ i+ Y$ T) R; T+ O
{
1 s0 f2 ]( O, q# ~$ k: h: V ApplyToAllMutations = true,
! {! g! }+ N5 Y: D: Z! N InputArgumentName = "input",6 {8 }0 _3 s; l9 r3 j; B
InputTypeNamePattern = "{MutationName}Input"," U, X8 m8 S* C9 F/ q& l' L; P2 _
PayloadTypeNamePattern = "{MutationName}Payload",) F4 u5 G4 q I$ i6 m5 X
PayloadErrorTypeNamePattern = "{MutationName}Error",
7 b" \( S1 w. E6 X* M1 N PayloadErrorsFieldName = "errors"
9 \( S# e" D; B9 e })
9 r G( {' Y% V4 H3 z. N" h' k .AddType<PostType>();' G9 _8 O6 m! E+ w
</code></pre>! g! }) K1 x2 s% x0 I9 @( H
<p>这样就实现了新增Post的需求,下面我们来验证一下。</p>4 ` P0 P) Z$ L
<h2 id="验证">验证</h2>
6 d; b2 E' L, X0 v. [0 I4 z% V<p>启动<code>Api</code>项目,调用接口:</p>
2 l! o \3 Y* Z* N9 L7 l$ G<p><img src="https://img2022.cnblogs.com/blog/2487237/202202/2487237-20220211104544617-1400586374.png" ></p>3 e: `/ ~+ a+ L$ b3 o9 ^9 K
<p>终端的日志输出如下:</p>
9 z! }( ?9 Z6 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']
' S0 l( Y' B1 l8 b- IINSERT INTO "Posts" ("Id", "Abstraction", "Author", "Content", "Created", "CreatedBy", "LastModified", "LastModifiedBy", "Link", "PublishedAt", "Title")
( i; w2 ?2 t: f' Z3 T3 f1 T) LVALUES (@p0, @p1, @p2, @p3, @p4, @p5, @p6, @p7, @p8, @p9, @p10);9 k/ H6 e/ F/ G8 N3 i. I. h' ]
[10:45:15 INF] Executed endpoint 'Hot Chocolate GraphQL Pipeline'
# _: C" v* o. r; z' n' q</code></pre>: j9 O+ X+ `5 r; |7 s+ M
<p>可以看到新建的Post已经存储到数据库中了,我们可以通过查询接口来获取详情:</p>' H2 D' A3 ]0 p9 ~
<p><img src="https://img2022.cnblogs.com/blog/2487237/202202/2487237-20220211104851825-1533915064.png" ></p>; H* ]0 D% s0 C4 r6 T' C
<h2 id="总结">总结</h2>
- s% `2 D8 m! s% @! B: ~3 U<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>/ V# L5 C# o. f) |5 P4 K% p- Q. T
<p>在下一篇文章中,我们通过<code>Mutation</code>对已有数据进行更新。</p> s0 T+ f0 w% k$ p
# C0 l1 p+ H( }. R
|
|