飞雪团队

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

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

[复制链接]

8131

主题

8219

帖子

2万

积分

管理员

Rank: 9Rank: 9Rank: 9

积分
26723
发表于 2022-2-12 14:35:41 | 显示全部楼层 |阅读模式
* B3 Y$ K& ?; v& H: u" Y
<h2 id="系列导航">系列导航</h2>
& c" z. F  }2 t; @1 M# d<p><a  href="https://www.cnblogs.com/code4nothing/p/graphql-net6-0.html">使用Hot Chocolate和.NET 6构建GraphQL应用文章索引</a></p>
. [9 ?: G/ _6 O4 s) _<h2 id="需求">需求</h2>
. W" y+ j" ^  ~; K, H' W% d<p>在讨论完GraphQL中的查询需求后,这篇文章我们将演示如何实现GraphQL中的数据添加任务。</p>
/ }8 H% k9 T$ \0 z7 g4 b9 k: O<h2 id="思路">思路</h2>
6 T6 }3 u4 W6 a# g0 L<p>在GraphQL中,对数据进行查询使用<code>query</code>,而对于修改数据则需要使用<code>mutation</code>,包括新增和修改数据。Hot Chocolate在使用Mutation的逻辑上和使用Query的基本一致,但是需要根据需要定义用于创建或者更新的数据对象,所以我们直接进行实现。</p>1 ?/ _* }- O' y9 f% I% @! h" I
<h2 id="实现">实现</h2>
* L* h1 d/ t0 w<p>为了保持简单,我们先定义以下两个类型:</p>
" p# e/ g+ _4 ]# Y0 ^/ K<pre><code class="language-c#">// 定义新增Post的参数
* H. Z+ Q, y7 k5 Ipublic record AddPostInput(string Title, string Author);
0 y6 j  T6 H" R9 F$ f0 s* w  ^* ^  c  \# ]
// 定义新增Post的返回对象8 x6 ^) p- A3 C, S) N5 N* D
public record AddPostPayload(Post Post);$ s/ e9 ?1 B" s# p3 r
</code></pre>
# |( O0 D" Q2 e* G<p>新建<code>Mutation.cs</code>用来定义相关接口:</p>
6 w9 A# W; q6 q) h<ul>
# x8 p5 n, D5 l4 J1 I1 u<li><code>Mutation.cs</code></li>( Y  e7 \. f! X: O4 ?  C
</ul>
2 o+ i" }9 `( |; U<pre><code class="language-c#">namespace PostGraphi.Api.GraphQL;
. {3 B  W# @/ q, p2 v- B; r
0 v7 K  A( F3 J0 l6 _( V- Cpublic class Mutation
* B) ]3 l) j, p8 T{
" ^0 D; [5 i0 W, [. n    public async Task&lt;AddPostPayload&gt; AddPostAsync(AddPostInput input, [Service] IRepository&lt;Post&gt; repository)
7 P: j% d  ~, }" y    {1 `4 n4 ^! l  V8 q# _
        return new AddPostPayload(await repository.AddAsync(new Post
5 u. n7 R* c/ W        {
1 P( t5 L3 \. z7 d) L4 S7 m' W            Title = input.Title,
7 s8 _+ a/ ?$ D1 U) M            Author = input.Author3 A9 l7 _9 D& I
        }));  C) t$ R3 T) v; i
    }3 w2 `( K6 V1 U0 n
}
+ R( Q$ l" F4 n. \8 D& l) }" a
  s1 U+ a+ g5 w* p% ~, J</code></pre>9 V, V7 L. V8 `; `, j
<p>最后在注入服务的地方进行配置:</p>
% N. Z- b+ {, w" z6 m! _<ul>2 r2 w) j8 n& V/ W$ f3 c& \6 G
<li><code>ProgramExtensions.cs</code></li>
& b& s2 ~2 J& W6 i( b; t- o</ul>
1 o! r/ H% q' R/ X, Q7 s# w<pre><code class="language-c#">builder.Services
" ~3 e4 h5 ^. B    .AddGraphQLServer()
4 s- ]( K+ Z4 K0 q    .SetPagingOptions(new PagingOptions, [* O7 C$ X+ ^. L
    {
3 w# c/ H. t' \! A        MaxPageSize = 50,
. @) c) `$ z4 o( q        IncludeTotalCount = true) W$ x$ S/ m$ T: z( x  T+ [6 Q
    })
2 V! Y; }) r% [( M3 N    .AddFiltering()9 {6 S% K0 O1 z. i
    .AddProjections()8 I0 C) ?/ H, ~" i) K/ _  U$ ^2 ~
    .AddSorting()
; j2 R/ Q6 h0 z. p$ D    .AddQueryType&lt;Query&gt;(). C& @# F+ o; Y
    .AddMutationType&lt;Mutation&gt;()
& M4 b" a' I. D: p( E    .AddMutationConventions(new MutationConventionOptions0 o9 E: _1 S5 y) u9 C
    {$ h" P4 M, e- ?2 Y# O8 I4 e
        ApplyToAllMutations = true,
% }& z; r, D6 v: Z, }1 Q        InputArgumentName = "input",4 u9 r8 D$ X7 O8 m& \
        InputTypeNamePattern = "{MutationName}Input",( _% p7 |2 A1 d, g' r& ?
        PayloadTypeNamePattern = "{MutationName}Payload",8 o& X3 O, R* R  I5 R' s( |( v
        PayloadErrorTypeNamePattern = "{MutationName}Error",: P- m& H% L5 H
        PayloadErrorsFieldName = "errors"5 g6 G5 }0 F! I* y% S, m
    })8 U: O# Y) J1 A! S1 H
    .AddType&lt;PostType&gt;();8 l/ D5 ?; B6 K; r* F* V. @1 Z
</code></pre>
) Z) S: y% d8 L( x8 u/ b<p>这样就实现了新增Post的需求,下面我们来验证一下。</p>* J3 N2 B" w, g" e; W
<h2 id="验证">验证</h2>
1 [6 u7 c- D7 j  r% ]% D<p>启动<code>Api</code>项目,调用接口:</p>% v) N2 }3 g: p$ ~) u
<p><img src="https://img2022.cnblogs.com/blog/2487237/202202/2487237-20220211104544617-1400586374.png" ></p>
) E/ S3 n+ S4 N) \<p>终端的日志输出如下:</p>
( S0 P, ~8 F" @1 v) @<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']
( W  F7 N; W7 p# {8 |* f3 o' mINSERT INTO "Posts" ("Id", "Abstraction", "Author", "Content", "Created", "CreatedBy", "LastModified", "LastModifiedBy", "Link", "PublishedAt", "Title")* p+ \2 |* i2 ]! H+ G
VALUES (@p0, @p1, @p2, @p3, @p4, @p5, @p6, @p7, @p8, @p9, @p10);# L0 Z, }6 S! z( J( h$ Y
[10:45:15 INF] Executed endpoint 'Hot Chocolate GraphQL Pipeline'; @$ V7 F" i6 L; U
</code></pre>
# _9 z8 n9 C; I/ O<p>可以看到新建的Post已经存储到数据库中了,我们可以通过查询接口来获取详情:</p>* e* T+ e& B  U' t$ t8 F9 a5 x4 P
<p><img src="https://img2022.cnblogs.com/blog/2487237/202202/2487237-20220211104851825-1533915064.png" ></p>
6 [( p$ |7 R1 s5 J, H. |: J6 K<h2 id="总结">总结</h2>, C1 Q& I. ~: e1 [
<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>7 ?% N" F9 y! p  L
<p>在下一篇文章中,我们通过<code>Mutation</code>对已有数据进行更新。</p>& ~2 d# V2 Y" ~! W& }* @
+ s8 Y  ^0 @# g
回复

使用道具 举报

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

本版积分规则

手机版|飞雪团队

GMT+8, 2025-12-27 08:02 , Processed in 0.060408 second(s), 21 queries , Gzip On.

Powered by Discuz! X3.4

Copyright © 2001-2021, Tencent Cloud.

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