Recently I’ve been tasked to provide date in a specific format from backend to the frontend and I’ve noticed a behavior that I’ve found a bit odd.

private static void OutputDateInfo(string value)
{
    Console.WriteLine($"Input: {value}");
    if (DateTime.TryParse(value, out DateTime dateTimeValue))
    {
        Console.WriteLine($"Setialized to universal format {dateTimeValue.ToString("yyyy-MM-dd'T'HHssZ")}");
    }
    Console.WriteLine();
}

//somewhere in main
OutputDateInfo("2021-04-15T2100.0000000Z");
OutputDateInfo("2021-04-15T2100.0000000");
Produces the output

Input: 2021-04-15T21:01:00.0000000Z
Setialized to universal format 2021-04-16T00:01:00Z

Input: 2021-04-15T21:01:00.0000000
Setialized to universal format 2021-04-15T21:01:00Z

So as I was in Kyiv which is in UTC+3 timezone it looked as letter Z forced ToString method to convert date to local time.

As documentation reads

If s contains no time zone information, result contains a DateTime value whose Kind property is DateTimeKind.Unspecified when the method returns. If the string to be parsed contains time zone information, result contains a DateTime value whose Kind property is DateTimeKind.Local when the method returns.

Is this DateTimeKind that makes ToString cast date to the local timezone? Let’s log more properties to see if it is the case

private static void OutputDateInfo(string value)
{
    Console.WriteLine($"Input: {value}");
    if (DateTime.TryParse(value, out DateTime dateTimeValue))
    {
        Console.WriteLine($"Setialized to universal format {dateTimeValue.ToString("yyyy-MM-dd'T'HHssZ")}");
        Console.WriteLine($"Setialized to default format {dateTimeValue}");
        Console.WriteLine($"Setialized with conversion to universal time {dateTimeValue.ToUniversalTime()}");
        Console.WriteLine($"Kind: {dateTimeValue.Kind}");
    }
    Console.WriteLine();
}
Now we see in a console:

Input: 2021-04-15T21:01:00.0000000Z
Setialized to universal format 2021-04-16T00:01:00Z
Setialized to default format 16.04.2021 0:01:00
Setialized with conversion to universal time 15.04.2021 21:01:00
Kind: Local

Input: 2021-04-15T21:01:00.0000000
Setialized to universal format 2021-04-15T21:01:00Z
Setialized to default format 15.04.2021 21:01:00
Setialized with conversion to universal time 15.04.2021 18:01:00
Kind: Unspecified

So indeed when we call ToString on a DateTimeKind.Local instance it will be adjusted according to server timezone.

But why letter Z is treated as timezone information? The documentation referred above has no example with Z letter. The answer is that format in question is ISO-8601 format and Z stands for “Zero UTC offset”. Does string of this format is treated just as any other string with timezone specified or is this some kind of special treatment? We’ll get an answer by comparing them.

OutputDateInfo("2021-04-15T2100.0000000Z");
OutputDateInfo("2021-04-15T2100.0000000");
OutputDateInfo("2021-04-15T1400.0000000 -7:00");
yeilds

Input: 2021-04-15T21:01:00.0000000Z
Setialized to universal format 2021-04-16T00:01:00Z
Setialized to default format 16.04.2021 0:01:00
Setialized with conversion to universal time 15.04.2021 21:01:00
Kind: Local

Input: 2021-04-15T21:01:00.0000000
Setialized to universal format 2021-04-15T21:01:00Z
Setialized to default format 15.04.2021 21:01:00
Setialized with conversion to universal time 15.04.2021 18:01:00
Kind: Unspecified

Input: 2021-04-15T14:01:00.0000000 -7:00
Setialized to universal format 2021-04-16T00:01:00Z
Setialized to default format 16.04.2021 0:01:00
Setialized with conversion to universal time 15.04.2021 21:01:00
Kind: Local

As you can see string of ISO-8601 format is treated just as any other date string with timezone specified.

My intent though wasn’t to treat it as local time but instead as UTC time. My goal was to provide universal time to front end so it will adjust it to end-user’s local time. How can we trick .NET into thinking that this is universal time without any timezone? The answer is the overload that accepts DateTimeStyles:

private static void OutputDateInfo(string value)
{
    Console.WriteLine($"Input: {value}");
    if (DateTime.TryParse(value, null, DateTimeStyles.AdjustToUniversal, out DateTime universalTime))
    {
        Console.WriteLine($"Adjusted to universal {universalTime}");
    }

    Console.WriteLine();
}
Outputs

Input: 2021-04-15T21:01:00.0000000Z
Adjusted to universal 15.04.2021 21:01:00

Input: 2021-04-15T21:01:00.0000000
Adjusted to universal 15.04.2021 21:01:00

Input: 2021-04-15T14:01:00.0000000 -7:00
Adjusted to universal 15.04.2021 21:01:00

Job done!

Conclusion

In my opinion, DateTimeKind is underrepresented in literature and blogs. However, this is the thing that should be definitely taken into account when parsing date strings and converting dates back to string. Also, it worth remembering that ISO-8601 format date strings are treated just as any other string with timezone info specified, although it might be unobvious at the first glance.