AspNetCore.Docs/aspnetcore/fundamentals/error-handling.md

136 lines
7.2 KiB
Markdown
Raw Normal View History

---
2017-07-01 07:47:15 +08:00
title: Error Handling in ASP.NET Core
2016-11-12 00:28:14 +08:00
author: ardalis
2016-11-30 08:35:18 +08:00
description: Explains how to handle errors in ASP.NET Core applications
keywords: ASP.NET Core, error handling, exception handling,
ms.author: tdykstra
2016-10-29 01:35:15 +08:00
manager: wpickett
2016-11-30 08:35:18 +08:00
ms.date: 11/30/2016
2016-10-29 01:35:15 +08:00
ms.topic: article
ms.assetid: 4db51023-c8a6-4119-bbbe-3917e272c260
2016-11-17 08:24:57 +08:00
ms.technology: aspnet
ms.prod: asp.net-core
uid: fundamentals/error-handling
2017-02-28 04:40:05 +08:00
ms.custom: H1Hack27Feb2017
2016-10-29 01:35:15 +08:00
---
2017-02-28 04:40:05 +08:00
# Introduction to Error Handling in ASP.NET Core
2016-10-29 01:35:15 +08:00
2016-11-30 08:35:18 +08:00
By [Steve Smith](http://ardalis.com) and [Tom Dykstra](https://github.com/tdykstra/)
2016-10-29 01:35:15 +08:00
2017-06-20 06:07:29 +08:00
This article covers common appoaches to handling errors in ASP.NET Core apps.
2016-10-29 01:35:15 +08:00
[View or download sample code](https://github.com/aspnet/Docs/tree/master/aspnetcore/fundamentals/error-handling/sample)
2016-10-29 01:35:15 +08:00
2016-11-30 08:35:18 +08:00
## The developer exception page
2016-10-29 01:35:15 +08:00
2016-11-30 08:35:18 +08:00
To configure an app to display a page that shows detailed information about exceptions, install the `Microsoft.AspNetCore.Diagnostics` NuGet package and add a line to the [Configure method in the Startup class](startup.md):
2016-10-29 01:35:15 +08:00
2016-11-30 08:35:18 +08:00
[!code-csharp[Main](error-handling/sample/Startup.cs?name=snippet_DevExceptionPage&highlight=7)]
2016-10-29 01:35:15 +08:00
2016-12-02 06:55:34 +08:00
Put `UseDeveloperExceptionPage` before any middleware you want to catch exceptions in, such as `app.UseMvc`.
2016-11-30 08:35:18 +08:00
>[!WARNING]
> Enable the developer exception page **only when the app is running in the Development environment**. You don't want to share detailed exception information publicly when the app runs in production. [Learn more about configuring environments](environments.md).
2016-10-29 01:35:15 +08:00
2016-12-02 06:55:34 +08:00
To see the developer exception page, run the sample application with the environment set to `Development`, and add `?throw=true` to the base URL of the app. The page includes several tabs with information about the exception and the request. The first tab includes a stack trace.
2016-10-29 01:35:15 +08:00
2016-11-30 08:35:18 +08:00
![Stack trace](error-handling/_static/developer-exception-page.png)
2016-10-29 01:35:15 +08:00
2016-12-02 06:55:34 +08:00
The next tab shows the query string parameters, if any.
2016-10-29 01:35:15 +08:00
2016-11-30 08:35:18 +08:00
![Query string parameters](error-handling/_static/developer-exception-page-query.png)
2016-10-29 01:35:15 +08:00
2016-11-30 08:35:18 +08:00
This request didn't have any cookies, but if it did, they would appear on the **Cookies** tab. You can see the headers that were passed in the last tab.
2016-10-29 01:35:15 +08:00
2016-11-30 08:35:18 +08:00
![Headers](error-handling/_static/developer-exception-page-headers.png)
2016-10-29 01:35:15 +08:00
2016-11-30 08:35:18 +08:00
## Configuring a custom exception handling page
It's a good idea to configure an exception handler page to use when the app is not running in the `Development` environment.
[!code-csharp[Main](error-handling/sample/Startup.cs?name=snippet_DevExceptionPage&highlight=11)]
In an MVC app, don't explicitly decorate the error handler action method with HTTP method attributes, such as `HttpGet`. Using explicit verbs could prevent some requests from reaching the method.
2016-10-29 01:35:15 +08:00
2016-11-18 13:03:07 +08:00
```csharp
2016-10-29 01:35:15 +08:00
[Route("/Error")]
public IActionResult Index()
{
// Handle error here
}
2016-11-18 13:03:07 +08:00
```
2016-10-29 01:35:15 +08:00
2016-11-30 08:35:18 +08:00
## Configuring status code pages
2016-10-29 01:35:15 +08:00
2016-12-02 06:55:34 +08:00
By default, your app will not provide a rich status code page for HTTP status codes such as 500 (Internal Server Error) or 404 (Not Found). You can configure the `StatusCodePagesMiddleware` by adding a line to the `Configure` method:
2016-10-29 01:35:15 +08:00
2016-11-18 13:03:07 +08:00
```csharp
2016-10-29 01:35:15 +08:00
app.UseStatusCodePages();
2016-11-18 13:03:07 +08:00
```
2016-10-29 01:35:15 +08:00
2016-12-02 06:55:34 +08:00
By default, this middleware adds simple, text-only handlers for common status codes, such as 404:
2016-10-29 01:35:15 +08:00
2016-11-30 08:35:18 +08:00
![404 page](error-handling/_static/default-404-status-code.png)
2016-10-29 01:35:15 +08:00
2016-11-30 08:35:18 +08:00
The middleware supports several different extension methods. One takes a lambda expression, another takes a content type and format string.
[!code-csharp[Main](error-handling/sample/Startup.cs?name=snippet_StatusCodePages)]
2016-10-29 01:35:15 +08:00
2016-11-18 13:03:07 +08:00
```csharp
2016-11-30 08:35:18 +08:00
app.UseStatusCodePages("text/plain", "Status code page, status code: {0}");
```
2016-10-29 01:35:15 +08:00
2016-11-30 08:35:18 +08:00
There are also redirect extension methods. One sends a 302 status code to the client, and one returns the original status code to the client but also executes the handler for the redirect URL.
2016-10-29 01:35:15 +08:00
2016-11-30 08:35:18 +08:00
[!code-csharp[Main](error-handling/sample/Startup.cs?name=snippet_StatusCodePagesWithRedirect)]
2016-10-29 01:35:15 +08:00
2016-11-18 13:03:07 +08:00
```csharp
2016-11-30 08:35:18 +08:00
app.UseStatusCodePagesWithReExecute("/error/{0}");
```
2016-10-29 01:35:15 +08:00
2016-11-30 08:35:18 +08:00
If you need to disable status code pages for certain requests, you can do so:
2016-10-29 01:35:15 +08:00
2016-11-18 13:03:07 +08:00
```csharp
2016-10-29 01:35:15 +08:00
var statusCodePagesFeature = context.Features.Get<IStatusCodePagesFeature>();
if (statusCodePagesFeature != null)
{
statusCodePagesFeature.Enabled = false;
}
2016-11-18 13:03:07 +08:00
```
2016-10-29 01:35:15 +08:00
2016-11-30 08:35:18 +08:00
## Exception-handling code
2016-10-29 01:35:15 +08:00
2016-11-30 08:35:18 +08:00
Code in exception handling pages can throw exceptions. It's often a good idea for production error pages to consist of purely static content.
2016-10-29 01:35:15 +08:00
2016-11-30 08:35:18 +08:00
Also, be aware that once the headers for a response have been sent, you can't change the response's status code, nor can any exception pages or handlers run. The response must be completed or the connection aborted.
2016-10-29 01:35:15 +08:00
2016-11-30 08:35:18 +08:00
## Server exception handling
2016-10-29 01:35:15 +08:00
2016-12-10 09:26:43 +08:00
In addition to the exception handling logic in your app, the [server](servers/index.md) hosting your app will perform some exception handling. If the server catches an exception before the headers have been sent it sends a 500 Internal Server Error response with no body. If it catches an exception after the headers have been sent, it closes the connection. Requests that are not handled by your app will be handled by the server, and any exception that occurs will be handled by the server's exception handling. Any custom error pages or exception handling middleware or filters you have configured for your app will not affect this behavior.
2016-10-29 01:35:15 +08:00
2016-11-30 08:35:18 +08:00
## Startup exception handling
2016-10-29 01:35:15 +08:00
2016-11-30 08:35:18 +08:00
Only the hosting layer can handle exceptions that take place during app startup. Exceptions that occur during app startup can impact server behavior. For example, if an exception happens before you call `KestrelServerOptions.UseHttps`, the hosting layer catches the exception, starts the server, and displays an error page on the non-SSL port. If an exception happens after that line executes, the error page is served over HTTPS instead.
2016-10-29 01:35:15 +08:00
You can [configure how the host will behave in response to errors during startup](hosting.md#configuring-a-host) using `CaptureStartupErrors` and the `detailedErrors` key.
## ASP.NET MVC error handling
2016-10-29 01:35:15 +08:00
2016-12-02 06:55:34 +08:00
[MVC](../mvc/index.md) apps have some additional options for handling errors, such as configuring exception filters and performing model validation.
2016-10-29 01:35:15 +08:00
### Exception Filters
2016-11-30 08:35:18 +08:00
Exception filters can be configured globally or on a per-controller or per-action basis in an MVC app. These filters handle any unhandled exception that occurs during the execution of a controller action or another filter, and are not called otherwise. Learn more about exception filters in [Filters](../mvc/controllers/filters.md).
2016-10-29 01:35:15 +08:00
>[!TIP]
> Exception filters are good for trapping exceptions that occur within MVC actions, but they're not as flexible as error handling middleware. Prefer middleware for the general case, and use filters only where you need to do error handling *differently* based on which MVC action was chosen.
### Handling Model State Errors
2016-11-30 08:35:18 +08:00
[Model validation](../mvc/models/validation.md) occurs prior to each controller action being invoked, and it is the action methods responsibility to inspect `ModelState.IsValid` and react appropriately.
Some apps will choose to follow a standard convention for dealing with model validation errors, in which case a [filter](../mvc/controllers/filters.md) may be an appropriate place to implement such a policy. You should test how your actions behave with invalid model states. Learn more in [Testing controller logic](../mvc/controllers/testing.md).
2016-10-29 01:35:15 +08:00