2

我正在使用带有 System.Text.Json 的自定义 JsonConverter 以使 DateTime 具有所需的时区

/// <summary>
/// Converts a string to/from DateTime, requiring the timezone
/// </summary>
public class JsonDateTimeWithTimezoneConverter : JsonConverter<DateTime>
{
    //
    // FOR EXPLANATIONS
    //
    // see https://stackoverflow.com/a/58103218/1545567 and https://stackoverflow.com/a/17752389/1545567
    //



    private readonly string[] FORMATS = { 
        // Basic formats
        "yyyyMMddTHHmmsszzz",
        "yyyyMMddTHHmmsszz",
        "yyyyMMddTHHmmssZ",
        // Extended formats
        "yyyy-MM-ddTHH:mm:sszzz",
        "yyyy-MM-ddTHH:mm:sszz",
        "yyyy-MM-ddTHH:mm:ssZ",
        // All of the above with reduced accuracy to minutes
        "yyyyMMddTHHmmzzz",
        "yyyyMMddTHHmmzz",
        "yyyyMMddTHHmmZ",
        "yyyy-MM-ddTHH:mmzzz",
        "yyyy-MM-ddTHH:mmzz",
        "yyyy-MM-ddTHH:mmZ",            
    };

    public override DateTime Read(ref Utf8JsonReader reader, Type typeToConvert, JsonSerializerOptions options)
    {
        //TODO: 400 BadRequest instead of error 500 InternalServerError when format not respected
        Debug.Assert(typeToConvert == typeof(DateTime));
        return DateTime.ParseExact(reader.GetString(), FORMATS, CultureInfo.InvariantCulture, DateTimeStyles.None);
    }

    public override void Write(Utf8JsonWriter writer, DateTime value, JsonSerializerOptions options)
    {
        writer.WriteStringValue(value.ToUniversalTime().ToString("yyyy-MM-ddTHH:mm:sszzz"));
    }
}

我这样注册ConfigureServices

        services.AddControllers()
            .AddJsonOptions(options => {
                options.JsonSerializerOptions.Converters.Add(new JsonDateTimeWithTimezoneConverter());
            });

就像您在代码中看到的那样,当收到没有时区的日期时,它会崩溃并出现异常,导致错误 500 InternalServerError

我如何抛出异常以返回 400 而不触及使用 app.UseExceptionHandler (因为代码在我的库中)?

请注意,抛出的异常是 FormatException 所以,对我来说,它应该被翻译成 BadRequest ...

4

1 回答 1

3

这个问题应该从 ASP.NET Core 3.1 开始修复:(FormatExceptionDateTime.ParseExact()方法抛出)现在被视为SystemTextJsonInputFormatter3.0 以来的默认输入格式化程序中的模型状态错误。这将导致返回 400 而不是 500 状态。请参阅Github 上的 PR ,以及3.03.1的相关代码部分;

如果您仍在使用 3.0,则可以捕获FormatException并重新抛出 a JsonException,这是 3.0 中唯一被视为模型状态错误的异常类型:

try
{
    return DateTime.ParseExact(reader.GetString(), FORMATS, CultureInfo.InvariantCulture, DateTimeStyles.None);
}
catch(FormatException e)
{
    throw new JsonException();
}
于 2019-12-17T14:26:14.267 回答