Search code examples
.netlocalizationinternationalizationdatetime-formatpseudolocalization

.NET Date to string gives invalid strings in Vista Pseudo-cultures


My computer is configured with a culture that is not en-US.

When using the native Win32 GetDateFormat function, i get correctly formatted dates:

  • 22//11//2011 4::42::53 P̰̃M]

This is correct; and is also how Windows renders it:

  • the taskbar

    enter image description here

  • Region and Language settings

    enter image description here

  • Windows Explorer

    enter image description here

  • Outlook

    enter image description here

When i try to convert a date to a string in .NET using my current locale, e.g.:

DateTime.Now.ToString();
DateTime.Now.ToString(CultureInfo.CurrentCulture);

i get an incorrect date:

  • 22////11////2011 4::::42::::53 P̰̃M]

This bug in .NET is evident anyplace in Windows that uses the buggy .NET code:

  • Windows Event Viewer:

    enter image description here

  • Task Scheduler:

    enter image description here

  • SQL Server Management Studio:

    enter image description here

How do i make .NET not buggy?

How do i convert dates and times to strings using the current culture (correctly)?

Note: The user is allowed to set their Windows to any locale preferences they want. As it is now, my program will not handle valid settings properly. Telling the user, "Don't do that" is pretty mean-spirited.

A similar example comes from Delphi, which assumes that a date separator can never be more than one character. When Windows is configured with a locale that uses multiple characters for the date separator, e.g.:

  • sk-SK (Slovak - Slovakia) : .

where dates should be formatted as:

22. 11. 2011

the code library fails to accept a date separator longer than one character, and falls back to:

22/11/2011

In the past some might suggest that you not to bother with such edge cases. Such suggestions carry no weight with me.

i'll avoid getting into a pissing match with someone who wants to alter the meaning of my question by changing the title. But the question is not limited to pseudo-locales, specifically designed to find bugs in applications.

Bonus Chatter

Here's a unique list of date formats from around the world:

  • 11.11.25
  • 11.25.2011
  • 11/25/2011
  • 2011.11.25
  • 2011.11.25.
  • 2011/11/25
  • 2011-11-25
      1. 2011
  • 25.11.11
  • 25.11.2011
  • 25.11.2011 г.
  • 25.11.2011.
  • 25//11//2011
  • 25/11 2011
  • 25/11/2011
  • 25/11/2554
  • 25-11-11
  • 25-11-2011
  • 29/12/32

Of particular interest is the last example which doesn't use the gregorian calendar:

  • Arabic (Saudi Arabia) ar-SA: 29/12/32 02:03:07 م
  • Divehi (Maldives) dv-MV: 29/12/32 14:03:07
  • Dari/Pashto (Afghanistan) prf-AF / ps-AF: 29/12/32 2:03:07 غ.و

Although those are edge cases that you'd never have to worry about.


Update 14//12//2011:

Another demonstration of the bug is that Datetime.Parse cannot parse DateTime.ToString:

String s = DateTime.Today.ToString("d");   //returns "14////12////2011"
DateTime d = DateTime.Parse(s);            //expects "dd//MM//yyyy"

The .Parse throws an exception.


Update 02//8, 2012 09::56'12:

Any use of a date separator is depricated, in addition to being incorrect. From MSDN:

LOCALE_SDATE

Windows Vista and later: This constant is deprecated. Use LOCALE_SSHORTDATE instead. A custom locale might not have a single, uniform separator character. For example, a format such as "12/31, 2006" is valid.

LOCALE_STIME

Windows Vista and later: This constant is deprecated. Use LOCALE_STIMEFORMAT instead. A custom locale might not have a single, uniform separator character. For example, a format such as "03:56'23" is valid.


Solution

  • This specific bug is due to the transformation of some special characters that aren't escaped in the patterns like ShortDatePattern.

    ShortDatePattern = "d//MM//yyyy";
    

    / in a pattern means "insert the date separator" but here the expansion is already done (at least on my system) when the string is copied from the system to the DateTimeFormat structure. Sadly it is missing an escaping (Obviously not visible on any language not using a special character as a separator and not visible in english as it is replaced with itself)

    The only solution seem to be to escape the separators in all the patterns of the DateTimeFormat instance :

    var c = new System.Globalization.CultureInfo("qps-ploc", true);
    c.DateTimeFormat.ShortDatePattern =
            c.DateTimeFormat.ShortDatePattern.Replace("/", "'/'");
    c.DateTimeFormat.LongTimePattern =
            c.DateTimeFormat.LongTimePattern.Replace(":", "':'");
    Console.WriteLine(DateTime.Now.ToString(c));
    

    Here's full code samples for all three common cases

    Date to string

    /// <summary>Convert a date to the short date string in the current locale (e.g. 30//11//2011)</summary>
    /// <param name="value">A DateTime to be converted to a short date string</param>
    /// <returns>A string containing the localized version of the date</returns>
    public static String DateToStr(DateTime value)
    {
        String format = CultureInfo.CurrentCulture.DateTimeFormat.ShortDatePattern;
    
        //The bug in .NET is that it assumes "/" in a date pattern means "the date separator"
        //What .NET doesn't realize is that the locale strings returned by Windows are the Windows format strings. 
        //The bug is exposed in locale's that use two slashes as for their date separator:
        //  dd//MM//yyyy
        // Which .NET misinterprets to give:
        //  30////11////2011
        // when really it should be taken literally to be:
        //  dd'//'MM'//'yyyy
        //which is what this fix does
        format = format.Replace("/", "'/'"); 
    
        return value.ToString(format);
    }
    

    Time to string

    /// <summary>
    /// Convert a time to string using the short time format in the current locale(e.g. 7::21 AM)
    /// </summary>
    /// <param name="value">A DateTime who's time portion will be converted to a localized string</param>
    /// <returns>A string containing the localized version of the time</returns>
    public static String TimeToStr(DateTime value)
    {
        String format = CultureInfo.CurrentCulture.DateTimeFormat.ShortTimePattern;
    
        //The bug in .NET is that it assumes ":" in a time pattern means "the time separator"
        //What .NET doesn't realize is that the locale strings returned by Windows are the Windows format strings. 
        //The bug is exposed in locale's that use two colons as their time separator:
        //  h::mm::ss tt
        // Which .NET misinterprets to give:
        //  11::::39::::17 AM
        // when really it should be taken literally to be:
        //  h'::'mm'::'ss tt
        //which is what this fix does
        format = format.Replace(":", "':'"); 
    
        return value.ToString(format);
    }
    

    Datetime to string

    /// <summary>
    /// Convert a datetime to a string in the current locale (e.g. 30//11//2001 7::21 AM) 
    /// </summary>
    /// <param name="datetime">A DateTime to be converted to a general string in the current locale</param>
    /// <returns>A string containing the localized version of the datetime</returns>
    public static String DateTimeToStr(DateTime datetime)
    {
        return DateToStr(datetime)+" "+TimeToStr(datetime);
    }