|
8 a* r! t4 S5 Q5 r
<h2 id="系列导航">系列导航</h2>7 z( u: q* u& q1 i* e+ M3 v# [
<p><a href="https://www.cnblogs.com/code4nothing/p/graphql-net6-0.html">使用Hot Chocolate和.NET 6构建GraphQL应用文章索引</a></p>
- |) |! T6 T- e% ]% J; w4 z<h2 id="需求">需求</h2>9 W' H' e# O0 a+ |5 {0 i, ]
<p>在讨论完GraphQL中的查询需求后,这篇文章我们将演示如何实现GraphQL中的数据添加任务。</p>
0 u: b3 Z6 c& P z M+ T. p<h2 id="思路">思路</h2>
3 L3 M& G: w+ p' @ w; E<p>在GraphQL中,对数据进行查询使用<code>query</code>,而对于修改数据则需要使用<code>mutation</code>,包括新增和修改数据。Hot Chocolate在使用Mutation的逻辑上和使用Query的基本一致,但是需要根据需要定义用于创建或者更新的数据对象,所以我们直接进行实现。</p>
, f" U2 K. }* ]<h2 id="实现">实现</h2>
% g6 e& J5 l; l @<p>为了保持简单,我们先定义以下两个类型:</p>
) ]0 W8 k) b# B) J# u% L<pre><code class="language-c#">// 定义新增Post的参数; y3 y! [( i, Q* m6 N9 C
public record AddPostInput(string Title, string Author);# i; H/ u" ^8 G) S
# ]3 t9 I* `5 L8 ^8 K// 定义新增Post的返回对象
b2 i: R! u" K' k lpublic record AddPostPayload(Post Post);6 B* Y& O/ S' Y0 J' H& b0 _. c4 M7 \
</code></pre>9 s' Y5 j* H& [* q( P
<p>新建<code>Mutation.cs</code>用来定义相关接口:</p>
9 u. J1 j9 N4 I4 w" T( V" R8 t0 ]3 v<ul>
2 C" ^+ e' G% ^1 X<li><code>Mutation.cs</code></li>/ v$ s- P, N* P" R# @% h
</ul>( S z$ e R% @- A, w
<pre><code class="language-c#">namespace PostGraphi.Api.GraphQL;
9 w9 j" |! m& O, n2 Q' C& p2 Z' U. H2 Z" p4 @$ t* V
public class Mutation* F ]2 ~0 V0 P2 A' N' _ |, g
{
% E0 U% ?% l$ g0 N public async Task<AddPostPayload> AddPostAsync(AddPostInput input, [Service] IRepository<Post> repository): ?' O3 r# F2 y6 c* K% Z
{9 Z {) y# H$ ] f7 s2 {3 _
return new AddPostPayload(await repository.AddAsync(new Post5 F0 n2 [4 }: Y3 L! u; ?$ p
{
0 y1 q% F0 J3 L Title = input.Title,
! y/ J! a# a7 q6 ]2 D! ? Author = input.Author
+ d) S* e" n% ^ }));2 |. v# B2 f( z: M& b% d2 f8 b2 D
}
" E/ a4 J. a: ^" A* Q}8 h) e2 E& m- A
0 q. K; e; g' k6 ]</code></pre>! a5 A, O8 [4 Q4 I/ a7 s
<p>最后在注入服务的地方进行配置:</p>
& i# o/ w3 r, ], L( e) D* Q<ul>
1 V# m# A l- ^7 X ~<li><code>ProgramExtensions.cs</code></li>! |6 v4 k) Q8 S3 m, U+ F# r
</ul>
& B; m/ D& |' E! _9 G<pre><code class="language-c#">builder.Services0 G1 ]" d2 F; t3 U
.AddGraphQLServer()
; l8 E0 h1 J# `. U+ N8 E. g( K! F .SetPagingOptions(new PagingOptions3 L% `& u9 W' D9 q8 |& \
{: j3 w9 E, U, l) J
MaxPageSize = 50,* K+ |; W1 j, a+ @
IncludeTotalCount = true T2 i& k$ A V
})
9 ~7 n a/ _' j! v7 S/ q, B2 f .AddFiltering()' W' E7 ~4 [- ^% r k% Z) u. U! O
.AddProjections()
. G e& ]8 q3 u8 W" u- a3 Q5 V .AddSorting()- \1 Q- G+ O+ ?* A! ]6 R
.AddQueryType<Query>()
- a% V& ?, k9 `2 X% P .AddMutationType<Mutation>()" t+ x& h; U6 O. j& R% y. e8 N
.AddMutationConventions(new MutationConventionOptions. Y1 m3 W3 _1 x% Q1 a
{' i* A( }4 ^+ b: l
ApplyToAllMutations = true,
9 Y* B' M. P* P8 N0 Y InputArgumentName = "input",
5 U# q& u. u* T1 H1 M; ~ L InputTypeNamePattern = "{MutationName}Input",/ e6 U! B' }) r8 S" _
PayloadTypeNamePattern = "{MutationName}Payload",$ J. ~7 b( m7 I) ^, p. Z4 p7 L
PayloadErrorTypeNamePattern = "{MutationName}Error",! L* |5 I& U* c. J" ~1 J0 ?
PayloadErrorsFieldName = "errors"! b; X# y7 i \( ]% l6 _ ~0 v; _
})3 ]. J- L4 P" d$ W
.AddType<PostType>();
( w8 {+ c9 M+ u& Y7 F$ L1 c1 c" N; `</code></pre>% X. T& P) F$ l( I- `7 q, }
<p>这样就实现了新增Post的需求,下面我们来验证一下。</p>2 P9 Y' D. x$ G2 {
<h2 id="验证">验证</h2>( m/ y% h/ T( E0 x/ t) ]7 s
<p>启动<code>Api</code>项目,调用接口:</p>
! g) W9 w0 }# N. a/ ^8 ?<p><img src="https://img2022.cnblogs.com/blog/2487237/202202/2487237-20220211104544617-1400586374.png" ></p>: I+ D8 o" R6 M7 |- T! D
<p>终端的日志输出如下:</p>
. ~: o7 l# E) j( o4 k4 V% b* 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']
! N9 K' N6 L# f* Q4 X7 _INSERT INTO "Posts" ("Id", "Abstraction", "Author", "Content", "Created", "CreatedBy", "LastModified", "LastModifiedBy", "Link", "PublishedAt", "Title")
& e3 V( X$ z$ e! [$ R4 eVALUES (@p0, @p1, @p2, @p3, @p4, @p5, @p6, @p7, @p8, @p9, @p10);
3 u" ~; g+ h$ R2 l& H[10:45:15 INF] Executed endpoint 'Hot Chocolate GraphQL Pipeline'
/ K: `6 A" `- j G, j/ C</code></pre>* y8 r# k) F9 V% d: ?8 ]7 r3 m- _
<p>可以看到新建的Post已经存储到数据库中了,我们可以通过查询接口来获取详情:</p>& X% R& z; n7 j/ a
<p><img src="https://img2022.cnblogs.com/blog/2487237/202202/2487237-20220211104851825-1533915064.png" ></p>
. C- y" z$ } G) }" b& i/ \<h2 id="总结">总结</h2>
& B% C% m$ o+ u" M/ E; u; e<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>
% P+ q+ ]( m" q<p>在下一篇文章中,我们通过<code>Mutation</code>对已有数据进行更新。</p>" ] z; r3 m# a( b0 f: ^) ` S
+ X' T2 W a* t }, h. v |
|