mirror of https://github.com/abpframework/abp
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
303 lines
11 KiB
303 lines
11 KiB
7 years ago
|
## Exception Handling
|
||
|
|
||
6 years ago
|
ABP provides a built-in infrastructure and offers a standard model for handling exceptions in a web application.
|
||
7 years ago
|
|
||
|
* Automatically **handles all exceptions** and sends a standard **formatted error message** to the client for an API/AJAX request.
|
||
7 years ago
|
* Automatically hides **internal infrastructure errors** and returns a standard error message.
|
||
6 years ago
|
* Provides a configurable way to **localize** exception messages.
|
||
|
* Automatically maps standard exceptions to **HTTP status codes** and provides a configurable option to map these to custom exceptions.
|
||
7 years ago
|
|
||
|
### Automatic Exception Handling
|
||
|
|
||
6 years ago
|
`AbpExceptionFilter` handles an exception if **any of the following conditions** are meet:
|
||
7 years ago
|
|
||
|
* Exception is thrown by a **controller action** which returns an **object result** (not a view result).
|
||
|
* The request is an AJAX request (`X-Requested-With` HTTP header value is `XMLHttpRequest`).
|
||
|
* Client explicitly accepts the `application/json` content type (via `accept` HTTP header).
|
||
|
|
||
|
If the exception is handled it's automatically **logged** and a formatted **JSON message** is returned to the client.
|
||
|
|
||
|
#### Error Message Format
|
||
|
|
||
|
Error Message is an instance of the `RemoteServiceErrorResponse` class. The simplest error JSON has a **message** property as shown below:
|
||
|
|
||
|
````json
|
||
|
{
|
||
|
"error": {
|
||
|
"message": "This topic is locked and can not add a new message"
|
||
|
}
|
||
|
}
|
||
|
````
|
||
|
|
||
6 years ago
|
There are **optional fields** those can be filled based upon the exception that has occured.
|
||
7 years ago
|
|
||
|
##### Error Code
|
||
|
|
||
|
Error **code** is an optional and unique string value for the exception. Thrown `Exception` should implement the `IHasErrorCode` interface to fill this field. Example JSON value:
|
||
|
|
||
|
````json
|
||
|
{
|
||
|
"error": {
|
||
|
"code": "App:010042",
|
||
|
"message": "This topic is locked and can not add a new message"
|
||
|
}
|
||
|
}
|
||
|
````
|
||
|
|
||
|
Error code can also be used to localize the exception and customize the HTTP status code (see the related sections below).
|
||
|
|
||
|
##### Error Details
|
||
|
|
||
|
Error **details** in an optional field of the JSON error message. Thrown `Exception` should implement the `IHasErrorDetails` interface to fill this field. Example JSON value:
|
||
|
|
||
|
```json
|
||
|
{
|
||
|
"error": {
|
||
|
"code": "App:010042",
|
||
|
"message": "This topic is locked and can not add a new message",
|
||
|
"details": "A more detailed info about the error..."
|
||
|
}
|
||
|
}
|
||
|
```
|
||
|
|
||
|
##### Validation Errors
|
||
|
|
||
|
**validationErrors** is a standard field that is filled if the thrown exception implements the `IHasValidationErrors` interface.
|
||
|
|
||
|
````json
|
||
|
{
|
||
|
"error": {
|
||
|
"code": "App:010046",
|
||
|
"message": "Your request is not valid, please correct and try again!",
|
||
|
"validationErrors": [{
|
||
7 years ago
|
"message": "Username should be minimum lenght of 3.",
|
||
7 years ago
|
"members": ["userName"]
|
||
|
},
|
||
|
{
|
||
|
"message": "Password is required",
|
||
|
"members": ["password"]
|
||
|
}]
|
||
|
}
|
||
|
}
|
||
|
````
|
||
|
|
||
6 years ago
|
`AbpValidationException` implements the `IHasValidationErrors` interface and it is automatically thrown by the framework when a request input is not valid. So, usually you don't need to deal with validation errors unless you have higly customised validation logic.
|
||
7 years ago
|
|
||
|
#### Logging
|
||
|
|
||
7 years ago
|
Caught exceptions are automatically logged.
|
||
7 years ago
|
|
||
|
##### Log Level
|
||
|
|
||
6 years ago
|
Exceptions are logged with the `Error` level by default. The Log level can be determined by the exception if it implements the `IHasLogLevel` interface. Example:
|
||
7 years ago
|
|
||
|
````C#
|
||
|
public class MyException : Exception, IHasLogLevel
|
||
|
{
|
||
|
public LogLevel LogLevel { get; set; } = LogLevel.Warning;
|
||
|
|
||
|
//...
|
||
|
}
|
||
|
````
|
||
|
|
||
|
##### Self Logging Exceptions
|
||
|
|
||
|
Some exception types may need to write additional logs. They can implement the `IExceptionWithSelfLogging` if needed. Example:
|
||
|
|
||
|
````C#
|
||
|
public class MyException : Exception, IExceptionWithSelfLogging
|
||
|
{
|
||
|
public void Log(ILogger logger)
|
||
|
{
|
||
|
//...log additional info
|
||
|
}
|
||
|
}
|
||
|
````
|
||
|
|
||
7 years ago
|
> `ILogger.LogException` extension methods is used to write exception logs. You can use the same extension method when needed.
|
||
7 years ago
|
|
||
7 years ago
|
### Business Exceptions
|
||
7 years ago
|
|
||
|
Most of your own exceptions will be business exceptions. The `IBusinessException` interface is used to mark an exception as a business exception.
|
||
|
|
||
6 years ago
|
`BusinessException` implements the `IBusinessException` interface in addition to the `IHasErrorCode`, `IHasErrorDetails` and `IHasLogLevel` interfaces. The default log level is `Warning`.
|
||
7 years ago
|
|
||
6 years ago
|
Usually you have an error code related to a particular business exception. For example:
|
||
7 years ago
|
|
||
7 years ago
|
````C#
|
||
7 years ago
|
throw new BusinessException(QaErrorCodes.CanNotVoteYourOwnAnswer);
|
||
7 years ago
|
````
|
||
7 years ago
|
|
||
6 years ago
|
`QaErrorCodes.CanNotVoteYourOwnAnswer` is just a `const string`. The following error code format is recommended:
|
||
7 years ago
|
|
||
7 years ago
|
````
|
||
|
<code-namespace>:<error-code>
|
||
|
````
|
||
|
|
||
|
**code-namespace** is a **unique value** specific to your module/application. Example:
|
||
|
|
||
|
````
|
||
|
Volo.Qa:010002
|
||
|
````
|
||
|
|
||
|
`Volo.Qa` is the code-namespace here. code-namespace is then will be used while **localizing** exception messages.
|
||
7 years ago
|
|
||
7 years ago
|
* You can **directly throw** a `BusinessException` or **derive** your own exception types from it when needed.
|
||
|
* All properties are optional for the `BusinessException` class. But you generally set either `ErrorCode` or `Message` property.
|
||
|
|
||
7 years ago
|
### Exception Localization
|
||
|
|
||
6 years ago
|
One problem with throwing exceptions is how to localize error messages while sending it to the client. ABP offers two models and their variants.
|
||
7 years ago
|
|
||
|
#### User Friendly Exception
|
||
|
|
||
6 years ago
|
If an exception implements the `IUserFriendlyException` interface, then ABP does not change it's `Message` and `Details` properties and directly send it to the client.
|
||
7 years ago
|
|
||
|
`UserFriendlyException` class is the built-in implementation of the `IUserFriendlyException` interface. Example usage:
|
||
|
|
||
|
````C#
|
||
|
throw new UserFriendlyException(
|
||
|
"Username should be unique!"
|
||
|
);
|
||
|
````
|
||
|
|
||
6 years ago
|
In this way, there is **no need for localization** at all. If you want to localize the message, you can inject and use the standard **string localizer** (see the [localization document](Localization.md)). Example:
|
||
7 years ago
|
|
||
|
````C#
|
||
|
throw new UserFriendlyException(_stringLocalizer["UserNameShouldBeUniqueMessage"]);
|
||
|
````
|
||
|
|
||
|
Then define it in the **localization resource** for each language. Example:
|
||
|
|
||
|
````json
|
||
|
{
|
||
|
"culture": "en",
|
||
|
"texts": {
|
||
|
"UserNameShouldBeUniqueMessage": "Username should be unique!"
|
||
|
}
|
||
|
}
|
||
|
````
|
||
|
|
||
6 years ago
|
String localizer already supports **parameterized messages**. For example:
|
||
7 years ago
|
|
||
|
````C#
|
||
7 years ago
|
throw new UserFriendlyException(_stringLocalizer["UserNameShouldBeUniqueMessage", "john"]);
|
||
|
````
|
||
|
|
||
|
Then the localization text can be:
|
||
|
|
||
|
````json
|
||
|
"UserNameShouldBeUniqueMessage": "Username should be unique! '{0}' is already taken!"
|
||
7 years ago
|
````
|
||
|
|
||
7 years ago
|
* The `IUserFriendlyException` interface is derived from the `IBusinessException` and the `UserFriendlyException` class is derived from the `BusinessException` class.
|
||
|
|
||
|
#### Using Error Codes
|
||
7 years ago
|
|
||
6 years ago
|
`UserFriendlyException` is fine, but it has a few problems in advanced usages:
|
||
7 years ago
|
|
||
6 years ago
|
* It requires you to **inject the string localizer** everywhere and always use it while throwing exceptions.
|
||
|
* However, in some of the cases, it may **not be possible** to inject the string localizer (in a static context or in an entity method).
|
||
7 years ago
|
|
||
7 years ago
|
Instead of localizing the message while throwing the exception, you can separate the process using **error codes**.
|
||
|
|
||
|
First, define the **code-namespace** to **localization resource** mapping in the module configuration:
|
||
7 years ago
|
|
||
|
````C#
|
||
|
services.Configure<ExceptionLocalizationOptions>(options =>
|
||
|
{
|
||
|
options.MapCodeNamespace("Volo.Qa", typeof(QaResource));
|
||
|
});
|
||
|
````
|
||
|
|
||
6 years ago
|
Then any of the exceptions with `Volo.Qa` namespace will be localized using their given localization resource. The localization resource should always have an entry with the error code key. Example:
|
||
7 years ago
|
|
||
|
````json
|
||
|
{
|
||
|
"culture": "en",
|
||
|
"texts": {
|
||
|
"Volo.Qa:010002": "You can not vote your own answer!"
|
||
|
}
|
||
|
}
|
||
|
````
|
||
|
|
||
7 years ago
|
Then a business exception can be thrown with the error code:
|
||
|
|
||
|
````C#
|
||
|
throw new BusinessException(QaDomainErrorCodes.CanNotVoteYourOwnAnswer);
|
||
|
````
|
||
7 years ago
|
|
||
7 years ago
|
* Throwing any exception implementing the `IHasErrorCode` interface behaves the same. So, the error code localization approach is not unique to the `BusinessException` class.
|
||
6 years ago
|
* Defining localized string is not required for an error message. If it's not defined, ABP sends the default error message to the client. It does not use the `Message` property of the exception! if you want that, use the `UserFriendlyException` (or use an exception type that implements the `IUserFriendlyException` interface).
|
||
7 years ago
|
|
||
|
##### Using Message Parameters
|
||
7 years ago
|
|
||
6 years ago
|
If you have a parameterized error message, then you can set it with the exception's `Data` property. For example:
|
||
7 years ago
|
|
||
|
````C#
|
||
|
throw new BusinessException("App:010046")
|
||
|
{
|
||
|
Data =
|
||
|
{
|
||
|
{"UserName", "john"}
|
||
|
}
|
||
|
};
|
||
|
|
||
|
````
|
||
|
|
||
6 years ago
|
Fortunately there is a shortcut way to code this:
|
||
7 years ago
|
|
||
|
````C#
|
||
|
throw new BusinessException("App:010046")
|
||
|
.WithData("UserName", "john");
|
||
|
````
|
||
|
|
||
6 years ago
|
Then the localized text can contain the `UserName` parameter:
|
||
7 years ago
|
|
||
|
````json
|
||
|
{
|
||
|
"culture": "en",
|
||
|
"texts": {
|
||
7 years ago
|
"App:010046": "Username should be unique. '{UserName}' is already taken!"
|
||
7 years ago
|
}
|
||
|
}
|
||
|
````
|
||
|
|
||
6 years ago
|
* `WithData` can be chained with more than one parameter (like `.WithData(...).WithData(...)`).
|
||
7 years ago
|
|
||
|
### HTTP Status Code Mapping
|
||
|
|
||
7 years ago
|
ABP tries to automatically determine the most suitable HTTP status code for common exception types by following these rules:
|
||
|
|
||
|
* For the `AbpAuthorizationException`:
|
||
|
* Returns `401` (unauthorized) if user has not logged in.
|
||
|
* Returns `403` (forbidden) if user has logged in.
|
||
|
* Returns `400` (bad request) for the `AbpValidationException`.
|
||
|
* Returns `404` (not found) for the `EntityNotFoundException`.
|
||
|
* Returns `403` (forbidden) for the `IBusinessException` (and `IUserFriendlyException` since it extends the `IBusinessException`).
|
||
|
* Returns `501` (not implemented) for the `NotImplementedException`.
|
||
|
* Returns `500` (internal server error) for other exceptions (those are assumed as infrastructure exceptions).
|
||
|
|
||
6 years ago
|
The `IHttpExceptionStatusCodeFinder` is used to automatically determine the HTTP status code. The default implementation is the `DefaultHttpExceptionStatusCodeFinder` class. It can be replaced or extended as needed.
|
||
7 years ago
|
|
||
|
#### Custom Mappings
|
||
|
|
||
6 years ago
|
Automatic HTTP status code determination can be overrided by custom mappings. For example:
|
||
7 years ago
|
|
||
|
````C#
|
||
|
services.Configure<ExceptionHttpStatusCodeOptions>(options =>
|
||
|
{
|
||
|
options.Map("Volo.Qa:010002", HttpStatusCode.Conflict);
|
||
|
});
|
||
|
````
|
||
7 years ago
|
|
||
|
### Built-In Exceptions
|
||
|
|
||
6 years ago
|
Some exception types are automatically thrown by the framework:
|
||
7 years ago
|
|
||
|
- `AbpAuthorizationException` is thrown if the current user has no permission to perform the requested operation. See authorization document (TODO: link) for more.
|
||
|
- `AbpValidationException` is thrown if the input of the current request is not valid. See validation document (TODO: link) for more.
|
||
|
- `EntityNotFoundException` is thrown if the requested entity is not available. This is mostly thrown by [repositories](Repositories.md).
|
||
|
|
||
6 years ago
|
You can also throw these type of exceptions in your code (although it's rarely needed).
|