飞雪团队

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

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

[复制链接]

7995

主题

8083

帖子

2万

积分

管理员

Rank: 9Rank: 9Rank: 9

积分
26315
发表于 2022-2-12 14:35:41 | 显示全部楼层 |阅读模式
) \* m3 t  S. v2 I* Y+ P% Q
<h2 id="系列导航">系列导航</h2>
; T+ x0 s7 d6 x" U. w<p><a  href="https://www.cnblogs.com/code4nothing/p/graphql-net6-0.html">使用Hot Chocolate和.NET 6构建GraphQL应用文章索引</a></p>
, E- ]7 l& [% F! t2 O( J. O<h2 id="需求">需求</h2>
) O$ Y& G  w: Z4 |0 L9 A! f2 i<p>在讨论完GraphQL中的查询需求后,这篇文章我们将演示如何实现GraphQL中的数据添加任务。</p>
, a- b3 `, Y7 ]4 n6 Y6 R$ s<h2 id="思路">思路</h2>) e# J: f9 `  _' E9 w- m( j/ F
<p>在GraphQL中,对数据进行查询使用<code>query</code>,而对于修改数据则需要使用<code>mutation</code>,包括新增和修改数据。Hot Chocolate在使用Mutation的逻辑上和使用Query的基本一致,但是需要根据需要定义用于创建或者更新的数据对象,所以我们直接进行实现。</p>4 v7 Z' H  k! |$ T
<h2 id="实现">实现</h2>
8 i9 w# V1 V$ W$ u' R7 U<p>为了保持简单,我们先定义以下两个类型:</p>/ J: O, h: Z( k( Y
<pre><code class="language-c#">// 定义新增Post的参数
  _  ]1 F" b( R. vpublic record AddPostInput(string Title, string Author);
$ P8 I# X, H) G6 b) |( I) [
, |, Q6 `( o. R1 _8 M// 定义新增Post的返回对象
6 s+ d/ y# B0 @public record AddPostPayload(Post Post);
' i9 p; A8 ~9 ~' d</code></pre>; y! q6 a+ Z4 G. X. z8 k
<p>新建<code>Mutation.cs</code>用来定义相关接口:</p>8 i& r, N8 ?* Y
<ul>
) A4 p0 R+ Z, {- G<li><code>Mutation.cs</code></li>
. d! \% r6 S( a1 ?: z</ul>
( D% I- }# Q2 \<pre><code class="language-c#">namespace PostGraphi.Api.GraphQL;
) ], ^3 O- }% @" ?, n( p7 T
; D. a0 e9 ~6 {9 u  o4 lpublic class Mutation
% g# P, l  w! T8 ?& e1 F" N{
2 S6 n; w5 y% \# [    public async Task&lt;AddPostPayload&gt; AddPostAsync(AddPostInput input, [Service] IRepository&lt;Post&gt; repository)' K% R3 Z1 g0 }
    {
+ X3 r" _4 O: I4 x  V0 G7 ^- T6 w' B        return new AddPostPayload(await repository.AddAsync(new Post1 S/ H0 |$ L! j) @, x3 k- Q
        {
& q+ V  t$ ^7 A* D2 N. Y+ k' ~4 n! t            Title = input.Title,( R  w# c0 ]* b% T! i! u3 v& _) s7 H+ M
            Author = input.Author6 t! N8 S9 d6 y  b0 j+ w- m
        }));6 J0 M/ I0 D% H
    }' @8 g" C- C- ?9 F$ {* C2 ?
}* Z8 x. [; ?- Z; g
; i2 q: D2 p" ~* `
</code></pre>% `) H& ~3 t$ \7 R7 W5 X6 i
<p>最后在注入服务的地方进行配置:</p>) r1 z& S* o7 {1 q1 h
<ul>* N# l+ W! I3 h4 g# a2 S, e! j
<li><code>ProgramExtensions.cs</code></li>0 d3 P1 h8 C) |: r: ?7 ~( f
</ul>5 _; }: H1 G1 f3 e" T/ [! H
<pre><code class="language-c#">builder.Services9 \' S( a4 I; v: s
    .AddGraphQLServer()
$ r' t8 H4 U0 Q; `: Q  z    .SetPagingOptions(new PagingOptions
- q$ c) S5 @- `$ z    {
5 k4 T( ^0 Q4 L" C7 H- `- H* x9 _        MaxPageSize = 50,
; i* M4 C8 n4 g4 W9 _5 p' U        IncludeTotalCount = true
7 x7 R( {3 R/ q( M, {    })
- ^; \) C% I) F5 Q9 e. j' m    .AddFiltering()
9 E- `: |5 U4 N+ L' I* X    .AddProjections()
/ j7 P+ ^; W4 P' [  d    .AddSorting()
/ E% K. X2 e' I. q) Y6 k: ]: e    .AddQueryType&lt;Query&gt;()$ J) y* a# [1 b* d1 Q; k7 y
    .AddMutationType&lt;Mutation&gt;(): D8 K' o% \; q. D
    .AddMutationConventions(new MutationConventionOptions, i" j6 b8 |0 `& J* t1 Q* S/ z
    {8 d. g! ]1 M% n9 ?0 e3 }: Q- L" X8 u1 {
        ApplyToAllMutations = true,4 L6 }4 O! g/ a8 a9 L6 ~: }
        InputArgumentName = "input",1 N. n* |0 T5 D
        InputTypeNamePattern = "{MutationName}Input",
* a7 F% ?# ]$ H7 F        PayloadTypeNamePattern = "{MutationName}Payload",
6 }/ t+ F. `3 `2 N        PayloadErrorTypeNamePattern = "{MutationName}Error",
+ i: K( o/ h8 `* b  ?2 h        PayloadErrorsFieldName = "errors"; m1 }* ]* F0 p& u% [
    })9 j% u# f6 z6 W2 W, k
    .AddType&lt;PostType&gt;();0 J& b' J' W9 d! i5 p
</code></pre>
! z5 ^9 |+ s# H. F  Q1 G) D, F% I<p>这样就实现了新增Post的需求,下面我们来验证一下。</p>2 n. V1 b1 m' ?' a! M( D+ O) G4 f
<h2 id="验证">验证</h2>
* B$ L+ `% W4 F) y8 d' M/ g: a<p>启动<code>Api</code>项目,调用接口:</p>
1 B( a* s7 V9 d0 k& [$ C<p><img src="https://img2022.cnblogs.com/blog/2487237/202202/2487237-20220211104544617-1400586374.png" ></p>2 c( q+ l! E) p+ G9 m' j# _8 ~
<p>终端的日志输出如下:</p>+ D8 x) z, s4 c) g6 `1 j/ L. k& `
<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" Y$ k) i1 V6 V6 v
INSERT INTO "Posts" ("Id", "Abstraction", "Author", "Content", "Created", "CreatedBy", "LastModified", "LastModifiedBy", "Link", "PublishedAt", "Title")7 P' Q$ ^5 H) }+ |0 l: `: w* K
VALUES (@p0, @p1, @p2, @p3, @p4, @p5, @p6, @p7, @p8, @p9, @p10);
' Q: k6 s6 s+ l+ a* @[10:45:15 INF] Executed endpoint 'Hot Chocolate GraphQL Pipeline'
6 f$ X' p' ?/ M- Y8 }  ]( Q& Z</code></pre>( |3 j7 S% \9 r- ~9 s2 C9 m% P
<p>可以看到新建的Post已经存储到数据库中了,我们可以通过查询接口来获取详情:</p>' v* I, G2 \8 A8 m7 Y" y7 ]
<p><img src="https://img2022.cnblogs.com/blog/2487237/202202/2487237-20220211104851825-1533915064.png" ></p>3 x# Q$ X' B1 P- t% W
<h2 id="总结">总结</h2>2 F% i& h6 S1 ~' _
<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 m% u" k0 w- v1 d. Y2 z) A8 }9 N9 B* K<p>在下一篇文章中,我们通过<code>Mutation</code>对已有数据进行更新。</p>
/ _: Q. l, U% U
' S- c8 y  G3 \& G! H9 T3 X7 _3 \+ X
回复

使用道具 举报

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

本版积分规则

手机版|飞雪团队

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

Powered by Discuz! X3.4

Copyright © 2001-2021, Tencent Cloud.

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