Sstv.DomainExceptions
3.0.0
dotnet add package Sstv.DomainExceptions --version 3.0.0
NuGet\Install-Package Sstv.DomainExceptions -Version 3.0.0
<PackageReference Include="Sstv.DomainExceptions" Version="3.0.0" />
paket add Sstv.DomainExceptions --version 3.0.0
#r "nuget: Sstv.DomainExceptions, 3.0.0"
// Install Sstv.DomainExceptions as a Cake Addin #addin nuget:?package=Sstv.DomainExceptions&version=3.0.0 // Install Sstv.DomainExceptions as a Cake Tool #tool nuget:?package=Sstv.DomainExceptions&version=3.0.0
Sstv.DomainExceptions
The main library in this repository.
Install
You can install using Nuget Package Manager:
Install-Package Sstv.DomainExceptions -Version 3.0.0
via the .NET CLI:
dotnet add package Sstv.DomainExceptions --version 3.0.0
or you can add package reference manually:
<PackageReference Include="Sstv.DomainExceptions" Version="3.0.0" />
How to use?
First of all you should decide, how you want to work with error codes.
Enum that decorates with ErrorDescriptionAttribute that can hold error code description, help link, error prefix.
// this attribute is common for all values
[ErrorDescription(Prefix = "SSTV", HelpLink = "https://help.myproject.ru/error-codes/{0}", Level = Level.Critical)]
[ExceptionConfig(ClassName = "FirstException")]
public enum ErrorCodes
{
[ErrorDescription(
Description = "Unhandled error code",
HelpLink = "https://help.myproject.ru/error-codes/nothing-here",
Level = Level.Fatal)]
Default = 0,
[ErrorDescription(
Description = "You have not enough money",
HelpLink = "https://help.myproject.ru/error-codes/not-enough-money",
Level = Level.NotError)]
NotEnoughMoney = 10001,
[ErrorDescription(Prefix = "DIF", Description = "Another prefix example")]
SomethingBadHappen = 10002,
[ErrorDescription(
Description = "Help link with template in enum member attribute",
HelpLink = "https://help.myproject.ru/{0}/error-code"
)]
WhateverElse = 10003,
}
Thats all that you need to do to start work with. After compile, source generator creates exception class and class with extensions methods for you:
public sealed partial class FirstException : DomainException
{
public static readonly IReadOnlyDictionary<ErrorCodes, ErrorDescription> ErrorDescriptions = new Dictionary<ErrorCodes, ErrorDescription>
{
[ErrorCodes.Default] = new ErrorDescription("SSTV00000", "Unhandled error code", Level.Fatal, "https://help.myproject.ru/error-codes/nothing-here"),
[ErrorCodes.NotEnoughMoney] = new ErrorDescription("SSTV10001", "You have not enough money", Level.NotError, "https://help.myproject.ru/error-codes/not-enough-money"),
[ErrorCodes.SomethingBadHappen] = new ErrorDescription("DIF10002", "Another prefix example", Level.Critical, "https://help.myproject.ru/error-codes/DIF10002"),
[ErrorCodes.WhateverElse] = new ErrorDescription("SSTV10003", "Help link with template in enum member attribute", Level.Critical, "https://help.myproject.ru/SSTV10003/error-code"),
}.ToFrozenDictionary();
public static IErrorCodesDescriptionSource ErrorCodesDescriptionSource { get; } = new ErrorCodesDescriptionInMemorySource(ErrorDescriptions.Values.ToFrozenDictionary(x => x.ErrorCode, x => x));
public FirstException(ErrorCodes errorCodes, Exception? innerException = null)
: base(ErrorDescriptions[errorCodes], innerException)
{
}
}
public static class ErrorCodesExtensions
{
public static ErrorDescription GetDescription(this ErrorCodes errorCodes)
{
return FirstException.ErrorDescriptions[errorCodes];
}
public static string GetErrorCode(this ErrorCodes errorCodes)
{
return FirstException.ErrorDescriptions[errorCodes].ErrorCode;
}
public static FirstException ToException(this ErrorCodes errorCodes, Exception? innerException = null)
{
return new FirstException(errorCodes, innerException);
}
}
Here usage example:
throw new FirstException(ErrorCodes.NotEnoughMoney)
.WithDetailedMessage("DetailedError")
.WithAdditionalData("123", 2);
// or more fluent api way:
throw ErrorCodes.NotEnoughMoney
.ToException()
.WithDetailedMessage("DetailedError")
.WithAdditionalData("123", 2);
If you don't like enums, you can also use simple class with constants in it and with separate additional dictionary (that can be loaded from appsettings.json, in memory dictionary, database etc) with error code description, help link, additional context data.
public static class DomainErrorCodes
{
public const string DEFAULT = "SSTV.10000";
public const string NOT_ENOUGH_MONEY = "SSTV.10004";
public const string SOMETHING_BAD_HAPPEN = "SSTV.10005";
}
public sealed class MyException : DomainException
{
public MyException(string errorCode, Exception? innerException = null)
: base(errorCode, innerException)
{
}
}
{
"DomainExceptionSettings": {
"ErrorCodes": {
"SSTV.10004": {
"Description": "You have not enough money",
"Level": "Low",
"HelpLink": "https://help.myproject.ru/error-codes/not-enough-money"
}
}
}
}
// somewhere in startup.cs
services.AddDomainException();
// or if you don't want to or can't use DI container
DomainExceptionSettings.Instance.ErrorCodesDescriptionSource = new ErrorCodesDescriptionFromConfigurationSource(configuration);
// and the usage
throw new MyException(DomainErrorCodes.NOT_ENOUGH_MONEY)
.WithDetailedMessage("DetailedError")
.WithAdditionalData("UserId", 2);
When to choose enums?
- If you want to keep error codes, description, help link on the same file
- Just want to use an enum 😃
Pros:
- All the things on the same file
- Library can be extended with static analyzer, or unit test that can validate enum and it's ErrorDescriptionAttribute for completeness. Snapshot testing can save to git all the error codes, so you and track changes.
- Default code static analyzer checks enum values overlapping
- Zero reflection usage (thanks for source generators)
- Source generators generates code and precompute all values that it needed, so and you can check them. Also it very fast because it ready to run.
Cons:
- All the changes should go through compile and release, even only description or help link was changed. Also you can't load this descriptions from external sources or appsettings.
- Can be challenging to change error code length at the future. Recommended length is 5, e.g. SSTV10000, so you have 9999 error codes per app or bounded context.
When to choose constants?
Pros:
- We see the full error code right when declaring the constant.
- Zero reflection usage.
- Easy to change description, link, make error code obsolete at runtime without release - just change configuration.
- Partial class with constants can help to reduce the number of Merge Conflicts while adding error codes in parallel.
Cons:
- Cant see all the things about on the same file (except output of ErrorDebugView), cause we need additional store for error description, help link etc.
- You can forgot to add error description. Need additional checks or validators.
Product | Versions Compatible and additional computed target framework versions. |
---|---|
.NET | net5.0 was computed. net5.0-windows was computed. net6.0 was computed. net6.0-android was computed. net6.0-ios was computed. net6.0-maccatalyst was computed. net6.0-macos was computed. net6.0-tvos was computed. net6.0-windows was computed. net7.0 was computed. net7.0-android was computed. net7.0-ios was computed. net7.0-maccatalyst was computed. net7.0-macos was computed. net7.0-tvos was computed. net7.0-windows was computed. net8.0 was computed. net8.0-android was computed. net8.0-browser was computed. net8.0-ios was computed. net8.0-maccatalyst was computed. net8.0-macos was computed. net8.0-tvos was computed. net8.0-windows was computed. net9.0 was computed. net9.0-android was computed. net9.0-browser was computed. net9.0-ios was computed. net9.0-maccatalyst was computed. net9.0-macos was computed. net9.0-tvos was computed. net9.0-windows was computed. |
.NET Core | netcoreapp2.0 was computed. netcoreapp2.1 was computed. netcoreapp2.2 was computed. netcoreapp3.0 was computed. netcoreapp3.1 was computed. |
.NET Standard | netstandard2.0 is compatible. netstandard2.1 was computed. |
.NET Framework | net461 was computed. net462 was computed. net463 was computed. net47 was computed. net471 was computed. net472 was computed. net48 was computed. net481 was computed. |
MonoAndroid | monoandroid was computed. |
MonoMac | monomac was computed. |
MonoTouch | monotouch was computed. |
Tizen | tizen40 was computed. tizen60 was computed. |
Xamarin.iOS | xamarinios was computed. |
Xamarin.Mac | xamarinmac was computed. |
Xamarin.TVOS | xamarintvos was computed. |
Xamarin.WatchOS | xamarinwatchos was computed. |
-
.NETStandard 2.0
- No dependencies.
NuGet packages (2)
Showing the top 2 NuGet packages that depend on Sstv.DomainExceptions:
Package | Downloads |
---|---|
Sstv.DomainExceptions.Extensions.DependencyInjection
Package Description |
|
Sstv.DomainExceptions.Extensions.SerilogEnricher
Package Description |
GitHub repositories
This package is not used by any popular GitHub repositories.