mirror of https://github.com/Squidex/squidex.git
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
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;
|
|
}
|
|
});
|
|
}
|
|
}
|
|
}
|
|
|