|
|
3 t' [- p4 t# i7 {
<h2 id="系列导航">系列导航</h2># r" a# @/ i3 }8 R, S; \
<p><a href="https://www.cnblogs.com/code4nothing/p/graphql-net6-0.html">使用Hot Chocolate和.NET 6构建GraphQL应用文章索引</a></p>
[- [/ r; Q/ I<h2 id="需求">需求</h2>7 T: C! f0 T( \6 |
<p>在讨论完GraphQL中的查询需求后,这篇文章我们将演示如何实现GraphQL中的数据添加任务。</p>
' v' M; i9 t" @- v9 Q/ O<h2 id="思路">思路</h2>" b* G. e8 J7 V- J) a% q$ Q
<p>在GraphQL中,对数据进行查询使用<code>query</code>,而对于修改数据则需要使用<code>mutation</code>,包括新增和修改数据。Hot Chocolate在使用Mutation的逻辑上和使用Query的基本一致,但是需要根据需要定义用于创建或者更新的数据对象,所以我们直接进行实现。</p>
( Q; e; Q2 x' k+ H$ H<h2 id="实现">实现</h2>% ]" V, O/ F0 l3 [* E7 n
<p>为了保持简单,我们先定义以下两个类型:</p>, _# x; C, y$ T
<pre><code class="language-c#">// 定义新增Post的参数
' P, {! \. Z0 p. G" B! |2 apublic record AddPostInput(string Title, string Author);
' Z1 t1 R1 M5 i% J5 R
1 F$ t n% A* Y e// 定义新增Post的返回对象
7 E6 }" o4 s6 Gpublic record AddPostPayload(Post Post);& A1 L8 y& j& Q: f2 a+ ^
</code></pre>
4 q9 W, a5 u& ]8 P, @/ R, p<p>新建<code>Mutation.cs</code>用来定义相关接口:</p>
: O2 t9 A3 m: i<ul>+ p% n- h, a, l/ C3 }3 x5 ^8 s2 C& l) [6 k
<li><code>Mutation.cs</code></li>5 j) g7 ^7 Q( Z9 @7 I
</ul>; P4 Q; y- Z* B# B" T
<pre><code class="language-c#">namespace PostGraphi.Api.GraphQL;0 ]% @2 u j. S( c9 u, W
5 u* {4 N# j" v2 X: W
public class Mutation5 u$ L& T& f1 y+ D7 J/ E& _- O% }' [
{9 J# K6 A* a; U% A8 c
public async Task<AddPostPayload> AddPostAsync(AddPostInput input, [Service] IRepository<Post> repository)2 ^5 g7 C8 p! E5 T, L4 s1 t- a
{
8 I/ F8 z7 Y% w8 }* p" \. A" q return new AddPostPayload(await repository.AddAsync(new Post( c& s% Z/ N$ j' o2 C9 U" Q* j A
{, Q$ P7 }% ^! \( s; w5 m/ u
Title = input.Title,( W8 H1 h! |- N0 T6 V5 Y1 I1 o
Author = input.Author `! `' @! c- D7 s- _; R* B
}));
1 q; k9 m2 O2 e i }0 b9 \) U2 T/ y6 m
}0 t1 H6 G& c# R+ k2 g
# C" A$ Y* h# V% b</code></pre>( i8 R4 @) w) C# Z$ `, @ V
<p>最后在注入服务的地方进行配置:</p>. `0 T+ ^$ ^' F3 L+ c. K$ j
<ul>8 H$ b: x0 Z1 C4 A: n0 j/ [
<li><code>ProgramExtensions.cs</code></li>
6 z, j9 \% W2 c8 D& a5 r</ul>6 }5 A. z9 ~) _0 f! F1 }: f
<pre><code class="language-c#">builder.Services8 a9 _3 ]) C1 I6 F/ t
.AddGraphQLServer()- U: B- V2 z G$ T
.SetPagingOptions(new PagingOptions
4 q5 t1 q4 _3 c2 O {1 e! t' e' q5 [' y% r$ _
MaxPageSize = 50,; m7 r/ O: s. p3 g( s$ B# _. j
IncludeTotalCount = true
4 d# s9 x/ U1 \. r })
y: d- ]' o; Z) W# T9 v( H1 \ .AddFiltering(); w" p+ M3 w, V: _* K
.AddProjections()/ g& F3 M3 M; d! D- P3 D. Z
.AddSorting()
0 c0 V5 V0 y% U2 O/ i; A .AddQueryType<Query>()$ H# H7 j+ T, l' p) x
.AddMutationType<Mutation>()7 O& `1 S9 e4 M( r
.AddMutationConventions(new MutationConventionOptions# M8 M, q& T5 T
{. I9 Q [. H1 q
ApplyToAllMutations = true,9 @2 Q) |$ f$ o! O, y8 q
InputArgumentName = "input",
8 U+ ~1 N4 K" R2 W& r0 a1 i InputTypeNamePattern = "{MutationName}Input",
& o+ y5 i% k0 Q; ?( U PayloadTypeNamePattern = "{MutationName}Payload",
9 O8 `5 ^+ F/ a( v* ] PayloadErrorTypeNamePattern = "{MutationName}Error",
6 Z" j, q7 o& k; o1 T! y% S PayloadErrorsFieldName = "errors"
* O5 b/ z; |# Z. q, O; o })
8 B; p: |1 l7 i .AddType<PostType>();
4 a3 A$ I7 a* D/ f" V# [</code></pre>
6 g! `0 y1 C6 k9 Y' e. Q3 D<p>这样就实现了新增Post的需求,下面我们来验证一下。</p>
8 f$ G* i) d& l! Q, L<h2 id="验证">验证</h2>
* A5 |$ I1 f! B* Z. Z3 @- x<p>启动<code>Api</code>项目,调用接口:</p>+ c. O) o1 L9 T1 m; ?1 E
<p><img src="https://img2022.cnblogs.com/blog/2487237/202202/2487237-20220211104544617-1400586374.png" ></p>
1 |; c' G& V: d4 ?- c. q! }$ ^7 c0 h<p>终端的日志输出如下:</p>
3 @" @8 e/ B3 N<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']
! g. m. r3 T8 h1 EINSERT INTO "Posts" ("Id", "Abstraction", "Author", "Content", "Created", "CreatedBy", "LastModified", "LastModifiedBy", "Link", "PublishedAt", "Title")
( z; e8 n9 I5 B' c$ V5 m2 TVALUES (@p0, @p1, @p2, @p3, @p4, @p5, @p6, @p7, @p8, @p9, @p10);8 X- |) \- l7 B
[10:45:15 INF] Executed endpoint 'Hot Chocolate GraphQL Pipeline'" L( E% H0 k( S; K# i; B
</code></pre>
9 ]( y; d# i, r% S: Z! N- y0 @. u<p>可以看到新建的Post已经存储到数据库中了,我们可以通过查询接口来获取详情:</p>
( r: X0 X" i# }$ |<p><img src="https://img2022.cnblogs.com/blog/2487237/202202/2487237-20220211104851825-1533915064.png" ></p>
; L( b/ ?3 v9 l! F: \% ^ Q<h2 id="总结">总结</h2>9 Z! o2 C+ H7 R& x
<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>
0 h$ H, Y5 b' b2 [. G2 S<p>在下一篇文章中,我们通过<code>Mutation</code>对已有数据进行更新。</p>
" f9 K4 I, j. W" g3 O, k8 |( E% }3 i; Y
|
|