ASP.NET Core使用GraphQL第二章之中間件
前言
在開(kāi)始本文之前,對(duì)GraphQL不熟悉的朋友們,可以看下下面這篇文章:
前文:ASP.NET Core中使用GraphQL - 第一章 Hello World
看完上面的文章,下面話不多說(shuō)了,來(lái)一起看看詳細(xì)的介紹吧
中間件
如果你熟悉ASP.NET Core的中間件,你可能會(huì)注意到之前的博客中我們已經(jīng)使用了一個(gè)中間件,
app.Run(async (context) =>
{
var result = await new DocumentExecuter()
.ExecuteAsync(doc =>
{
doc.Schema = schema;
doc.Query = @"
query {
hello
}
";
}).ConfigureAwait(false);
var json = new DocumentWriter(indent: true)
.Write(result)
await context.Response.WriteAsync(json);
});
這個(gè)中間件負(fù)責(zé)輸出了當(dāng)前查詢的結(jié)果。
中間件的定義:
中間件是裝載在應(yīng)用程序管道中的組件,負(fù)責(zé)處理請(qǐng)求和響應(yīng),每一個(gè)中間件
可以選擇是否傳遞請(qǐng)求到應(yīng)用程序管道中的下一個(gè)組件
可以在應(yīng)用程序管道中下一個(gè)組件運(yùn)行前和運(yùn)行后進(jìn)行一些操作
來(lái)源: Microsoft Documentation
實(shí)際上中間件是一個(gè)委托,或者更精確的說(shuō)是一個(gè)請(qǐng)求委托(Request Delegate)。 正如他的名字一樣,中間件會(huì)處理請(qǐng)求,并決定是否將他委托到應(yīng)用程序管道中的下一個(gè)中間件中。在我們前面的例子中,我們使用IApplicationBuilder類的Run()方法配置了一個(gè)請(qǐng)求委托。
使用動(dòng)態(tài)查詢體替換硬編碼查詢體
在我們之前的例子中,中間件中的代碼非常簡(jiǎn)單,它僅是返回了一個(gè)固定查詢的結(jié)果。然而在現(xiàn)實(shí)場(chǎng)景中,查詢應(yīng)該是動(dòng)態(tài)的,因此我們必須從請(qǐng)求中讀取查詢體。
在服務(wù)器端,每一個(gè)請(qǐng)求委托都可以接受一個(gè)HttpContext參數(shù)。如果一個(gè)查詢體是通過(guò)POST請(qǐng)求發(fā)送到服務(wù)器的,你可以很容易的使用如下代碼獲取到請(qǐng)求體中的內(nèi)容。
string body;
using (var streamReader = new StreamReader(httpContext.Request.Body))
{
body = await streamReader.ReadToEndAsync();
}
在獲取請(qǐng)求體內(nèi)容之前,為了不引起任何問(wèn)題,我們需要先檢測(cè)一些當(dāng)前請(qǐng)求
- 是否是一個(gè)POST請(qǐng)求
- 是否使用了特定的Url, 例如 /api/graphql
因此我們需要對(duì)代碼進(jìn)行調(diào)整。
if(context.Request.Path.StartsWithSegments("/api/graphql")
&& string.Equals(context.Request.Method,
"POST",
StringComparison.OrdinalIgnoreCase))
{
string body;
using (var streamReader = new StreamReader(context.Request.Body))
{
body = await streamReader.ReadToEndAsync();
}
....
....
....
一個(gè)請(qǐng)求體可以包含很多字段,這里我們約定傳入graphql查詢體字段名稱是query。因此我們可以將請(qǐng)求體中的JSON字符串轉(zhuǎn)換成一個(gè)包含Query屬性的復(fù)雜類型。
這個(gè)復(fù)雜類型代碼如下:
public class GraphQLRequest
{
public string Query { get; set; }
}
下一步我們要做的就是,反序列化當(dāng)前請(qǐng)求體的內(nèi)容為一個(gè)GraphQLRequest類型的實(shí)例。這里我們需要使用Json.Net中的靜態(tài)方法JsonConvert.DeserializeObjct來(lái)替換之前的硬編碼的查詢體。
var request = JsonConvert.DeserializeObject<GraphQLRequest>(body);
var result = await new DocumentExecuter().ExecuteAsync(doc =>
{
doc.Schema = schema;
doc.Query = request.Query;
}).ConfigureAwait(false);
在完成以上修改之后,Startup.cs文件的Run方法應(yīng)該是這個(gè)樣子的。
app.Run(async (context) =>
{
if (context.Request.Path.StartsWithSegments("/api/graphql")
&& string.Equals(context.Request.Method,
"POST",
StringComparison.OrdinalIgnoreCase))
{
string body;
using (var streamReader = new StreamReader(context.Request.Body))
{
body = await streamReader.ReadToEndAsync();
var request = JsonConvert.DeserializeObject<GraphQLRequest>(body);
var schema = new Schema { Query = new HelloWorldQuery() };
var result = await new DocumentExecuter()
.ExecuteAsync(doc =>
{
doc.Schema = schema;
doc.Query = request.Query;
}).ConfigureAwait(false);
var json = new DocumentWriter(indent: true)
.Write(result);
await context.Response.WriteAsync(json);
}
}
});
最終效果
現(xiàn)在我們可以使用POSTMAN來(lái)創(chuàng)建一個(gè)POST請(qǐng)求, 請(qǐng)求結(jié)果如下:

結(jié)果正確返回了。
本篇源代碼: https://github.com/lamondlu/GraphQL_Blogs/tree/master/Part%20II
總結(jié)
以上就是這篇文章的全部?jī)?nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,如果有疑問(wèn)大家可以留言交流,謝謝大家對(duì)腳本之家的支持。
相關(guān)文章
.NET Core Dapper操作mysql數(shù)據(jù)庫(kù)的實(shí)現(xiàn)方法
這篇文章主要介紹了.NET Core Dapper操作mysql數(shù)據(jù)庫(kù)的實(shí)現(xiàn)方法,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-04-04
.net中string類型可以作為lock的鎖對(duì)象嗎
lock 關(guān)鍵字是用于在多線程編程中實(shí)現(xiàn)同步和互斥訪問(wèn)的關(guān)鍵字,它的作用是確保共享資源在任意時(shí)刻只能被一個(gè)線程訪問(wèn),從而避免出現(xiàn)競(jìng)態(tài)條件(race condition)和數(shù)據(jù)不一致的問(wèn)題,這篇文章主要介紹了string類型可以作為lock的鎖對(duì)象嗎,需要的朋友可以參考下2023-06-06
ASP.NET MVC5網(wǎng)站開(kāi)發(fā)概述(一)
這篇文章主要內(nèi)容是ASP.NET MVC5網(wǎng)站開(kāi)發(fā)實(shí)踐的整體概述,分析了開(kāi)發(fā)環(huán)境、使用的技術(shù)以及項(xiàng)目的整體結(jié)構(gòu),感興趣的小伙伴們可以參考一下2015-09-09
.Net語(yǔ)言Smobiler開(kāi)發(fā)利用Gridview控件設(shè)計(jì)較復(fù)雜的表單
這篇文章主要為大家詳細(xì)介紹了.Net語(yǔ)言Smobiler開(kāi)發(fā)利用Gridview控件設(shè)計(jì)較復(fù)雜的表單,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2016-09-09
asp.net UpdatePanel實(shí)現(xiàn)無(wú)刷新上傳圖片
UpdatePanel實(shí)現(xiàn)無(wú)刷新上傳圖片實(shí)現(xiàn)代碼,需要的朋友可以參考下。2010-03-03
利用.net控件實(shí)現(xiàn)下拉導(dǎo)航菜單制作的具體方法
這篇文章介紹了利用.net控件實(shí)現(xiàn)下拉導(dǎo)航菜單制作的具體方法,有需要的朋友可以參考一下,希望對(duì)你有所幫助2013-07-07
aspnetcore 實(shí)現(xiàn)簡(jiǎn)單的偽靜態(tài)化功能
這篇文章主要介紹了aspnetcore 實(shí)現(xiàn)簡(jiǎn)單的偽靜態(tài)化功能,本文給大家介紹的非常詳細(xì),具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2019-07-07
AspNet Core上實(shí)現(xiàn)web定時(shí)任務(wù)實(shí)例
在本篇文章里小編給大家分享了關(guān)于AspNet Core上實(shí)現(xiàn)web定時(shí)任務(wù)的實(shí)例內(nèi)容,有興趣的朋友們學(xué)習(xí)參考下。2019-02-02
Visual Studio 2015 配置 Opencv3.2的圖文詳解
這篇文章主要介紹了Visual Studio 2015 配置 Opencv3.2的圖文詳解,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2018-05-05
EF?Core項(xiàng)目中不同數(shù)據(jù)庫(kù)需要的安裝包介紹
這篇文章介紹了EF?Core項(xiàng)目中不同數(shù)據(jù)庫(kù)需要的安裝包,對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2022-05-05

