This site is the archived OWASP Foundation Wiki and is no longer accepting Account Requests.
To view the new OWASP Foundation website, please visit https://owasp.org

Difference between revisions of "Exception Handling"

From OWASP
Jump to: navigation, search
(Created page with "= DRAFT DOCUMENT - WORK IN PROGRESS = ==Description== ==Risk Factors== ==Related Attacks== ==Related Vulnerabilities== ==Related Controls== ==Related ...")
 
Line 2: Line 2:
  
 
==Description==
 
==Description==
 +
An exception is any error condition or unexpected behavior that is encountered by your application. There are several key security concerns to be aware of when handling exceptions in .NET web applications:
 +
*Preventing system information leakage that could aid a malicious user.
 +
*Ensuring the application fails securely under all circumstances, both expected and not expected.
 +
*Using a centralized error strategy to reduce points of failure and promote consistency.
 +
*Log when exceptions are thrown and include sufficient detail for security auditing.
  
 +
==Displaying Errors==
 +
When your application displays error messages, it should not give away information that a malicious user might find helpful in attacking your system. For example, if your application unsuccessfully tries to log in to a database, it should not display an error message that includes the connection string being used.
  
==Risk Factors==
+
The default display mechanism for unhandled exceptions in ASP.NET is controlled via the ''<customErrors>'' element in the web.config file. This element includes a ''mode'' attribute that can be set to: ''Off'', ''On'', or ''RemoteOnly''. This mode setting should never be set to ''Off'' in a production environment because it will cause exception details to be displayed to the end user which may include sensitive information about the system.
  
 +
<pre><customErrors mode="RemoteOnly" defaultRedirect="/Error">
 +
  <error statusCode="404" redirect="/Error?id=404" />
 +
  <error statusCode="403" redirect="/Error?id=403" />
 +
</customErrors></pre>
  
==Related [[Attacks]]==
+
===HTTP Status Codes===
 +
When reviewing error responses, be careful that a malicious user is not able to glean any potentially sensitive information about the system from the status code returned. For example: an API search that takes a tenant ID parameter that returns a HTTP 404 response for no results, and an HTTP 500 response if the tenant ID does not exist. An attacker could potentially leverage this information to determine if a tenant ID is valid or not. For this scenario one solution would be to catch the exception thrown when a supplied tenant ID does not exist, and return an HTTP 404 status code with the error response.
  
 +
==Logging Exception Details==
 +
There are a variety of places within a typical web application where exceptions can be caught and logged. Ideally, you should handle exceptions in as many of the locations described below as are applicable to your application.
  
==Related [[Vulnerabilities]]==
+
===Application Level===
 +
The following code example shows how to create an error handler in the Global.asax file that will catch and log all unhandled ASP.NET errors at the application level. In this example ''Server.ClearError()'' is not called so that the exception will continue to be processed by the error display mechanism defined by the ''<customErrors>'' element in web.config. Unhandled exceptions must be logged within ''Application_Error'' rather than in a subsequent page defined by ''<customErrors>'', because the exception details are only available at this point.
  
 +
<pre>void Application_Error(object sender, EventArgs e)
 +
{
 +
  // Get the exception object.
 +
  var exception = Server.GetLastError();
  
==Related [[Controls]]==
+
  // Log exception details. Be sure to include items like client IP
 +
  // address and current authenticated user if applicable.
 +
  System.Diagnostics.Trace.TraceError(exception.ToString());
 +
}</pre>
  
 +
===ASP.NET Web Forms===
 +
ASP.NET web form pages provide the ability to capture exceptions at the page level in a similar fashion to the application level. This may be useful if you need to perform failure handling logic or log additional detail unique to the page.
  
==Related [[Technical Impacts]]==
+
<pre>void Page_Error(object sender, EventArgs e)
 +
{
 +
  // Get the exception object.
 +
  var exception = Server.GetLastError();
  
 +
  // Log exception details. Be sure to include items like client IP
 +
  // address and current authenticated user if applicable.
 +
  System.Diagnostics.Trace.TraceError(exception.ToString());
 +
 +
  // Calling Server.ClearError allows the page to continue processing,
 +
  // otherwise control is passed to application level error handling.
 +
  Server.ClearError();
 +
}</pre>
 +
 +
===ASP.NET MVC===
 +
The MVC Controller class provides an ''OnException'' method similar to the ''Page_Error'' method in ASP.NET Web Forms. It allows you to process all unhandled exceptions thrown by actions of the controller. One option for implementing exception logging is to create a base controller class and defining the ''OnException'' method in this one central location.
 +
 +
<pre>protected override void OnException(ExceptionContext filterContext)
 +
{
 +
  base.OnException(filterContext);
 +
  System.Diagnostics.Trace.TraceError(filterContext.Exception.ToString());
 +
}</pre>
 +
 +
A second (and arguably better) approach is to utilize the ''System.Web.Mvc.HandleErrorAttribute'' which can be used to handle exceptions at the application, controller, or action level. To apply this at the application level you add an instance of ''HandleErrorAttribute'' to the ''GlobalFilterCollection'' at application startup. This takes affect before the ''Application_Error'' method in Global.asax.cs, and by default will redirect to the ''Error'' view in the ~/Views/Shared folder. This will bypass any custom error pages defined via the ''<customErrors>'' element in web.config.
 +
 +
<pre>public static void RegisterGlobalFilters(GlobalFilterCollection filters)
 +
{
 +
  filters.Add(new HandleErrorAttribute());
 +
}</pre>
 +
 +
You can log the exception details right within the ''Error.cshtml'' view using properties from the view model which is of type ''System.Web.Mvc.HandleErrorInfo''.
 +
 +
<pre>@model System.Web.Mvc.HandleErrorInfo
 +
 +
@{
 +
    ViewBag.Title = "Error";
 +
   
 +
    // Log exception details
 +
    System.Diagnostics.Trace.TraceError(Model.Exception.ToString());
 +
}
 +
 +
<h1 class="text-danger">Error.</h1>
 +
<h2 class="text-danger">An error occurred while processing your request.</h2></pre>
 +
 +
===ASP.NET Web API===
 +
When handling exceptions for your API endpoints you generally want to return a JSON or XML response rather than redirect to an HTML page. ASP.NET Web API provides exception filters to customize how exceptions are handled. A good way of adding exception logging is to derive your own class from ''System.Web.Http.Filters.ExceptionFilterAttribute'' and override the ''OnException'' method as shown.
 +
 +
<pre>public class LogExceptionFilterAttribute : ExceptionFilterAttribute
 +
{
 +
  public override void OnException(HttpActionExecutedContext context)
 +
  {
 +
    System.Diagnostics.Trace.TraceError(context.Exception.ToString());
 +
    base.OnException(context);
 +
  }
 +
}</pre>
 +
 +
You can apply the ''[LogExceptionFilter]'' attribute to any ''ApiController'', or add it at a global level using the following code at application startup.
 +
 +
<pre>System.Web.Http.GlobalConfiguration.Configuration.Filters.Add(new LogExceptionFilterAttribute());</pre>
 +
 +
===Web API Global Error Handling (Added in Web API 2.1)===
 +
There are a number of cases in Web API where an exception can occur that falls outside of what exception filters can handle. These include exceptions in controller construction, message handlers, routing, and response content serialization. To handle these exceptions you will need to register a class that implements the ''IExceptionLogger'' interface, or derives from the ''ExceptionLogger'' base class.
 +
 +
<pre>public interface IExceptionLogger
 +
{
 +
  Task LogAsync(ExceptionLoggerContext context, CancellationToken cancellationToken);
 +
}</pre>
 +
 +
You then register your class at application startup to ensure that all exceptions raised during a Web API request life cycle are captured and logged.
 +
 +
<pre>System.Web.Http.GlobalConfiguration.Services.Add(typeof(IExceptionLogger), new MyExceptionLogger());</pre>
 +
 +
==Logging Sensitive Data==
 +
Be careful when logging exception details that you do not inadvertently expose sensitive data to a logging system or external reports that may have relaxed access controls compared to the primary system. For example if a ''SqlException'' is thrown and the details contain a connection string, you may want to mask the password attribute before logging the event. Use care when logging exception details that may contain the following types of data:
 +
 +
*Passwords
 +
*Authentication tokens
 +
*Personally identifiable information (PII)
 +
*Personal health information (PHI)
 +
*Credit card data
  
 
==References==
 
==References==
 
+
*[http://www.asp.net/web-forms/tutorials/deployment/deploying-web-site-projects/displaying-a-custom-error-page-cs Displaying a Custom Error Page]
 +
*[http://msdn.microsoft.com/en-us/library/h0hfz6fc(v=vs.100).aspx customErrors Element (ASP.NET Settings Schema)]
 +
*[http://msdn.microsoft.com/en-us/library/ed577840(v=vs.100).aspx How to: Handle Page-Level Errors]
 +
*[http://msdn.microsoft.com/en-us/library/system.web.mvc.controller.onexception(v=vs.118).aspx Controller.OnException Method]
 +
*[http://msdn.microsoft.com/en-us/library/system.web.mvc.handleerrorattribute(v=vs.118).aspx HandleErrorAtttribute Class]
 +
*[http://www.asp.net/web-api/overview/testing-and-debugging/exception-handling Exception Handling in ASP.NET Web API]
 +
*[http://www.asp.net/web-api/overview/testing-and-debugging/web-api-global-error-handling Web API 2 .1 Global Error Handling]
  
 
[[Category:OWASP .NET Project]][[Category:Stub]]
 
[[Category:OWASP .NET Project]][[Category:Stub]]

Revision as of 13:50, 2 October 2014

DRAFT DOCUMENT - WORK IN PROGRESS

Description

An exception is any error condition or unexpected behavior that is encountered by your application. There are several key security concerns to be aware of when handling exceptions in .NET web applications:

  • Preventing system information leakage that could aid a malicious user.
  • Ensuring the application fails securely under all circumstances, both expected and not expected.
  • Using a centralized error strategy to reduce points of failure and promote consistency.
  • Log when exceptions are thrown and include sufficient detail for security auditing.

Displaying Errors

When your application displays error messages, it should not give away information that a malicious user might find helpful in attacking your system. For example, if your application unsuccessfully tries to log in to a database, it should not display an error message that includes the connection string being used.

The default display mechanism for unhandled exceptions in ASP.NET is controlled via the <customErrors> element in the web.config file. This element includes a mode attribute that can be set to: Off, On, or RemoteOnly. This mode setting should never be set to Off in a production environment because it will cause exception details to be displayed to the end user which may include sensitive information about the system.

<customErrors mode="RemoteOnly" defaultRedirect="/Error">
  <error statusCode="404" redirect="/Error?id=404" />
  <error statusCode="403" redirect="/Error?id=403" />
</customErrors>

HTTP Status Codes

When reviewing error responses, be careful that a malicious user is not able to glean any potentially sensitive information about the system from the status code returned. For example: an API search that takes a tenant ID parameter that returns a HTTP 404 response for no results, and an HTTP 500 response if the tenant ID does not exist. An attacker could potentially leverage this information to determine if a tenant ID is valid or not. For this scenario one solution would be to catch the exception thrown when a supplied tenant ID does not exist, and return an HTTP 404 status code with the error response.

Logging Exception Details

There are a variety of places within a typical web application where exceptions can be caught and logged. Ideally, you should handle exceptions in as many of the locations described below as are applicable to your application.

Application Level

The following code example shows how to create an error handler in the Global.asax file that will catch and log all unhandled ASP.NET errors at the application level. In this example Server.ClearError() is not called so that the exception will continue to be processed by the error display mechanism defined by the <customErrors> element in web.config. Unhandled exceptions must be logged within Application_Error rather than in a subsequent page defined by <customErrors>, because the exception details are only available at this point.

void Application_Error(object sender, EventArgs e)
{
  // Get the exception object.
  var exception = Server.GetLastError();

  // Log exception details. Be sure to include items like client IP 
  // address and current authenticated user if applicable.
  System.Diagnostics.Trace.TraceError(exception.ToString());
}

ASP.NET Web Forms

ASP.NET web form pages provide the ability to capture exceptions at the page level in a similar fashion to the application level. This may be useful if you need to perform failure handling logic or log additional detail unique to the page.

void Page_Error(object sender, EventArgs e)
{
  // Get the exception object.
  var exception = Server.GetLastError();

  // Log exception details. Be sure to include items like client IP 
  // address and current authenticated user if applicable.
  System.Diagnostics.Trace.TraceError(exception.ToString());

  // Calling Server.ClearError allows the page to continue processing,
  // otherwise control is passed to application level error handling.
  Server.ClearError();
}

ASP.NET MVC

The MVC Controller class provides an OnException method similar to the Page_Error method in ASP.NET Web Forms. It allows you to process all unhandled exceptions thrown by actions of the controller. One option for implementing exception logging is to create a base controller class and defining the OnException method in this one central location.

protected override void OnException(ExceptionContext filterContext)
{
  base.OnException(filterContext);
  System.Diagnostics.Trace.TraceError(filterContext.Exception.ToString());
}

A second (and arguably better) approach is to utilize the System.Web.Mvc.HandleErrorAttribute which can be used to handle exceptions at the application, controller, or action level. To apply this at the application level you add an instance of HandleErrorAttribute to the GlobalFilterCollection at application startup. This takes affect before the Application_Error method in Global.asax.cs, and by default will redirect to the Error view in the ~/Views/Shared folder. This will bypass any custom error pages defined via the <customErrors> element in web.config.

public static void RegisterGlobalFilters(GlobalFilterCollection filters)
{
  filters.Add(new HandleErrorAttribute());
}

You can log the exception details right within the Error.cshtml view using properties from the view model which is of type System.Web.Mvc.HandleErrorInfo.

@model System.Web.Mvc.HandleErrorInfo

@{
    ViewBag.Title = "Error";
    
    // Log exception details
    System.Diagnostics.Trace.TraceError(Model.Exception.ToString());
}

<h1 class="text-danger">Error.</h1>
<h2 class="text-danger">An error occurred while processing your request.</h2>

ASP.NET Web API

When handling exceptions for your API endpoints you generally want to return a JSON or XML response rather than redirect to an HTML page. ASP.NET Web API provides exception filters to customize how exceptions are handled. A good way of adding exception logging is to derive your own class from System.Web.Http.Filters.ExceptionFilterAttribute and override the OnException method as shown.

public class LogExceptionFilterAttribute : ExceptionFilterAttribute 
{
  public override void OnException(HttpActionExecutedContext context)
  {
    System.Diagnostics.Trace.TraceError(context.Exception.ToString());
    base.OnException(context);
  }
}

You can apply the [LogExceptionFilter] attribute to any ApiController, or add it at a global level using the following code at application startup.

System.Web.Http.GlobalConfiguration.Configuration.Filters.Add(new LogExceptionFilterAttribute());

Web API Global Error Handling (Added in Web API 2.1)

There are a number of cases in Web API where an exception can occur that falls outside of what exception filters can handle. These include exceptions in controller construction, message handlers, routing, and response content serialization. To handle these exceptions you will need to register a class that implements the IExceptionLogger interface, or derives from the ExceptionLogger base class.

public interface IExceptionLogger
{
  Task LogAsync(ExceptionLoggerContext context, CancellationToken cancellationToken);
}

You then register your class at application startup to ensure that all exceptions raised during a Web API request life cycle are captured and logged.

System.Web.Http.GlobalConfiguration.Services.Add(typeof(IExceptionLogger), new MyExceptionLogger());

Logging Sensitive Data

Be careful when logging exception details that you do not inadvertently expose sensitive data to a logging system or external reports that may have relaxed access controls compared to the primary system. For example if a SqlException is thrown and the details contain a connection string, you may want to mask the password attribute before logging the event. Use care when logging exception details that may contain the following types of data:

  • Passwords
  • Authentication tokens
  • Personally identifiable information (PII)
  • Personal health information (PHI)
  • Credit card data

References