From 188b2fe0eb163b058cba7b7463a9b791ae8e23d9 Mon Sep 17 00:00:00 2001 From: Carl-Hugo Marcotte Date: Mon, 12 Feb 2024 23:53:08 -0500 Subject: [PATCH 1/7] Minor code format update --- .../Serialization/Json/ProblemDetailsSerializationHandler.cs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/ForEvolve.ExceptionMapper/Serialization/Json/ProblemDetailsSerializationHandler.cs b/src/ForEvolve.ExceptionMapper/Serialization/Json/ProblemDetailsSerializationHandler.cs index 87dcd6c..e35b369 100644 --- a/src/ForEvolve.ExceptionMapper/Serialization/Json/ProblemDetailsSerializationHandler.cs +++ b/src/ForEvolve.ExceptionMapper/Serialization/Json/ProblemDetailsSerializationHandler.cs @@ -80,7 +80,7 @@ public async Task ExecuteAsync(ExceptionHandlingContext ctx) { var traceIdKey = "traceId"; problemDetails.Extensions.Remove(traceIdKey); - problemDetails.Extensions.Add(FormatName(traceIdKey), traceId); + problemDetails.Extensions.Add(FormatName(traceIdKey), traceId); } // Transfer non-excluded and non-JsonIgnored properties to the problem details. From 84dc18a7d5a06c10c7678ad9b1706fb726b5023f Mon Sep 17 00:00:00 2001 From: Carl-Hugo Marcotte Date: Tue, 13 Feb 2024 00:01:51 -0500 Subject: [PATCH 2/7] Correct typos and improve README --- README.md | 64 ++++++++++++++++++++++++------------------------------- 1 file changed, 28 insertions(+), 36 deletions(-) diff --git a/README.md b/README.md index 4d561c4..deedc31 100644 --- a/README.md +++ b/README.md @@ -5,24 +5,15 @@ [![feedz.io](https://img.shields.io/badge/endpoint.svg?url=https%3A%2F%2Ff.feedz.io%2Fforevolve%2Fexception-mapper%2Fshield%2FForEvolve.ExceptionMapper%2Flatest)](https://f.feedz.io/forevolve/exception-mapper/packages/ForEvolve.ExceptionMapper/latest/download) A simple ASP.NET Core middleware that intercepts and reacts to `Exception`. -You can map specific exception types to HTTP Status Code, use predefined handlers, or create your own. +You can map specific exception types to HTTP Status Codes, use predefined handlers, or create your own. You can throw an exception from anywhere in your codebase and ExceptionMapper will handle it according to your specifications. This makes it a breeze to uniformize exception handling in a REST API. -All of the handlers are iterated through, in order, so you can build a pipeline to handle exceptions where multiple handlers have a single responsibility. +All the handlers are iterated through, in order, so you can build a pipeline to handle exceptions where multiple handlers have a single responsibility. For example, you could have handlers that respond to certain exception types, then one or more fallback handlers that react only if no previous handler handled the exception. -Finally, there is a serializer that convert handled exceptions to JSON, in the format of your choice, making your API linear between endpoints and exception types without much effort. The default serializer converts the errors to [Problem Details for HTTP APIs](https://datatracker.ietf.org/doc/html/rfc7807). - -## Versioning - -The packages follows _semantic versioning_ and use `Nerdbank.GitVersioning` to automatically version packages based on git commits. - -### Pre-released - -Prerelease packages are packaged code not yet merged to `master`. -The prerelease CI builds are packaged and hosted at [feedz.io](feedz.io), thanks to their "Open Source" subscription. +Finally, there is a serializer that converts handled exceptions to JSON, in the format of your choice, making your API linear between endpoints and exception types without much effort. The default serializer converts the errors to [Problem Details for HTTP APIs](https://datatracker.ietf.org/doc/html/rfc7807). ## How to install @@ -36,7 +27,7 @@ _You can take a look at the `samples/WebApiSample` project for a working example ## Getting started -You must register the services, and optionally configure/register handlers, and use the middleware that catches exceptions (and that handles the logic). +You must register the services, optionally configure/register handlers, and use the middleware that catches exceptions (and that handles the logic). **Program.cs** @@ -76,7 +67,7 @@ public class Startup ## Extending the existing exception -An easy way to manage your custom exceptions is to inherit from the one that are already mapped. +An easy way to manage your custom exceptions is to inherit from the ones that are already mapped. For example, you could create and throw the following `DroidNotFoundException` and ExceptionMapper will associate it with a 404 Not Found status code because it inherits from `NotFoundException`: ```csharp @@ -91,7 +82,7 @@ public class DroidNotFoundException : NotFoundException ## Mapping Exception types to status code -If you do not want or can't inherit the provided exceptions, you can map any Exception types to specific status code, like this: +If you do not want or can't inherit the provided exceptions, you can map any Exception types to a specific status code, like this: ```csharp builder.AddExceptionMapper(builder => @@ -188,7 +179,7 @@ ExceptionMapper implements different common exceptions and their handlers, like # Fallback handler -ExceptionMapper also comes with a fallback handler that convert unhandled exceptions to `500 InternalServerError`. This is an opt-out feature, configured by the `FallbackExceptionHandlerOptions`. +ExceptionMapper also comes with a fallback handler that converts unhandled exceptions to `500 InternalServerError`. This is an opt-out feature, configured by the `FallbackExceptionHandlerOptions`. You can also configure the `FallbackExceptionHandlerOptions` like the following or under the `ExceptionMapper:FallbackExceptionHandler` key in your settings: @@ -234,11 +225,11 @@ You can also customize the options from the `appsettings.json` file: ``` Note that the serializer displays the debug information when in development. -Use the `DisplayDebugInformation` function to display the debug info in other environment, like staging or production. +Use the `DisplayDebugInformation` function to display the debug info in other environments, like staging or production. ## Property names -To change the way the property name are serialized, you can configure the `JsonOptions` and change the `PropertyNamingPolicy` property. +To change the way the property names are serialized, you can configure the `JsonOptions` and change the `PropertyNamingPolicy` property. **.NET 8+** @@ -258,7 +249,7 @@ builder.Services.Configure(options => { ### Dictionary keys -For .NET 7+ projects, ExceptionMapper sets the `DictionaryKeyPolicy` property to the `PropertyNamingPolicy` property value so dictionaries are serialized the same way as the normal properties. +For .NET 7+ projects, ExceptionMapper sets the `DictionaryKeyPolicy` property to the `PropertyNamingPolicy` property value, so dictionaries are serialized the same way as the normal properties. ## Ensuring a property is not serialized @@ -289,6 +280,15 @@ services.AddSingleton(); builder.AddExceptionMapper(); ``` +## Versioning + +The package follows _semantic versioning_ and uses `Nerdbank.GitVersioning` to automatically version packages based on git commits. + +### Pre-released + +Prerelease packages are packaged code not yet merged to the `main` branch. +The prerelease CI builds are packaged and hosted at [feedz.io](feedz.io), thanks to their "Open Source" subscription. + # Release notes ## 3.0 @@ -298,42 +298,34 @@ The version 3 of ExceptionMapper is a major rewrite that simplifies the codebase - Add support to .NET 7 and .NET 8. - Remove transitive dependency on JSON.NET (`Newtonsoft.Json`). - Drop support for .NET Standard 2.0 because `ExceptionMapper` depends on the `HttpContext` class which requires a `` which is not compatible with `netstandard2.0`. -- Merge all assemblies in `ForEvolve.ExceptionMapper` but `ForEvolve.ExceptionMapper.Scrutor` and removed `ForEvolve.ExceptionMapper.Scrutor` althogether. +- Merge all assemblies in `ForEvolve.ExceptionMapper` but `ForEvolve.ExceptionMapper.Scrutor` and removed `ForEvolve.ExceptionMapper.Scrutor` altogether. - Replace the `AddMvcCore` call by registering a copy of the `DefaultProblemDetailsFactory` using a `TryAddSingleton` call, so you must register your custom `ProblemDetailsFactory` implementation before `AddExceptionMapper`. The good news is, if you are using a custom factory, the `ProblemDetailsSerializationHandler` will use it! > Removing the copy of the `DefaultProblemDetailsFactory` class could be resolved by https://github.com/dotnet/aspnetcore/issues/49982 - Calling `AddExceptionMapper()` now registers the common exceptions and the serializer automatically. - The `Order` property was removed from the `IExceptionHandler` interface. The system uses the registration order instead. -- The interface now leverage a serialzier implementing the `IExceptionSerializer` interface. The serializer no longer implements the `IExceptionHandler` interface. +- The interface now leverages a serializer implementing the `IExceptionSerializer` interface. The serializer no longer implements the `IExceptionHandler` interface. - By default, `ProblemDetailsSerializationOptions` is bound to the section `ExceptionMapper:ProblemDetailsSerialization` and `FallbackExceptionHandlerOptions` is bound to the section `ExceptionMapper:FallbackExceptionHandler`. ### Breaking changes .NET 7+ - Remove the `ContentType` and `JsonSerializerOptions` properties from the `ProblemDetailsSerializationOptions` class (`ForEvolve.ExceptionMapper.Serialization.Json`). -- The `ProblemDetailsSerializationHandler` class now leverages the `IProblemDetailsService` interface to write the `ProblemDetails` object to the response stream instead of serializing it with the `JsonSerializer`, relinguishing the control of the process to .NET. +- The `ProblemDetailsSerializationHandler` class now leverages the `IProblemDetailsService` interface to write the `ProblemDetails` object to the response stream instead of serializing it with the `JsonSerializer`, relinquishing the control of the process to .NET. - The `ProblemDetailsSerializationHandler` leverages the `JsonOptions` class to ensure the names are formatted according to the `PropertyNamingPolicy` object. The default is `camelCase`. - ExceptionMapper sets the `DictionaryKeyPolicy` property to the `PropertyNamingPolicy` property value so dictionaries are serialized the same way as the normal properties. -## 2.0 +## 2.0 (deprecated) - Drop .NET Core 3.1 support - Add support for .NET 6.0 -## 1.1 +## 1.1 (deprecated) - Add a handler that serializes exceptions to `ProblemDetails` (JSON) - Add the `ForEvolve.ExceptionMapper.Serialization.Json` project -## 1.0 - -- Initial release (no yet released) - -# Future/To do - -Here is a list of what I want to do: +## 1.0 (deprecated) -- [ ] Improve overall test coverage. -- [ ] Implement custom type converter. The serializer would either use the converter or fall back to the reflection-based code if no type converter is available. -- [ ] Create a more "real-life" code sample. +- Initial release (not yet released). # Found a bug or have a feature request? @@ -341,8 +333,8 @@ Please open an issue and be as clear as possible; see _How to contribute?_ for m # How to contribute? -If you would like to contribute to the project, first, thank you for your interest, and please read [Contributing to ForEvolve open source projects](https://github.com/ForEvolve/ForEvolve.DependencyInjection/tree/master/CONTRIBUTING.md) for more information. +If you would like to contribute to the project, first, thank you for your interest, and please read [Contributing to ForEvolve open source projects](https://github.com/ForEvolve/Toc/blob/master/CONTRIBUTING.md) for more information. ## Contributor Covenant Code of Conduct -Also, please read the [Contributor Covenant Code of Conduct](https://github.com/ForEvolve/ForEvolve.DependencyInjection/tree/master/CODE_OF_CONDUCT.md) that applies to all ForEvolve repositories. +Also, please read the [Contributor Covenant Code of Conduct](https://github.com/ForEvolve/Toc/blob/master/CODE_OF_CONDUCT.md) that applies to all ForEvolve repositories. From 31224deddb1ccdcb3a183378506139cba87428a9 Mon Sep 17 00:00:00 2001 From: Carl-Hugo Marcotte Date: Tue, 13 Feb 2024 00:02:00 -0500 Subject: [PATCH 3/7] Add README to NuGet package --- .../ForEvolve.ExceptionMapper.csproj | 5 ++++- 1 file changed, 4 insertions(+), 1 deletion(-) diff --git a/src/ForEvolve.ExceptionMapper/ForEvolve.ExceptionMapper.csproj b/src/ForEvolve.ExceptionMapper/ForEvolve.ExceptionMapper.csproj index 23b5384..c5a227f 100644 --- a/src/ForEvolve.ExceptionMapper/ForEvolve.ExceptionMapper.csproj +++ b/src/ForEvolve.ExceptionMapper/ForEvolve.ExceptionMapper.csproj @@ -3,9 +3,12 @@ $(FETargetFrameworks) - + + + + From c510876195fddd247114ac05ac5150ccf7c75fad Mon Sep 17 00:00:00 2001 From: Carl-Hugo Marcotte Date: Tue, 13 Feb 2024 00:02:24 -0500 Subject: [PATCH 4/7] Typo --- README.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/README.md b/README.md index deedc31..21851bd 100644 --- a/README.md +++ b/README.md @@ -293,7 +293,7 @@ The prerelease CI builds are packaged and hosted at [feedz.io](feedz.io), thanks ## 3.0 -The version 3 of ExceptionMapper is a major rewrite that simplifies the codebase and usage of the library. Here are a few important changes: +Version 3 of ExceptionMapper is a major rewrite that simplifies the codebase and usage of the library. Here are a few important changes: - Add support to .NET 7 and .NET 8. - Remove transitive dependency on JSON.NET (`Newtonsoft.Json`). From d604644dfdf0d3095bb445a45d5a49435b64a137 Mon Sep 17 00:00:00 2001 From: Carl-Hugo Marcotte Date: Tue, 13 Feb 2024 00:10:24 -0500 Subject: [PATCH 5/7] Add license file to NuGet package --- src/ForEvolve.ExceptionMapper/ForEvolve.ExceptionMapper.csproj | 1 + 1 file changed, 1 insertion(+) diff --git a/src/ForEvolve.ExceptionMapper/ForEvolve.ExceptionMapper.csproj b/src/ForEvolve.ExceptionMapper/ForEvolve.ExceptionMapper.csproj index c5a227f..ad637dd 100644 --- a/src/ForEvolve.ExceptionMapper/ForEvolve.ExceptionMapper.csproj +++ b/src/ForEvolve.ExceptionMapper/ForEvolve.ExceptionMapper.csproj @@ -10,5 +10,6 @@ + From 7b2b3456968759c511ac3260c0804be39ea4cad6 Mon Sep 17 00:00:00 2001 From: Carl-Hugo Marcotte Date: Tue, 13 Feb 2024 00:10:54 -0500 Subject: [PATCH 6/7] Update copyright line --- LICENSE | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/LICENSE b/LICENSE index e9741b8..0651cc8 100644 --- a/LICENSE +++ b/LICENSE @@ -1,6 +1,6 @@ MIT License -Copyright (c) 2020 ForEvolve +Copyright (c) 2024 Carl-Hugo Marcotte Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal From e067ba43f84853d00985f91608fde6b7f77bc185 Mon Sep 17 00:00:00 2001 From: Carl-Hugo Marcotte Date: Tue, 13 Feb 2024 00:11:00 -0500 Subject: [PATCH 7/7] Make CI builds deterministic --- src/Directory.Build.props | 3 +++ 1 file changed, 3 insertions(+) diff --git a/src/Directory.Build.props b/src/Directory.Build.props index 33a40bc..8d523a5 100644 --- a/src/Directory.Build.props +++ b/src/Directory.Build.props @@ -15,6 +15,9 @@ enable enable + + true +