|
|
3 Z2 j3 h: G- ]. L<h2 id="系列导航">系列导航</h2>
]: F7 H& \& \/ V: T<p><a href="https://www.cnblogs.com/code4nothing/p/graphql-net6-0.html">使用Hot Chocolate和.NET 6构建GraphQL应用文章索引</a></p>
- @ T' _: } s1 U<h2 id="需求">需求</h2>
& {) J9 z6 V0 C6 d<p>在讨论完GraphQL中的查询需求后,这篇文章我们将演示如何实现GraphQL中的数据添加任务。</p>
# g3 j* Y& v! l! C6 b% U) `<h2 id="思路">思路</h2>) q0 \1 D) V( F& |- _) k r Q& f
<p>在GraphQL中,对数据进行查询使用<code>query</code>,而对于修改数据则需要使用<code>mutation</code>,包括新增和修改数据。Hot Chocolate在使用Mutation的逻辑上和使用Query的基本一致,但是需要根据需要定义用于创建或者更新的数据对象,所以我们直接进行实现。</p>) l1 O$ C+ p" B
<h2 id="实现">实现</h2>
" ~9 v; t/ {6 J" j, [<p>为了保持简单,我们先定义以下两个类型:</p>/ k1 W3 W8 M. j; C
<pre><code class="language-c#">// 定义新增Post的参数
: M$ _7 i) \% L# |, u+ `public record AddPostInput(string Title, string Author);8 v2 q6 B) Q. o) U: F
9 ~" {2 H; i& Z) B// 定义新增Post的返回对象$ h( I4 V( a$ _4 I7 J
public record AddPostPayload(Post Post);5 y2 i: k; x9 ?) m8 g( o
</code></pre>$ ` F7 P4 O- b" m, Q
<p>新建<code>Mutation.cs</code>用来定义相关接口:</p>$ d- V, B/ g4 N; e7 q9 Y) D& [" A
<ul>" `6 u$ R8 h8 w
<li><code>Mutation.cs</code></li>' u# [ @& C- J" O! Y+ t2 W
</ul>* |5 e( m, b- p& n
<pre><code class="language-c#">namespace PostGraphi.Api.GraphQL;
1 {% Q& k% `# j" ]' M( g" K( K- j/ D
public class Mutation8 Z& B" N* k- f. r/ A# `! Q( n- d
{! O2 \" }. y' m2 S- S
public async Task<AddPostPayload> AddPostAsync(AddPostInput input, [Service] IRepository<Post> repository)
% C7 x8 P8 ]0 N# S {. P) G6 h/ g# ~& g8 w. x
return new AddPostPayload(await repository.AddAsync(new Post% z4 V/ v0 D* \( Q2 l% R7 Q' k0 q
{
7 x4 k& x+ c) K% U a Title = input.Title,; S' G' g+ D! h* t% ^; V; A' K. G
Author = input.Author! j3 q! w* \! T5 W; x% B
}));
, B" I5 E4 ~ p2 _7 J) V* c: Z4 Y$ c }
- v% K* G3 q; B8 h}1 V/ o) O9 m! t
' ^/ _) h- C8 L2 Q</code></pre>
" G9 L" o5 J [: s, v: s- i# k" k<p>最后在注入服务的地方进行配置:</p>; [4 ~! `: ^$ n$ s4 Q k- A$ c
<ul>
. ]! K( H1 r7 A0 y! Q<li><code>ProgramExtensions.cs</code></li>( G+ v$ z* H7 {4 C1 G, O
</ul>
2 _3 \+ A0 ?- Y! J<pre><code class="language-c#">builder.Services1 ^4 z; T1 [; J9 c7 r
.AddGraphQLServer()- v/ X' l" n# f
.SetPagingOptions(new PagingOptions+ T; G7 s/ K8 Q/ s& q0 h
{
/ g* |9 t% }6 { MaxPageSize = 50,
5 \& U# f0 x) R' ^; Z IncludeTotalCount = true
( |4 I" {5 Z* F3 w& |0 D) }( T })8 b6 h6 U" U) X1 t
.AddFiltering(); f! M& o9 V& ] l* a3 D" f
.AddProjections()
3 H. |* k( f9 U .AddSorting()
" @8 h1 d; ~ M( T% H, w .AddQueryType<Query>()$ d" B2 P& K- B3 d# o \
.AddMutationType<Mutation>()
) `( _0 L' l4 { .AddMutationConventions(new MutationConventionOptions, k- D0 `3 l" D8 G5 [8 t
{
+ z+ _ ?9 k5 W) \ ApplyToAllMutations = true,
8 u/ n5 V( q4 @ u InputArgumentName = "input",
' R, [& w+ e$ E( @ InputTypeNamePattern = "{MutationName}Input",+ \8 u9 v" h* P/ W5 k3 I
PayloadTypeNamePattern = "{MutationName}Payload",
; F" m9 t4 h, K8 `# M9 ?# r PayloadErrorTypeNamePattern = "{MutationName}Error",
9 M* T! I+ Q" z PayloadErrorsFieldName = "errors"
9 n/ j6 P! {; W& p3 T })
1 ]& }/ L% I2 [( ^& Y .AddType<PostType>();: @% Q2 B1 k+ k5 R$ E
</code></pre>- ?. N8 B" g& U
<p>这样就实现了新增Post的需求,下面我们来验证一下。</p>; I+ M+ }& d' `; v
<h2 id="验证">验证</h2>9 R2 l# }9 ?$ u( z* a% p
<p>启动<code>Api</code>项目,调用接口:</p>" Q4 C8 K7 u; |" A7 d- l
<p><img src="https://img2022.cnblogs.com/blog/2487237/202202/2487237-20220211104544617-1400586374.png" ></p>
4 _% y2 R. O; O) r2 ?7 ~$ q: K3 G<p>终端的日志输出如下:</p>
5 a6 A8 B" ^& K9 L9 u' |. y; P8 q<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 L, c% \+ Y! ]( ?$ Q
INSERT INTO "Posts" ("Id", "Abstraction", "Author", "Content", "Created", "CreatedBy", "LastModified", "LastModifiedBy", "Link", "PublishedAt", "Title")
( ^7 o. {8 s, v: FVALUES (@p0, @p1, @p2, @p3, @p4, @p5, @p6, @p7, @p8, @p9, @p10);
$ b. E$ u3 S" g+ P, ][10:45:15 INF] Executed endpoint 'Hot Chocolate GraphQL Pipeline'4 P5 V$ b* J" \7 Z7 G3 N7 F" _# j+ a* F
</code></pre>+ k {0 Q4 l# N( M
<p>可以看到新建的Post已经存储到数据库中了,我们可以通过查询接口来获取详情:</p>
/ B* j% B2 i; O% Z<p><img src="https://img2022.cnblogs.com/blog/2487237/202202/2487237-20220211104851825-1533915064.png" ></p>
# Z. n9 C4 L6 A/ Y6 O$ A; y<h2 id="总结">总结</h2>2 J/ S) q. A; t+ T$ 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>8 ^! Y! B- A: I1 g" ^" e
<p>在下一篇文章中,我们通过<code>Mutation</code>对已有数据进行更新。</p>
; M2 m6 i9 G2 D1 @. v. m5 P: f/ q- f4 X" \( b: g
|
|