我已将自定义中间件添加到我的Startup类(以下示例here https://learn.microsoft.com/en-us/aspnet/core/fundamentals/middleware?tabs=aspnetcore2x#writing-middleware).
public void Configure(IApplicationBuilder app, IHostingEnvironment env)
{
app.Use((context, next) =>
{
try { return next(); }
catch (Exception exception) { return new Task(() => { }); }
});
app.UseCors("Welcome All");
app.UseMvc();
app.UseSwagger();
app.UseSwaggerUI(_ => _.SwaggerEndpoint("/swagger/v0.1/swagger.json", "Version 0.1"));
}
据我了解,在控制器中的方法中抛出异常应该被捕获试着抓的自定义中间件。但.NET Core 似乎与我的期望不符。通过断点我可以看到返回下一个()被调用but在控制器的方法抛出异常后,该异常未被捕获,并且黄色标记完全跳过中间件并落在其末尾的大括号处。
我缺少什么?
如果它有任何意义或兴趣,我的目标是将异常处理从我的方法移至横切中间,以简化代码(并且我不想应用过滤器,因为我的目标是没有 MVC 的纯 WebApi )。所以控制器和方法看起来像这样。
[Route("api/test")]
public class TestController : Controller
{
public TestController(...) { ... }
...
[HttpPost]
public IActionResult CrashBoom([FromBody] Thing input)
{
if (input.isCrashworthy)
throw new Exception("Boom")
else
return Ok();
}
}
这里的问题是请求委托(中间件的可执行部分)异步运行。如果你看一下类型next
你可以看到它实际上是一个Func<Task>
,所以它是一个返回任务的函数。或者换句话说:它是一个没有返回值的异步函数。
这意味着为了能够捕获异常,您需要保留异步上下文。因此您的自定义中间件也应该异步执行。如果这样做,您会注意到您可以捕获异常,然后相应地响应它们,例如通过写出纯文本响应(作为示例):
app.Use(async (context, next) =>
{
try
{
await next();
}
catch (Exception ex)
{
// let’s log the exception
var logger = context.RequestServices.GetService<ILogger<Startup>>();
logger.LogWarning(ex, "Encountered an exception");
// write a response
context.Response.StatusCode = 500;
context.Response.ContentType = "text/plain";
await context.Response.WriteAsync("Sorry, something went wrong!");
}
});
现在,如果在中间件管道中进一步引发异常,那么您的自定义管道将能够捕获它并对其做出响应。
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)