|
|
) T/ [% s- @8 H7 j8 K<h2 id="系列导航">系列导航</h2>0 v' c0 \2 a9 \8 y
<p><a href="https://www.cnblogs.com/code4nothing/p/graphql-net6-0.html">使用Hot Chocolate和.NET 6构建GraphQL应用文章索引</a></p>
1 `" {; v* ?& _- }<h2 id="需求">需求</h2>- l7 o) Q, m$ g
<p>在讨论完GraphQL中的查询需求后,这篇文章我们将演示如何实现GraphQL中的数据添加任务。</p>
3 A T* n" Q- {, j5 {8 N<h2 id="思路">思路</h2>0 W1 e3 g; o) F
<p>在GraphQL中,对数据进行查询使用<code>query</code>,而对于修改数据则需要使用<code>mutation</code>,包括新增和修改数据。Hot Chocolate在使用Mutation的逻辑上和使用Query的基本一致,但是需要根据需要定义用于创建或者更新的数据对象,所以我们直接进行实现。</p>
; R+ {# `0 N* C' X' Q5 d$ r# I! b$ s<h2 id="实现">实现</h2>3 Z" X& n! b6 X y2 x" u: t; a
<p>为了保持简单,我们先定义以下两个类型:</p>
) e- T9 n' D- Y' X8 d, h) \; [7 p<pre><code class="language-c#">// 定义新增Post的参数" a8 H2 T5 ?" b, k: [) @
public record AddPostInput(string Title, string Author);
1 O- ]0 o( P( Y& ^% |" Z
, x4 A4 C6 W! x: k// 定义新增Post的返回对象
) C5 [. Q! S; Kpublic record AddPostPayload(Post Post);% ~3 E7 L& r" {4 [5 |2 c0 j9 n
</code></pre>: w* k* E7 k; O$ a5 n7 `9 T
<p>新建<code>Mutation.cs</code>用来定义相关接口:</p>
8 D( f+ b2 Y4 l' s8 g/ c7 j+ t<ul>; B% c: q- v" {4 j
<li><code>Mutation.cs</code></li>
3 }- j- o( g9 H( y</ul>
4 Y: Y0 A- Z$ f3 ^ X<pre><code class="language-c#">namespace PostGraphi.Api.GraphQL;9 L& Q ?' m" K, c
: P% `" S5 i0 b
public class Mutation
" a% X5 B* e- @9 O{
: u7 ^: Z1 m4 E' J public async Task<AddPostPayload> AddPostAsync(AddPostInput input, [Service] IRepository<Post> repository)1 x/ r* ?( \7 b/ `
{
/ D) \/ z) g( c) }5 q return new AddPostPayload(await repository.AddAsync(new Post
G6 A/ g, p! x# s0 J* M' y8 V {
6 `* V) h) n! d* n. e- Z, N Title = input.Title,
) W/ Y8 l! ^9 s4 _9 {; D: B Author = input.Author
2 A/ f. l0 {; N1 b% E8 P& Z% ^% y% s7 r }));
$ G0 R! M# L& u7 `% b- } }8 j% O) s7 M" _, ^7 P6 ~5 [7 J
}, S/ N1 u' N) |+ x! W" D7 ~
0 e0 Q; h& w# }) G
</code></pre>
C7 F4 N, Q8 Z- N4 [ j<p>最后在注入服务的地方进行配置:</p>, g' Q; f; L" O0 s2 o" R7 r% i
<ul>- G9 s% v; k. r- K
<li><code>ProgramExtensions.cs</code></li>
/ e2 G s S& S</ul># n0 i3 z9 F$ t0 N6 M5 K3 }+ v/ H
<pre><code class="language-c#">builder.Services
: q H, X. h, z9 A- ]0 N6 j .AddGraphQLServer()' \& a3 {/ c; t9 Z
.SetPagingOptions(new PagingOptions7 ^$ l; m E$ V' w7 J
{, q5 a4 ?) w# D
MaxPageSize = 50,/ Z' R* v! W1 e' U# i; D
IncludeTotalCount = true
t) X6 m% v6 e( N4 u })8 T, j7 h* F2 F+ L) M& E: ]
.AddFiltering()
, F; Z/ {) s- e+ N: c3 g0 @ .AddProjections(); U0 ~) [; |" G
.AddSorting()1 G/ @5 h, Y- c. E$ y# ~& T; e% I
.AddQueryType<Query>()
$ _5 j0 C3 u( \. ]- e .AddMutationType<Mutation>()
& A u* k& t' O( ^4 H, A9 a; R .AddMutationConventions(new MutationConventionOptions! E* y7 J0 d" K2 y# x- @4 P4 x
{
( q# C/ C9 [# X3 n1 h8 }8 H6 u ApplyToAllMutations = true,
2 [: \6 k; d3 Q5 L1 l2 J InputArgumentName = "input",# D7 k8 ?) c6 a3 p4 m
InputTypeNamePattern = "{MutationName}Input",8 C9 r f {9 ~. q0 i
PayloadTypeNamePattern = "{MutationName}Payload",
+ ^6 b- \, _7 P9 S" {3 v PayloadErrorTypeNamePattern = "{MutationName}Error", M0 \! t9 U/ g' ~
PayloadErrorsFieldName = "errors"
% A( j+ M* y& s3 { })3 ?& l( r" V2 Y3 ?
.AddType<PostType>();
) c. m# @: v2 `8 {5 \' k</code></pre>5 v' i9 a, j( b1 Q% H5 o
<p>这样就实现了新增Post的需求,下面我们来验证一下。</p>
; \, z$ t4 b9 {6 X6 |0 h8 x<h2 id="验证">验证</h2>$ B* [2 A" ?- D0 P& e! l
<p>启动<code>Api</code>项目,调用接口:</p>
. t$ U. N+ E, q<p><img src="https://img2022.cnblogs.com/blog/2487237/202202/2487237-20220211104544617-1400586374.png" ></p># Y" j- Y* [1 l" {
<p>终端的日志输出如下:</p># L- y, S* `) _; W n: |& 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']1 b/ s/ M- R9 F4 x
INSERT INTO "Posts" ("Id", "Abstraction", "Author", "Content", "Created", "CreatedBy", "LastModified", "LastModifiedBy", "Link", "PublishedAt", "Title")
6 E6 v% `2 Z8 M% g+ SVALUES (@p0, @p1, @p2, @p3, @p4, @p5, @p6, @p7, @p8, @p9, @p10);
) j* u4 @9 q; X2 G- \[10:45:15 INF] Executed endpoint 'Hot Chocolate GraphQL Pipeline'
- U. H2 {' {5 L: f+ l: m</code></pre>' {8 v/ B& q$ V' \3 E/ \
<p>可以看到新建的Post已经存储到数据库中了,我们可以通过查询接口来获取详情:</p>
; y i' v2 q6 i8 q8 }: y<p><img src="https://img2022.cnblogs.com/blog/2487237/202202/2487237-20220211104851825-1533915064.png" ></p>
7 S/ [' B* w/ Y! m7 t8 X3 u<h2 id="总结">总结</h2>
$ u9 g1 q! N$ e* w7 [+ l8 p4 Q<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>
! g4 F$ k( v3 I- q$ p0 C, T<p>在下一篇文章中,我们通过<code>Mutation</code>对已有数据进行更新。</p># o- O' d/ y4 @3 T+ R. W
! G2 A* U1 P5 J( y4 H
|
|