Due to other priorities this project is currently not being supported. There are no planned releases at this time. No new features are planned and no new issues are being picked up.
Middleware and extensions for returning exceptions over HTTP, e.g. as ASP.NET Core Problem Details. Problem Details are a machine-readable format for specifying errors in HTTP API responses based on https://tools.ietf.org/html/rfc7807. But you are not limited to returning exception results as Problem Details, but you can create your own mappers for your own custom formats.
You can install Opw.HttpExceptions.AspNetCore from the console.
> dotnet add package Opw.HttpExceptions.AspNetCore
Add the HttpExceptions services and the middleware in the Startup.cs
of your application. First add HttpExceptions using the IMvcBuilder
of IMvcCoreBuilder
.
public void ConfigureServices(IServiceCollection services)
{
...
services.AddMvc().AddHttpExceptions();
// or services.AddMvcCore().AddHttpExceptions();
...
}
Then you can add the HttpExceptions middleware using the application builder. UseHttpExceptions
should be the first middleware
component added to the pipeline. That way the UseHttpExceptions
Middleware catches any exceptions that occur in later calls. When
using HttpExceptions you don't need to use UseExceptionHandler
or UseDeveloperExceptionPage
.
public void Configure(IApplicationBuilder app)
{
app.UseHttpExceptions(); // this is the first middleware component added to the pipeline
...
}
HttpExceptions overrides the default Microsoft.AspNetCore.Mvc.ApiBehaviorOptions.InvalidModelStateResponseFactory
and related settings and
will use the configured ExceptionMappers.
You can extend or override the default behavior through the configuration options, HttpExceptionsOptions
.
Whether or not to include the full exception details in the response. The default behavior is only to include exception details in a development environment.
mvcBuilder.AddHttpExceptions(options =>
{
// This is the same as the default behavior; only include exception details in a development environment.
options.IncludeExceptionDetails = context => context.RequestServices.GetRequiredService<IHostingEnvironment>().IsDevelopment();
});
You can include extra public exception properties in exception details, by adding the ProblemDetailsAttribute
to them.
mvcBuilder.AddHttpExceptions(options =>
{
// The default is also true.
options.IsProblemDetailsAttributeEnabled = true;
});
public class CustomHttpException : HttpException
{
[ProblemDetails]
public string PropertyA { get; set; }
[ProblemDetails]
public int PropertyB { get; set; }
public long PropertyC { get; set; }
}
Is the response an exception and should it be handled by the HttpExceptions middleware.
mvcBuilder.AddHttpExceptions(options =>
{
// This is a simplified version of the default behavior; only map exceptions for 4xx and 5xx responses.
options.IsExceptionResponse = context => (context.Response.StatusCode >= 400 && context.Response.StatusCode < 600);
});
Should an exception be logged by the HttpExceptions middleware or not, default behavior is to log all exceptions (all status codes).
mvcBuilder.AddHttpExceptions(options =>
{
// Only log the when it has a status code of 500 or higher, or when it is not a HttpException.
options.ShouldLogException = exception => {
if ((exception is HttpExceptionBase httpException && (int)httpException.StatusCode >= 500) || !(exception is HttpExceptionBase))
return true;
return false;
};
});
You can inject your own mappings for the ProblemDetails
properties using functions on the HttpExceptionsOptions
, or by creating your own IExceptionMapper
and/or IHttpResponseMapper
. If you inject your own function that will be tried first, and if no result is returned the defaults will be used.
In the following example we will create a custom function mapping the ProblemDetails.Type
property. By default the ProblemDetails.Type
property will be set by:
- Either the
Exception.HelpLink
or the HTTP status code information link. - Or the
DefaultHelpLink
will be used. - Or an URI with the HTTP status name ("error:[status:slug]") will be used.
When the ExceptionTypeMapping
or HttpContextTypeMapping
are set the result of those functions will be tried first, if no result is returned the defaults will be used.
mvcBuilder.AddHttpExceptions(options =>
{
ExceptionTypeMapping = exception => {
// This is a example, you can implement your own logic here.
return "My Exception Type Mapping";
},
HttpContextTypeMapping = context => {
// This is a example, you can implement your own logic here.
return "My HttpContext Type Mapping";
}
});
Set the ExceptionMapper collection that will be used during mapping. You can override and/or add ExceptionMappers for specific exception types. The ExceptionMappers are called in order so make sure you add them in the right order.
By default there is one ExceptionMapper configured, that ExceptionMapper catches all exceptions.
mvcBuilder.AddHttpExceptions(options =>
{
// Override and or add ExceptionMapper for specific exception types, the default ExceptionMapper catches all exceptions.
options.ExceptionMapper<BadRequestException, BadRequestExceptionMapper>();
options.ExceptionMapper<ArgumentException, ExceptionMapper<ArgumentException>>();
// The last ExceptionMapper should be a catch all, for type Exception.
options.ExceptionMapper<Exception, MyCustomExceptionMapper>();
});
Serialize the HTTP content to ProblemDetails.
ProblemDetails problemDetails = ((HttpContent)response.Content).ReadAsProblemDetails();
Try to get the exception details from the ProblemDetails.
SerializableException exception;
((ProblemDetails)problemDetails).TryGetExceptionDetails(out exception);
Try to get the errors dictionary from the ProblemDetails.
var IDictionary<string, object[]> errors;
((ProblemDetails)problemDetails).TryGetErrors(out errors);
See the samples/Opw.HttpExceptions.AspNetCore.Sample project for a sample implementation. This project contains examples on how to use the HttpExceptions middleware.
Please see the code 🤓
HTTP-specific exception classes that enable ASP.NET to generate exception information. These classes can be used by themselves or as base classes for your own HttpExceptions.
You can install Opw.HttpExceptions from the console.
> dotnet add package Opw.HttpExceptions
- 400 BadRequestException
- 400 InvalidModelException
- 400 ValidationErrorException<T>
- 401 UnauthorizedException
- 402 PaymentRequiredException
- 403 ForbiddenException
- 404 NotFoundException
- 404 NotFoundException<T>
- 409 ConflictException
- 415 UnsupportedMediaTypeException
- 500 HttpException (default exception with status code 500)
- 503 ServiceUnavailableException
The above exception are just a few of the most common exceptions, you can create your own HttpExceptions by inheriting from the HttpExceptionBase
class. For instance to create a NotAcceptableException
(406).
public class NotAcceptableException : HttpExceptionBase
{
public override HttpStatusCode StatusCode { get; protected set; } = HttpStatusCode.NotAcceptable;
public override string HelpLink { get; set; } = ResponseStatusCodeLink.NotAcceptable;
public NotAcceptableException(string message) : base(message) { }
}
throw new new NotAcceptableException("Totally unacceptable!");
We accept fixes and features! Here are some resources to help you get started on how to contribute code or new content.
Copyright © 2021, Of Pine Wood. Created by Peter van den Hout. Released under the terms of the MIT license.