首页 > 代码库 > [转]Create Custom Exception Filter in ASP.NET Core

[转]Create Custom Exception Filter in ASP.NET Core

本文转自:http://www.binaryintellect.net/articles/5df6e275-1148-45a1-a8b3-0ba2c7c9cea1.aspx

 

In my previous article I explained how errors in an ASP.NET Core web application can be dealt with  using middleware. I also mentioned that time that you can also use exception  filters to handle errors. In this article I will explain how to do just that.

In ASP.NET MVC 5 you used the [HandleError] attribute and OnException()  controller method to deal with exceptions that arise in the actions of a  controller. In ASP.NET Core the process is bit different but the overall concept  is still the same. As you are probable aware of, filters sit in the ASP.NET Core  pipeline and allow you to execute a piece of code before or after an action  execution. To understand how errors can be handled in ASP.NET Core let‘s create  an exception filter called HandleException and then try to mimic the behavior of  [HandleError] of ASP.NET MVC 5.

Begin by creating a new ASP.NET web application. Then add a class to it named  HandleExceptionAttribute. This class needs to inherit from  ExceptionFilterAttribute base class. The complete code of this class is shown  below:

using Microsoft.AspNetCore.Mvc;using Microsoft.AspNetCore.Mvc.Filters;using Microsoft.AspNetCore.Mvc.ModelBinding;using Microsoft.AspNetCore.Mvc.ViewFeatures;........public class HandleExceptionAttribute : ExceptionFilterAttribute{    public override void OnException(ExceptionContext context)    {      var result = new ViewResult { ViewName = "Error" };      var modelMetadata = http://www.mamicode.com/new EmptyModelMetadataProvider();"HandleException",               context.Exception);      context.Result = result;      context.ExceptionHandled = true;    }            }}

The HandleExceptionAttribute class overrides the OnException() method of t he  base class. The OnException() method is called only when there is an unhandled  exception in an action. Inside, we create a ViewResult object that represents  the custom error page. It is assumed that the view representing the custom error  page is Error.cshtml. Later we will write some code to remove this hard-coding.

Then the code creates an empty object of EmptyModelMetadataProvider class.  This is required because we need to initialize the ViewData of the request and  add exception details in it. Then ViewData property of the ViewResult is  assigned to a new ViewDataDictionary object. While creating the  ViewDataDictionary the IModelMetadataProvider object created earlier is passed  to the constructor along with the context‘s ModelState.

We would like to store the exception details into the ViewData so that the  Error view can retrieve them and possibly display on the page. To facilitate  this HandleException key is added to the ViewData. The exception being thrown  can be obtained using context‘s Exception property.

Then Result property of context is set to the ViewResult we just configured.  Finally, ExceptionHandled property is set to true to indicate that the exception  need not be bubbled up any further.

Now open the HomeController and decorate the Index() action with the [HandleException]  attribute.

[HandleException]public IActionResult Index(){    throw new Exception("This is some exception!!!");    return View();}

Also add Error.cshtml in Shared folder and write the following code to it:

@{     Exception ex = ViewData["HandleException"] as Exception;}<h1>@ex.Message</h1><div>@ex.StackTrace</div>

As you can see the Error view can grab the exception details using the  HandleException key stored in the ViewData. The Message and StackTrace  properties of the Exception are then outputted on the page.

If you run the application you should see the error page like this:

技术分享

So far so good. Now let‘s add a couple of features to our [HandleException]  attribute:

  • The ViewName should be customizable
  • There should be provision to handle only specific type of exceptions

Go ahead and add two public properties to the HandleExceptionAttribute class.  These properties are shown below:

public class HandleExceptionAttribute : ExceptionFilterAttribute{    public string ViewName { get; set; } = "Error";    public Type ExceptionType { get; set; } = null;    ....    ....}

The ViewName property allows you to specify a custom view name that is then  used while forming the ViewResult. The ExceptionType property holds the Type of  the exception you wish to handle (say DivideByZeroException or FormatException).

Then modify the OnException() as shown below:

 public override void OnException(ExceptionContext context){  if(this.ExceptionType != null)  {    if(context.Exception.GetType() == this.ExceptionType)    {      ....      ....    }  }  else  {    ....    ....  }}

The OnException() method now checks whether a specific ExceptionType is to be  handled (non null value). If yes then it checks whether the exception being  thrown matches with that type (inner if block) and accordingly ViewResult is  configured. If no specific ExceptionType is to be checked then control directly  goes in the else block. The code in both the cases is shown below:

var result = new ViewResult { ViewName = this.ViewName };var modelMetadata = http://www.mamicode.com/new EmptyModelMetadataProvider();"HandleException", context.Exception);context.Result = result;context.ExceptionHandled = true;

This code is quite similar to what you wrote earlier. But this time it uses  the ViewName property while forming the ViewResult.

Now open the HomeController and change the [HandleException] attribute like  this:

[HandleException(ViewName ="CustomError",ExceptionType =typeof(DivideByZeroException))]public IActionResult Index(){    throw new Exception("This is some exception!!!");    return View();}

Now the [HandleException] attribute specifies two properties - ViewName and  ExceptionType. Since ExceptionType is set to the type of DivideByZeroException  the [HandleException] won‘t handle the error (because Index() throws a custom  Exception). So, rename Error.cshtml to CustomError.cshtml and throw a  DivideByZeroException. This time error gets trapped as expected.

It should be noted that [HandleException] can be applied on top of the  controller class itself rather than to individual actions.

[HandleException]public class HomeController : Controller{........}

More details about the ASP.NET Core filters can be found here.

That‘s it for now! Keep coding!!

Bipin Joshi is a software consultant, trainer, author and a yogi having 21+ years of experience in software development. He conducts online courses in ASP.NET MVC / Core, jQuery, AngularJS, and Design Patterns. He is a published author and has authored or co-authored books for Apress and Wrox press. Having embraced Yoga way of life he also teaches Ajapa Meditation to interested individuals. To know more about him click here.

[转]Create Custom Exception Filter in ASP.NET Core