飞雪团队

 找回密码
 立即注册
搜索
热搜: 活动 交友 discuz
查看: 12389|回复: 0

使用Hot Chocolate和.NET 6构建GraphQL应用(8) —— 实现Mutate添加数据 ...

[复制链接]

7903

主题

7991

帖子

2万

积分

管理员

Rank: 9Rank: 9Rank: 9

积分
26039
发表于 2022-2-12 14:35:41 | 显示全部楼层 |阅读模式
0 O1 G1 [# e2 @
<h2 id="系列导航">系列导航</h2>  i  @6 N- f0 F% X* ^3 \
<p><a  href="https://www.cnblogs.com/code4nothing/p/graphql-net6-0.html">使用Hot Chocolate和.NET 6构建GraphQL应用文章索引</a></p>  ^) E% T$ H7 B" K
<h2 id="需求">需求</h2>" L( {6 B" c/ J2 Y
<p>在讨论完GraphQL中的查询需求后,这篇文章我们将演示如何实现GraphQL中的数据添加任务。</p>
7 A& Z) Z/ x" C5 _0 z: Y" @<h2 id="思路">思路</h2>) v: H) N/ D2 Y$ l$ V9 Y
<p>在GraphQL中,对数据进行查询使用<code>query</code>,而对于修改数据则需要使用<code>mutation</code>,包括新增和修改数据。Hot Chocolate在使用Mutation的逻辑上和使用Query的基本一致,但是需要根据需要定义用于创建或者更新的数据对象,所以我们直接进行实现。</p>2 F& x0 P3 |) q3 N' d3 Q$ k/ Y
<h2 id="实现">实现</h2>/ |$ T+ T2 U0 r
<p>为了保持简单,我们先定义以下两个类型:</p>, E2 m4 G7 Y* m; x( C" J
<pre><code class="language-c#">// 定义新增Post的参数
! M* i+ F: G; X0 J4 Tpublic record AddPostInput(string Title, string Author);
6 Z- p: V/ O2 _+ w3 r& q6 j1 d& a5 `6 u! e
// 定义新增Post的返回对象
( M- L. r- S) e9 {! T0 Epublic record AddPostPayload(Post Post);+ T7 ~# j- E1 k: Z3 @" z
</code></pre>
1 u7 h& i, Q5 F# Y- N0 S: k4 _<p>新建<code>Mutation.cs</code>用来定义相关接口:</p>% K! |7 ?8 B' {
<ul>0 S% m8 H7 O9 L; H- S
<li><code>Mutation.cs</code></li>, b& v0 C* o- m7 C' w
</ul>
8 w6 B7 L1 t% e* N<pre><code class="language-c#">namespace PostGraphi.Api.GraphQL;. n4 Y3 u0 {8 E" n9 d' w- _
$ G* @/ D1 f( N  o- k+ {9 B
public class Mutation) o5 V' x0 R% R; [5 X& \% E
{; |: X1 n* P9 {% p
    public async Task&lt;AddPostPayload&gt; AddPostAsync(AddPostInput input, [Service] IRepository&lt;Post&gt; repository)  U7 \' y* j% |5 k
    {% b; m: x" @6 a& l( v
        return new AddPostPayload(await repository.AddAsync(new Post$ V. o( M2 B( \2 K' }
        {
& g6 W2 m; Q8 Y* G% G2 p            Title = input.Title,
+ G0 _) H" _" A  w            Author = input.Author. \1 ~: @; L; j2 u; S: z; h
        }));" z. I9 v& z* @/ n
    }: p$ c( X  O1 U" r# _
}
! |' I' \2 x2 k4 s) v
1 g) y$ L' S. t1 W, E/ w: U</code></pre>4 D/ ], y3 [" }- Z. o8 {% I2 P; Z
<p>最后在注入服务的地方进行配置:</p>
0 s! a/ ~; l1 \<ul>
1 K) R# Z2 l; _) V5 F. V<li><code>ProgramExtensions.cs</code></li>: t% v7 T$ g" q( y" ~: t+ f
</ul>
6 u1 ]4 L5 l9 q$ z" Y<pre><code class="language-c#">builder.Services
' o6 m- a$ m! b    .AddGraphQLServer()2 y! r2 I5 c& B; |. i$ k( ?
    .SetPagingOptions(new PagingOptions
4 m: n0 b7 q! a6 H    {; Y- k+ v4 f/ {4 \+ `, o. c6 i: a
        MaxPageSize = 50,- O& ?# c* {6 H. T
        IncludeTotalCount = true
5 ?- X* v% D. {7 i    })
% k+ |4 Q0 c5 m: u4 f" t4 P- D# G9 _0 C    .AddFiltering()
; O% P8 C9 m% n/ i" n    .AddProjections()
' z" {$ r+ \' A0 k    .AddSorting()8 t: M$ A5 W7 N) c$ t4 v6 m
    .AddQueryType&lt;Query&gt;()
8 _% k2 F$ i9 V3 x7 d) D0 L  t    .AddMutationType&lt;Mutation&gt;()+ m5 B" d4 h$ a/ R
    .AddMutationConventions(new MutationConventionOptions
6 j6 S: z- `1 }    {
! }( O4 h9 N6 w- T) @9 q& \* G        ApplyToAllMutations = true,
* g% O: [& H, _/ |5 J; H        InputArgumentName = "input",
8 {1 M7 r) o- }5 K        InputTypeNamePattern = "{MutationName}Input",
: I# z2 G3 m; ?' J! G& @/ g8 Z        PayloadTypeNamePattern = "{MutationName}Payload",
- g( a8 \% I6 I% V) S1 A4 `        PayloadErrorTypeNamePattern = "{MutationName}Error",
( O  K2 z+ b, Y. V        PayloadErrorsFieldName = "errors"# g! z8 H# Q5 a5 f( q
    })5 ?" H4 R0 K# A
    .AddType&lt;PostType&gt;();3 P1 Z3 Z2 ]! P( M( _6 |
</code></pre>0 G7 F  s+ s7 ]; O1 i
<p>这样就实现了新增Post的需求,下面我们来验证一下。</p>
7 D: U) c$ |$ q+ H4 U2 R<h2 id="验证">验证</h2>! e9 j- b$ x6 e$ R: q* s5 i
<p>启动<code>Api</code>项目,调用接口:</p>
6 b, d- A) H. j  t/ E+ O<p><img src="https://img2022.cnblogs.com/blog/2487237/202202/2487237-20220211104544617-1400586374.png" ></p>) B; l& y4 @; {6 ^  C
<p>终端的日志输出如下:</p>! h& L0 d' \/ p8 d
<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']
$ P) M8 J: |5 n5 U+ q0 L* N5 ^! ~INSERT INTO "Posts" ("Id", "Abstraction", "Author", "Content", "Created", "CreatedBy", "LastModified", "LastModifiedBy", "Link", "PublishedAt", "Title")* i* z' c1 `" x
VALUES (@p0, @p1, @p2, @p3, @p4, @p5, @p6, @p7, @p8, @p9, @p10);: i* r" `) z( K# g: K
[10:45:15 INF] Executed endpoint 'Hot Chocolate GraphQL Pipeline'8 V. ?& c& Z6 e; S
</code></pre>
9 e7 j5 L. O7 x. h* o<p>可以看到新建的Post已经存储到数据库中了,我们可以通过查询接口来获取详情:</p>9 y& j2 Z; _7 T# r$ u" e6 ]
<p><img src="https://img2022.cnblogs.com/blog/2487237/202202/2487237-20220211104851825-1533915064.png" ></p>3 a' Q" A/ `" |* c0 B+ @6 D
<h2 id="总结">总结</h2>' f0 O2 h6 {0 ?" m
<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>
5 {* O1 A2 D5 A" I  x; K3 J8 Y<p>在下一篇文章中,我们通过<code>Mutation</code>对已有数据进行更新。</p>8 g  j$ a9 ^8 Y/ ~: A& B+ g. G" O* L+ T
5 L0 G: p5 R; A4 t
回复

使用道具 举报

懒得打字嘛,点击右侧快捷回复 【右侧内容,后台自定义】
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

手机版|飞雪团队

GMT+8, 2025-11-21 22:29 , Processed in 0.065876 second(s), 21 queries , Gzip On.

Powered by Discuz! X3.4

Copyright © 2001-2021, Tencent Cloud.

快速回复 返回顶部 返回列表