飞雪团队

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

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

[复制链接]

8053

主题

8141

帖子

2万

积分

管理员

Rank: 9Rank: 9Rank: 9

积分
26489
发表于 2022-2-12 14:35:41 | 显示全部楼层 |阅读模式
: R- R* x8 u. M. u2 ^. Q0 t
<h2 id="系列导航">系列导航</h2>
& @, F1 N8 b6 L' `<p><a  href="https://www.cnblogs.com/code4nothing/p/graphql-net6-0.html">使用Hot Chocolate和.NET 6构建GraphQL应用文章索引</a></p>
4 ~" W  D9 M* J" Z! ~% Z. v<h2 id="需求">需求</h2>* B% d" X, U6 J( ~
<p>在讨论完GraphQL中的查询需求后,这篇文章我们将演示如何实现GraphQL中的数据添加任务。</p>2 V" X, l3 E6 a! x, z8 V
<h2 id="思路">思路</h2>6 L7 b  A- g# a+ V
<p>在GraphQL中,对数据进行查询使用<code>query</code>,而对于修改数据则需要使用<code>mutation</code>,包括新增和修改数据。Hot Chocolate在使用Mutation的逻辑上和使用Query的基本一致,但是需要根据需要定义用于创建或者更新的数据对象,所以我们直接进行实现。</p>
! k/ U4 o" S( ?. T0 V<h2 id="实现">实现</h2>5 S) E9 G, h* T* q2 E- e9 Q
<p>为了保持简单,我们先定义以下两个类型:</p>) _0 G( n) {3 _' X) T. t# R( S
<pre><code class="language-c#">// 定义新增Post的参数4 o' E! _4 ]$ E/ U8 W+ D/ Y: V
public record AddPostInput(string Title, string Author);
& V7 o( ~  a& N& ^' y' c2 ~! g4 h0 B3 Z  A& s+ U/ g
// 定义新增Post的返回对象
2 Y  B9 b$ H7 T7 o0 _8 b, l/ npublic record AddPostPayload(Post Post);
, S1 m7 b0 I* L1 K- @% B( t</code></pre>9 a+ R, U' A+ |9 \
<p>新建<code>Mutation.cs</code>用来定义相关接口:</p>
" V/ p3 L: R; k<ul>9 U* t4 C8 W! d5 S2 b) y
<li><code>Mutation.cs</code></li>
! ]" Y# W( E" p8 O- r3 M! y</ul>& d) V. V" g! S- O" C% W, p
<pre><code class="language-c#">namespace PostGraphi.Api.GraphQL;
8 z/ ?8 S3 i8 U; ^, N+ \
0 y( `2 |4 X2 ]4 Wpublic class Mutation
6 F1 J$ h# `/ X6 J: [! Z- H{$ t2 _7 S) \) K: z
    public async Task&lt;AddPostPayload&gt; AddPostAsync(AddPostInput input, [Service] IRepository&lt;Post&gt; repository)
) m) W8 e$ B& \    {
* l/ x2 t, ^' N9 f; b5 H        return new AddPostPayload(await repository.AddAsync(new Post, {7 L+ u1 G% d7 ]6 Z" m
        {* ^" D0 E( C3 r+ g" {- @4 A
            Title = input.Title,5 U6 n  T! L! q. f5 K
            Author = input.Author
' \  s( d2 b+ {, b$ D6 V' y        }));
( H* A# q# a: L" L    }
) a, E3 E- t' [" m}# P1 |" y1 w  O' {, {5 T

& D, v2 h# B( J( B4 `</code></pre>7 ~, L1 u/ R. v" z' @4 ~+ M) v$ u
<p>最后在注入服务的地方进行配置:</p>
2 v' `4 X( t3 V* N9 |. h( G<ul>
9 k$ S* f: j6 S  S* v<li><code>ProgramExtensions.cs</code></li>
+ c$ U9 U% x1 k5 E5 ^. Z</ul>$ e& E& o3 a9 R  J& q1 j' D
<pre><code class="language-c#">builder.Services/ V9 n# u. X  i& F! `( e; h8 t
    .AddGraphQLServer()# X" `$ X8 i$ H# q
    .SetPagingOptions(new PagingOptions, W" c' g  i- F4 Y" j& |6 D
    {
9 ?6 j' m3 y# F5 x$ i. q        MaxPageSize = 50,
$ R9 W9 |. `( t; U        IncludeTotalCount = true# ^$ e2 Z5 X3 B$ P; E- R
    })+ |0 a+ q4 O6 s9 Z8 p. j9 d
    .AddFiltering()
5 W5 ?% v1 E8 V1 `& M    .AddProjections()
+ o# R  y  y( I- u! Z6 `" Z9 R# K    .AddSorting(): ]9 t8 ^( e% l2 ^
    .AddQueryType&lt;Query&gt;()
* s3 K2 I) v" z    .AddMutationType&lt;Mutation&gt;()2 _) B% r: L$ ]0 J
    .AddMutationConventions(new MutationConventionOptions
0 i1 |. ]2 Q( v% p$ B8 I) B$ v3 \    {' _# a4 v( k' Q; C
        ApplyToAllMutations = true,. p5 N* ~% ]* M% E/ x* y
        InputArgumentName = "input",
' V9 y- `4 L+ [/ |5 O! A7 D        InputTypeNamePattern = "{MutationName}Input",! o7 R- }5 g. K. H: ~4 w( X
        PayloadTypeNamePattern = "{MutationName}Payload",, O, A/ b5 C2 s1 E7 M! p/ ^
        PayloadErrorTypeNamePattern = "{MutationName}Error",
/ {3 u( Y4 t" c4 t; W: k* C        PayloadErrorsFieldName = "errors"
* A5 [0 ]9 P. W+ x- r0 L    })1 N/ Y3 \4 f" S( I
    .AddType&lt;PostType&gt;();
6 W  t. c. O( S' H% w</code></pre>
3 B, y) a, r" s) Q) Q, ]<p>这样就实现了新增Post的需求,下面我们来验证一下。</p>
- a6 `& V; s0 k( X<h2 id="验证">验证</h2>
: a2 F: W+ w4 \9 V4 d+ S<p>启动<code>Api</code>项目,调用接口:</p>
8 a0 h3 X1 c" |4 I<p><img src="https://img2022.cnblogs.com/blog/2487237/202202/2487237-20220211104544617-1400586374.png" ></p>1 l0 F5 f: y2 I- n+ f: g' b8 ^
<p>终端的日志输出如下:</p>$ L0 g1 B5 K8 _) |+ W, b
<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']
8 k8 k5 ]1 k+ F/ P* |9 m3 C5 qINSERT INTO "Posts" ("Id", "Abstraction", "Author", "Content", "Created", "CreatedBy", "LastModified", "LastModifiedBy", "Link", "PublishedAt", "Title")
; A7 }, G8 T6 i: ~6 Z: |$ aVALUES (@p0, @p1, @p2, @p3, @p4, @p5, @p6, @p7, @p8, @p9, @p10);; d3 {2 w8 d4 k' n, \
[10:45:15 INF] Executed endpoint 'Hot Chocolate GraphQL Pipeline'
( B0 _( n& L2 Q; H/ y5 W5 K$ j</code></pre>
( o7 i/ O; [) j9 _2 l<p>可以看到新建的Post已经存储到数据库中了,我们可以通过查询接口来获取详情:</p>
: M! R8 j  r3 K/ a+ g5 z<p><img src="https://img2022.cnblogs.com/blog/2487237/202202/2487237-20220211104851825-1533915064.png" ></p>
. [9 b/ k8 I4 n- s' }4 k8 q% {<h2 id="总结">总结</h2>
; L8 B% \  Z* Y- Y! h# 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>2 W/ M/ B7 u9 |( X% |2 J
<p>在下一篇文章中,我们通过<code>Mutation</code>对已有数据进行更新。</p>
% a" W* C( T4 g8 }
9 ~  `1 x. h; k/ B1 k
回复

使用道具 举报

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

本版积分规则

手机版|飞雪团队

GMT+8, 2025-12-11 19:00 , Processed in 0.063302 second(s), 21 queries , Gzip On.

Powered by Discuz! X3.4

Copyright © 2001-2021, Tencent Cloud.

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