|
|
6 J1 O6 \! j" o8 `' i" J; o* }<h2 id="系列导航">系列导航</h2>
2 G3 j$ o |9 b+ a: W4 \<p><a href="https://www.cnblogs.com/code4nothing/p/graphql-net6-0.html">使用Hot Chocolate和.NET 6构建GraphQL应用文章索引</a></p> r) T" g1 `/ B( }) u, R6 S- x
<h2 id="需求">需求</h2>
+ p" c& R. g2 H2 F. ? H<p>在讨论完GraphQL中的查询需求后,这篇文章我们将演示如何实现GraphQL中的数据添加任务。</p>
2 u$ a2 |6 t# q2 C! U1 M. f<h2 id="思路">思路</h2>
: y# r: f' r2 i* K<p>在GraphQL中,对数据进行查询使用<code>query</code>,而对于修改数据则需要使用<code>mutation</code>,包括新增和修改数据。Hot Chocolate在使用Mutation的逻辑上和使用Query的基本一致,但是需要根据需要定义用于创建或者更新的数据对象,所以我们直接进行实现。</p>. O0 {# X4 |+ V: F5 m% S
<h2 id="实现">实现</h2>
7 p$ e1 E, T9 P, M4 `<p>为了保持简单,我们先定义以下两个类型:</p>0 ^+ z( |! H: d, q- B! ^" r
<pre><code class="language-c#">// 定义新增Post的参数
& g6 o& C4 O; ?' |" H7 o% i9 {7 N3 Ppublic record AddPostInput(string Title, string Author);
2 G, j. i$ c/ J( o0 t4 p8 J R& [0 {
// 定义新增Post的返回对象/ a+ `7 B9 i& e/ S1 B: d( f
public record AddPostPayload(Post Post);
s3 [& Z: t* M# f0 T; o" U</code></pre>
5 c' `- r' y/ @" K. z4 p<p>新建<code>Mutation.cs</code>用来定义相关接口:</p>$ ~6 \6 C0 g6 C T! Y2 Q: J
<ul>* a' S' D. ?$ B
<li><code>Mutation.cs</code></li>8 a/ y# b* p9 o# ?$ h8 P
</ul>9 a5 m) t$ r a2 V6 K
<pre><code class="language-c#">namespace PostGraphi.Api.GraphQL;
; X1 W3 n+ @& d" j& s- Z: |& j! V; E$ d' j
public class Mutation
0 i: H6 e! H+ k" z( u{/ I: ?& I, y* h* y
public async Task<AddPostPayload> AddPostAsync(AddPostInput input, [Service] IRepository<Post> repository)1 [& l" t9 m+ r" g. ?: ^8 ^
{0 d0 {1 W5 m6 @8 K* R3 F7 F6 [
return new AddPostPayload(await repository.AddAsync(new Post$ L4 F: i$ q/ r' S: ^
{
; R6 i5 |6 I9 a* c: y ^- c Title = input.Title,4 A0 j8 z7 [7 u: O
Author = input.Author, T4 H9 X. G1 E0 a# c
}));" I! T4 G ~- V6 ^* x \) Y7 E
}
4 f$ j Q$ U2 M4 I5 p; j3 x8 f- h/ R}
$ h2 z$ W; C+ o& D2 `, Y" i- f6 q. h0 u3 h n! z% _, E( ]
</code></pre>( M; M2 w, y' r- e( k
<p>最后在注入服务的地方进行配置:</p>
2 [0 e$ k" H( Q<ul>
( }5 I* I0 v7 f7 Z/ ^4 q# C<li><code>ProgramExtensions.cs</code></li>
8 u' P; I( t* o</ul>4 ^* d, R1 [ ?5 U2 `
<pre><code class="language-c#">builder.Services
! ]; Z; [1 d9 o9 Y .AddGraphQLServer()! g( g4 ~3 ~5 U m+ ~) u9 u
.SetPagingOptions(new PagingOptions
, L i: n8 v* H4 x0 B2 \ {
5 X) [( [1 L6 d3 E% w: X4 L MaxPageSize = 50,; ?* K- z# G7 k# A1 f
IncludeTotalCount = true4 k N% n$ J' c
}): ~1 i2 ]( [' n2 }: m9 e4 U4 q
.AddFiltering()
! d0 g7 \2 D2 S% h; _ O .AddProjections()& j) r9 p0 F9 F* J4 o
.AddSorting()* E! A* w* R# q: C5 F
.AddQueryType<Query>()( N# A. E. r0 b
.AddMutationType<Mutation>()
, n- T% q* j! L) I9 `. o .AddMutationConventions(new MutationConventionOptions$ k' Z# A, q2 h2 ?* R+ X0 C
{
& k9 W& m1 }) P+ r2 b' y ApplyToAllMutations = true,2 U( p( s W. D
InputArgumentName = "input",
- X6 P1 E& W. J( S0 ? InputTypeNamePattern = "{MutationName}Input",
1 p; W- e3 \. w- h/ R8 l/ M* B( Z- c PayloadTypeNamePattern = "{MutationName}Payload",+ ]% k: m1 b7 R" D% p1 z
PayloadErrorTypeNamePattern = "{MutationName}Error",) S& r2 p/ A0 d* I: g
PayloadErrorsFieldName = "errors": b' S# c( h/ \2 q) U8 Q5 S0 N
})
/ `+ \- U5 z h S .AddType<PostType>();
6 I3 e* G9 H$ ^1 w, K* Z7 y8 I</code></pre>9 P; e# k- p1 A. Q9 K7 P1 z# P8 ?5 h
<p>这样就实现了新增Post的需求,下面我们来验证一下。</p>4 U$ C+ b6 ]7 D b
<h2 id="验证">验证</h2>$ T' v% ]' m3 @+ [* r/ k1 ]! o, q
<p>启动<code>Api</code>项目,调用接口:</p>" f& {8 P8 R, ~- d6 r% \4 l
<p><img src="https://img2022.cnblogs.com/blog/2487237/202202/2487237-20220211104544617-1400586374.png" ></p>
3 E' @! X$ j# s; v9 k L<p>终端的日志输出如下:</p>
# O# q' A5 E, Z: O<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']
9 Q$ q4 a+ ]7 N# Q. ^: S3 u* eINSERT INTO "Posts" ("Id", "Abstraction", "Author", "Content", "Created", "CreatedBy", "LastModified", "LastModifiedBy", "Link", "PublishedAt", "Title")
t0 v, X/ V# b2 ]8 P* |# mVALUES (@p0, @p1, @p2, @p3, @p4, @p5, @p6, @p7, @p8, @p9, @p10);
! H& e, M; Z. N. \3 C8 B1 M `[10:45:15 INF] Executed endpoint 'Hot Chocolate GraphQL Pipeline'' {* v8 ~. u9 S' U
</code></pre>1 p( m2 L7 Q" m& k
<p>可以看到新建的Post已经存储到数据库中了,我们可以通过查询接口来获取详情:</p>
5 I% M- ^7 H! x9 D% g8 b+ l<p><img src="https://img2022.cnblogs.com/blog/2487237/202202/2487237-20220211104851825-1533915064.png" ></p>
9 G) f5 }0 Y0 [8 T! e<h2 id="总结">总结</h2>
! S8 @$ S. E" \+ d" ^<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>& f* T5 I6 L. c- g7 b
<p>在下一篇文章中,我们通过<code>Mutation</code>对已有数据进行更新。</p>
D6 L P5 e6 G7 E+ j: B
, z1 X# k4 M6 _0 v @* d |
|