|
# g K: s4 I: {, k) S
<h2 id="系列导航">系列导航</h2>
* @7 D" H1 H% B8 J1 W) ?<p><a href="https://www.cnblogs.com/code4nothing/p/graphql-net6-0.html">使用Hot Chocolate和.NET 6构建GraphQL应用文章索引</a></p>
( K8 t+ j9 D0 n* g# n6 ^<h2 id="需求">需求</h2>; v9 P) e7 n5 I8 h& A9 c! E6 N) c E
<p>在讨论完GraphQL中的查询需求后,这篇文章我们将演示如何实现GraphQL中的数据添加任务。</p>
/ T$ T( o, h# k B1 g7 E( @- h<h2 id="思路">思路</h2>5 B& {; P+ j+ G2 m9 @9 o' A: O
<p>在GraphQL中,对数据进行查询使用<code>query</code>,而对于修改数据则需要使用<code>mutation</code>,包括新增和修改数据。Hot Chocolate在使用Mutation的逻辑上和使用Query的基本一致,但是需要根据需要定义用于创建或者更新的数据对象,所以我们直接进行实现。</p>, ^3 o5 ?) _. u+ @
<h2 id="实现">实现</h2>. }2 E2 N+ e) a% n, U
<p>为了保持简单,我们先定义以下两个类型:</p>
8 e) ~1 I0 a1 b8 i<pre><code class="language-c#">// 定义新增Post的参数
( Y3 w3 f% P" G" N# J# J# |public record AddPostInput(string Title, string Author);
1 v N. j: \: b5 o& h1 R+ k6 S! u/ E s) q9 d3 h6 q( p! |. G
// 定义新增Post的返回对象
' C) P; k4 a- b" {public record AddPostPayload(Post Post);- m1 N7 T6 F; S* y0 Q+ L
</code></pre>
" V0 e0 Q/ R- v9 v6 J<p>新建<code>Mutation.cs</code>用来定义相关接口:</p>8 H+ e( p. f: R1 v
<ul>
( a" `0 q1 }" H0 Z Z<li><code>Mutation.cs</code></li>
: G" N: T, R& D8 c1 i e</ul>
3 |/ u4 O( \5 j' Y<pre><code class="language-c#">namespace PostGraphi.Api.GraphQL;
; [" C3 c1 @2 } l% r/ p; a' z$ A+ c9 D2 w! R( l" D
public class Mutation
; ^; ]% q; A* k9 g{
% x& u9 W* B3 {' d( y: G# A public async Task<AddPostPayload> AddPostAsync(AddPostInput input, [Service] IRepository<Post> repository)5 K2 ^) q6 a% y& D8 Y% i
{% N" r7 H: D2 o- z. M: c0 o3 h
return new AddPostPayload(await repository.AddAsync(new Post% ?- H" s/ r8 h& w: F2 h2 M# d7 n
{# h# S; `0 F( Z) j
Title = input.Title,
( q( `" A# r# `/ r, _, I- U' [ Author = input.Author
! u7 b5 j% m# |7 i: e5 y }));' g$ S+ t% L8 R' F
}- B; k% J* O4 I
}, v6 h0 l6 D6 M
! L n2 a) Y* y/ e1 U1 J7 g</code></pre>
T. g3 |+ t' K0 V<p>最后在注入服务的地方进行配置:</p>
, K: k, U; a: U) j' U/ j<ul>
" C" i$ H2 k, [. w' ]5 I; Y3 |<li><code>ProgramExtensions.cs</code></li>3 i8 I8 q2 ~$ B! b6 u
</ul>
+ P9 U8 V# [6 |$ V<pre><code class="language-c#">builder.Services
+ s8 c! U7 b5 |5 \& K4 ] .AddGraphQLServer()
! s( e/ V) F t" M! F .SetPagingOptions(new PagingOptions
/ B% v' B9 J0 U+ _" J) H2 C {* i1 C) Z, Y* w- o
MaxPageSize = 50,- |2 _! O3 n+ F' f0 J
IncludeTotalCount = true% Z7 A. f$ F6 _6 k! J
})
; M6 H! R) @2 Y& v( \7 j* d .AddFiltering()
. b4 s% J; S1 Y3 E# ~9 [ .AddProjections()0 r# K& q# n3 H) L; w4 I
.AddSorting()7 ^2 H3 b0 T0 o% r4 U& V# f
.AddQueryType<Query>()3 w/ t. P% \" b, d6 a9 O" _
.AddMutationType<Mutation>()! J4 x7 h- t3 Z* p- \8 P
.AddMutationConventions(new MutationConventionOptions( ]8 B% `7 a& v+ n4 q1 |; P0 [4 P" a
{+ j8 ^* h: P$ N/ Y+ G$ k4 K8 v
ApplyToAllMutations = true," [9 z, `, h% R& H
InputArgumentName = "input",
! K7 W0 n# x6 X InputTypeNamePattern = "{MutationName}Input",1 r. ~; i! m( r; F. w- e* C& h
PayloadTypeNamePattern = "{MutationName}Payload",
9 ~* N: E6 t* i5 B1 @8 {0 Q7 w PayloadErrorTypeNamePattern = "{MutationName}Error",
! R! } l6 U" ]- { PayloadErrorsFieldName = "errors"
7 g5 f$ W" J. G }): F R; F( e; X+ _% ?
.AddType<PostType>();4 O* g3 d) S: f
</code></pre>
4 B- W4 i+ f3 w+ t; k* Z<p>这样就实现了新增Post的需求,下面我们来验证一下。</p>0 t0 v# ^0 m' y+ V4 y
<h2 id="验证">验证</h2>
/ ^; _: H' r& Z" R& p<p>启动<code>Api</code>项目,调用接口:</p>2 ?; @2 @4 b: o9 U* D( t0 q
<p><img src="https://img2022.cnblogs.com/blog/2487237/202202/2487237-20220211104544617-1400586374.png" ></p>2 U: |' n4 b) c, ]9 @ D
<p>终端的日志输出如下:</p>
/ o8 ?1 C3 {. b) }8 a, B' E$ H: 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']' Z/ C6 a* _! i1 N$ @
INSERT INTO "Posts" ("Id", "Abstraction", "Author", "Content", "Created", "CreatedBy", "LastModified", "LastModifiedBy", "Link", "PublishedAt", "Title")5 o. O& s! _# C6 n n
VALUES (@p0, @p1, @p2, @p3, @p4, @p5, @p6, @p7, @p8, @p9, @p10);4 a4 H7 e. O- L- d0 b
[10:45:15 INF] Executed endpoint 'Hot Chocolate GraphQL Pipeline'
* V0 {, r0 p8 [" v- n- [</code></pre>
' F* v. Y' b; O2 [<p>可以看到新建的Post已经存储到数据库中了,我们可以通过查询接口来获取详情:</p>. O% P, ~ Q$ S, J1 z
<p><img src="https://img2022.cnblogs.com/blog/2487237/202202/2487237-20220211104851825-1533915064.png" ></p>
" u: w9 ~" W: M# a' x7 C<h2 id="总结">总结</h2>, W& i; }# a3 Y, v% z( Y# q( _# ~" y
<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>9 K# k V* D6 o2 A* |5 @6 V
<p>在下一篇文章中,我们通过<code>Mutation</code>对已有数据进行更新。</p>
9 ~: v2 u. @3 L& ^. F# e2 t
3 E' H% N! I# o" j |
|