Headless CMS and Content Managment Hub
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.
 
 
 
 
 

200 lines
6.7 KiB

// ==========================================================================
// Squidex Headless CMS
// ==========================================================================
// Copyright (c) Squidex UG (haftungsbeschraenkt)
// All rights reserved. Licensed under the MIT license.
// ==========================================================================
using System.Diagnostics;
using System.Security;
using System.Text;
using Microsoft.AspNetCore.Http;
using Microsoft.AspNetCore.Mvc;
using Squidex.Infrastructure;
using Squidex.Infrastructure.Translations;
using Squidex.Infrastructure.Validation;
namespace Squidex.Web
{
public static class ApiExceptionConverter
{
private static readonly Dictionary<int, string> Links = new Dictionary<int, string>
{
[400] = "https://tools.ietf.org/html/rfc7231#section-6.5.1",
[401] = "https://tools.ietf.org/html/rfc7235#section-3.1",
[403] = "https://tools.ietf.org/html/rfc7231#section-6.5.3",
[404] = "https://tools.ietf.org/html/rfc7231#section-6.5.4",
[406] = "https://tools.ietf.org/html/rfc7231#section-6.5.6",
[409] = "https://tools.ietf.org/html/rfc7231#section-6.5.8",
[410] = "https://tools.ietf.org/html/rfc7231#section-6.5.9",
[412] = "https://tools.ietf.org/html/rfc7231#section-6.5.10",
[415] = "https://tools.ietf.org/html/rfc7231#section-6.5.13",
[422] = "https://tools.ietf.org/html/rfc4918#section-11.2",
[500] = "https://tools.ietf.org/html/rfc7231#section-6.6.1"
};
public static (ErrorDto Error, Exception? Unhandled) ToErrorDto(int statusCode, HttpContext? httpContext)
{
var error = new ErrorDto { StatusCode = statusCode };
Enrich(httpContext, error);
return (error, null);
}
public static (ErrorDto Error, Exception? Unhandled) ToErrorDto(this ProblemDetails problem, HttpContext? httpContext)
{
Guard.NotNull(problem, nameof(problem));
var error = CreateError(problem.Status ?? 500, problem.Title);
Enrich(httpContext, error);
return (error, null);
}
public static (ErrorDto Error, Exception? Unhandled) ToErrorDto(this Exception exception, HttpContext? httpContext)
{
Guard.NotNull(exception, nameof(exception));
var result = CreateError(exception);
Enrich(httpContext, result.Error);
return result;
}
private static void Enrich(HttpContext? httpContext, ErrorDto error)
{
error.TraceId = Activity.Current?.Id ?? httpContext?.TraceIdentifier;
if (error.StatusCode == 0)
{
error.StatusCode = 500;
}
error.Type = Links.GetOrDefault(error.StatusCode);
}
private static (ErrorDto Error, Exception? Unhandled) CreateError(Exception exception)
{
switch (exception)
{
case ValidationException ex:
{
var message = T.Get("common.httpValidationError");
return (CreateError(400, message, null, ToErrors(ex.Errors)), GetInner(exception));
}
case DomainObjectNotFoundException ex:
return (CreateError(404, ex.ErrorCode), GetInner(exception));
case DomainObjectVersionException ex:
return (CreateError(412, ex.Message, ex.ErrorCode), GetInner(exception));
case DomainObjectDeletedException ex:
return (CreateError(410, ex.Message, ex.ErrorCode), GetInner(exception));
case DomainObjectConflictException ex:
return (CreateError(409, ex.Message, ex.ErrorCode), GetInner(exception));
case DomainForbiddenException ex:
return (CreateError(403, ex.Message, ex.ErrorCode), GetInner(exception));
case DomainException ex:
return (CreateError(400, ex.Message, ex.ErrorCode), GetInner(exception));
case SecurityException:
return (CreateError(403), exception);
case DecoderFallbackException ex:
return (CreateError(400, ex.Message), null);
case BadHttpRequestException ex:
return (CreateError(ex.StatusCode, ex.Message), null);
default:
return (CreateError(500), exception);
}
}
private static Exception? GetInner(Exception exception)
{
var current = exception;
while (current != null)
{
if (current is not DomainException)
{
return current;
}
current = current.InnerException;
}
return null;
}
private static ErrorDto CreateError(int status, string? message = null, string? errorCode = null, IEnumerable<string>? details = null)
{
var error = new ErrorDto { StatusCode = status, Message = message };
if (!string.IsNullOrWhiteSpace(errorCode))
{
error.ErrorCode = errorCode;
}
error.Details = details?.ToArray();
return error;
}
public static IEnumerable<string> ToErrors(IEnumerable<ValidationError> errors)
{
static string FixPropertyName(string property)
{
property = property.Trim();
if (property.Length == 0)
{
return property;
}
var prevChar = 0;
var builder = new StringBuilder(property.Length);
builder.Append(char.ToLowerInvariant(property[0]));
foreach (var character in property.Skip(1))
{
if (prevChar == '.')
{
builder.Append(char.ToLowerInvariant(character));
}
else
{
builder.Append(character);
}
prevChar = character;
}
return builder.ToString();
}
return errors.Select(e =>
{
if (e.PropertyNames?.Any() == true)
{
return $"{string.Join(", ", e.PropertyNames.Select(FixPropertyName))}: {e.Message}";
}
else
{
return e.Message;
}
});
}
}
}