飞雪团队

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

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

[复制链接]

8116

主题

8204

帖子

2万

积分

管理员

Rank: 9Rank: 9Rank: 9

积分
26678
发表于 2022-2-12 14:35:41 | 显示全部楼层 |阅读模式
; v7 S' D: M" N! x* `/ X4 z2 F$ K
<h2 id="系列导航">系列导航</h2>7 j! G/ b+ I0 Y3 V7 ]/ U; A
<p><a  href="https://www.cnblogs.com/code4nothing/p/graphql-net6-0.html">使用Hot Chocolate和.NET 6构建GraphQL应用文章索引</a></p>4 A% r) B% o5 }+ b0 _
<h2 id="需求">需求</h2>* v6 @" e6 u) L! F. x* b
<p>在讨论完GraphQL中的查询需求后,这篇文章我们将演示如何实现GraphQL中的数据添加任务。</p>) a  [) T7 k. m+ `
<h2 id="思路">思路</h2>9 `6 b: ^4 {9 A+ q7 e+ d9 F# R
<p>在GraphQL中,对数据进行查询使用<code>query</code>,而对于修改数据则需要使用<code>mutation</code>,包括新增和修改数据。Hot Chocolate在使用Mutation的逻辑上和使用Query的基本一致,但是需要根据需要定义用于创建或者更新的数据对象,所以我们直接进行实现。</p>
& H* d; R. A( U. [6 @+ Z/ o<h2 id="实现">实现</h2>
+ @% R6 C  q3 m" a& L* p9 j: e<p>为了保持简单,我们先定义以下两个类型:</p>
5 }$ j, F: E6 P<pre><code class="language-c#">// 定义新增Post的参数7 g: A* w- y- Q' Q2 Y$ D+ Y
public record AddPostInput(string Title, string Author);" G5 A. \* s) B6 j! K
! G7 I, u& J# N7 `
// 定义新增Post的返回对象1 f& P8 A/ b0 Q6 j+ B4 z0 n& d
public record AddPostPayload(Post Post);
  p6 [8 B# t. x. E' h! k</code></pre>
$ O' ^3 R. }& D1 D( q/ Y; Y% }6 j<p>新建<code>Mutation.cs</code>用来定义相关接口:</p>
- c, N) t- b1 {! a& {0 N<ul>
8 n! H' {4 t- F; k5 I) z1 M  l<li><code>Mutation.cs</code></li>
) a" p; {8 @8 @! C) G% U* X& l</ul>1 P+ e. L/ |! L3 f% Y! v( t
<pre><code class="language-c#">namespace PostGraphi.Api.GraphQL;
; @1 V( L* `1 n' Z5 {+ F3 b3 ^
2 F+ }  n# h/ T  `public class Mutation* j% L$ Z0 f1 _  e6 s& p
{. U3 R* X& U8 T0 b- z- o+ s
    public async Task&lt;AddPostPayload&gt; AddPostAsync(AddPostInput input, [Service] IRepository&lt;Post&gt; repository)
8 W8 e1 Q& U$ ~3 Y5 X' x    {. C. T. V% h0 Z) V" h3 _7 B
        return new AddPostPayload(await repository.AddAsync(new Post
! d9 H* T, _3 {+ U        {
- ~+ Y9 {1 R  y. R: Y* Q& D. I# }            Title = input.Title,
) D3 R! T% h: K  G/ R$ y% q* l            Author = input.Author
' q. Z. s; L# F' x" W        }));: }$ b: G) \& b; H
    }
( ^5 \' t/ T3 G) }% Q- m% l- @+ `}
  S5 K* P5 d; n3 X9 Q5 v
) L( B2 @; Q, \) i5 G2 ], M2 C</code></pre>( U5 p& U5 ^: D) E
<p>最后在注入服务的地方进行配置:</p>
+ b( J( X% c/ Z# D<ul>
/ \/ `6 A5 o/ Y& D" p<li><code>ProgramExtensions.cs</code></li>. p0 X5 X3 \0 {+ d6 p
</ul>
' O8 P9 s3 Q6 ~) v# Y3 B( v) \<pre><code class="language-c#">builder.Services' c& f8 g) v9 [8 o; W
    .AddGraphQLServer()1 h0 ?; l6 b5 y$ k0 h& w
    .SetPagingOptions(new PagingOptions4 e4 P/ [( k' v% H5 M* `5 W
    {
/ O- v% z9 K/ X: `2 B' k) j        MaxPageSize = 50,
& p; Y  m+ ]; {* x9 a! H# l        IncludeTotalCount = true
3 U$ z* s4 a- a: B, h3 o    })
9 ^' z7 d0 N; t0 k( m    .AddFiltering()6 `3 w! A( V" X6 ?
    .AddProjections()
% `- G$ |7 f" r& H$ p    .AddSorting()
4 ^) z& A/ M4 }4 F    .AddQueryType&lt;Query&gt;()
6 B/ x; N* [1 G% `* p; V    .AddMutationType&lt;Mutation&gt;()
& e3 ?+ \3 v! U( J9 K; i) _! [* C    .AddMutationConventions(new MutationConventionOptions
& |( l6 g0 T/ N: ]; D    {! B0 S' l# I' p/ k  o
        ApplyToAllMutations = true,
0 W0 Z6 \$ H- B* n        InputArgumentName = "input",5 _- d7 x2 \8 K1 E, P
        InputTypeNamePattern = "{MutationName}Input",
3 D9 J: N2 s5 K0 |6 B) R+ d' A        PayloadTypeNamePattern = "{MutationName}Payload",
" S* A" e0 |! n5 o/ U6 {        PayloadErrorTypeNamePattern = "{MutationName}Error",
0 i' R. \6 F6 L* O, C" T        PayloadErrorsFieldName = "errors"/ M3 f0 C* N8 z3 P  x4 |/ m% J( F
    })% q# W% B8 z+ ?. r* u6 h
    .AddType&lt;PostType&gt;();% D' p4 L2 P+ A3 d$ Z
</code></pre>! l' q7 K* x5 A
<p>这样就实现了新增Post的需求,下面我们来验证一下。</p>
/ E  x4 K8 C; B8 D* _' ^$ C<h2 id="验证">验证</h2>
5 e7 K: T  _2 K1 |9 P<p>启动<code>Api</code>项目,调用接口:</p>9 I0 M* N' [; a! x9 f
<p><img src="https://img2022.cnblogs.com/blog/2487237/202202/2487237-20220211104544617-1400586374.png" ></p>$ ^3 w0 G4 i8 W& [' g
<p>终端的日志输出如下:</p>/ z  S# h3 ]7 k/ ~# f
<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']! O3 J1 `% T! o. w1 b  G, B& C
INSERT INTO "Posts" ("Id", "Abstraction", "Author", "Content", "Created", "CreatedBy", "LastModified", "LastModifiedBy", "Link", "PublishedAt", "Title"). z1 r4 U- u9 C0 J6 ]9 {3 f
VALUES (@p0, @p1, @p2, @p3, @p4, @p5, @p6, @p7, @p8, @p9, @p10);8 R# ?4 z2 o/ B! m, K
[10:45:15 INF] Executed endpoint 'Hot Chocolate GraphQL Pipeline'
3 E8 Z, c$ Z2 w6 [! S! _</code></pre>  f) g5 O6 k! {6 u* i0 Q
<p>可以看到新建的Post已经存储到数据库中了,我们可以通过查询接口来获取详情:</p>! S0 ^6 V' C" w: Z  |
<p><img src="https://img2022.cnblogs.com/blog/2487237/202202/2487237-20220211104851825-1533915064.png" ></p>
# _$ `/ l& Y2 B* q: {) J<h2 id="总结">总结</h2>
" s& q: r1 G8 C4 ^  X- f" }  Z<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>
& ]; ~# N6 V1 z. ?3 A4 T<p>在下一篇文章中,我们通过<code>Mutation</code>对已有数据进行更新。</p>$ v5 {/ ~% F9 `" w

1 U3 p' ?# T: D- M! a
回复

使用道具 举报

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

本版积分规则

手机版|飞雪团队

GMT+8, 2025-12-21 16:24 , Processed in 0.060330 second(s), 21 queries , Gzip On.

Powered by Discuz! X3.4

Copyright © 2001-2021, Tencent Cloud.

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