這篇文章給大家分享的是有關(guān)Asp.Net Core控制器怎么接收原始請求正文內(nèi)容的內(nèi)容。小編覺得挺實(shí)用的,因此分享給大家做個參考,一起跟隨小編過來看看吧。
創(chuàng)新互聯(lián)公司專注于桑日企業(yè)網(wǎng)站建設(shè),成都響應(yīng)式網(wǎng)站建設(shè)公司,購物商城網(wǎng)站建設(shè)。桑日網(wǎng)站建設(shè)公司,為桑日等地區(qū)提供建站服務(wù)。全流程定制制作,專業(yè)設(shè)計,全程項(xiàng)目跟蹤,創(chuàng)新互聯(lián)公司專業(yè)和態(tài)度為您提供的服務(wù)主要目標(biāo)
在Asp.net Core控制器中,通過自定義格式化程序來映射自定義處理控制器中的“未知”內(nèi)容。本文將給大家詳細(xì)介紹關(guān)于Asp.Net Core控制器接收原始請求正文內(nèi)容的相關(guān)內(nèi)容,分享出來供大家參考學(xué)習(xí),下面話不多說了,來一起看看詳細(xì)的介紹吧
簡單案例
為了演示這個問題,我們用VS2017創(chuàng)建一個默認(rèn)的Asp.net Core Web Api項(xiàng)目。
[Route("api/[controller]")] [ApiController] public class ValuesController : ControllerBase{ [HttpGet] public ActionResultGet() { return "ok"; } [HttpPost] [Route("PostX")] public ActionResult Post([FromBody] string value) { return value; } }
Json請求
我們從最常見的json輸入請求開始。
User-Agent: Fiddler Host: localhost:5000 Content-Type: application/json Content-Length: 16
請求body:
{"123456"}
通過后臺調(diào)試和fiddler抓包,我們可以看到請求輸入和返回。
后臺調(diào)試,查看請求輸入結(jié)果
fiddler查看請求header
fiddler查看返回結(jié)果
注意!!
別忘了[FromBody],有時候會忘的。
后臺action接收類型為string的時候,請求body只能是字符串,不能傳json對象。我演示這個例子時,被這點(diǎn)坑了。如果接收對象是一個類的時候,才可以傳json對象。
沒有JSON
雖然傳輸json數(shù)據(jù)是最常用的,但有時候我們需要支持普通的文本或者二進(jìn)制信息。我們將Content-Type改為
text/plain
User-Agent: Fiddler Host: localhost:5000 Content-Type:text/plain Content-Length: 16
請求body:
{"123456"}
悲劇的事情來,報404!
不支持text/plain
事情到此就變得稍微復(fù)雜了一些,因?yàn)閍sp.netcore只處理它認(rèn)識的類型,如json和formdata。默認(rèn)情況下,原始數(shù)據(jù)不能直接映射到控制器參數(shù)。這是個小坑,不知你踩到過沒有?仔細(xì)想想,這是有道理的。MVC具有特定內(nèi)容類型的映射,如果您傳遞的數(shù)據(jù)不符合這些內(nèi)容類型,則無法轉(zhuǎn)換數(shù)據(jù),因此它假定沒有匹配的端點(diǎn)可以處理請求。
那么怎么支持原始的請求映射呢?
支持原始正文請求
不幸的是,ASP.NET Core不允許您僅通過方法參數(shù)以任何有意義的方式捕獲“原始”數(shù)據(jù)。無論如何,您需要對其進(jìn)行一些自定義處理Request.Body以獲取原始數(shù)據(jù),然后對其進(jìn)行反序列化。
您可以捕獲原始數(shù)據(jù)Request.Body并從中直接讀取原始緩沖區(qū)。
最簡單,最不易侵入,但不那么明顯的方法是使用一個方法接受沒有參數(shù)的 POST或PUT數(shù)據(jù),然后從Request.Body以下位置讀取原始數(shù)據(jù):
讀取字符串緩沖區(qū)
[HttpPost] [Route("PostText")] public async TaskPostText() { using (StreamReader reader = new StreamReader(Request.Body, Encoding.UTF8)) { return await reader.ReadToEndAsync(); } }
這適用于一下Http和文本
User-Agent: Fiddler Host: localhost:5000 Content-Type: text/plain Content-Length: 6
要讀取二進(jìn)制數(shù)據(jù),你可以使用以下內(nèi)容:
讀取byte緩沖區(qū)
[HttpPost] [Route("PostBinary")] public async TaskPostBinary() { using (var ms = new MemoryStream(2048)) { await Request.Body.CopyToAsync(ms); return ms.ToArray(); // returns base64 encoded string JSON result } }
查看執(zhí)行結(jié)果
接收文本內(nèi)容
接收二進(jìn)制數(shù)據(jù)
HttpRequest靜態(tài)擴(kuò)展
如果你為了方便,寫了很多HttpRequest的擴(kuò)展,接收參數(shù)時,可以看起來更簡潔一些。
public static class HttpRequestExtension { ////// /// /// /// ///public static async Task GetRawBodyStringFormater(this HttpRequest httpRequest, Encoding encoding) { if (encoding == null) { encoding = Encoding.UTF8; } using (StreamReader reader = new StreamReader(httpRequest.Body, encoding)) { return await reader.ReadToEndAsync(); } } /// /// 二進(jìn)制 /// /// /// ///public static async Task GetRawBodyBinaryFormater(this HttpRequest httpRequest, Encoding encoding) { if (encoding == null) { encoding = Encoding.UTF8; } using (StreamReader reader = new StreamReader(httpRequest.Body, encoding)) { using (var ms = new MemoryStream(2048)) { await httpRequest.Body.CopyToAsync(ms); return ms.ToArray(); // returns base64 encoded string JSON result } } } }
[HttpPost] [Route("PostTextX")] public async TaskPostTextX() { return await Request.GetRawBodyStringAsyn(); } /// /// 接收 /// ///[HttpPost] [Route("PostBinaryX")] public async Task PostBinaryX() { return await Request.GetRawBodyBinaryAsyn(); }
自動轉(zhuǎn)換文本和二進(jìn)制值
上面雖然解決了原始參數(shù)轉(zhuǎn)換問題,但不夠友好。如果你打算像原生MVC那樣自動映射參數(shù)的話,你需要做一些自定義格式化適配。
創(chuàng)建一個Asp.net MVC InputFormatter
ASP.NET Core使用一種干凈且更通用的方式來處理內(nèi)容的自定義格式InputFormatter。輸入格式化程序掛鉤到請求處理管道,讓您查看特定類型的內(nèi)容以確定是否要處理它。然后,您可以閱讀請求正文并對入站內(nèi)容執(zhí)行自己的反序列化。
InputFormatter有幾個要求
您需要使用[FromBody]去獲取
您必須能夠查看請求并確定是否以及如何處理內(nèi)容。
在這個例子中,對于“原始內(nèi)容”,我想查看具有以下類型的請求:
text/plain(文本)
appliaction/octet-stream(byte[])
沒有內(nèi)容類型(string)
要創(chuàng)建格式化程序,你可以實(shí)現(xiàn)IInputFormatter或者從InputFormatter繼承。
public class RawRequestBodyFormatter : IInputFormatter { public RawRequestBodyFormatter() { } public bool CanRead(InputFormatterContext context) { if (context == null) throw new ArgumentNullException("argument is Null"); var contentType = context.HttpContext.Request.ContentType; if (string.IsNullOrEmpty(contentType) || contentType == "text/plain" || contentType == "application/octet-stream") return true; return false; } public async TaskReadAsync(InputFormatterContext context) { var request = context.HttpContext.Request; var contentType = context.HttpContext.Request.ContentType; if (string.IsNullOrEmpty(contentType) || contentType.ToLower() == "text/plain") { using (StreamReader reader = new StreamReader(request.Body, Encoding.UTF8)) { var content = await reader.ReadToEndAsync(); return await InputFormatterResult.SuccessAsync(content); } } if (contentType == "application/octet-stream") { using (StreamReader reader = new StreamReader(request.Body, Encoding.UTF8)) { using (var ms = new MemoryStream(2048)) { await request.Body.CopyToAsync(ms); var content = ms.ToArray(); return await InputFormatterResult.SuccessAsync(content); } } } return await InputFormatterResult.FailureAsync(); } }
格式化程序用于CanRead()檢查對內(nèi)容類型的請求以支持,然后將ReadRequestBodyAsync()內(nèi)容讀取和反序列化為應(yīng)在控制器方法的參數(shù)中返回的結(jié)果類型。
InputFormatter必須在ConfigureServices()啟動代碼中注冊MVC :
public void ConfigureServices(IServiceCollection services) { services.AddMvc(o=>o.InputFormatters.Insert(0,new RawRequestBodyFormatter())).SetCompatibilityVersion(CompatibilityVersion.Version_2_1); }
接受原始輸入
[HttpPost] [Route("PostTextPlus")] public string PostTextPlus([FromBody] string value) { return value; }
然后你就可以發(fā)送post請求,像這樣:
User-Agent: Fiddler Host: localhost:5000 Content-Length: 6
或者
User-Agent: Fiddler Host: localhost:5000 Content-Type:text/plain Content-Length: 6
請注意,您可以使用內(nèi)容類型調(diào)用相同的控制器方法application/json并傳遞JSON字符串,這也將起作用。在RawRequestBodyFormatter 簡單地增加它支持的附加內(nèi)容類型的支持。
二進(jìn)制數(shù)據(jù)
[HttpPost] [Route("PostBinaryPlus")] public byte[] PostBinaryPlus([FromBody] byte[] value) { return value; }
請求內(nèi)容如下:
User-Agent: Fiddler Host: localhost:5000 Content-Length: 6 Content-Type: application/octet-stream
感謝各位的閱讀!關(guān)于“Asp.Net Core控制器怎么接收原始請求正文內(nèi)容”這篇文章就分享到這里了,希望以上內(nèi)容可以對大家有一定的幫助,讓大家可以學(xué)到更多知識,如果覺得文章不錯,可以把它分享出去讓更多的人看到吧!