Tuesday, October 30, 2012

Handling Exceptions in ASP.NET Web API

Introduction

Whenever there is any unhandled exception in Web API controller class, most of the actual exceptions are indicated as HTTP status code 500, i.e. Internal Server Error. This generic error message is of little use to the client as it doesn't reveal the actual cause of the server side error. However, Web API allows you to fine tune and customize the HTTP errors that are sent to the client browser. This article discusses how this is done.

Throwing Exceptions from an API Controller

Consider a Web API controller named CustomerController as shown below:
  1. public class CustomerController : ApiController
  2. {
  3. public Customer Get(string id)
  4. {
  5. NorthwindEntities db=new NorthwindEntities();
  6. var data = from item in db.Customers
  7. where item.CustomerID == id
  8. select item;
  9. Customer obj = data.SingleOrDefault();
  10. if (obj == null)
  11. {
  12. throw new Exception("CustomerID Not Found in Database!");
  13. }
  14. else
  15. {
  16. return obj;
  17. }
  18. }
  19. ...
  20. }
The CustomerController class uses the Customers table of the Northwind database. The above code shows only the Get() method of the CustomerController class. The Get() method accepts a CustomerID and returns a Customer object matching the specified CustomerID. If there is no Customer entry for the specified CustomerID, an exception is thrown with message "CustomerID Not Found in Database!".
To invoke the Get() method from the client side you can use the jQuery $.ajax() method as shown below:
  1. $.ajax({
  2. type: "GET",
  3. url: '/api/Customer',
  4. data: {id:$("#txtCustomerID").val()},
  5. contentType: "application/json; charset=utf-8",
  6. dataType: "json",
  7. success: function (result) {
  8. alert(result.CustomerID + " - " + result.CompanyName);
  9. },
  10. error: function (err,type,httpStatus) {
  11. alert(err.status + " - " + err.statusText + " - " + httpStatus);
  12. }
  13. })
Since you are invoking the Get() method, the $.ajax() method uses GET as the request type. The url is specified as /api/Customer. Based on the request type and the URL, the Web API framework will invoke the Get() method of Customer API controller. The Get() method takes one parameter - CustomerID. The CustomerID is grabbed from a textbox (txtCustomerID) in the above example and is wrapped in an object literal with key id. This key name must be the same as the parameter name. If the Get() method completes successfully, a function as specified by the success setting is called. The success function receives the Customer object returned by the Get() method. The success method simply displays the CustomerID and CompanyName property values of the Customer object. The error function is called in case there is any error while invoking the Get() method. The error function receives three parameters, viz. error object, type of error and HTTP status.
If you run the above code and supply some CustomerID that doesn't exist, you will get an error message as shown below:
500 Error - Internal Server Error
500 Error - Internal Server Error
As you can see, though the server side exception message is "CustomerID Not Found in Database!" the client always shows the error message as "Internal Server Error".

Using HttpResponseException Class to Throw Exceptions

To emit some meaningful error messages to the client you can use the HttpResponseException class. Let's see how the Get() method can be modified using the HttpResponseException class. Have a look at the code below:
  1. public Customer Get(string id)
  2. {
  3. NorthwindEntities db=new NorthwindEntities();
  4. var data = from item in db.Customers
  5. where item.CustomerID == id
  6. select item;
  7. Customer obj = data.SingleOrDefault();
  8. if (obj == null)
  9. {
  10. HttpResponseMessage msg = new HttpResponseMessage(HttpStatusCode.NotFound)
  11. {
  12. Content = new StringContent(string.Format("No customer with ID = {0}", id)),
  13. ReasonPhrase = "CustomerID Not Found in Database!"
  14. };
  15. throw new HttpResponseException(msg);
  16. }
  17. else
  18. {
  19. return obj;
  20. }
  21. }
Notice the code marked in bold letters. The code creates an instance of HttpResponseMessage class by specifying HTTP status code as NotFound (i.e. 404). It also sets the Content and ReasonPhrase properties to a custom error message. The Content property specifies the response text whereas the ReasonPhrase property controls the descriptive text accompanying the status code. Finally, an HttpResponseException is thrown by passing the HttpResponseMessage instance that you just created. If you run the modified version of Get() method and supply some nonexistent CustomerID you will get an error message like this:
404 Error
404 Error
As you can see, this time the client gets the custom error message as specified by the ResponsePhrase property in the code instead of the standard message - "Internal Server Error". If you observe the HTTP response in Chrome Developer Tools you will see something like this:
HTTP response in Chrome Developer Tool
HTTP response in Chrome Developer Tool

Writing a Custom Exception Filter to Process Unhandled Exceptions

In the modified version of the Get() method you checked some condition and based on the outcome of the condition HttpResponseException was thrown. However, at times there can be unhandled exceptions in the code. These unhandled exceptions too don't generate any meaningful error message for the client. To deal with such unhandled exceptions you can create an Exception Filter. An exception filter is a class that implements the IExceptionFilter interface. To create a custom exception filter you can either implement the IExceptionFilter interface yourself or create a class that inherits from the inbuilt ExceptionFilterAttribute class. In the later approach all you need to do is override the OnException() method and plug-in some custom implementation. The following code shows how this is done.
  1. public class MyExceptionFilter:ExceptionFilterAttribute
  2. {
  3. public override void OnException(HttpActionExecutedContext context)
  4. {
  5. HttpResponseMessage msg = new HttpResponseMessage(HttpStatusCode.InternalServerError)
  6. {
  7. Content = new StringContent("An unhandled exception was thrown by Customer Web API controller."),
  8. ReasonPhrase = "An unhandled exception was thrown by Customer Web API controller."
  9. };
  10. context.Response = msg;
  11. }
  12. }
As you can see the MyExceptionFilter class inherits from ExceptionFilterAttribute class and overrides the OnException() method. The OnException() method is called whenever there is any unhandled exception. The OnException() method essentially creates a new HttpResponseMessage class as discussed earlier and then sets the Response property of the HttpActionExecutedContext  to the newly created HttpResponseMessage object.

Using a Custom Exception Filter

Once you develop a custom exception filter you can put it to use in two ways:
  • You can register it in the Application_Start event of Global.asax
  • You can decorate the Get()  method with the exception filter attribute
To register your custom exception filter using the first technique, add the the following piece of code to the Global.asax file.
  1. public class WebApiApplication : System.Web.HttpApplication
  2. {
  3. protected void Application_Start()
  4. {
  5. GlobalConfiguration.Configuration.Filters.Add(new WebAPIExceptionsDemo.MyExceptionFilter());
  6. AreaRegistration.RegisterAllAreas();
  7. ...
  8. }
  9. }
Notice the line of code marked in bold letters. The custom exception filter is added to the Filters collection using the GlobalConfiguration class. This way all the methods automatically use the MyExceptionFilter exception filter for dealing with unhandled exceptions.
Alternatively, you can also mark the individual methods with the custom exception filter attribute as shown below:
  1. [MyExceptionFilter]
  2. public Customer Get(string id)
  3. {
  4. ...
  5. }
As you can see the Get() method is now decorated with [MyExceptionFilter] attribute. To test the exception filter you can modify the Get() method to throw an unhandled exception.
  1. [MyExceptionFilter]
  2. public Customer Get(string id)
  3. {
  4. ...
  5. if (obj == null)
  6. {
  7. if (id.Length != 5)
  8. {
  9. throw new Exception("Invalid Customer ID!");
  10. }
  11. HttpResponseMessage msg = new HttpResponseMessage(HttpStatusCode.NotFound)
  12. {
  13. Content = new StringContent(string.Format("No customer with ID = {0}", id)),
  14. ReasonPhrase = "CustomerID Not Found in Database!"
  15. };
  16. throw new HttpResponseException(msg);
  17. }
  18. ...
  19. }
The following figure shows a sample run of the Get() method when there is an unhandled exception.
A sample run of the Get() method
A sample run of the Get() method

Summary

By default any unhandled exceptions thrown inside Web API are propagated to the client as "Internal Server Error". This generic error message is of little use to the client. Using HttpResponseException class you can return meaningful error messages to the client. You can also create an exception filter to deal with unhandled exceptions and to return custom error messages to the client.

No comments:

Post a Comment