飞雪团队

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

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

[复制链接]

8087

主题

8175

帖子

2万

积分

管理员

Rank: 9Rank: 9Rank: 9

积分
26591
发表于 2022-2-12 14:35:41 | 显示全部楼层 |阅读模式

$ R7 z+ H9 Z( M0 M5 N<h2 id="系列导航">系列导航</h2>0 I% l, x7 T" k# |  @( W
<p><a  href="https://www.cnblogs.com/code4nothing/p/graphql-net6-0.html">使用Hot Chocolate和.NET 6构建GraphQL应用文章索引</a></p>
  F( M; N! R" k! n<h2 id="需求">需求</h2>2 U2 h- Q6 c) c* I7 d+ T+ j" L
<p>在讨论完GraphQL中的查询需求后,这篇文章我们将演示如何实现GraphQL中的数据添加任务。</p>) `& z3 V1 H  M( ?+ ]0 C( \
<h2 id="思路">思路</h2>- Q) S; t( M$ ]/ D/ q
<p>在GraphQL中,对数据进行查询使用<code>query</code>,而对于修改数据则需要使用<code>mutation</code>,包括新增和修改数据。Hot Chocolate在使用Mutation的逻辑上和使用Query的基本一致,但是需要根据需要定义用于创建或者更新的数据对象,所以我们直接进行实现。</p>- q9 \9 G/ K: S0 u! z8 Z5 h
<h2 id="实现">实现</h2>
1 T" l/ u, J" V  e& y( q' F<p>为了保持简单,我们先定义以下两个类型:</p>. L2 \! A; K( ?4 C
<pre><code class="language-c#">// 定义新增Post的参数
. S& j6 g: c6 Z' V# r/ g& o2 q7 c/ Hpublic record AddPostInput(string Title, string Author);
3 k1 M5 V6 `: ~2 r! b' l7 a" @3 F- ^+ e& d+ i# W6 y% N1 J
// 定义新增Post的返回对象+ @( m  m# _' p0 ~0 p0 @, T- P
public record AddPostPayload(Post Post);7 ?! i- m& J, a  X) j
</code></pre>
6 K/ x/ P* @3 g& |- C2 \<p>新建<code>Mutation.cs</code>用来定义相关接口:</p>2 q6 c3 v3 M5 a7 G, \
<ul>
" M3 W3 f& E0 I- w# @<li><code>Mutation.cs</code></li>
( J: [! s" A1 I8 x, z. }2 U</ul>' h) Z  {  e) q- J
<pre><code class="language-c#">namespace PostGraphi.Api.GraphQL;
" [0 L2 c2 k3 }, W% f1 M. B& h
& U5 x% c# F/ M0 \0 F$ y7 ^public class Mutation# Q2 ?) k; c7 K
{# b$ }9 _& ~; {! a
    public async Task&lt;AddPostPayload&gt; AddPostAsync(AddPostInput input, [Service] IRepository&lt;Post&gt; repository)# P8 ^% _% i: q& f" ~& O
    {
7 M6 q" g1 |! P! P* l8 V        return new AddPostPayload(await repository.AddAsync(new Post( R% S" {* Q; ~2 u; u8 I" X% z
        {
1 H( x) N% ^- D0 K  K9 C( o            Title = input.Title,
1 O  T# m' i# G% L2 K            Author = input.Author
0 k% Z# J  n6 y2 o% U        }));
$ Q+ z7 e3 s/ O4 j    }
  K( y2 I% W% _/ Q, Z  e' t$ ]3 C}
) |6 t2 ~$ V: ]" H# p" c6 |6 w6 j9 X# E
</code></pre>
. ]/ _, o% v( i6 g<p>最后在注入服务的地方进行配置:</p>7 b! F: W# C" \+ `  S3 L1 z' [5 i) w
<ul>: H9 F- {) X6 f1 c
<li><code>ProgramExtensions.cs</code></li>
' P. S: T. r% H; @* K; Z6 P. p</ul>0 {9 h* o, N0 x) E6 o& n* B
<pre><code class="language-c#">builder.Services
1 b, ~* I' t& X1 o  r& y    .AddGraphQLServer()
' K/ Q- B7 b! `, J! `7 a: c3 _- q    .SetPagingOptions(new PagingOptions
: {$ p9 j0 J+ @: K% w    {
  @2 X8 e8 ?) m# i        MaxPageSize = 50,7 |# T  O/ Y9 B# h: q" Q2 H. y) {
        IncludeTotalCount = true
3 I) k8 e7 _0 h1 W0 G9 K    }): M. V. x0 N) V/ k7 o' ?2 j
    .AddFiltering()0 D0 Z( Y8 B3 V
    .AddProjections()4 Q, ^6 N0 u- u. z% v1 s
    .AddSorting()
# K! Z# m9 E8 P+ ~  j    .AddQueryType&lt;Query&gt;()) z# ]! a% f: a. }6 Y1 v3 a2 h' d
    .AddMutationType&lt;Mutation&gt;()
1 j9 A4 x3 F7 ~# }$ G0 j    .AddMutationConventions(new MutationConventionOptions  I0 n! Z( m: s- k! H6 c" [2 w
    {
& |  |1 b! x; E- X1 i        ApplyToAllMutations = true,
$ B0 q9 S8 U4 W7 H% L$ |, F: }        InputArgumentName = "input",
7 [, N/ s. K6 J2 P# X( d        InputTypeNamePattern = "{MutationName}Input"," T4 `2 s; W% C- ~8 g3 t& \
        PayloadTypeNamePattern = "{MutationName}Payload",
0 u- n0 e8 |! D, t8 P5 e; a        PayloadErrorTypeNamePattern = "{MutationName}Error",
+ A, @8 o7 B* \* K        PayloadErrorsFieldName = "errors"
- |  E; w" y4 g$ W    })) ?1 i6 k. T  U9 P0 K
    .AddType&lt;PostType&gt;();9 a4 D2 `8 M  S* B  X& L: [9 z6 q: w
</code></pre>
; \. B0 x. a4 E! L5 V' E$ ?<p>这样就实现了新增Post的需求,下面我们来验证一下。</p>* @, J: i% x7 C4 K: k
<h2 id="验证">验证</h2>
6 O  p/ i. C1 s* J; f, R<p>启动<code>Api</code>项目,调用接口:</p>" q/ L% y6 `8 M+ |3 u8 j
<p><img src="https://img2022.cnblogs.com/blog/2487237/202202/2487237-20220211104544617-1400586374.png" ></p>7 z5 b% D: e4 |. T. [7 Q' V
<p>终端的日志输出如下:</p>" S2 L# w* D8 a( z. r+ K1 c
<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']
" o1 n; K5 h) |6 w* qINSERT INTO "Posts" ("Id", "Abstraction", "Author", "Content", "Created", "CreatedBy", "LastModified", "LastModifiedBy", "Link", "PublishedAt", "Title")
2 l; H- I6 O5 O" d% OVALUES (@p0, @p1, @p2, @p3, @p4, @p5, @p6, @p7, @p8, @p9, @p10);! ?$ Z3 T( Z" Q2 {. u
[10:45:15 INF] Executed endpoint 'Hot Chocolate GraphQL Pipeline'% z" Z: A0 i. d4 K
</code></pre>. v, G* m; i1 D& U1 A- J/ |
<p>可以看到新建的Post已经存储到数据库中了,我们可以通过查询接口来获取详情:</p>
" L8 q' `6 N: v( ^& }$ g- L0 z<p><img src="https://img2022.cnblogs.com/blog/2487237/202202/2487237-20220211104851825-1533915064.png" ></p>" w* `7 d3 K9 D7 w, n; }
<h2 id="总结">总结</h2>
4 K: f, K3 _+ n& n! t- T/ @<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 K8 |! V; i4 i4 O% F1 b/ u, e
<p>在下一篇文章中,我们通过<code>Mutation</code>对已有数据进行更新。</p>
9 o3 s  F" O: Y! L6 C  N
' L) c4 {# o* F
回复

使用道具 举报

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

本版积分规则

手机版|飞雪团队

GMT+8, 2025-12-17 18:39 , Processed in 0.068344 second(s), 21 queries , Gzip On.

Powered by Discuz! X3.4

Copyright © 2001-2021, Tencent Cloud.

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