|
1 e0 D7 N& A1 a; h I" o, G
<h2 id="系列导航">系列导航</h2>
, C) R1 ]+ k/ \" {& |3 n<p><a href="https://www.cnblogs.com/code4nothing/p/graphql-net6-0.html">使用Hot Chocolate和.NET 6构建GraphQL应用文章索引</a></p>
; M5 Z3 i2 z: {" W( J: }. a* S<h2 id="需求">需求</h2>
& p. I; t! n/ ^2 w& h5 P C9 [<p>在讨论完GraphQL中的查询需求后,这篇文章我们将演示如何实现GraphQL中的数据添加任务。</p>
: L& I+ V3 f+ H<h2 id="思路">思路</h2>
8 F$ i' N# B8 L9 `" W5 `<p>在GraphQL中,对数据进行查询使用<code>query</code>,而对于修改数据则需要使用<code>mutation</code>,包括新增和修改数据。Hot Chocolate在使用Mutation的逻辑上和使用Query的基本一致,但是需要根据需要定义用于创建或者更新的数据对象,所以我们直接进行实现。</p>
4 [6 O8 }4 \: j' }" ~: \! z<h2 id="实现">实现</h2>
6 [) e( X2 w4 m& p5 X<p>为了保持简单,我们先定义以下两个类型:</p> t/ ~2 N( i* q2 O
<pre><code class="language-c#">// 定义新增Post的参数5 U) `, t" C ~$ _2 u7 B+ ?
public record AddPostInput(string Title, string Author);. `, }8 @+ l* N
) S, ^9 q b3 a" ]// 定义新增Post的返回对象8 q2 e% ]; R( N
public record AddPostPayload(Post Post);6 _) J! h- }3 M7 i
</code></pre>
" y$ s7 v. @' M6 w<p>新建<code>Mutation.cs</code>用来定义相关接口:</p>
6 j: A9 m3 `2 o. E$ ` R<ul>3 L+ r) }- g4 T& z! K
<li><code>Mutation.cs</code></li>( ~! w1 s+ @: y& p* J @ X% s
</ul>7 C; Z+ [" k9 G# j
<pre><code class="language-c#">namespace PostGraphi.Api.GraphQL;
/ l3 v* V( k z0 {% g r, R5 j- L, m$ H- ^' d* ?; g
public class Mutation
; \7 @9 M! S9 ]2 G5 w1 h: M{
~* e5 m9 ~8 e! T i! T public async Task<AddPostPayload> AddPostAsync(AddPostInput input, [Service] IRepository<Post> repository)1 B% \* o" s9 `$ I! ~/ s; L. f2 B+ i" V
{
3 h) G0 T3 K2 V* X. v2 Z0 }! H; }* X2 Z return new AddPostPayload(await repository.AddAsync(new Post
$ G4 w" Q+ H& i# M1 c' G0 ?2 S {+ q5 Q& h( {" Y$ T; O: f
Title = input.Title,
, ]- S) ?+ B) V1 V Author = input.Author
& u+ F& K; h! i3 u }));: J/ q1 @" S) j
}/ Z; p, U" J5 |* v, k
}
9 X1 P, u+ }7 b7 B @& j. g6 ^0 Z1 K/ H+ u
</code></pre>
% F6 [6 f2 I6 y<p>最后在注入服务的地方进行配置:</p>
: t+ }; ?9 J! f2 h& h<ul>( X* l0 G6 G5 m! G
<li><code>ProgramExtensions.cs</code></li>
% U; Y2 l1 G0 r0 C4 Q</ul>0 E7 Y) e& Q- W5 F4 G% d2 r: c
<pre><code class="language-c#">builder.Services
/ n& b: O# ~) V0 [1 d .AddGraphQLServer()4 v. V$ o& A( w/ v* N
.SetPagingOptions(new PagingOptions
3 i4 d* R1 ~6 B {
: w# I6 E: w5 Y0 R& k9 o/ A MaxPageSize = 50,
/ w& p$ b; ]9 s0 z3 D5 J IncludeTotalCount = true9 @0 T+ G0 h- g4 M/ H3 A
})1 J ~0 r" i- M2 ~
.AddFiltering()
) X6 y. i# P! j, L, m5 A& ] .AddProjections() p; l" G2 i" x- w! I" `! J% ?% p, s- H
.AddSorting()
! Y, P& r) t" W; y% d .AddQueryType<Query>()( _. ^& U! Y8 u m6 l
.AddMutationType<Mutation>()
5 @2 L; ^- ?6 m \& O( j2 m* ? .AddMutationConventions(new MutationConventionOptions
D# d$ |, f+ o4 l3 v2 E4 ? {
) e# R; o9 f+ O. J. {% A ApplyToAllMutations = true,
; t* R+ B* T2 u1 k InputArgumentName = "input",$ E7 s" g( A$ s7 B
InputTypeNamePattern = "{MutationName}Input",6 m9 W" Z) d4 w2 [
PayloadTypeNamePattern = "{MutationName}Payload",
) W/ z, V: g {5 s: { PayloadErrorTypeNamePattern = "{MutationName}Error",! q0 u! R4 K7 ^
PayloadErrorsFieldName = "errors"
! _( J. v- j! c* N" _. D* f })- k2 Y( ~' G3 m
.AddType<PostType>();
0 C z! C1 ~& } P5 l D7 F</code></pre>5 t( @, j, ]" x. }+ ]( F
<p>这样就实现了新增Post的需求,下面我们来验证一下。</p>
G; E9 j; ^: ]% Z9 `( v! \& p<h2 id="验证">验证</h2>/ } C% q( m0 I- E6 M Y
<p>启动<code>Api</code>项目,调用接口:</p> f' x2 N r; U7 B+ h; Z1 L
<p><img src="https://img2022.cnblogs.com/blog/2487237/202202/2487237-20220211104544617-1400586374.png" ></p>
2 O9 x3 |9 }1 Q" X/ }3 @<p>终端的日志输出如下:</p>
2 l7 `9 d8 z: Z, [2 |1 ~8 F<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']) S4 r8 b5 P0 p& j
INSERT INTO "Posts" ("Id", "Abstraction", "Author", "Content", "Created", "CreatedBy", "LastModified", "LastModifiedBy", "Link", "PublishedAt", "Title")
% x: V1 q/ Z" P2 K, \' DVALUES (@p0, @p1, @p2, @p3, @p4, @p5, @p6, @p7, @p8, @p9, @p10);
, z5 P; z* y7 a) X[10:45:15 INF] Executed endpoint 'Hot Chocolate GraphQL Pipeline'/ I- z+ k( Y' Q' B- E) t) n8 N
</code></pre>
) V) d( t* O3 N, D# k<p>可以看到新建的Post已经存储到数据库中了,我们可以通过查询接口来获取详情:</p>
; z! n- N9 D- V<p><img src="https://img2022.cnblogs.com/blog/2487237/202202/2487237-20220211104851825-1533915064.png" ></p>6 e( g; c8 u U: j; F, |0 B- J
<h2 id="总结">总结</h2>6 Z; `: [3 C c2 I8 x. u" h
<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>! I$ |' |: P9 u8 w
<p>在下一篇文章中,我们通过<code>Mutation</code>对已有数据进行更新。</p>* `5 U* o& o4 i2 m. T, C0 f ]
5 K& k" y6 G0 ? H
|
|