小弟初来乍到,分享一些工作学习中遇到的问题和解决方式,如有不准确或是有错误的地方,希望不吝赐教,谢过了。  --Dogtwo

起因:
ABP 中异常处理的思路是很清晰的。一共五种类型的异常类。

AbpInitializationException用于封装ABP初始化过程中出现的异常,只要抛出AbpInitializationException异常就可以,无须做额外处理。这类异常往往是需要维护人员介入分析的。

其他四个异常都在AbpController中被集中处理,处理分为两步:一,通过EventBus触发异常事件,相应的异常处理函数则处理异常。二,针对AbpValidationException,UserFriendlyException和AbpAuthorizationException异常,Abp会将异常信息转换为ErrorInfo,并以view或Json的形式返回给客户端。
(以上内容摘自 HK Zhand大大的博客

我们应使用UserFriendlyException来包装我们自定义的异常,但UserFriendlyException抛出的异常存在一个问题:无法指定HttpStatus code,这样前端收到的response中HttpStatus code均为500。对一部分前端语言或框架来说,这个状态码难以处理或不便于处理,因为约定5xx指示服务器异常,不应再由前端进行Handle。

所以,我们希望更改Response中的HttpStatus code。

思路很简单,我们利用filter去拦截异常,判断其类型,若为我们自定义的异常则更改HttpStatus code。

Filter部分代码

 public class MyExceptionFilter : IExceptionFilter
{
public ILogger Logger { get; set; } public MyExceptionFilter()
{
Logger = NullLogger.Instance;
} public void OnException(ExceptionContext context)
{
if (!(context.Exception is MyException))
{
return;
} HandleAndWrapException(context);
} private void HandleAndWrapException(ExceptionContext context)
{
context.HttpContext.Response.StatusCode = (int)HttpStatusCode.BadRequest; var myException = (MyException)context.Exception; context.Result = new ObjectResult(
  new AjaxResponse(
    new ErrorInfo(myException.ErrorCode, myException.Message)
        )
     ); LogHelper.LogException(Logger, context.Exception); context.Exception = null; //Handled!
}
}

然后再StartUp文件中添加filter

 services.Configure(mvcOptions =>
{
mvcOptions.Filters.AddService(typeof(MyExceptionFilter ));
});

运行发现,filter并不能catch住我们抛出的异常,甚至OnException方法体都没进入。
原来,ABP本身实现了几个filter,包括Authorization Filter, Audit Action Filter, Validation Action Filter, Unit of Work Action Filter, Exception Filter和Result Filter.
造成我们的自定义filter无法正常执行的原因就是Exception Filter.

ABP官方文档介绍为:
AbpExceptionFilter is used to handle exceptions thrown from controller actions. It handles and logs exceptions and returns a wrapped response to the client.

This only handles object results, and not view results. Actions returning any object, JsonResult or ObjectResult will be handled. Actions are not handled if they return a view or any other result type implementing IActionsResult. It is recommend that you use the built-in UseExceptionHandler extension method defined in the Microsoft.AspNetCore.Diagnostics package to handle view exceptions.
Exception handling and logging behaviour can be changed using the WrapResult and DontWrapResult attributes for methods and classes.

除此之外, ABPExceptionFilter还会触发AbpHandledExceptionData eventbus event.且,经由ABPExceptionFilter处理之后,会将异常信息转换为ErrorInfo,并以view或Json的形式返回给客户端。所以当response通过ABPExceptionFilter之后便不再包含Exception了,自然我们的Filter捕捉不到了。

由此,想到两个解决方法
一 经由Event bus再次将异常抛出(很明显,如果是为了解决本问题的话,逻辑不通,很差的解决方式。已经被catch的异常再抛出来被自定义的filter去处理,七擒孟获)
但为了熟悉这部分的代码逻辑还是做了一下实现。

 public class MyExceptionHandler : IEventHandler, ITransientDependency
{
public ILogger Logger { get; set; } public MyExceptionHandler()
{
Logger = NullLogger.Instance;
} public void HandleEvent(AbpHandledExceptionData eventData)
{
Logger.Info(eventData.Exception.ToString());
throw eventData.Exception;
}
}

如果是为了别的一些功能,上面利用EventBus来处理的方式也可以借鉴。

二是禁用ABPExceptionFilter改为使用我们自己的filter

禁用ABPExceptionFilter
除上面声明Filter之外还需要在Configure中app.UseMvc之后添加

// Remove AbpExceptionFilter
var ops = app.ApplicationServices.GetRequiredService<IOptions<MvcOptions>>().Value;
var abpExceptionFilter = ops.Filters.FirstOrDefault(f =>
(f as ServiceFilterAttribute)?.ServiceType == (typeof(AbpExceptionFilter)));
ops.Filters.Remove(abpExceptionFilter);

此方法也有不好的地方,即除我们自定义的异常外,其他异常并不能再由ABPExceptionFilter来处理。

改良最佳版:
我们的最根本的目的其实是更改HttpStatusCode,所以即便Exception被ABPExceptionFilter 封装成ErrorInfo来说对我们影响并不大,只要能让我们的自定义Filter catch住异常即可,因此,在Configuresevice中注册filter时声明顺序即可。

options.Filters.AddService(typeof(myExceptionFilter), order: );

另:由ABPExceptionFilter介绍可知,它并不会catch住所有类型的异常,如果想对任意类型的异常进行处理,使用middleware可能会更好。

参考内容:

github:
https://github.com/aspnetboilerplate/aspnetboilerplate/issues/1550
https://github.com/aspnetboilerplate/aspnetboilerplate/issues/3280

sessionliang大佬的博客

HK Zhang大佬的博客

ABP Changing Httpcode status的更多相关文章

  1. clear read-only status问题的解决

    IDEA系工具可能会报出的错误. 解决方法见官方文档吧:Changing Read-Only Status of Files  : https://www.jetbrains.com/help/ide ...

  2. words

    conscious[英][ˈkɒnʃəs][美][ˈkɑnʃəs]consensus[英][kənˈsensəs][美][kənˈsɛnsəs] scious sensuswaterflood; de ...

  3. SQLSERVER全文搜索

    SQLSERVER全文搜索 看这篇文章之前请先看一下下面我摘抄的全文搜索的MSDN资料,基本上MSDN上关于全文搜索的资料的我都copy下来了 并且非常认真地阅读和试验了一次,并且补充了一些SQL语句 ...

  4. xp+WinDBG+VMware调试内核

    呵呵,搞点突兀的标题而已.其实说的还是如何使用WinDBG和VMware来搭建调试内核的环境而已,这些网上已经有数不清的教程了,不过我喜欢自己亲手写一下.第一,把这个过程写一遍能加深印象,就算以后忘记 ...

  5. oracle11g中SQL优化(SQL TUNING)新特性之SQL Plan Management(SPM)

    1.   简介 Oracle Database11gR1引进了SQL PlanManagement(简称SPM),一套允许DBA捕获和保持任意SQL语句执行计划最优的新工具,这样,限制了刷新优化器统计 ...

  6. APPCORE Routine APIs

    Introduction to APPCORE Routine APIs This chapter provides you with specifications for calling many ...

  7. Mysql 5.7 基于组复制(MySQL Group Replication) - 运维小结

    之前介绍了Mysq主从同步的异步复制(默认模式).半同步复制.基于GTID复制.基于组提交和并行复制 (解决同步延迟),下面简单说下Mysql基于组复制(MySQL Group Replication ...

  8. CloudSim源代码学习——任务单元(Cloudlet)

    /* * Title: CloudSim Toolkit * Description: CloudSim (Cloud Simulation) Toolkit for Modeling and Sim ...

  9. SQL Server全文搜索

    SQL Server全文搜索 看这篇文章之前请先看一下下面我摘抄的全文搜索的MSDN资料,基本上MSDN上关于全文搜索的资料的我都copy下来了 并且非常认真地阅读和试验了一次,并且补充了一些SQL语 ...

随机推荐

  1. 在windows上安装VTK

    看了很多教程,花了1天半的时间装上了,记录下. 前置条件:我安装了VS2015,用来编译工程. 参考资料 官方:http://www.vtk.org/Wiki/VTK/Building 安装:http ...

  2. idea 中dao层自动生成接口

    1.在生成接口的类上右键 2.选中要生成的接口方法 3.点击Yes 4.出现(? reference in ? file)即生成成功

  3. jupyter notebook常用快捷键

    阅读目录 命令模式 (按键 Esc 开启) 编辑模式 ( Enter 键启动) Jupyter Notebook 的快捷键 使用前需要进行安装: pip install jupyter   (前提是你 ...

  4. NLP文本相似度

    NLP文本相似度 相似度 相似度度量:计算个体间相似程度 相似度值越小,距离越大,相似度值越大,距离越小 最常用--余弦相似度:​ 一个向量空间中两个向量夹角的余弦值作为衡量两个个体之间差异的大小 余 ...

  5. Android中弹出dialog后无法捕捉back键

    一.需求 在Android开发过程中,弹出dialog后无法捕捉back键,点击back按键无响应. 二.解决方案 原因:弹出dialog后,activity失去焦点,dialog获得当前焦点. 解决 ...

  6. Hystrix 熔断机制

    熔断机制相当于电路的跳闸功能,即在一定时间内,错误比例达到一定数目时业务从原来流程转移到另外的流程处理.在一段时间后,恢复到原业务逻辑. 测试代码如下 /** * @author zimu * @de ...

  7. [转] OpenStack — nova image-create, under the hood

    I was trying to understand what kind of image nova image-create creates. It's not entirely obvious f ...

  8. 逆水行舟 —— MyBatis

    第一轮总结性笔记 这是一个很漫长的过程,我买了套课程,将在日后记录学习笔记,取名为逆水行舟系列 MyBatis的基础 根据MyBatis的官方介绍: 整个测试项目结构如下:使用Maven架构项目 po ...

  9. jquery获取radio选中值及遍历

    使用jquery获取radio的值,最重要的是掌握jquery选择器的使用,在一个表单中我们通常是要获取被选中的那个radio项的值,所以要加checked来筛选,比如有以下的一些radio项:1.& ...

  10. selenium python 一些操作和定位收集

    (—)滚动条操作 python中selenium操作下拉滚动条方法汇总 selenium_webdriver(python)控制浏览器滚动条 selenium+Python(select定位) Sel ...