|
|
, u# y* i- s" G" N5 d<h2 id="系列导航">系列导航</h2>3 `0 b8 y% }/ A# n& V c
<p><a href="https://www.cnblogs.com/code4nothing/p/graphql-net6-0.html">使用Hot Chocolate和.NET 6构建GraphQL应用文章索引</a></p>
5 B3 j/ x3 ^! w. c( |& i<h2 id="需求">需求</h2>! y3 @& S( m9 `) n) G: j' l
<p>在讨论完GraphQL中的查询需求后,这篇文章我们将演示如何实现GraphQL中的数据添加任务。</p>& ^2 ]4 e3 @/ U/ O- y/ e1 @2 O- s
<h2 id="思路">思路</h2>
) D2 _. U M) M; b' x' I<p>在GraphQL中,对数据进行查询使用<code>query</code>,而对于修改数据则需要使用<code>mutation</code>,包括新增和修改数据。Hot Chocolate在使用Mutation的逻辑上和使用Query的基本一致,但是需要根据需要定义用于创建或者更新的数据对象,所以我们直接进行实现。</p>
j& b3 }, |, B: ?" j<h2 id="实现">实现</h2>$ j- D/ X, e3 l' F: U. ~
<p>为了保持简单,我们先定义以下两个类型:</p>
6 { r1 o; B/ d( M/ A<pre><code class="language-c#">// 定义新增Post的参数8 l& r- ~5 C% I" x6 V) z2 B
public record AddPostInput(string Title, string Author);
" G- X2 s+ `( I+ v ~" x# c8 E2 w4 v+ i% q
// 定义新增Post的返回对象
8 B( g. V" Q- g# z# V# npublic record AddPostPayload(Post Post);5 k) N# B( e/ Z8 Y; T& S2 l
</code></pre>
: w. i; o4 l* M% H M, [<p>新建<code>Mutation.cs</code>用来定义相关接口:</p>
# a0 s4 [) p% y: J/ S<ul>
- R' t* |% J, O' K2 l9 D<li><code>Mutation.cs</code></li>
1 L4 g7 l5 I" V% x; k- U</ul>" f3 x/ _0 O; z8 F1 X1 e
<pre><code class="language-c#">namespace PostGraphi.Api.GraphQL;
0 g; f7 Q* R5 y1 P! Z# Y- f5 \) s4 B1 b
public class Mutation
l6 l+ T3 o/ Y( G2 o1 Q{4 x; P# \7 @) n
public async Task<AddPostPayload> AddPostAsync(AddPostInput input, [Service] IRepository<Post> repository)2 D! Q# f7 y" \# ^3 t1 z
{
5 O$ |2 P1 W: A$ j _) [ return new AddPostPayload(await repository.AddAsync(new Post
7 }) u7 ~0 m H' l- J3 y {5 a- O/ {" z/ Z+ w4 O; p# k: A
Title = input.Title,
Y* W+ t6 Q8 @5 b Author = input.Author. G: k; z5 m; g4 c9 e
}));
& C# l$ o1 R$ _/ s }
/ s% S9 \( e" R. `2 ~}
o4 M/ _. I! a$ [! E# q- v5 ~* v) E4 Y( @( L6 ]
</code></pre>
6 t+ u+ i! F* A0 i2 N* W' h2 Z<p>最后在注入服务的地方进行配置:</p>
* K8 A ^% Q/ [$ I/ h4 B3 @<ul>
% J& N7 @; \1 m8 K1 {<li><code>ProgramExtensions.cs</code></li>
1 i! b6 c' x8 R" d, g</ul>
6 z7 D; ]' x9 q' B, m' }<pre><code class="language-c#">builder.Services
3 ?2 I4 U$ V/ J) B .AddGraphQLServer()$ G7 L8 G- f5 U$ Z1 w
.SetPagingOptions(new PagingOptions
2 M$ _! {# _! T# G! b {: \7 w ?& O2 a1 F2 v: X" B
MaxPageSize = 50,3 W' i3 | x% k4 t5 f K% |( G( f
IncludeTotalCount = true6 ?+ [& [. p- y" U, D
})( ?3 O) _9 z9 ~" @
.AddFiltering()
5 O9 @: D+ C z .AddProjections()
; ^; V0 C* C5 Z$ J) v; ]/ E .AddSorting()
3 ^8 W& G/ J( j& R .AddQueryType<Query>()* A9 @$ _$ K7 Z( |
.AddMutationType<Mutation>()$ Q" H* ]) M5 n9 N5 z2 ?; k% ]
.AddMutationConventions(new MutationConventionOptions
* ^7 |( `+ S! W5 C: r! V {9 U% v+ V: h* T( Y
ApplyToAllMutations = true,
) D/ K. h2 S' F InputArgumentName = "input",
; P( q. p8 G. {5 Z, m, n' r/ k InputTypeNamePattern = "{MutationName}Input"," x) z' Y) e" @1 Z3 d% O V
PayloadTypeNamePattern = "{MutationName}Payload",
L3 W; d) A* e: ~3 z PayloadErrorTypeNamePattern = "{MutationName}Error",
7 [ ]7 N) Q8 |; @; _+ V. R" o PayloadErrorsFieldName = "errors"# @9 X0 R6 P) \0 D7 d: ^
})9 m# Y( R8 p* |: w
.AddType<PostType>();5 U1 |7 _' q" r. R$ Y
</code></pre>
0 @! [- [* l6 X/ z* c, J<p>这样就实现了新增Post的需求,下面我们来验证一下。</p>
' l, Y- t* N! k. j% b5 d<h2 id="验证">验证</h2>
% g! q t1 e' u! U' m( M- I0 D<p>启动<code>Api</code>项目,调用接口:</p>
. ]: Q& R. G$ v' j<p><img src="https://img2022.cnblogs.com/blog/2487237/202202/2487237-20220211104544617-1400586374.png" ></p>2 @! P" }. @' D' i1 ?; v2 `3 e1 o9 w, G
<p>终端的日志输出如下:</p>
6 D. J1 W0 v5 C/ g/ E6 ]- 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']
- u. K2 P+ n7 c$ MINSERT INTO "Posts" ("Id", "Abstraction", "Author", "Content", "Created", "CreatedBy", "LastModified", "LastModifiedBy", "Link", "PublishedAt", "Title")
: \$ l6 k9 e7 m b9 D: Q* }VALUES (@p0, @p1, @p2, @p3, @p4, @p5, @p6, @p7, @p8, @p9, @p10);! f# y$ T' y l1 [1 x3 e3 r8 h3 o
[10:45:15 INF] Executed endpoint 'Hot Chocolate GraphQL Pipeline'+ Y2 E1 K- s ^8 G' j I
</code></pre>( V! T- V! L Z1 ~) G; z3 V% R$ G
<p>可以看到新建的Post已经存储到数据库中了,我们可以通过查询接口来获取详情:</p>
* [1 d% P! w& F- d* P, t$ t k- n<p><img src="https://img2022.cnblogs.com/blog/2487237/202202/2487237-20220211104851825-1533915064.png" ></p>
5 \8 i/ d6 b, ~7 @<h2 id="总结">总结</h2>
% ]8 I( w& c" J m6 ^4 S/ \<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>- Q+ W6 U7 M) M9 S
<p>在下一篇文章中,我们通过<code>Mutation</code>对已有数据进行更新。</p>
1 o: R4 X L6 q, @7 ]/ K+ y' [+ e. K2 h0 O+ k3 m& T1 U0 R
|
|