/errors_aspnetcore

Errors and exceptions.

Primary LanguageC#OtherNOASSERTION

Errors Toolbox

This toolbox contains objects and exceptions for error handling.

Table of Contents

Installation

Adding the DataAccess Toolbox to a project is as easy as adding it to the project.json file :

 "dependencies": {
    "Digipolis.Errors":  "3.0.0", 
 }

Alternatively, it can also be added via the NuGet Package Manager interface.

ExceptionMapper

ExceptionMapper is an abstract class that needs to be implemented in your project to be able to map certain Exceptions to a specific Error object. This is not limited to the exceptions contained in the toolbox. But can be any exception the inherits from System.Exception!

Configure mappings

To map exceptions to Error objects you need to implement two functions:

protected abstract void Configure();

protected abstract void CreateDefaultMap(Error error, Exception exception);
  • CreateDefaultMap is needed to resolve an Error object for exceptions that are not mapped in the Configure method.

  • Configure is where the mappings are made between Exceptions and Error objects.

Mapping exceptions

To configure mappings you can use two method:

/// <summary>
/// Used to map an exception to an Error object were only Status is set
/// </summary>
protected void CreateMap<TException>(int statusCode) where TException : Exception

/// <summary>
/// Used to map an exception to an Error object
/// </summary>
protected void CreateMap<TException>(Action<Error, TException> configError) where TException : Exception

Example

protected override void Configure()
{
    CreateMap<NotImplementedException>((error, ex) =>
    {
        error.Status = (int) HttpStatusCode.NotFound;
        error.Title = "Methode call not allowed";
        error.Code = "NOTF001";
    });

    CreateMap<UnauthorizedAccessException>((int)HttpStatusCode.Forbidden);
}

Default mappings

There are already default mappings provided for NotFoundException, UnauthorizedException and ValidationException also contained in this toolbox. These default mapping can be overriden.

protected override void CreateNotFoundMap(Error error, NotFoundException exception)
{
    error.Title = "not Found";
    error.Code = "NOTF001";
    error.Status = (int)HttpStatusCode.NotFound;
    error.ExtraParameters = exception.Messages.ToDictionary(ms => ms.Key, ms => (object)ms.Value);
}

protected override void CreateUnauthorizedMap(Error error, UnauthorizedException exception)
{
    error.Title = "Access restricted";
    error.Code = "NAUTH001";
    error.Status = (int)HttpStatusCode.Forbidden;
    error.ExtraParameters = exception.Messages.ToDictionary(ms => ms.Key, ms => (object)ms.Value);
}

protected override void CreateValidationMap(Error error, ValidationException exception)
{
    error.Title = "Invalid model in the request";
    error.Code = "NVAL001";
    error.Status = (int)HttpStatusCode.BadRequest;
    error.ExtraParameters = exception.Messages.ToDictionary(ms => ms.Key, ms => (object)ms.Value);
}

Resolving exceptions

This class also provides a Resolve function that will translate an exception to an Error object. This function can also be overriden to implement additional or custom logic.

try
{
    ...
}
catch (Exception ex)
{
    var mapper = new ExceptionMapperTester();
    var error = mapper.Resolve(ex)
}

Error object

The Error object contains the following properties.

public class Error
{
    /// <summary>
    /// A unique id to identify error messages in the logs
    /// </summary>
    public Guid Identifier { get; private set; }

    /// <summary>
    /// A URI to an absolute or relative html resource to identify the problem.
    /// </summary>
    public Uri Type { get; set; }

    /// <summary>
    /// A short description of the error
    /// </summary>
    public string Title { get; set; }

    /// <summary>
    /// The http Status code 
    /// </summary>
    public int Status { get; set; }

    /// <summary>
    /// A code to identify what error it is.
    /// </summary>
    public string Code { get; set; }

    /// <summary>
    /// Extra parameters to clarify the error
    /// </summary>
    public Dictionary<string, object> ExtraParameters { get; set; }
}

An Error object can be instantiated by using one of both constructors

public Error(Dictionary<string, object> extraParameters = null)
   : this(Guid.NewGuid(), extraParameters)
{ }

public Error(Guid identifier, Dictionary<string, object> extraParameters = null)
{
   if (identifier == default(Guid))
       throw new ArgumentException("An empty Guid is not allowed", nameof(identifier));

   Identifier = identifier;
   ExtraParameters = extraParameters ?? new Dictionary<string, object>();
}

As you can see, you can instantiate it by specifying a custom Identifier or by letting the class generate one for you. This way you can use this Identifier to link several error objects for e.g. logging purposes.

var error = new Error();
Console.WriteLine("id = {0}", error.Id);
//outputs: id = 22699ff7-8496-49b2-8772-4c9e952a39fd

The constructor also optionally accepts a dictionary of extra parameters :

var extraParameters = new Dictionary<string, object>();
extraParameters.Add("key1", "message1");
extraParameters.Add("key2", "message2");

//constructors
var error1 = new Error(extraParameters);
var error2 = new Error(Guid.NewGuid(), extraParameters);

ToString

The ToString method returns the contents of the object as a string. This can be handy when tracing and debugging.

Exceptions

BaseException

A base class for exceptions. It inherits from the standard Exception class and has the following extra field :

public Dictionary<string, IEnumerable<string>> Messages { get; protected set; }
  • This Messages property will contain all additional messages you want to copy over to the ExtraParameters property of the Error object when resolving it through the Resolve function of the ExceptionMapper.

AddMessage

A string message or a key and string message can be given as arguments. If only a string message is given, the Key field will be String.Empty.

var error = new Error();
error.AddMessage("aMessage");
error.AddMessage("aKey", "anotherMessage");

AddMessages

Adds multiple string messages to the ErrorMessage list. The key for each message will be String.Empty unless a key is given in the parameters

var message1 = "message1";
var message2 = "messages2";
var messages = new string[] { message1, message2 };

var error = new Error();
error.AddMessages(messages);
error.AddMessages("aKey", messages);

NotFoundException

To be used when a requested resource is not found or does not exist.

//constructor
public NotFoundException(string message = "Not found.", Exception exception = null, Dictionary < string, IEnumerable<string>> messages = null )
    : base(message, exception, messages)
{}

Inherits from BaseException and has all the functionality of the base.

UnauthorizedException

Used when the user does not have sufficient rights.

//constructor
public NotFoundException(string message = "Access denied.", Exception exception = null, Dictionary < string, IEnumerable<string>> messages = null )
    : base(message, exception, messages)
{}

Inherits from BaseException and has all the functionality of the base.

ValidationException

Can be used when input validation fails.

//constructor
public NotFoundException(string message = "Access denied.", Exception exception = null, Dictionary < string, IEnumerable<string>> messages = null )
    : base(message, exception, messages)
{}

Inherits from BaseException and has all the functionality of the base.