|
|
7 Y/ l1 z* l+ j3 q* K
<h2 id="系列导航">系列导航</h2>$ i, A: w9 L: \) j$ F8 x
<p><a href="https://www.cnblogs.com/code4nothing/p/graphql-net6-0.html">使用Hot Chocolate和.NET 6构建GraphQL应用文章索引</a></p>
/ w. s+ \0 r0 O n# y<h2 id="需求">需求</h2>
+ ^7 _& G2 ~1 ? m" R, j2 T<p>在讨论完GraphQL中的查询需求后,这篇文章我们将演示如何实现GraphQL中的数据添加任务。</p> u+ Y( U0 I) F7 u
<h2 id="思路">思路</h2>
5 r4 A' E& n# T4 R<p>在GraphQL中,对数据进行查询使用<code>query</code>,而对于修改数据则需要使用<code>mutation</code>,包括新增和修改数据。Hot Chocolate在使用Mutation的逻辑上和使用Query的基本一致,但是需要根据需要定义用于创建或者更新的数据对象,所以我们直接进行实现。</p>5 ]# Z1 h' k. F) a2 j* U
<h2 id="实现">实现</h2>
/ w/ [7 i6 P( Y. G& N% m<p>为了保持简单,我们先定义以下两个类型:</p>7 Y0 }% W: L/ ^6 i# C
<pre><code class="language-c#">// 定义新增Post的参数' P! r& N) d U8 A8 @
public record AddPostInput(string Title, string Author);% k9 _7 t }9 ^& F+ q8 X
5 h, W+ S3 m \% t9 L4 o' s2 L// 定义新增Post的返回对象
- Q+ j! t/ b! A; P( N: A+ G fpublic record AddPostPayload(Post Post);4 f! {; a/ @3 s3 ?
</code></pre>
* @: i: y6 U$ c$ y6 P* H<p>新建<code>Mutation.cs</code>用来定义相关接口:</p>$ P/ C- Y2 e4 W+ J% W
<ul>% s0 `! Q0 B- [& e
<li><code>Mutation.cs</code></li>
3 s! V/ I7 g( }6 A</ul>1 v1 w/ e* q. A6 W& E+ F
<pre><code class="language-c#">namespace PostGraphi.Api.GraphQL;) A( W6 c7 W" q" h
* F3 I+ T# E3 n' Cpublic class Mutation
: A' m1 ~1 V2 L! R& b; d0 o{
/ i, R. o$ q2 h5 a9 H! f( x public async Task<AddPostPayload> AddPostAsync(AddPostInput input, [Service] IRepository<Post> repository)1 L- E* U- P' O6 M' ]
{# ^4 `6 M" }( M; ]5 ?# P, x' `5 H
return new AddPostPayload(await repository.AddAsync(new Post, {6 w6 s; `8 I# ^/ X! t
{* z# e6 o9 u5 F- g6 v6 u1 C8 G
Title = input.Title,0 l* n& k& P! b' l- t7 S: S4 J6 [
Author = input.Author q- u! T1 F$ g9 C
}));
8 t+ b: [1 N. V( T) y }5 S6 H# W; x# ^* F1 c
}' Y* q7 o" B9 H4 @1 n1 _5 f9 P
2 Y6 B+ c6 x- Y
</code></pre>
4 i1 b! h q, F1 p) `, T<p>最后在注入服务的地方进行配置:</p>
8 F8 Q$ y8 W% u<ul>
3 i* v' k2 f* o1 @* u<li><code>ProgramExtensions.cs</code></li>
6 G+ g& t8 ~: r/ n3 G5 K# T B</ul>
: W5 q! J3 Y6 R; }. v* `$ |<pre><code class="language-c#">builder.Services* [+ ?" o3 Z- W/ a# Y# n& P
.AddGraphQLServer(), |+ ~' K7 S1 s5 I8 R7 O9 H) h& Z
.SetPagingOptions(new PagingOptions, A5 J2 S. _" ]$ e) m2 m
{
4 r0 a* E# M+ W MaxPageSize = 50,
l" w" M3 U" a9 J IncludeTotalCount = true6 v/ R8 n) M# J; N: }" U, X
})2 W& g8 \8 g6 i5 G1 |) ^
.AddFiltering()
7 t" s2 r% r% ~$ D& ? .AddProjections()8 k8 l5 A$ [" k0 u
.AddSorting()
1 d c' |$ E. a; V! h% @( C7 {6 o* ]9 x g .AddQueryType<Query>()* v- W3 m9 o% |* ?* A& O: h
.AddMutationType<Mutation>()
# g9 f6 i! P* B8 W0 ^! O; } .AddMutationConventions(new MutationConventionOptions
8 W+ h8 R+ X: m& W0 ^( q {/ M- ?4 [6 \$ m1 o
ApplyToAllMutations = true,2 z7 Y. g8 Z( l+ z" ~: } s) G
InputArgumentName = "input",
" D ^0 I- @1 @* E. q4 W7 r7 s InputTypeNamePattern = "{MutationName}Input",
; j0 _# j0 u- c7 W PayloadTypeNamePattern = "{MutationName}Payload",
% E, o) D$ p" |- E7 k! a PayloadErrorTypeNamePattern = "{MutationName}Error",
: P' ^4 N7 B7 G1 e" J PayloadErrorsFieldName = "errors"
, V: Y* |. V# }6 g })
4 H: D: B4 ]6 c6 O3 ^( G .AddType<PostType>();1 T% t9 w' l; a' q
</code></pre>5 j' N% Y. u2 v |" V. E
<p>这样就实现了新增Post的需求,下面我们来验证一下。</p>
9 X [9 O1 ?# G<h2 id="验证">验证</h2> c% v7 s: E% t" A
<p>启动<code>Api</code>项目,调用接口:</p>
0 X1 B3 {3 n8 U<p><img src="https://img2022.cnblogs.com/blog/2487237/202202/2487237-20220211104544617-1400586374.png" ></p>& a4 u5 A1 B9 l2 W+ L6 U7 I
<p>终端的日志输出如下:</p>
9 f, J% Y: O3 d. ^$ U) 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']
( J6 J+ Q) E+ [) [INSERT INTO "Posts" ("Id", "Abstraction", "Author", "Content", "Created", "CreatedBy", "LastModified", "LastModifiedBy", "Link", "PublishedAt", "Title")
% A5 k1 {! B; z: d# P) oVALUES (@p0, @p1, @p2, @p3, @p4, @p5, @p6, @p7, @p8, @p9, @p10);: ]1 n7 }. m: Q; d8 y
[10:45:15 INF] Executed endpoint 'Hot Chocolate GraphQL Pipeline'0 E7 m' I* M+ Z% `
</code></pre>
6 z$ Z7 h0 `; Q# y- }<p>可以看到新建的Post已经存储到数据库中了,我们可以通过查询接口来获取详情:</p>
2 Y0 F/ k, G3 t! K% m2 B<p><img src="https://img2022.cnblogs.com/blog/2487237/202202/2487237-20220211104851825-1533915064.png" ></p>5 W/ T/ ?/ X! w0 t' t, e6 W' z
<h2 id="总结">总结</h2>
; W$ C7 K0 g8 Z) _0 [$ a<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( F# h3 | R) ^<p>在下一篇文章中,我们通过<code>Mutation</code>对已有数据进行更新。</p>3 U$ h, P& g6 z/ N, d2 V; Y- ~
- S$ u, P( h H6 ~" h& d3 ?
|
|