飞雪团队

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

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

[复制链接]

6806

主题

6894

帖子

2万

积分

管理员

Rank: 9Rank: 9Rank: 9

积分
22748
发表于 2022-2-12 14:35:41 | 显示全部楼层 |阅读模式
4 c6 l- i. g/ {! I$ n1 o
<h2 id="系列导航">系列导航</h2>
" }1 n. R& v5 e<p><a  href="https://www.cnblogs.com/code4nothing/p/graphql-net6-0.html">使用Hot Chocolate和.NET 6构建GraphQL应用文章索引</a></p>. j9 N( `$ V1 @. ~
<h2 id="需求">需求</h2>
8 g* Z( p6 @( k3 C6 c$ V: Q/ m<p>在讨论完GraphQL中的查询需求后,这篇文章我们将演示如何实现GraphQL中的数据添加任务。</p>$ u7 f; \0 `% w, ^2 C) z
<h2 id="思路">思路</h2>
+ Y$ m! D7 e  Z<p>在GraphQL中,对数据进行查询使用<code>query</code>,而对于修改数据则需要使用<code>mutation</code>,包括新增和修改数据。Hot Chocolate在使用Mutation的逻辑上和使用Query的基本一致,但是需要根据需要定义用于创建或者更新的数据对象,所以我们直接进行实现。</p>4 V: a, s8 q. v* X' {& Q
<h2 id="实现">实现</h2>
# [3 s$ z8 q# b+ R0 N& g2 ~  n" @<p>为了保持简单,我们先定义以下两个类型:</p>
, w& l# i) l* x<pre><code class="language-c#">// 定义新增Post的参数
( J6 I; K+ Z* H& f7 A5 r6 ?# }public record AddPostInput(string Title, string Author);$ Q9 D: N( t% o  p

" t# i( W( i7 @0 j, E! ~4 H// 定义新增Post的返回对象
8 F1 U. F1 T- T6 t1 V, J) a, Vpublic record AddPostPayload(Post Post);, S1 W0 _) e: Q* k# f- E: Q5 p
</code></pre>: x2 ~* v# |  v% q
<p>新建<code>Mutation.cs</code>用来定义相关接口:</p>
1 g& ~( g! h8 z3 F/ g7 |# _# z<ul>) k1 V: Z' F/ h( R' H! j9 W
<li><code>Mutation.cs</code></li>
" W( r2 |7 h% G% L2 D! k# l) V</ul>
7 Y7 T+ [" |2 F<pre><code class="language-c#">namespace PostGraphi.Api.GraphQL;
5 s0 E5 I8 n9 ?3 l1 F, f1 B- B4 Z% z9 o. k
public class Mutation
, N: g7 m. Q5 h; ?- g7 D# j# A{  O8 j, v! v" A6 g$ U; s
    public async Task&lt;AddPostPayload&gt; AddPostAsync(AddPostInput input, [Service] IRepository&lt;Post&gt; repository)" j: i2 \5 ]& I$ N/ u
    {
/ F8 ~2 B( u$ V8 ^1 `# m" }        return new AddPostPayload(await repository.AddAsync(new Post* d/ W$ H7 \# X& ^
        {
2 w( F9 s: k6 y4 {0 A+ A1 m! z% @            Title = input.Title," W% f% w$ N' y' u$ w! l
            Author = input.Author' ?. k2 q: M' ?4 l, c- f8 i5 D
        }));  N$ q7 c/ c$ _8 T* K
    }
) N/ A2 B% r* R  Y( h* D}
# E* m$ s3 V! v/ k$ d
+ z: m% A* N5 S' @0 L</code></pre>
- T/ R! y" G# ]" H9 M<p>最后在注入服务的地方进行配置:</p>
" V5 y# M' H5 H<ul>% n! V0 u) d/ Q0 ]' E$ S  O
<li><code>ProgramExtensions.cs</code></li>
9 P( N. ^: W. n3 `/ u8 h1 O+ z  ?</ul>
! I# N9 c& k: X<pre><code class="language-c#">builder.Services4 X$ n9 X6 D0 w
    .AddGraphQLServer()
% ?9 D1 Q( a3 Q8 A    .SetPagingOptions(new PagingOptions
7 \$ O8 u3 N* ~( I5 w    {
# _# k, q9 V" U0 T6 D, ^        MaxPageSize = 50,
2 t" }: p; X4 H/ b/ X        IncludeTotalCount = true& V9 R% r& X2 X3 C
    })
' V4 C. H: e- c( P# r    .AddFiltering()7 z9 [2 I" ]. e: ~: t4 z
    .AddProjections()
: y9 V9 P( m4 b9 k3 ]3 l0 m    .AddSorting()
. p# ]9 X; X" B    .AddQueryType&lt;Query&gt;()9 M! H/ @- [( i9 e! d+ E+ ]
    .AddMutationType&lt;Mutation&gt;()
+ F2 s  ^) ^' _" ?: J& N8 ]1 f    .AddMutationConventions(new MutationConventionOptions" D0 e/ E! }+ p' {2 d8 O* [5 ~; v
    {9 _2 ^  x: h0 _7 c) y9 H9 [# v: t
        ApplyToAllMutations = true,$ q! Y( V7 g6 V) L* B
        InputArgumentName = "input",) |) [% f6 |, u
        InputTypeNamePattern = "{MutationName}Input",
/ e: \$ _6 _6 M        PayloadTypeNamePattern = "{MutationName}Payload",
' T/ Z" w- }) _& V+ Q( j        PayloadErrorTypeNamePattern = "{MutationName}Error",
6 D* s; j1 W5 z& u, A4 K        PayloadErrorsFieldName = "errors"
% j, Q1 @0 O7 {9 U: R    }), V9 W# C% A; c# u) T
    .AddType&lt;PostType&gt;();
: R) r. _4 C5 @. X; ]# f* e3 N</code></pre>
1 D& A3 y: E* ~' c7 Q$ G1 I  v3 w- D<p>这样就实现了新增Post的需求,下面我们来验证一下。</p>3 x1 e6 q- L/ {  c2 l
<h2 id="验证">验证</h2>
0 ^- x- ?/ O  ^+ i/ a<p>启动<code>Api</code>项目,调用接口:</p>% J% s9 G+ E: i. ^3 {
<p><img src="https://img2022.cnblogs.com/blog/2487237/202202/2487237-20220211104544617-1400586374.png" ></p>+ O2 _/ h" ~+ i/ i4 K
<p>终端的日志输出如下:</p>2 C. {6 @" D, _# `4 \
<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']
/ J- ^3 }3 p& QINSERT INTO "Posts" ("Id", "Abstraction", "Author", "Content", "Created", "CreatedBy", "LastModified", "LastModifiedBy", "Link", "PublishedAt", "Title")" W, f6 Y, {% o, z- ^% R
VALUES (@p0, @p1, @p2, @p3, @p4, @p5, @p6, @p7, @p8, @p9, @p10);5 U8 c' a- x( V
[10:45:15 INF] Executed endpoint 'Hot Chocolate GraphQL Pipeline'
# I% q! T' }" ]" r9 I  W3 h* M$ [</code></pre># L# o# ]7 O4 g# }; R4 K9 p
<p>可以看到新建的Post已经存储到数据库中了,我们可以通过查询接口来获取详情:</p>
2 i* @, o5 s( o* c: y% ?/ O& r<p><img src="https://img2022.cnblogs.com/blog/2487237/202202/2487237-20220211104851825-1533915064.png" ></p>& c4 L* d  g/ J3 U2 `
<h2 id="总结">总结</h2>$ H# o, t( T& l  ?) Z3 c5 ^2 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>
1 Q" k- N! `/ v' I( F: V<p>在下一篇文章中,我们通过<code>Mutation</code>对已有数据进行更新。</p>( j8 l7 ?3 a( v! w  a

% {' d* i0 n8 _9 \' B
回复

使用道具 举报

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

本版积分规则

手机版|飞雪团队

GMT+8, 2025-6-29 18:29 , Processed in 0.062069 second(s), 21 queries , Gzip On.

Powered by Discuz! X3.4

Copyright © 2001-2021, Tencent Cloud.

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