How to get JSON.NET to serialize date/time to ISO 8601?
Asked Answered
G

4

27

I have a Web API application that returns JSON to consumers who may not be using Microsoft technologies. When my controller returns an object with DateTime properties as JSON, it serializes the date in this format:

2017-03-15T00:00:00-04:00

This is giving the consumer a bit of a headache as they're expect it to be in ISO 8601 format. Some research has told me that JSON.NET now uses ISO 8601 by default (I am using 9.0.1). When I run this code...

Clipboard.Copy(JsonConvert.SerializeObject(DateTime.Now));

...I get this:

2017-03-15T09:10:13.8105498-04:00

Wikipedia shows these as valid ISO 8601 formats when expressing full date and time:

2017-03-15T11:45:42+00:00
2017-03-15T11:45:42Z
20170315T114542Z

However, the output I got above doesn't exactly match any of those. I want the formatter to use 2017-03-15T11:45:42Z.

And probably worthy of another question altogether, adding the below line in my Web API config seems to be ignored as it continues to return JSON in the date originally shown above:

config.Formatters.JsonFormatter.SerializerSettings.Converters.Add(new IsoDateTimeConverter());

I assume that once I figure out the core issue, the Web API issue may also be resolved.

Gymnosophist answered 15/3, 2017 at 13:36 Comment(1)
I wish I could upvote this multiple times. More people should be concerned with JSON dates with ISO 8601 formatting.Braided
U
34

The format you are getting is ISO 8601 format (read the section on Times and Time Zone Designators in Wikipedia), it's just that your dates are apparently not adjusted to UTC time, so you are getting a timezone offset appended to the date rather than the Z Zulu timezone indicator.

The IsoDateTimeConverter has settings you can use to customize its output. You can make it automatically adjust dates to UTC by setting DateTimeStyles to AdjustToUniversal. You can also customize the output format to omit the fractional seconds if you don't want them. By default, the converter does not adjust to UTC time and includes as many decimals of precision as there are available for the seconds.

Try this:

IsoDateTimeConverter converter = new IsoDateTimeConverter
{
    DateTimeStyles = DateTimeStyles.AdjustToUniversal,
    DateTimeFormat = "yyyy'-'MM'-'dd'T'HH':'mm':'ssK"
};

config.Formatters.JsonFormatter.SerializerSettings.Converters.Add(converter);

If your dates are already UTC, but the DateTimeKind on them is not set to Utc like it should be (e.g. it is Unspecified), then ideally you should fix your code so that this indicator is set correctly prior to serializing. However, if you can't (or don't want to) do that, you can work around it by changing the converter settings to always include the Z indicator in the date format (instead of using the K specifier which looks at the DateTimeKind on the date) and removing the AdjustToUniversal directive.

IsoDateTimeConverter converter = new IsoDateTimeConverter
{
    DateTimeFormat = "yyyy'-'MM'-'dd'T'HH':'mm':'ss'Z'"
};
Unconditioned answered 15/3, 2017 at 17:33 Comment(2)
Good answer, thanks. I used part of your initial example and ended up just using DateTimeStyles = DateTimeStyles.AdjustToUniversal without the format and it outputs as 2017-03-15T04:00:00Z just as I was wanting.Gymnosophist
Very good answer. Just to note to add this to the WebApiConfig.cs file. I also used Culture = System.Globalization.CultureInfo.InvariantCulture as well as DateTimeFormat = "o" which produced serialized dates like this 2014-12-10T17:32:44.4930000ZBraided
L
4

Adding to @Brian Rogers' answer, for ASP Core, add in Startup.cs:

services.AddMvc()
  .SetCompatibilityVersion(CompatibilityVersion.Version_2_2)
  .AddJsonOptions(options =>
    options.SerializerSettings.Converters.Add(new IsoDateTimeConverter
    {
      DateTimeStyles = DateTimeStyles.AdjustToUniversal
    }));
Leyes answered 15/7, 2019 at 8:25 Comment(1)
sorry, doesn't work .. also, if you want specific to core, maybe use .AddMvcCore() ?Banditry
D
3

Another option:

string json = JsonConvert.SerializeObject(DateTime.Now, new JsonSerializerSettings
{
     DateTimeZoneHandling = DateTimeZoneHandling.Utc
});
Dimidiate answered 28/6, 2020 at 9:46 Comment(0)
O
0

To handle DateTime serialization and deserialization as ISO 8601 strings in JSON, you can create a custom JsonConverter and register it in Program.cs using AddJsonOptions.

Handler Class:

/// <summary>
/// Converts DateTime to and from ISO 8601 strings in JSON.
/// </summary>
public class JsonIsoDateTimeConverter : JsonConverter<DateTime>
{
    private const string IsoFormat = "yyyy-MM-ddTHH:mm:ss.fffffffZ";

    public override DateTime Read(ref Utf8JsonReader reader, Type typeToConvert, JsonSerializerOptions options)
    {
        return DateTime.Parse(reader.GetString());
    }

    public override void Write(Utf8JsonWriter writer, DateTime value, JsonSerializerOptions options)
    {
        writer.WriteStringValue(value.ToUniversalTime().ToString(IsoFormat));
    }
}

Configuration in Program.cs:

builder.Services.AddControllers()
    .AddJsonOptions(options =>
    {
        options.JsonSerializerOptions.Converters.Add(new JsonIsoDateTimeConverter());
    });

By using this custom JsonIsoDateTimeConverter, all DateTime values will be serialized to and deserialized from the ISO 8601 format (e.g., "2024-07-18T09:30:00.0000000Z"), ensuring compliance with the ISO-8601 standard.

Olnay answered 4/7, 2024 at 22:31 Comment(0)

© 2022 - 2025 — McMap. All rights reserved.