Changed TryPeek to handle race condition with Dequeue.
[mono.git] / mcs / class / corlib / Test / System / DateTimeTest.cs
1 //
2 // DateTimeTest.cs - NUnit Test Cases for the System.DateTime struct
3 //
4 // author:
5 //   Martin Baulig (martin@gnome.org)
6 //
7 //   (C) 2002 Free Software Foundation
8 // Copyright (C) 2004 Novell (http://www.novell.com)
9 //
10
11 using System;
12 using System.IO;
13 using System.Threading;
14 using System.Globalization;
15
16 using NUnit.Framework;
17
18 namespace MonoTests.System
19 {
20         [TestFixture]
21         public class DateTimeTest
22         {
23                 [Flags]
24                 internal enum Resolution : ushort
25                 {
26                         Year = 64,
27                         Month = 96,
28                         Day = 112,
29                         Hour = 120,
30                         Minute = 124,
31                         Second = 126,
32                         Millisecond = 127,
33                         _Month = 32,
34                         _Day = 16,
35                         _Hour = 8,
36                         _Minute = 4,
37                         _Second = 2,
38                         _Millisecond = 1
39                 }
40
41                 internal void DTAssertEquals (DateTime actual, DateTime expected, Resolution resolution)
42                 {
43                         DTAssertEquals (actual, expected, resolution, "");
44                 }
45
46                 internal void DTAssertEquals (DateTime expected, DateTime actual, Resolution resolution, string message)
47                 {
48                         if ((resolution & Resolution.Year) != 0)
49                                 Assert.AreEqual (expected.Year, actual.Year, message);
50                         if ((resolution & Resolution._Month) != 0)
51                                 Assert.AreEqual (expected.Month, actual.Month, message);
52                         if ((resolution & Resolution._Day) != 0)
53                                 Assert.AreEqual (expected.Day, actual.Day, message);
54                         if ((resolution & Resolution._Hour) != 0)
55                                 Assert.AreEqual (expected.Hour, actual.Hour, message);
56                         if ((resolution & Resolution._Minute) != 0)
57                                 Assert.AreEqual (expected.Minute, actual.Minute, message);
58                         if ((resolution & Resolution._Second) != 0)
59                                 Assert.AreEqual (expected.Second, actual.Second, message);
60                         if ((resolution & Resolution._Millisecond) != 0)
61                                 Assert.AreEqual (expected.Millisecond, actual.Millisecond, message);
62                 }
63
64                 private CultureInfo oldcult;
65         
66                 long[] myTicks = {
67                         631501920000000000L,    // 25 Feb 2002 - 00:00:00
68                         631502475130080000L,    // 25 Feb 2002 - 15:25:13,8
69                         631502115130080000L,    // 25 Feb 2002 - 05:25:13,8
70                         631502115000000000L,    // 25 Feb 2002 - 05:25:00
71                         631502115130000000L,    // 25 Feb 2002 - 05:25:13
72                         631502079130000000L,    // 25 Feb 2002 - 04:25:13
73                         629197085770000000L,    // 06 Nov 1994 - 08:49:37 
74                         631796544000000000L,    // 01 Feb 2003 - 00:00:00
75                 };
76
77                 [SetUp]
78                 public void SetUp() 
79                 {
80                         // the current culture determines the result of formatting
81                         oldcult = Thread.CurrentThread.CurrentCulture;
82                         Thread.CurrentThread.CurrentCulture = new CultureInfo ("");
83                 }
84         
85                 [TearDown]
86                 public void TearDown ()
87                 {
88                         Thread.CurrentThread.CurrentCulture = oldcult;
89                 }
90
91                 [Test]
92                 public void TestCtors ()
93                 {
94                         DateTime t1 = new DateTime (2002,2,25);
95                         Assert.AreEqual (myTicks [0], t1.Ticks, "A01");
96                         DateTime t2 = new DateTime (2002,2,25,15,25,13,8);
97                         Assert.AreEqual (myTicks [1], t2.Ticks, "A02");
98                         Assert.AreEqual (myTicks [0], t2.Date.Ticks, "A03");
99                         Assert.AreEqual (2002, t2.Year, "A04");
100                         Assert.AreEqual (2, t2.Month, "A05");
101                         Assert.AreEqual (25, t2.Day, "A06");
102                         Assert.AreEqual (15, t2.Hour, "A07");
103                         Assert.AreEqual (25, t2.Minute, "A08");
104                         Assert.AreEqual (13, t2.Second, "A09");
105                         Assert.AreEqual (8, t2.Millisecond, "A10");
106                         DateTime t3 = new DateTime (2002,2,25,5,25,13,8);
107                         Assert.AreEqual (myTicks [2], t3.Ticks, "A11");
108                 }
109
110                 [Test]
111                 public void Constructor_Max ()
112                 {
113                         Assert.AreEqual (3155378975999990000, new DateTime (9999, 12, 31, 23, 59, 59, 999).Ticks, "Max");
114                 }
115
116                 [Test]
117                 [ExpectedException (typeof (ArgumentOutOfRangeException))]
118                 public void Constructor_Milliseconds_Negative () 
119                 {
120                         new DateTime (9999, 12, 31, 23, 59, 59, -1);
121                 }
122
123                 [Test]
124                 [ExpectedException (typeof (ArgumentOutOfRangeException))]
125                 public void Constructor_Milliseconds_1000 () 
126                 {
127                         new DateTime (9999, 12, 31, 23, 59, 59, 1000);
128                 }
129                 
130                 [Test]
131                 public void Fields ()
132                 {
133                         Assert.AreEqual (3155378975999999999L, DateTime.MaxValue.Ticks, "#1");
134                         Assert.AreEqual (0, DateTime.MinValue.Ticks, "#2");
135                 }
136                 
137                 [Test]
138                 public void Add ()
139                 {
140                         DateTime t1 = new DateTime (myTicks [1]);
141                         TimeSpan span = new TimeSpan (3, 54, 1);
142                         DateTime t2 = t1.Add (span);
143
144                         Assert.AreEqual (25, t2.Day, "#1");
145                         Assert.AreEqual (19, t2.Hour, "#2");
146                         Assert.AreEqual (19, t2.Minute, "#3");
147                         Assert.AreEqual (14, t2.Second, "#4");
148
149                         Assert.AreEqual (25, t1.Day, "#5");
150                         Assert.AreEqual (15, t1.Hour, "#6");
151                         Assert.AreEqual (25, t1.Minute, "#7");
152                         Assert.AreEqual (13, t1.Second, "#8");
153                 }
154                 
155                 [Test]
156                 [ExpectedException(typeof (ArgumentOutOfRangeException))]
157                 public void AddOutOfRangeException1 ()
158                 {
159                         DateTime t1 = new DateTime (myTicks [1]);
160                         t1.Add (TimeSpan.MaxValue);
161                 }
162
163                 [Test]
164                 [ExpectedException(typeof (ArgumentOutOfRangeException))]
165                 public void AddOutOfRangeException2 ()
166                 {
167                         DateTime t1 = new DateTime (myTicks [1]);
168                         t1.Add (TimeSpan.MinValue);
169                 }
170         
171                 [Test]
172                 public void AddDays ()
173                 {
174                         DateTime t1 = new DateTime (myTicks [1]);
175                         t1 = t1.AddDays (3);
176                         Assert.AreEqual (28, t1.Day, "#A1");
177                         Assert.AreEqual (15, t1.Hour, "#A2");
178                         Assert.AreEqual (25, t1.Minute, "#A3");
179                         Assert.AreEqual (13, t1.Second, "#A4");
180                         
181                         t1 = t1.AddDays (1.9);
182                         Assert.AreEqual (2, t1.Day, "#B1");
183                         Assert.AreEqual (13, t1.Hour, "#B2");
184                         Assert.AreEqual (1, t1.Minute, "#B3");
185                         Assert.AreEqual (13, t1.Second, "#B4");
186
187                         t1 = t1.AddDays (0.2);
188                         Assert.AreEqual (2, t1.Day, "#C1");
189                         Assert.AreEqual (17, t1.Hour, "#C2");
190                         Assert.AreEqual (49, t1.Minute, "#C3");
191                         Assert.AreEqual (13, t1.Second, "#C4");
192                 }
193                 
194                 [Test]
195                 [ExpectedException(typeof (ArgumentOutOfRangeException))]
196                 public void AddDaysOutOfRangeException1 ()
197                 {
198                         DateTime t1 = new DateTime (myTicks [1]);
199                         t1.AddDays (10000000);
200                 }
201
202                 [Test]
203                 [ExpectedException(typeof (ArgumentOutOfRangeException))]
204                 public void AddDaysOutOfRangeException2 ()
205                 {
206                         DateTime t1 = new DateTime (myTicks [1]);
207                         t1.AddDays (-10000000);
208                 }
209
210                 [Test]
211                 public void AddHours ()
212                 {
213                         DateTime t1 = new DateTime (myTicks [1]);
214                         t1 = t1.AddHours (10);
215                         Assert.AreEqual (26, t1.Day, "#A1");
216                         Assert.AreEqual (1, t1.Hour, "#A2");
217                         Assert.AreEqual (25, t1.Minute, "#A3");
218                         Assert.AreEqual (13, t1.Second, "#A4");
219                         
220                         t1 = t1.AddHours (-3.7);
221                         Assert.AreEqual (25, t1.Day, "#B1");
222                         Assert.AreEqual (21, t1.Hour, "#B2");
223                         Assert.AreEqual (43, t1.Minute, "#B3");
224                         Assert.AreEqual (13, t1.Second, "#B4");
225
226                         t1 = t1.AddHours (3.732);
227                         Assert.AreEqual (26, t1.Day, "#C1");
228                         Assert.AreEqual (1, t1.Hour, "#C2");
229                         Assert.AreEqual (27, t1.Minute, "#C3");
230                         Assert.AreEqual (8, t1.Second, "#C4");
231                 }
232                 
233                 [Test]
234                 [ExpectedException (typeof (ArgumentOutOfRangeException))]
235                 public void AddHoursOutOfRangeException1 ()
236                 {
237                         DateTime t1 = new DateTime (myTicks [1]);
238                         t1.AddHours (9E100);
239                 }
240
241                 [Test]
242                 [ExpectedException (typeof (ArgumentOutOfRangeException))]
243                 public void AddHoursOutOfRangeException2 ()
244                 {
245                         DateTime t1 = new DateTime (myTicks [1]);
246                         t1.AddHours (-9E100);
247                 }
248
249                 [Test]
250                 public void AddMilliseconds ()
251                 {
252                         DateTime t1 = new DateTime (myTicks [1]);
253                         t1 = t1.AddMilliseconds (1E10);
254                         Assert.AreEqual (21, t1.Day, "#A1");
255                         Assert.AreEqual (9, t1.Hour, "#A2");
256                         Assert.AreEqual (11, t1.Minute, "#A3");
257                         Assert.AreEqual (53, t1.Second, "#A4");
258                         
259                         t1 = t1.AddMilliseconds (-19E10);
260                         Assert.AreEqual (13, t1.Day, "#B1");
261                         Assert.AreEqual (7, t1.Hour, "#B2");
262                         Assert.AreEqual (25, t1.Minute, "#B3");
263                         Assert.AreEqual (13, t1.Second, "#B4");
264
265                         t1 = t1.AddMilliseconds (15.623);
266                         Assert.AreEqual (13, t1.Day, "#C1");
267                         Assert.AreEqual (7, t1.Hour, "#C2");
268                         Assert.AreEqual (25, t1.Minute, "#C3");
269                         Assert.AreEqual (13, t1.Second, "#C4");
270                 }
271
272                 [Test]
273                 [ExpectedException (typeof (ArgumentOutOfRangeException))]
274                 public void AddMillisecondsOutOfRangeException1 ()
275                 {
276                         DateTime t1 = new DateTime (myTicks [1]);
277                         t1.AddMilliseconds (9E100);
278                 }
279
280                 [Test]
281                 [ExpectedException (typeof (ArgumentOutOfRangeException))]
282                 public void AddMillisecondsOutOfRangeException2 ()
283                 {
284                         DateTime t1 = new DateTime (myTicks [1]);
285                         t1.AddMilliseconds (-9E100);
286                 }
287
288                 [Test]
289                 public void TestToString ()
290                 {
291                         DateTime t1 = new DateTime (myTicks[2]);
292                         DateTime t2 = new DateTime (myTicks[1]);
293                         DateTime t3 = new DateTime (999, 1, 2, 3, 4, 5);
294                         // Standard patterns
295                         Assert.AreEqual ("02/25/2002", t1.ToString ("d"), "#A1");
296                         Assert.AreEqual ("Monday, 25 February 2002", t1.ToString ("D"), "#A2");
297                         Assert.AreEqual ("Monday, 25 February 2002 05:25", t1.ToString ("f"), "#A3");
298                         Assert.AreEqual ("Monday, 25 February 2002 05:25:13", t1.ToString ("F"), "#A4");
299                         Assert.AreEqual ("02/25/2002 05:25", t1.ToString ("g"), "#A5");
300                         Assert.AreEqual ("02/25/2002 05:25:13", t1.ToString ("G"), "#A6");
301                         Assert.AreEqual ("February 25", t1.ToString ("m"), "#A7");
302                         Assert.AreEqual ("February 25", t1.ToString ("M"), "#A8");
303                         Assert.AreEqual ("Mon, 25 Feb 2002 05:25:13 GMT", t1.ToString ("r"), "#A9");
304                         Assert.AreEqual ("Mon, 25 Feb 2002 05:25:13 GMT", t1.ToString ("R"), "#A10");
305                         Assert.AreEqual ("2002-02-25T05:25:13", t1.ToString ("s"), "#A11");
306                         Assert.AreEqual ("05:25", t1.ToString ("t"), "#A12");
307                         Assert.AreEqual ("05:25:13", t1.ToString ("T"), "#A13");
308                         Assert.AreEqual ("2002-02-25 05:25:13Z", t1.ToString ("u"), "#A14");
309                         // FIXME: this test is timezone dependent
310                         // Assert.AreEqual ("Sunday, 24 February 2002 11:25:13", t1.ToString ("U"), "#A15");
311                         Assert.AreEqual ("2002 February", t1.ToString ("y"), "#A16");
312                         Assert.AreEqual ("2002 February", t1.ToString ("Y"), "#A17");
313                         Assert.AreEqual ("02/25/2002 05:25:13", t1.ToString (""), "#A18");
314
315                         // Custom patterns
316                         Assert.AreEqual ("25", t1.ToString ("%d"), "#B1");
317                         Assert.AreEqual ("25", t1.ToString ("dd"), "#B2");
318                         Assert.AreEqual ("Mon", t1.ToString ("ddd"), "#B3");
319                         Assert.AreEqual ("Monday", t1.ToString ("dddd"), "#B4");
320                         Assert.AreEqual ("2", t1.ToString ("%M"), "#B5");
321                         Assert.AreEqual ("02", t1.ToString ("MM"), "#B6");
322                         Assert.AreEqual ("Feb", t1.ToString ("MMM"), "#B7");
323                         Assert.AreEqual ("February", t1.ToString ("MMMM"), "#B8");
324                         Assert.AreEqual ("2", t1.ToString ("%y"), "#B9");
325                         Assert.AreEqual ("02", t1.ToString ("yy"), "#B10");
326                         Assert.AreEqual ("2002", t1.ToString ("yyyy"), "#B11");
327                         Assert.AreEqual ("5", t1.ToString ("%h"), "#B12");
328                         Assert.AreEqual ("05", t1.ToString ("hh"), "#B13");
329                         Assert.AreEqual ("3", t2.ToString ("%h"), "#B14");
330                         Assert.AreEqual ("03", t2.ToString ("hh"), "#B15");
331                         Assert.AreEqual ("15", t2.ToString ("%H"), "#B16");
332                         Assert.AreEqual ("15", t2.ToString ("HH"), "#B17");
333                         Assert.AreEqual ("25", t2.ToString ("%m"), "#B18");
334                         Assert.AreEqual ("25", t2.ToString ("mm"), "#B19");
335                         Assert.AreEqual ("13", t2.ToString ("%s"), "#B20");
336                         Assert.AreEqual ("13", t2.ToString ("ss"), "#B21");
337                         Assert.AreEqual ("A", t1.ToString ("%t"), "#B22");
338                         Assert.AreEqual ("P", t2.ToString ("%t"), "#B23");
339                         Assert.AreEqual ("AM", t1.ToString ("tt"), "#B24");
340                         Assert.AreEqual ("PM", t2.ToString ("tt"), "#B25");
341                         long offset = TimeZone.CurrentTimeZone.GetUtcOffset(t1).Ticks / 36000000000;
342                         // Must specify '+0' for GMT
343                         Assert.AreEqual (offset.ToString ("+#;-#;+0"), t1.ToString ("%z"), "#B26");
344                         Assert.AreEqual (offset.ToString ("+00;-00;+00"), t1.ToString ("zz"), "#B28");
345                         // This does not work in, eg banglore, because their timezone has an offset of
346                         // +05:30
347                         //Assert.AreEqual (offset.ToString("+00;-00;00") + ":00", t1.ToString ("zzz"), "#B28");
348                         Assert.AreEqual (" : ", t1.ToString (" : "), "#B29");
349                         Assert.AreEqual (" / ", t1.ToString (" / "), "#B30");
350                         Assert.AreEqual (" yyy ", t1.ToString (" 'yyy' "), "#B31");
351                         Assert.AreEqual (" d", t1.ToString (" \\d"), "#B32");
352                         Assert.AreEqual ("2002", t1.ToString ("yyy"), "#B33");
353                         Assert.AreEqual ("0002002", t1.ToString ("yyyyyyy"), "#B34");
354                         Assert.AreEqual ("999", t3.ToString ("yyy"), "#B33");
355                         Assert.AreEqual ("0999", t3.ToString ("yyyy"), "#B33");
356                 }
357
358                 [Test]
359                 public void TestToStringGenitive ()
360                 {
361                         DateTime dt = new DateTime (2010, 1, 2, 3, 4, 5);
362                         var dtf = new CultureInfo ("cs-cz").DateTimeFormat;
363
364                         Assert.AreEqual ("2 ledna", dt.ToString ("d MMMM", dtf), "#A1");
365                         Assert.AreEqual ("2. ledna", dt.ToString ("d. MMMM", dtf), "#A2");
366                         Assert.AreEqual ("leden", dt.ToString ("MMMM", dtf), "#A4");
367                         Assert.AreEqual ("leden", dt.ToString ("MMMMMMM", dtf), "#A5");
368                 }
369
370                 [Test]
371                 public void ParseExact_Format_Empty ()
372                 {
373                         try {
374                                 DateTime.ParseExact ("2002-02-25 04:25:13Z", string.Empty, null);
375                                 Assert.Fail ("#A1");
376                         } catch (FormatException ex) {
377                                 // Format specifier was invalid
378                                 Assert.AreEqual (typeof (FormatException), ex.GetType (), "#B2");
379                                 Assert.IsNull (ex.InnerException, "#B3");
380                                 Assert.IsNotNull (ex.Message, "#B4");
381                         }
382
383                         try {
384                                 DateTime.ParseExact ("2002-02-25 04:25:13Z", string.Empty, null,
385                                         DateTimeStyles.None);
386                                 Assert.Fail ("#B1");
387                         } catch (FormatException ex) {
388                                 // Format specifier was invalid
389                                 Assert.AreEqual (typeof (FormatException), ex.GetType (), "#B2");
390                                 Assert.IsNull (ex.InnerException, "#B3");
391                                 Assert.IsNotNull (ex.Message, "#B4");
392                         }
393                 }
394
395                 [Test]
396                 public void ParseExact_Format_Null ()
397                 {
398                         try {
399                                 DateTime.ParseExact ("2002-02-25 04:25:13Z", (string) null, null);
400                                 Assert.Fail ("#A1");
401                         } catch (ArgumentNullException ex) {
402                                 Assert.AreEqual (typeof (ArgumentNullException), ex.GetType (), "#A2");
403                                 Assert.IsNull (ex.InnerException, "#A3");
404                                 Assert.IsNotNull (ex.Message, "#A4");
405                                 Assert.IsNotNull (ex.ParamName, "#A5");
406                                 Assert.AreEqual ("format", ex.ParamName, "#A6");
407                         }
408
409                         try {
410                                 DateTime.ParseExact ("2002-02-25 04:25:13Z", (string) null, null,
411                                         DateTimeStyles.None);
412                                 Assert.Fail ("#B1");
413                         } catch (ArgumentNullException ex) {
414                                 Assert.AreEqual (typeof (ArgumentNullException), ex.GetType (), "#B2");
415                                 Assert.IsNull (ex.InnerException, "#B3");
416                                 Assert.IsNotNull (ex.Message, "#B4");
417                                 Assert.IsNotNull (ex.ParamName, "#B5");
418                                 Assert.AreEqual ("format", ex.ParamName, "#B6");
419                         }
420                 }
421
422                 [Test]
423                 public void ParseExact_Formats_Empty ()
424                 {
425                         try {
426                                 DateTime.ParseExact ("2002-02-25 04:25:13Z", new string [0],
427                                         null, DateTimeStyles.None);
428                                 Assert.Fail ("#A1");
429                         } catch (FormatException ex) {
430                                 // Format specifier was invalid
431                                 Assert.AreEqual (typeof (FormatException), ex.GetType (), "#A2");
432                                 Assert.IsNull (ex.InnerException, "#A3");
433                                 Assert.IsNotNull (ex.Message, "#A4");
434                         }
435
436                         string [] formats = new string [] { "G", string.Empty, "d" };
437                         try {
438                                 DateTime.ParseExact ("2002-02-25 04:25:13Z", formats, null,
439                                         DateTimeStyles.None);
440                                 Assert.Fail ("#B1");
441                         } catch (FormatException ex) {
442                                 // Format specifier was invalid
443                                 Assert.AreEqual (typeof (FormatException), ex.GetType (), "#B2");
444                                 Assert.IsNull (ex.InnerException, "#B3");
445                                 Assert.IsNotNull (ex.Message, "#B4");
446                         }
447                 }
448
449                 [Test]
450                 public void ParseExact_Formats_Null ()
451                 {
452                         try {
453                                 DateTime.ParseExact ("2002-02-25 04:25:13Z", (string []) null,
454                                         null, DateTimeStyles.None);
455                                 Assert.Fail ("#A1");
456                         } catch (ArgumentNullException ex) {
457                                 Assert.AreEqual (typeof (ArgumentNullException), ex.GetType (), "#A2");
458                                 Assert.IsNull (ex.InnerException, "#A3");
459                                 Assert.IsNotNull (ex.Message, "#A4");
460                                 Assert.IsNotNull (ex.ParamName, "#A5");
461                                 Assert.AreEqual ("formats", ex.ParamName, "#A6");
462                         }
463
464                         string [] formats = new string [] { "G", null, "d" };
465                         try {
466                                 DateTime.ParseExact ("2002-02-25 04:25:13Z", formats, null,
467                                         DateTimeStyles.None);
468                                 Assert.Fail ("#B1");
469                         } catch (FormatException ex) {
470                                 // Format specifier was invalid
471                                 Assert.AreEqual (typeof (FormatException), ex.GetType (), "#B2");
472                                 Assert.IsNull (ex.InnerException, "#B3");
473                                 Assert.IsNotNull (ex.Message, "#B4");
474                         }
475                 }
476
477                 [Test]
478                 public void ParseExact_String_Empty ()
479                 {
480                         try {
481                                 DateTime.ParseExact (string.Empty, "G", null);
482                                 Assert.Fail ("#A1");
483                         } catch (FormatException ex) {
484                                 // Format specifier was invalid
485                                 Assert.AreEqual (typeof (FormatException), ex.GetType (), "#A2");
486                                 Assert.IsNull (ex.InnerException, "#A3");
487                                 Assert.IsNotNull (ex.Message, "#A4");
488                         }
489
490                         try {
491                                 DateTime.ParseExact (string.Empty, "G", null, DateTimeStyles.None);
492                                 Assert.Fail ("#B1");
493                         } catch (FormatException ex) {
494                                 // Format specifier was invalid
495                                 Assert.AreEqual (typeof (FormatException), ex.GetType (), "#B2");
496                                 Assert.IsNull (ex.InnerException, "#B3");
497                                 Assert.IsNotNull (ex.Message, "#B4");
498                         }
499
500                         try {
501                                 DateTime.ParseExact (string.Empty, new string [] { "G" }, null,
502                                         DateTimeStyles.None);
503                                 Assert.Fail ("#C1");
504                         } catch (FormatException ex) {
505                                 // Format specifier was invalid
506                                 Assert.AreEqual (typeof (FormatException), ex.GetType (), "#C2");
507                                 Assert.IsNull (ex.InnerException, "#C3");
508                                 Assert.IsNotNull (ex.Message, "#C4");
509                         }
510                 }
511
512                 [Test]
513                 public void ParseExact_String_Null ()
514                 {
515                         try {
516                                 DateTime.ParseExact ((string) null, "G", null);
517                                 Assert.Fail ("#A1");
518                         } catch (ArgumentNullException ex) {
519                                 Assert.AreEqual (typeof (ArgumentNullException), ex.GetType (), "#A2");
520                                 Assert.IsNull (ex.InnerException, "#A3");
521                                 Assert.IsNotNull (ex.Message, "#A4");
522                                 Assert.IsNotNull (ex.ParamName, "#A5");
523                                 Assert.AreEqual ("s", ex.ParamName, "#A6");
524                         }
525
526                         try {
527                                 DateTime.ParseExact ((string) null, "G", null, DateTimeStyles.None);
528                                 Assert.Fail ("#B1");
529                         } catch (ArgumentNullException ex) {
530                                 Assert.AreEqual (typeof (ArgumentNullException), ex.GetType (), "#B2");
531                                 Assert.IsNull (ex.InnerException, "#B3");
532                                 Assert.IsNotNull (ex.Message, "#B4");
533                                 Assert.IsNotNull (ex.ParamName, "#B5");
534                                 Assert.AreEqual ("s", ex.ParamName, "#B6");
535                         }
536
537                         try {
538                                 DateTime.ParseExact ((string) null, new string [] { "G" }, null,
539                                         DateTimeStyles.None);
540                                 Assert.Fail ("#C1");
541                         } catch (ArgumentNullException ex) {
542                                 Assert.AreEqual (typeof (ArgumentNullException), ex.GetType (), "#C2");
543                                 Assert.IsNull (ex.InnerException, "#C3");
544                                 Assert.IsNotNull (ex.Message, "#C4");
545                                 Assert.IsNotNull (ex.ParamName, "#C5");
546                                 Assert.AreEqual ("s", ex.ParamName, "#C6");
547                         }
548                 }
549
550                 [Test]
551                 public void TestParseExact3 ()
552                 {
553                         DateTime t1 = DateTime.ParseExact ("2002-02-25 04:25:13Z", "u", null);
554                         Assert.AreEqual (2002, t1.Year, "#1");
555                         Assert.AreEqual (02, t1.Month, "#2");
556                         Assert.AreEqual (25, t1.Day, "#3");
557                         Assert.AreEqual (04, t1.Hour, "#4");
558                         Assert.AreEqual (25, t1.Minute, "#5");
559                         Assert.AreEqual (13, t1.Second, "#6");
560                 }
561
562                 [Test]
563                 public void TestParseExact4 ()
564                 {
565                         // Checks 24hours format used together with the AM/PM designator
566                         string s = "6/28/2004 13:00:00 AM";
567                         string f = "M/d/yyyy HH':'mm':'ss tt";
568                         try {
569                                 DateTime.ParseExact (s, f, CultureInfo.InvariantCulture);
570                                 Assert.Fail ("#1");
571                         } catch (FormatException) {
572                         }
573
574                         s = "6/28/2004 04:00:00 AM";
575                         DateTime.ParseExact (s, f, CultureInfo.InvariantCulture);
576
577                         s = "6/28/2004 06:00:00 PM";
578                         try {
579                                 DateTime.ParseExact (s, f, CultureInfo.InvariantCulture);
580                                 Assert.Fail ("#2");
581                         } catch (FormatException) {
582                         }
583
584                         s = "6/28/2004 17:00:00 PM";
585                         DateTime.ParseExact (s, f, CultureInfo.InvariantCulture);
586                 }
587                 
588                 [Test]
589                 public void TestParseExact4_2 ()
590                 {
591                         // bug #63137
592                         DateTime.ParseExact ("Wed, 12 May 2004 20:51:09 +0200",
593                                 @"ddd, d MMM yyyy H:m:s zzz",
594                                 CultureInfo.CreateSpecificCulture("en-us"),
595                                 DateTimeStyles.AllowInnerWhite);
596                 }
597
598                 [Test]
599                 public void TestParseExact5 ()
600                 {
601                         DateTime dt = DateTime.ParseExact ("Wed, 12 May 2004 20:51:09 -02:30",
602                                                         @"ddd, d MMM yyyy H:m:s zzz",
603                                                         CultureInfo.CreateSpecificCulture("en-us"),
604                                                         DateTimeStyles.AllowInnerWhite);
605                         dt = dt.ToUniversalTime ();
606                         Assert.AreEqual (23, dt.Hour, "Hour");
607                         Assert.AreEqual (21, dt.Minute, "Minute");
608                 }
609
610                 [Test]
611                 public void TestParseExactMiliseconds ()
612                 {
613                         string s = "1984-09-17T09:00:00.6Z";
614                         DateTime dt = DateTime.ParseExact (s, "yyyy-MM-dd'T'HH:mm:ss.FFF'Z'", CultureInfo.InvariantCulture);
615                         Assert.AreEqual (new DateTime (1984, 9, 17, 9, 0, 0, 600), dt, "#1");
616
617                         s = "1984-09-17T09:00:00Z";
618                         dt = DateTime.ParseExact (s, "yyyy-MM-dd'T'HH:mm:ss.FFF'Z'", CultureInfo.InvariantCulture);
619                         Assert.AreEqual (new DateTime (1984, 9, 17, 9, 0, 0, 0), dt, "#2");
620
621                         s = "1984-09-17T09:00:00Z";
622                         dt = DateTime.ParseExact (s, "yyyy-MM-dd'T'HH:mm:ssFFF'Z'", CultureInfo.InvariantCulture);
623                         Assert.AreEqual (new DateTime (1984, 9, 17, 9, 0, 0, 0), dt, "#3");
624                 }
625
626                 [Test]
627                 public void TestParseExact ()
628                 {
629                         // Standard patterns
630                         DateTime t1 = DateTime.ParseExact ("02/25/2002", "d", null);
631                         Assert.AreEqual (myTicks [0], t1.Ticks, "#A1");
632                         t1 = DateTime.ParseExact ("Monday, 25 February 2002", "D", null);
633                         Assert.AreEqual (myTicks [0], t1.Ticks, "#A2");
634                         t1 = DateTime.ParseExact ("Monday, 25 February 2002 05:25", "f", null);
635                         Assert.AreEqual (myTicks [3], t1.Ticks, "#A3");
636                         t1 = DateTime.ParseExact ("Monday, 25 February 2002 05:25:13", "F", null);
637                         Assert.AreEqual (myTicks [4], t1.Ticks, "#A4");
638                         t1 = DateTime.ParseExact ("02/25/2002 05:25", "g", null);
639                         Assert.AreEqual (myTicks [3], t1.Ticks, "#A5");
640                         t1 = DateTime.ParseExact ("02/25/2002 05:25:13", "G", null);
641                         Assert.AreEqual (myTicks [4], t1.Ticks, "#A6");
642                         t1 = DateTime.ParseExact ("Monday, 25 February 2002 04:25:13", "U", null);
643                         t1 = TimeZone.CurrentTimeZone.ToUniversalTime(t1);
644                         Assert.AreEqual (2002, t1.Year, "#A7");
645                         Assert.AreEqual (02, t1.Month, "#A8");
646                         Assert.AreEqual (25, t1.Day, "#A9");
647                         Assert.AreEqual (04, t1.Hour, "#A10");
648                         Assert.AreEqual (25, t1.Minute, "#A11");
649                         Assert.AreEqual (13, t1.Second, "#A12");
650                         t1 = DateTime.ParseExact ("Monday, 25 February 2002 04:25:13", "U", null);
651                         Assert.AreEqual ("Monday, 25 February 2002 04:25:13", t1.ToString ("U"), "#A13");
652
653                         DateTime t2 = new DateTime (DateTime.Today.Year, 2, 25);
654                         t1 = DateTime.ParseExact ("February 25", "m", null);
655                         Assert.AreEqual (t2.Ticks, t1.Ticks, "#B1");
656
657                         t2 = new DateTime (DateTime.Today.Year, 2, 25);
658                         t1 = DateTime.ParseExact ("February 25", "M", null);
659                         Assert.AreEqual (t2.Ticks, t1.Ticks, "#B2");
660
661                         t1 = DateTime.ParseExact ("Mon, 25 Feb 2002 04:25:13 GMT", "r", null);
662                         Assert.AreEqual (2002, t1.Year, "#C1");
663                         Assert.AreEqual (02, t1.Month, "#C2");
664                         Assert.AreEqual (25, t1.Day, "#C3");
665                         Assert.AreEqual (04, t1.Hour, "#C4");
666                         Assert.AreEqual (25, t1.Minute, "#C5");
667                         Assert.AreEqual (13, t1.Second, "#C6");
668
669                         t1 = DateTime.ParseExact ("Mon, 25 Feb 2002 04:25:13 GMT", "R", null);
670                         Assert.AreEqual (2002, t1.Year, "#D1");
671                         Assert.AreEqual (02, t1.Month, "#D2");
672                         Assert.AreEqual (25, t1.Day, "#D3");
673                         Assert.AreEqual (04, t1.Hour, "#D4");
674                         Assert.AreEqual (25, t1.Minute, "#D5");
675                         Assert.AreEqual (13, t1.Second, "#D6");
676
677                         t1 = DateTime.ParseExact ("2002-02-25T05:25:13", "s", null);
678                         Assert.AreEqual (myTicks [4], t1.Ticks, "#E1");
679
680                         t2 = DateTime.Today + new TimeSpan (5,25,0);
681                         t1 = DateTime.ParseExact ("05:25", "t", null);
682                         Assert.AreEqual (t2.Ticks, t1.Ticks, "#E2");
683
684                         t2 = DateTime.Today + new TimeSpan (5,25,13);
685                         t1 = DateTime.ParseExact ("05:25:13", "T", null);
686                         Assert.AreEqual (t2.Ticks, t1.Ticks, "#E3");
687
688                         t2 = new DateTime (2002, 2, 1);
689                         t1 = DateTime.ParseExact ("2002 February", "y", null);
690                         Assert.AreEqual (t2.Ticks, t1.Ticks, "#E4");
691
692                         t2 = new DateTime (2002, 2, 1);
693                         t1 = DateTime.ParseExact ("2002 February", "Y", null);
694                         Assert.AreEqual (t2.Ticks, t1.Ticks, "#E5");
695
696                         // Custom patterns
697                         t2 = new DateTime (DateTime.Now.Year, 1, 25);
698                         t1 = DateTime.ParseExact ("25", "%d", null);
699                         Assert.AreEqual (t2.Ticks, t1.Ticks, "#F1");
700                         t1 = DateTime.ParseExact ("25", "dd", null);
701                         Assert.AreEqual (t2.Ticks, t1.Ticks, "#F2");
702
703                         t2 = new DateTime (DateTime.Today.Year, 2, 1);
704                         t1 = DateTime.ParseExact ("2", "%M", null);
705                         Assert.AreEqual (t2.Ticks, t1.Ticks, "#G1");
706                         t1 = DateTime.ParseExact ("02", "MM", null);
707                         Assert.AreEqual (t2.Ticks, t1.Ticks, "#G2");
708                         t1 = DateTime.ParseExact ("Feb", "MMM", null);
709                         Assert.AreEqual (t2.Ticks, t1.Ticks, "#G3");
710                         t1 = DateTime.ParseExact ("February", "MMMM", null);
711                         Assert.AreEqual (t2.Ticks, t1.Ticks, "#G4");
712
713                         t2 = new DateTime (2005, 1, 1);
714                         t1 = DateTime.ParseExact ("5", "%y", null);
715                         Assert.AreEqual (t2.Ticks, t1.Ticks, "#G5");
716                         t1 = DateTime.ParseExact ("05", "yy", null);
717                         Assert.AreEqual (t2.Ticks, t1.Ticks, "#G6");
718                         t1 = DateTime.ParseExact ("2005", "yyyy", null);
719                         Assert.AreEqual (t2.Ticks, t1.Ticks, "#G7");
720
721                         t2 = DateTime.Today + new TimeSpan (5, 0, 0);
722                         t1 = DateTime.ParseExact ("5A", "ht", null);
723                         Assert.AreEqual (t2.Ticks, t1.Ticks, "#G8");
724                         t1 = DateTime.ParseExact ("05A", "hht", null);
725                         Assert.AreEqual (t2.Ticks, t1.Ticks, "#G9");
726
727                         t2 = DateTime.Today + new TimeSpan (15, 0, 0);
728                         t1 = DateTime.ParseExact ("3P", "ht", null);
729                         Assert.AreEqual (t2.Ticks, t1.Ticks, "#G10");
730                         t1 = DateTime.ParseExact ("03P", "hht", null);
731                         Assert.AreEqual (t2.Ticks, t1.Ticks, "#G11");
732
733                         t2 = DateTime.Today + new TimeSpan (5, 0, 0);
734                         t1 = DateTime.ParseExact ("5", "%H", null);
735                         Assert.AreEqual (t2.Ticks, t1.Ticks, "#G12");
736
737                         t2 = DateTime.Today + new TimeSpan (15, 0, 0);
738                         t1 = DateTime.ParseExact ("15", "%H", null);
739                         Assert.AreEqual (t2.Ticks, t1.Ticks, "#G13");
740                         t1 = DateTime.ParseExact ("15", "HH", null);
741                         Assert.AreEqual (t2.Ticks, t1.Ticks, "#G14");
742
743                         // Time zones
744 #if false
745                         // Fails durring DST for msft and mono
746                         t2 = DateTime.Today + new TimeSpan (17, 18, 0);
747                         t1 = DateTime.ParseExact ("11:18AM -5", "h:mmtt z", null);
748                         t1 = TimeZone.CurrentTimeZone.ToUniversalTime(t1);
749                         if (!TimeZone.CurrentTimeZone.IsDaylightSavingTime(t1))
750                                 t1 += new TimeSpan(1, 0, 0);
751                         Assert.AreEqual (t2.Ticks, t1.Ticks, "#H1");
752                         
753                         t1 = DateTime.ParseExact ("11:18AM -05:00", "h:mmtt zzz", null);
754                         t1 = TimeZone.CurrentTimeZone.ToUniversalTime(t1);
755                         if (!TimeZone.CurrentTimeZone.IsDaylightSavingTime(t1))
756                                 t1 += new TimeSpan(1, 0, 0);
757                         Assert.AreEqual (t2.Ticks, t1.Ticks, "#H2");
758
759                         t1 = DateTime.ParseExact ("7:18PM +03", "h:mmtt zz", null);
760                         t1 = TimeZone.CurrentTimeZone.ToUniversalTime(t1);
761                         if (!TimeZone.CurrentTimeZone.IsDaylightSavingTime(t1))
762                                 t1 += new TimeSpan(1, 0, 0);
763                         Assert.AreEqual (t2.Ticks, t1.Ticks, "#H3");
764
765                         t1 = DateTime.ParseExact ("7:48PM +03:30", "h:mmtt zzz", null);
766                         t1 = TimeZone.CurrentTimeZone.ToUniversalTime(t1);
767                         if (!TimeZone.CurrentTimeZone.IsDaylightSavingTime(t1))
768                                 t1 += new TimeSpan(1, 0, 0);
769                         Assert.AreEqual (t2.Ticks, t1.Ticks, "#H4");
770 #endif
771
772                         // Options
773                         t2 = DateTime.Today + new TimeSpan (16, 18, 0);
774                         t1 = DateTime.ParseExact ("11:18AM -5", "h:mmtt z",
775                                                   null, DateTimeStyles.AdjustToUniversal);
776                         Assert.AreEqual (t2.Ticks, t1.Ticks, "#I1");
777
778                         t1 = DateTime.ParseExact ("Monday, 25 February 2002 05:25:13", "F",
779                                                   null, DateTimeStyles.AdjustToUniversal);
780                         Assert.AreEqual (myTicks [4], t1.Ticks, "#I2");
781                         t1 = DateTime.ParseExact ("Monday, 25 February 2002 05:25:13",
782                                                   "dddd, dd MMMM yyyy HH:mm:ss",
783                                                   null, DateTimeStyles.AdjustToUniversal);
784                         Assert.AreEqual (myTicks [4], t1.Ticks, "#I3");
785
786                         t1 = DateTime.ParseExact ("02/25/2002", "d", null,
787                                                   DateTimeStyles.AllowWhiteSpaces);
788                         Assert.AreEqual (myTicks [0], t1.Ticks, "#I4");
789
790                         t1 = DateTime.ParseExact ("    02/25/2002", "d", null,
791                                                   DateTimeStyles.AllowLeadingWhite);
792                         Assert.AreEqual (myTicks [0], t1.Ticks, "#I5");
793
794                         t1 = DateTime.ParseExact ("02/25/2002    ", "d", null,
795                                                   DateTimeStyles.AllowTrailingWhite);
796                         Assert.AreEqual (myTicks [0], t1.Ticks, "#I6");
797
798                         t1 = DateTime.ParseExact ("  02 / 25 / 2002    ", "d", null,
799                                                   DateTimeStyles.AllowWhiteSpaces);
800                         Assert.AreEqual (myTicks [0], t1.Ticks, "#I7");
801
802                         // Multi Custom Patterns
803                         string rfc1123_date = "r";
804                         string rfc850_date = "dddd, dd'-'MMM'-'yy HH':'mm':'ss 'GMT'";
805                         string asctime_date = "ddd MMM d HH':'mm':'ss yyyy";
806                         string [] formats = new string [] {rfc1123_date, rfc850_date, asctime_date};
807                         CultureInfo enUS = new CultureInfo("en-US", false);
808                         t1 = DateTime.ParseExact ("Sun, 06 Nov 1994 08:49:37 GMT", formats[0], enUS, 
809                                                 DateTimeStyles.AllowWhiteSpaces);
810                         Assert.AreEqual (myTicks [6], t1.Ticks, "#J1");
811                         t1 = DateTime.ParseExact ("Sunday, 06-Nov-94 08:49:37 GMT", formats[1], enUS, 
812                                                 DateTimeStyles.AllowWhiteSpaces);
813                         Assert.AreEqual (myTicks [6], t1.Ticks, "#J2");
814                         t1 = DateTime.ParseExact ("Sun Nov  6 08:49:37 1994", formats[2], enUS, 
815                                                 DateTimeStyles.AllowWhiteSpaces);
816                         Assert.AreEqual (myTicks [6], t1.Ticks, "#J3");
817                         t1 = DateTime.ParseExact ("Sun, 06 Nov 1994 08:49:37 GMT", formats, enUS, 
818                                                 DateTimeStyles.AllowWhiteSpaces);
819                         Assert.AreEqual (myTicks [6], t1.Ticks, "#J4");
820                         t1 = DateTime.ParseExact ("Sunday, 06-Nov-94 08:49:37 GMT", formats, enUS, 
821                                                 DateTimeStyles.AllowWhiteSpaces);
822                         Assert.AreEqual (myTicks [6], t1.Ticks, "#J5");
823                         t1 = DateTime.ParseExact ("Sun Nov  6 08:49:37 1994", formats, enUS, 
824                                                 DateTimeStyles.AllowWhiteSpaces);
825                         Assert.AreEqual (myTicks [6], t1.Ticks, "#J6");
826                         t1 = DateTime.ParseExact ("Monday, 25 February 2002 05:25:13",
827                                                 "ddddddd, dd MMMMMMM yyyy HHHHH:mmmmm:sssss",
828                                                 null, DateTimeStyles.AdjustToUniversal);
829                         Assert.AreEqual (myTicks[4], t1.Ticks, "#J7");
830                 
831                         // Bug 52274
832                         t1 = DateTime.ParseExact ("--12--", "--MM--" , null);
833                         Assert.AreEqual (12, t1.Month, "#K1");
834                         t1=DateTime.ParseExact ("--12-24", "--MM-dd" , null);
835                         Assert.AreEqual (24, t1.Day, "#K2");
836                         Assert.AreEqual (12, t1.Month, "#K3");
837                         t1=DateTime.ParseExact ("---24", "---dd" , null);
838                         Assert.AreEqual (24, t1.Day, "#K4");
839
840                         // Bug 63376
841                         t1 = DateTime.ParseExact ("18Aug2004 12:33:00", "ddMMMyyyy hh:mm:ss", new CultureInfo ("en-US"));
842                         Assert.AreEqual (0, t1.Hour, "hh allows 12, though it's useless");
843
844                         // Bug 74775
845                         DateTime.ParseExact ("Tue, 12 Apr 2005 10:10:04 +0100",
846                                 "Tue, 12 Apr 2005 10:10:04 +0100", enUS);
847                         try {
848                                 DateTime.ParseExact ("Tue, 12 Apr 2005 10:10:04 +00000",
849                                         "ddd, dd MMM yyyy HH':'mm':'ss zzz", enUS);
850                                 Assert.Fail ("#L1");
851                         } catch (FormatException) {
852                         }
853
854                         // Bug #75213 : literal escaping.
855                         t1 = DateTime.ParseExact ("20050707132527Z",
856                                 "yyyyMMddHHmmss\\Z", CultureInfo.InvariantCulture);
857                         Assert.AreEqual (632563395270000000, t1.Ticks, "#L2");
858
859                         // XAttributeTest.CastDateTimeOffsets():#6b
860                         // It is said broken, probably due to timezone difference.
861                         //t1 = DateTime.ParseExact ("2039-10-31T12:34:56.7552+00:00", "yyyy-MM-ddTHH:mm:ss.FFFFFFFzzz",
862                         //                        CultureInfo.InvariantCulture, DateTimeStyles.AssumeUniversal);
863                         //Assert.AreEqual (643393064967552000, t1.Ticks, "#M");
864                 }
865
866                 [Test]
867                 public void TestParse2 ()
868                 {
869                         DateTime t1 = DateTime.Parse ("Mon, 25 Feb 2002 04:25:13 GMT");
870                         t1 = TimeZone.CurrentTimeZone.ToUniversalTime(t1);
871                         Assert.AreEqual (4, t1.Hour);
872                 }
873
874                 [Test]
875                 public void TestUtcOffset_Novell710512 ()
876                 {
877                         // test for bug Novell #710512
878                         if (TimeZoneInfo.Local.BaseUtcOffset == TimeSpan.Zero)
879                                 Assert.Ignore("Test doesn't apply with current time zone");
880
881                         var localTime = DateTime.Parse ("Mon, 25 Feb 2002 04:25:13 GMT");
882                         var utcTime = TimeZone.CurrentTimeZone.ToUniversalTime(localTime);
883                         Assert.AreEqual (TimeSpan.Zero, TimeZone.CurrentTimeZone.GetUtcOffset (utcTime));
884                 }
885
886                 [Test]
887                 public void TestParseDateFirst ()
888                 {
889                         // Standard patterns
890                         CultureInfo USCultureInfo = new CultureInfo("en-US");
891                         DateTime t1 = DateTime.Parse ("02/25/2002", USCultureInfo);
892                         Assert.AreEqual (myTicks [0], t1.Ticks, "#A1");
893                         t1 = DateTime.Parse ("2002-02-25", USCultureInfo);
894                         Assert.AreEqual (myTicks [0], t1.Ticks, "#A2");
895                         t1 = DateTime.Parse ("Monday, 25 February 2002");
896                         Assert.AreEqual (myTicks [0], t1.Ticks, "#A3");
897                         t1 = DateTime.Parse ("Monday, 25 February 2002 05:25");
898                         Assert.AreEqual (myTicks [3], t1.Ticks, "#A4");
899                         t1 = DateTime.Parse ("Monday, 25 February 2002 05:25:13");
900                         Assert.AreEqual (myTicks [4], t1.Ticks, "#A5");
901                         t1 = DateTime.Parse ("02/25/2002 05:25", USCultureInfo);
902                         Assert.AreEqual (myTicks [3], t1.Ticks, "#A6");
903                         t1 = DateTime.Parse ("02/25/2002 05:25:13", USCultureInfo);
904                         Assert.AreEqual (myTicks [4], t1.Ticks, "#A7");
905                         t1 = DateTime.Parse ("2002-02-25 04:25:13Z");
906                         t1 = TimeZone.CurrentTimeZone.ToUniversalTime(t1);
907                         Assert.AreEqual (2002, t1.Year, "#A8");
908                         Assert.AreEqual (02, t1.Month, "#A9");
909                         Assert.AreEqual (25, t1.Day, "#A10");
910                         Assert.AreEqual (04, t1.Hour, "#A11");
911                         Assert.AreEqual (25, t1.Minute, "#A12");
912                         Assert.AreEqual (13, t1.Second, "#A13");
913                         t1 = DateTime.Parse ("Mon,02/25/2002", USCultureInfo);
914                         Assert.AreEqual (myTicks [0], t1.Ticks, "#A14");
915                         DateTime t2 = new DateTime (1999, 1, 2, 0, 3, 4);
916                         t1 = DateTime.Parse (t2.ToLongTimeString ());
917                         Assert.AreEqual (0, t1.Hour, "#A14");
918
919                         // parsed as UTC string
920                         t1 = DateTime.Parse ("Mon, 25 Feb 2002 04:25:13 GMT");
921                         t1 = TimeZone.CurrentTimeZone.ToUniversalTime (t1);
922                         Assert.AreEqual (2002, t1.Year, "#C1");
923                         Assert.AreEqual (02, t1.Month, "#C2");
924                         Assert.AreEqual (25, t1.Day, "#C3");
925                         Assert.AreEqual (4, t1.Hour, "#C4");
926                         Assert.AreEqual (25, t1.Minute, "#C5");
927                         Assert.AreEqual (13, t1.Second, "#C6");
928
929                         // Some date 'T' time formats
930                         t1 = DateTime.Parse ("2002-02-25T05:25");
931                         Assert.AreEqual (myTicks [3], t1.Ticks, "#D1");
932
933                         t1 = DateTime.Parse ("2002-02-25T05:25:13");
934                         Assert.AreEqual (myTicks [4], t1.Ticks, "#D1");
935                         t1 = DateTime.Parse ("2002-02-25T05:25:13.008");
936                         Assert.AreEqual (myTicks [2], t1.Ticks, "#D1");
937                         t1 = DateTime.Parse ("02-2002-25T05:25:13");
938                         Assert.AreEqual (myTicks [4], t1.Ticks, "#D1");
939
940                         // Day month
941                         t2 = new DateTime (DateTime.Today.Year, 2, 25);
942                         t1 = DateTime.Parse ("February 25", USCultureInfo);
943                         Assert.AreEqual (t2.Ticks, t1.Ticks, "#B1");
944
945                         t2 = new DateTime (DateTime.Today.Year, 2, 8);
946                         t1 = DateTime.Parse ("February 08", USCultureInfo);
947                         Assert.AreEqual (t2.Ticks, t1.Ticks, "#B2");
948
949                         t2 = new DateTime (DateTime.Today.Year, 2, 8);
950                         t1 = DateTime.Parse ("February 8", USCultureInfo);
951                         Assert.AreEqual (t2.Ticks, t1.Ticks, "#D6");
952
953                         // Month year
954                         t2 = new DateTime (2002, 2, 1);
955                         t1 = DateTime.Parse ("2002 February");
956                         Assert.AreEqual (t2.Ticks, t1.Ticks, "#D4");
957
958                         t2 = new DateTime (2002, 2, 1);
959                         t1 = DateTime.Parse ("2002 February", new CultureInfo ("ja-JP"));
960                         Assert.AreEqual (t2.Ticks, t1.Ticks, "#D5");
961
962                         // bug #72132
963                         t2 = new DateTime (2002, 2, 25, 5, 25, 22);
964                         t1 = DateTime.Parse ("Monday, 25 February 2002 05:25:22",
965                                 new CultureInfo ("hi-IN"));
966                         Assert.AreEqual (t2.Ticks, t1.Ticks, "#D7");
967                         t2 = new DateTime (2002, 2, 25, 5, 25, 0);
968                         t1 = DateTime.Parse ("Monday, 25 February 2002 05:25",
969                                 new CultureInfo ("hi-IN"));
970                         Assert.AreEqual (t2.Ticks, t1.Ticks, "#D8");
971
972                         // MM-yyyy-dd + different time formats
973                         t1 = DateTime.Parse ("02-2002-25 05:25", USCultureInfo);
974                         Assert.AreEqual (myTicks[3], t1.Ticks, "#E1");
975                         t1 = DateTime.Parse ("02-2002-25 05:25:13", USCultureInfo);
976                         Assert.AreEqual (myTicks[4], t1.Ticks, "#E1");
977                         t1 = DateTime.Parse ("02-2002-25 05:25:13 Mon", USCultureInfo);
978                         Assert.AreEqual (myTicks[4], t1.Ticks, "#E2");
979                         t1 = DateTime.Parse ("02-2002-25 05:25:13 Monday", USCultureInfo);
980                         Assert.AreEqual (myTicks[4], t1.Ticks, "#E3");
981                         t1 = DateTime.Parse ("02-2002-25 05:25:13.008", USCultureInfo);
982                         Assert.AreEqual (myTicks[2], t1.Ticks, "#E4");
983
984                         // Formats with timezone
985                         long offset = TimeZone.CurrentTimeZone.GetUtcOffset(t1).Ticks;
986                         long hourTicks = 36000000000L;
987                         long halfHourTicks = hourTicks / 2;
988                         t1 = DateTime.Parse ("02-2002-25 05:25+01", USCultureInfo);
989                         Assert.AreEqual (myTicks[3], t1.Ticks + hourTicks - offset, "#F1");
990                         t1 = DateTime.Parse ("02-2002-25 05:25-01", USCultureInfo);
991                         Assert.AreEqual (myTicks[3], t1.Ticks - hourTicks - offset, "#F2");
992                         t1 = DateTime.Parse ("02-2002-25 05:25+00:30", USCultureInfo);
993                         Assert.AreEqual (myTicks[3], t1.Ticks + hourTicks/2 - offset, "#F3");
994                         t1 = DateTime.Parse ("02-2002-25 05:25:13+02", USCultureInfo);
995                         Assert.AreEqual (myTicks[4], t1.Ticks + 2*hourTicks - offset, "#F4");
996
997                         // NET 1.0 doesn't accept second fractions and time zone.
998                         t1 = DateTime.Parse ("2002-02-25 05:25:13.008-02");
999                         Assert.AreEqual (myTicks[2], t1.Ticks - 2*hourTicks - offset, "#F5");
1000                         // NET 1.0 doesn't parse well time zone with AM afterwards.
1001                         t1 = DateTime.Parse ("02-25-2002 05:25:13-02 AM", USCultureInfo);
1002                         Assert.AreEqual (myTicks[4], t1.Ticks - 2*hourTicks - offset, "#F6");
1003                         t1 = DateTime.Parse ("25 Feb 2002 05:25:13-02 AM", USCultureInfo);
1004                         Assert.AreEqual (myTicks[4], t1.Ticks - 2*hourTicks - offset, "#F6");
1005                 }
1006
1007                 [Test]
1008                 public void TestParseTimeFirst ()
1009                 {
1010                         CultureInfo USCultureInfo = new CultureInfo("en-US");
1011
1012                         // Hour only patterns
1013                         DateTime t2 = DateTime.Today + new TimeSpan (5,25,0);
1014                         DateTime t1 = DateTime.Parse ("05:25");
1015                         Assert.AreEqual (t2.Ticks, t1.Ticks, "#C1");
1016                         t2 = DateTime.Today + new TimeSpan (5,25,13);
1017                         t1 = DateTime.Parse ("05:25:13");
1018                         Assert.AreEqual (t2.Ticks, t1.Ticks, "#B2");
1019
1020                         // Test with different date formats
1021                         t1 = DateTime.Parse ("05:25 02/25/2002", USCultureInfo);
1022                         Assert.AreEqual (myTicks[3], t1.Ticks, "#B1");
1023                         t1 = DateTime.Parse ("05:25:13 2002-02-25");
1024                         Assert.AreEqual (myTicks[4], t1.Ticks, "#B2");
1025                         t1 = DateTime.Parse ("05:25:13.008 02-2002-25");
1026                         Assert.AreEqual (myTicks[2], t1.Ticks, "#B3");
1027                         t1 = DateTime.Parse ("05:25:13.008 Feb 25 2002");
1028                         Assert.AreEqual (myTicks[2], t1.Ticks, "#B4");
1029                         t1 = DateTime.Parse ("05:25:13.008 25 Feb 2002");
1030                         Assert.AreEqual (myTicks[2], t1.Ticks, "#B5");
1031
1032                         // Add AM and day of the week
1033                         t1 = DateTime.Parse ("AM 05:25:13 2002-02-25");
1034                         Assert.AreEqual (myTicks[4], t1.Ticks, "#C1");
1035                         t1 = DateTime.Parse ("Monday05:25 02/25/2002", USCultureInfo);
1036                         Assert.AreEqual (myTicks[3], t1.Ticks, "#C2");
1037                         t1 = DateTime.Parse ("Mon 05:25 AM 02/25/2002", USCultureInfo);
1038                         Assert.AreEqual (myTicks[3], t1.Ticks, "#C3");
1039                         t1 = DateTime.Parse ("AM 05:25 Monday, 02/25/2002", USCultureInfo);
1040                         Assert.AreEqual (myTicks[3], t1.Ticks, "#C4");
1041                         t1 = DateTime.Parse ("05:25 02/25/2002 Monday", USCultureInfo);
1042                         Assert.AreEqual (myTicks[3], t1.Ticks, "#C5");
1043                         t1 = DateTime.Parse ("PM 03:25:13.008 02-2002-25");
1044                         Assert.AreEqual (myTicks[1], t1.Ticks, "#C6");
1045
1046                         // ASP.NET QuickStarts
1047                         t2 = new DateTime (2002, 10, 7, 15, 6, 0);
1048                         t1 = DateTime.Parse ("3:06 PM 10/7/2002", USCultureInfo);
1049                         Assert.AreEqual (t2.Ticks, t1.Ticks, "#D1");
1050                         t2 = new DateTime (2002, 10, 7, 15, 6, 0);
1051                         t1 = DateTime.Parse ("3:06 pm 10/7/2002", USCultureInfo);
1052                         Assert.AreEqual (t2.Ticks, t1.Ticks, "#D2");
1053                         t2 = new DateTime (2002, 10, 7, 3, 6, 0);
1054                         t1 = DateTime.Parse ("3:06 AM 10/7/2002", USCultureInfo);
1055                         Assert.AreEqual (t2.Ticks, t1.Ticks, "#D3");
1056                         t2 = new DateTime (2002, 10, 7, 3, 6, 0);
1057                         t1 = DateTime.Parse ("3:06 am 10/7/2002", USCultureInfo);
1058                         Assert.AreEqual (t2.Ticks, t1.Ticks, "#D4");
1059                 }
1060
1061                 [Test]
1062                 public void TestParseWithDifferentShortDatePatterns ()
1063                 {
1064                         CultureInfo cultureInfo = new CultureInfo("en-US");
1065                         DateTimeFormatInfo dateFormatInfo = cultureInfo.DateTimeFormat;
1066                         DateTime t1 = DateTime.Parse ("02/01/2003", cultureInfo);
1067                         Assert.AreEqual (myTicks[7], t1.Ticks, "#A1");
1068
1069                         // Day, month year behaviour
1070                         dateFormatInfo.ShortDatePattern = "dd/MM/yyyy";
1071                         t1 = DateTime.Parse ("01/02/03", cultureInfo);
1072                         Assert.AreEqual (myTicks[7], t1.Ticks, "#B1");
1073                         t1 = DateTime.Parse ("01/02/2003", cultureInfo);
1074                         Assert.AreEqual (myTicks[7], t1.Ticks, "#B2");
1075                         t1 = DateTime.Parse ("2003/02/01", cultureInfo);
1076                         Assert.AreEqual (myTicks[7], t1.Ticks, "#B3");
1077                         t1 = DateTime.Parse ("01/Feb/03", cultureInfo);
1078                         Assert.AreEqual (myTicks[7], t1.Ticks, "#B4");
1079                         t1 = DateTime.Parse ("Feb/01/03", cultureInfo);
1080                         Assert.AreEqual (myTicks[7], t1.Ticks, "#B5");
1081
1082                         // Month, day year behaviour
1083                         dateFormatInfo.ShortDatePattern = "MM/dd/yyyy";
1084                         t1 = DateTime.Parse ("02/01/03", cultureInfo);
1085                         Assert.AreEqual (myTicks[7], t1.Ticks, "#C1");
1086                         t1 = DateTime.Parse ("02/01/2003", cultureInfo);
1087                         Assert.AreEqual (myTicks[7], t1.Ticks, "#C2");
1088                         t1 = DateTime.Parse ("2003/02/01", cultureInfo);
1089                         Assert.AreEqual (myTicks[7], t1.Ticks, "#C3");
1090                         t1 = DateTime.Parse ("01/Feb/03", cultureInfo);
1091                         Assert.AreEqual (myTicks[7], t1.Ticks, "#C4");
1092                         t1 = DateTime.Parse ("Feb/01/03", cultureInfo);
1093                         Assert.AreEqual (myTicks[7], t1.Ticks, "#C5");
1094
1095                         // Year, month day behaviour
1096                         dateFormatInfo.ShortDatePattern = "yyyy/MM/dd";
1097                         t1 = DateTime.Parse ("03/02/01", cultureInfo);
1098                         Assert.AreEqual (myTicks[7], t1.Ticks, "#D1");
1099                         t1 = DateTime.Parse ("02/01/2003", cultureInfo);
1100                         Assert.AreEqual (myTicks[7], t1.Ticks, "#D2");
1101                         t1 = DateTime.Parse ("2003/02/01", cultureInfo);
1102                         Assert.AreEqual (myTicks[7], t1.Ticks, "#D3");
1103                         t1 = DateTime.Parse ("03/Feb/01", cultureInfo);
1104                         Assert.AreEqual (myTicks[7], t1.Ticks, "#D4");
1105                         t1 = DateTime.Parse ("Feb/03/01", cultureInfo);
1106                         Assert.AreEqual (myTicks[7], t1.Ticks, "#D5");
1107
1108                         // Year, day month behaviour
1109                         // Note that no culture I am aware of has this pattern, and indeed
1110                         dateFormatInfo.ShortDatePattern = "yyyy/dd/MM";
1111                         t1 = DateTime.Parse ("03/01/02", cultureInfo);
1112                         Assert.AreEqual (myTicks[7], t1.Ticks, "#E1");
1113                         t1 = DateTime.Parse ("01/02/2003", cultureInfo);
1114                         Assert.AreEqual (myTicks[7], t1.Ticks, "#E2");
1115
1116                         t1 = DateTime.Parse ("2003/01/02", cultureInfo);
1117                         Assert.AreEqual (myTicks[7], t1.Ticks, "#E3");
1118
1119                         // For some reason the following throws an exception on .Net
1120                         // t1 = DateTime.Parse ("03/Feb/01", cultureInfo);
1121                         // Assert.AreEqual (myTicks[7], t1.Ticks, "#E4");
1122                         // t1 = DateTime.Parse ("03/01/Feb", cultureInfo);
1123                         // Assert.AreEqual (myTicks[7], t1.Ticks, "#E5");
1124                         // t1 = DateTime.Parse ("Feb/01/03", cultureInfo);
1125                         // Assert.AreEqual (myTicks[7], t1.Ticks, "#E6");
1126                 }
1127
1128                 [Test]
1129                 public void TestParseWithDifferentMonthDayPatterns ()
1130                 {
1131                         CultureInfo cultureInfo = new CultureInfo("en-US");
1132                         DateTimeFormatInfo dateFormatInfo = cultureInfo.DateTimeFormat;
1133                         DateTime t1 = DateTime.Parse ("Feb 03", cultureInfo);
1134                         Assert.AreEqual (2, t1.Month, "#A1");
1135                         Assert.AreEqual (3, t1.Day, "#A2");
1136
1137                         // Day month behaviour
1138                         dateFormatInfo.MonthDayPattern = "dd/MM";
1139
1140                         t1 = DateTime.Parse ("Feb 03", cultureInfo);
1141                         Assert.AreEqual (2, t1.Month, "#B1");
1142                         Assert.AreEqual (1, t1.Day, "#B2");
1143                         Assert.AreEqual (2003, t1.Year, "#B3");
1144
1145                         t1 = DateTime.Parse ("03/02", cultureInfo);
1146                         Assert.AreEqual (2, t1.Month, "#B6");
1147                         Assert.AreEqual (3, t1.Day, "#B7");
1148                         t1 = DateTime.Parse ("03 Feb", cultureInfo);
1149                         Assert.AreEqual (2, t1.Month, "#B8");
1150                         Assert.AreEqual (3, t1.Day, "#B9");
1151
1152                         // Month day behaviour
1153                         dateFormatInfo.MonthDayPattern = "MM/dd";
1154                         t1 = DateTime.Parse ("Feb 03", cultureInfo);
1155                         Assert.AreEqual (2, t1.Month, "#C1");
1156                         Assert.AreEqual (3, t1.Day, "#C2");
1157                         t1 = DateTime.Parse ("02/03", cultureInfo);
1158                         Assert.AreEqual (2, t1.Month, "#C3");
1159                         Assert.AreEqual (3, t1.Day, "#C4");
1160                         t1 = DateTime.Parse ("03 Feb", cultureInfo);
1161                         Assert.AreEqual (2, t1.Month, "#C5");
1162                         Assert.AreEqual (3, t1.Day, "#C6");
1163                 }
1164
1165                 [Test]
1166                 public void TestParse3 ()
1167                 {
1168                         string s = "Wednesday, 09 June 2004";
1169                         DateTime.ParseExact (s, "dddd, dd MMMM yyyy", CultureInfo.InvariantCulture);
1170                         try {
1171                                 DateTime.ParseExact (s, "dddd, dd MMMM yyyy", new CultureInfo ("ja-JP"));
1172                                 Assert.Fail ("ja-JP culture does not support format \"dddd, dd MMMM yyyy\"");
1173                         } catch (FormatException) {
1174                         }
1175
1176                         // Ok, now we can assume ParseExact() works expectedly.
1177
1178                         DateTime.Parse (s, CultureInfo.InvariantCulture, DateTimeStyles.AllowWhiteSpaces);
1179                         DateTime.Parse (s, new CultureInfo ("ja-JP"), DateTimeStyles.AllowWhiteSpaces);
1180                         //DateTime.Parse (s, null); currently am not sure if it works for _every_ culture.
1181                 }
1182
1183
1184                 [Test] // bug #74936
1185                 public void TestParse4 ()
1186                 {
1187                         try {
1188                                 DateTime.Parse("1");
1189                                 Assert.Fail ("#1");
1190                         } catch (FormatException) {
1191                         }
1192
1193                         try {
1194                                 DateTime.Parse("1000");
1195                                 Assert.Fail ("#2");
1196                         } catch (FormatException) {
1197                         }
1198
1199                         try {
1200                                 DateTime.Parse("8:");
1201                                 Assert.Fail ("#3");
1202                         } catch (FormatException) {
1203                         }
1204                 }
1205
1206                 [Test] // bug #71289
1207                 public void Parse_Bug71289a ()
1208                 {
1209                         DateTime.Parse ("Sat,,,,,, 01 Oct 1994 03:00:00", CultureInfo.InvariantCulture);
1210                 }
1211
1212                 [Test]
1213                 public void Parse_Bug71289b ()
1214                 {
1215                         // more example...
1216                         DateTime.Parse ("Sat,,, 01,,, Oct,,, ,,,1994 03:00:00", CultureInfo.InvariantCulture);
1217                 }
1218
1219                 [Test]
1220                 [ExpectedException (typeof (FormatException))]
1221                 public void Parse_CommaAfterHours ()
1222                 {
1223                         // ',' after 03 is not allowed.
1224                         DateTime.Parse ("Sat,,, 01,,, Oct,,, ,,,1994 03,:00:00", CultureInfo.InvariantCulture);
1225                 }
1226
1227                 [Test] // bug #72788
1228                 public void Parse_Bug72788 ()
1229                 {
1230                         DateTime dt = DateTime.Parse ("21/02/05", new CultureInfo ("fr-FR"));
1231                         Assert.AreEqual (2005, dt.Year, "#1");
1232                         Assert.AreEqual (02, dt.Month, "#2");
1233                         Assert.AreEqual (21, dt.Day, "#3");
1234                 }
1235
1236                 [Test] // bug #322510
1237                 public void Parse_HourDesignator ()
1238                 {
1239                         DateTime dt;
1240                         DateTime now = DateTime.Now;
1241
1242                         dt = DateTime.Parse ("12:00:00 AM", new CultureInfo ("en-US"));
1243                         Assert.AreEqual (now.Year, dt.Year, "#A1");
1244                         Assert.AreEqual (now.Month, dt.Month, "#A2");
1245                         Assert.AreEqual (now.Day, dt.Day, "#A3");
1246                         Assert.AreEqual (0, dt.Hour, "#A4");
1247                         Assert.AreEqual (0, dt.Minute, "#A5");
1248                         Assert.AreEqual (0, dt.Second, "#A6");
1249                         Assert.AreEqual (0, dt.Millisecond, "#A7");
1250
1251                         dt = DateTime.Parse ("12:00:00 PM", new CultureInfo ("en-US"));
1252                         Assert.AreEqual (now.Year, dt.Year, "#B1");
1253                         Assert.AreEqual (now.Month, dt.Month, "#B2");
1254                         Assert.AreEqual (now.Day, dt.Day, "#B3");
1255                         Assert.AreEqual (12, dt.Hour, "#B4");
1256                         Assert.AreEqual (0, dt.Minute, "#B5");
1257                         Assert.AreEqual (0, dt.Second, "#B6");
1258                         Assert.AreEqual (0, dt.Millisecond, "#B7");
1259                 }
1260
1261                 [Test]
1262                 public void Parse_Bug53023b ()
1263                 {
1264                         foreach (CultureInfo ci in CultureInfo.GetCultures (CultureTypes.SpecificCultures)) {
1265                                 try {
1266                                         DateTime.Parse ("01-Sep-05", ci);
1267
1268                                         // FIXME: Our UmAlQuraCalendar/HijriCalendar calendars support month days - 1 only (fail on last day in month)
1269                                         if (ci.Calendar is UmAlQuraCalendar || ci.Calendar is HijriCalendar)
1270                                                 continue;
1271
1272                                         DateTime.Parse ("4:35:35 AM", ci);
1273                                 } catch {
1274                                         Assert.Fail (ci.Name);
1275                                 }
1276                         }
1277                 }
1278
1279                 [Test]
1280                 [ExpectedException (typeof (FormatException))]
1281                 public void Parse_RequireSpaceSeparator ()
1282                 {
1283                         DateTime.Parse ("05:25:132002-02-25", CultureInfo.InvariantCulture);
1284                 }
1285
1286                 [Test]
1287                 [ExpectedException (typeof (FormatException))]
1288                 public void Parse_DontAccept2DigitsYears ()
1289                 {
1290                         // don't allow 2 digit years where we require 4.
1291                         DateTime.ParseExact ("05", "yyyy", CultureInfo.InvariantCulture);
1292                 }
1293
1294                 [Test]
1295                 [ExpectedException (typeof (FormatException))]
1296                 public void Parse_DontAcceptEmptyHours ()
1297                 {
1298                         DateTime.ParseExact (":05", "H:m", CultureInfo.InvariantCulture);
1299                 }
1300
1301                 [Test]
1302                 [ExpectedException (typeof (FormatException))]
1303                 public void Parse_DontAcceptEmptyMinutes ()
1304                 {
1305                         DateTime.ParseExact ("0::0", "H:m:s", CultureInfo.InvariantCulture);
1306                 }
1307
1308                 [Test]
1309                 public void ParseCOMDependentFormat ()
1310                 {
1311                         // Japanese format.
1312                         DateTime.Parse (String.Format (
1313                                 "{0}\u5E74{1}\u6708{2}\u65E5 {3}\u6642{4}\u5206{5}\u79D2",
1314                                 2006, 3, 1, 15, 32, 42), new CultureInfo (""));
1315
1316                         try {
1317                                 // incorrect year mark.
1318                                 DateTime.Parse (String.Format (
1319                                         "{0}\u4E00{1}\u6708{2}\u65E5 {3}\u6642{4}\u5206{5}\u79D2",
1320                                         2006, 3, 1, 15, 32, 42), new CultureInfo (""));
1321                                 Assert.Fail ();
1322                         } catch (FormatException) {
1323                         }
1324                 }
1325
1326                 [Test]
1327                 [ExpectedException(typeof (FormatException))]
1328                 public void ParseFormatException1 ()
1329                 {
1330                         // Following string is not a correct French date i.e.
1331                         // MM/dd/yyyy HH:mm:ss since it expects d/M/yyyy HH:mm:ss
1332                         // instead (however fr-FR accepts both MM/dd/yyyy and
1333                         // dd/MM/yyyy, which means that we can't just throw exceptions 
1334                         // on overflow).
1335                         String frDateTime = "11/13/2003 11:28:15";
1336                         IFormatProvider format = new CultureInfo("fr-FR", true);
1337                         DateTime t1 = DateTime.Parse(frDateTime, format);
1338                 }
1339         
1340                 [Test]
1341                 [ExpectedException(typeof (FormatException))]
1342                 public void ParseFormatExceptionForInvalidYear ()
1343                 {
1344                         // Bug #77633.  In .NET 1..1, the expected exception is ArgumentOutOfRangeException
1345                         // In .NET 2.0, the expected exception is FormatException
1346                         // build a string with the year of 5 digits
1347                         string s = "1/1/10000";
1348                         DateTime dt = DateTime.Parse (s);
1349                 }
1350         
1351                 [Test]
1352                 public void TestOA ()
1353                 {
1354                         double number=5000.41443;
1355                         DateTime d = DateTime.FromOADate(number);
1356                         DTAssertEquals (d, new DateTime(1913, 9, 8, 9, 56, 46, 0), Resolution.Second, "#1");
1357                         Assert.AreEqual (d.ToOADate(), number, "#2");
1358                 }
1359
1360                 [Test]
1361                 public void ParseAllowsQueerString ()
1362                 {
1363                         DateTime.Parse ("Sat,,,,,, 01 Oct 1994 03:00:00", CultureInfo.InvariantCulture);
1364                 }
1365
1366                 [Test]
1367                 public void ParseUtcNonUtc ()
1368                 {
1369                         Thread.CurrentThread.CurrentCulture = new CultureInfo ("es-ES");
1370
1371                         CultureInfo ci;
1372                         string s, s2, s3, d;
1373                         DateTime dt;
1374                         DateTimeFormatInfo dfi = DateTimeFormatInfo.InvariantInfo;
1375                         s = dfi.UniversalSortableDateTimePattern;
1376                         s2 = "r";
1377
1378                         s3 = "s";
1379
1380                         long tick1 = 631789220960000000; // 2003-01-23 12:34:56 as is
1381                         long tick2 = TimeZone.CurrentTimeZone.ToLocalTime (new DateTime (tick1)).Ticks; // adjusted to local time
1382
1383                         // invariant
1384                         ci = CultureInfo.InvariantCulture;
1385
1386                         d = "2003/01/23 12:34:56";
1387                         dt = DateTime.Parse (d, ci);
1388                         Assert.AreEqual (tick1, dt.Ticks, "#1:" + d);
1389
1390                         d = "2003/01/23 12:34:56 GMT";
1391                         dt = DateTime.Parse (d, ci);
1392                         Assert.AreEqual (tick2, dt.Ticks, "#2:" + d);
1393
1394                         d = "Thu, 23 Jan 2003 12:34:56 GMT";
1395                         dt = DateTime.ParseExact (d, s2, ci);
1396                         Assert.AreEqual (tick1, dt.Ticks, "#3:" + d);
1397
1398                         d = "2003-01-23 12:34:56Z";
1399                         dt = DateTime.ParseExact (d, s, ci);
1400                         Assert.AreEqual (tick1, dt.Ticks, "#4:" + d);
1401
1402                         d = "2003-01-23T12:34:56";
1403                         dt = DateTime.ParseExact (d, s3, ci);
1404                         Assert.AreEqual (tick1, dt.Ticks, "#5:" + d);
1405
1406                         // ja-JP ... it should be culture independent
1407                         ci = new CultureInfo ("ja-JP");
1408
1409                         d = "2003/01/23 12:34:56";
1410                         dt = DateTime.Parse (d, ci);
1411                         Assert.AreEqual (tick1, dt.Ticks, "#6:" + d);
1412
1413                         d = "2003/01/23 12:34:56 GMT";
1414                         dt = DateTime.Parse (d, ci);
1415                         Assert.AreEqual (tick2, dt.Ticks, "#7:" + d);
1416
1417                         d = "Thu, 23 Jan 2003 12:34:56 GMT";
1418                         dt = DateTime.ParseExact (d, s2, ci);
1419                         Assert.AreEqual (tick1, dt.Ticks, "#8:" + d);
1420
1421                         d = "2003-01-23 12:34:56Z";
1422                         dt = DateTime.ParseExact (d, s, ci);
1423                         Assert.AreEqual (tick1, dt.Ticks, "#9:" + d);
1424
1425                         d = "2003-01-23T12:34:56";
1426                         dt = DateTime.ParseExact (d, s3, ci);
1427                         Assert.AreEqual (tick1, dt.Ticks, "#10:" + d);
1428                 }
1429
1430                 [Test]
1431                 public void TimeZoneAdjustment ()
1432                 {
1433                         CultureInfo ci = Thread.CurrentThread.CurrentCulture;
1434                         try {
1435                                 Thread.CurrentThread.CurrentCulture = new CultureInfo ("en-US");
1436                                 DateTime d1 = DateTime.ParseExact ("2004/06/30", "yyyy/MM/dd", null);
1437                                 DateTime d2 = DateTime.ParseExact ("2004/06/30Z", "yyyy/MM/dd'Z'", null);
1438                                 DateTime d3 = DateTime.ParseExact ("Wed, 30 Jun 2004 00:00:00 GMT", "ddd, dd MMM yyyy HH':'mm':'ss 'GMT'", null);
1439                                 DateTime d4 = DateTime.ParseExact ("2004-06-30 00:00:00Z", "yyyy'-'MM'-'dd HH':'mm':'ss'Z'", null);
1440                                 StringWriter sw = new StringWriter ();
1441                                 sw.Write ("{0} {1}", d1.Ticks, d1);
1442                                 Assert.AreEqual ("632241504000000000 6/30/2004 12:00:00 AM", sw.ToString (), "#1");
1443                                 sw.GetStringBuilder ().Length = 0;
1444                                 sw.Write ("{0} {1}", d2.Ticks, d2);
1445                                 Assert.AreEqual ("632241504000000000 6/30/2004 12:00:00 AM", sw.ToString (), "#2");
1446                                 sw.GetStringBuilder ().Length = 0;
1447                                 sw.Write ("{0} {1}", d3.Ticks, d3);
1448                                 Assert.AreEqual ("632241504000000000 6/30/2004 12:00:00 AM", sw.ToString (), "#3");
1449                                 sw.GetStringBuilder ().Length = 0;
1450                                 sw.Write ("{0} {1}", d4.Ticks, d4);
1451                                 Assert.AreEqual ("632241504000000000 6/30/2004 12:00:00 AM", sw.ToString (), "#4");
1452                         } finally {
1453                                 Thread.CurrentThread.CurrentCulture = ci;
1454                         }
1455
1456                         // bug #76082
1457                         Assert.AreEqual (DateTime.MinValue, DateTime.ParseExact ("00010101T00:00:00",
1458                                         "yyyyMMdd'T'HH':'mm':'ss", DateTimeFormatInfo.InvariantInfo), "#5");
1459                 }
1460
1461                 [Test]
1462                 public void DateTimeStylesAdjustToUniversal ()
1463                 {
1464                         // bug #75995 : AdjustToUniversal
1465                         DateTime t1 = DateTime.Parse ("2005-09-05T22:29:00Z",
1466                                 CultureInfo.InvariantCulture,
1467                                 DateTimeStyles.AdjustToUniversal);
1468                         Assert.AreEqual ("2005-09-05 22:29:00Z", t1.ToString ("u"));
1469                 }
1470
1471                 [Test]
1472                 [ExpectedException (typeof (ArgumentException))]
1473                 public void FromOADate_Min () 
1474                 {
1475                         // minimum documented value isn't inclusive
1476                         DateTime.FromOADate (-657435.0d);
1477                 }
1478
1479                 [Test]
1480                 [ExpectedException (typeof (ArgumentException))]
1481                 public void FromOADate_Max () 
1482                 {
1483                         // maximum documented value isn't inclusive
1484                         DateTime.FromOADate (2958466.0d);
1485                 }
1486
1487                 [Test]
1488                 public void FromOADate ()
1489                 {
1490                         // Note: OA (OLE Automation) dates aren't timezone sensitive
1491                         Assert.AreEqual (599264352000000000, DateTime.FromOADate (0.0d).Ticks, "#1");
1492                         Assert.AreEqual (31242239136000000, DateTime.FromOADate (-657434.999d).Ticks, "#2");
1493                         Assert.AreEqual (3155378975136000000, DateTime.FromOADate (2958465.999d).Ticks, "#3");
1494                 }
1495
1496                 [Test]
1497                 public void ToOADate ()
1498                 {
1499                         // Note: OA (OLE Automation) dates aren't timezone sensitive
1500                         DateTime d = new DateTime (0);
1501                         Assert.AreEqual (0.0d, d.ToOADate (), "#1");
1502                         d = new DateTime (599264352000000000);
1503                         Assert.AreEqual (0.0d, d.ToOADate (), "#2");
1504                         d = new DateTime (31242239136000000);
1505                         Assert.AreEqual (-657434.999d, d.ToOADate (), "#3");
1506                         d = new DateTime (3155378975136000000);
1507                         Assert.AreEqual (2958465.999d, d.ToOADate (), "#4");
1508                 }
1509
1510                 [Test]
1511                 public void ToOADate_OverMax ()
1512                 {
1513                         DateTime d = new DateTime (3155378975136000001);
1514                         Assert.AreEqual (2958465.999d, d.ToOADate ());
1515                 }
1516
1517                 [Test]
1518                 [Ignore ("This test is probably geo location dependent, at least fails on .NET 4.0 in Japan")]
1519                 public void ToOADate_MaxValue ()
1520                 {
1521                         Assert.AreEqual (2958465.99999999d, DateTime.MaxValue.ToOADate ());
1522                 }
1523
1524                 [Test]
1525                 public void ToOADate_UnderMin ()
1526                 {
1527                         DateTime d = new DateTime (31242239135999999);
1528                         Assert.AreEqual (-657434.999d, d.ToOADate ());
1529                 }
1530
1531                 [Test]
1532                 public void ToOADate_MinValue ()
1533                 {
1534                         Assert.AreEqual (0, DateTime.MinValue.ToOADate ());
1535                 }
1536
1537                 [Test] // bug52075
1538                 public void MaxValueYear ()
1539                 {
1540                         Assert.AreEqual ("9999", DateTime.MaxValue.Year.ToString ());
1541                 }
1542
1543                 [Test]
1544                 public void X509Certificate () 
1545                 {
1546                         // if this test fails then *ALL* or *MOST* X509Certificate tests will also fails
1547                         DateTime dt = DateTime.ParseExact ("19960312183847Z", "yyyyMMddHHmmssZ", null);
1548                         Assert.AreEqual (DateTimeKind.Local, dt.Kind, "#1");
1549                         dt = dt.ToUniversalTime ();
1550                         Assert.AreEqual (DateTimeKind.Utc, dt.Kind, "#2");
1551                         Assert.AreEqual ("03/12/1996 18:38:47", dt.ToString (), "#3");
1552
1553                         // technically this is invalid (PKIX) because of the missing seconds but it exists so...
1554                         dt = DateTime.ParseExact ("9602231915Z", "yyMMddHHmmZ", null);
1555                         Assert.AreEqual (DateTimeKind.Local, dt.Kind, "#4");
1556                         dt = dt.ToUniversalTime ();
1557                         Assert.AreEqual (DateTimeKind.Utc, dt.Kind, "#5");
1558                         Assert.AreEqual ("02/23/1996 19:15:00", dt.ToString (), "#6");
1559
1560                         dt = DateTime.ParseExact ("19960312183847Z", "yyyyMMddHHmmssZ", null, DateTimeStyles.AdjustToUniversal);
1561                         Assert.AreEqual (DateTimeKind.Utc, dt.Kind, "#7");
1562                 }
1563
1564                 [Test]
1565                 public void ZLiteral ()
1566                 {
1567                         // However, "Z" and "'Z'" are different.
1568                         DateTime dt = DateTime.ParseExact ("19960312183847Z", "yyyyMMddHHmmss'Z'", null);
1569                         DateTime dtz = DateTime.ParseExact ("19960312183847Z", "yyyyMMddHHmmssZ", null);
1570                         Assert.AreEqual (DateTimeKind.Unspecified, dt.Kind, "#1");
1571                         dt = dt.ToLocalTime ();
1572                         Assert.AreEqual (DateTimeKind.Local, dt.Kind, "#2");
1573                         Assert.AreEqual (DateTimeKind.Local, dtz.Kind, "#3");
1574                         Assert.AreEqual (dt, dtz, "#4");
1575                 }
1576
1577                 [Test] // bug 56436
1578                 public void QuotedFormat ()
1579                 {
1580                         string date = "28/Mar/2004:19:12:37 +0200";
1581                         string [] expectedFormats = {"dd\"/\"MMM\"/\"yyyy:HH:mm:ss zz\"00\""};
1582                         DateTime.ParseExact (date, expectedFormats, null, DateTimeStyles.AllowWhiteSpaces);
1583                 }
1584
1585                 [Test]
1586                 public void CultureIndependentTests ()
1587                 {
1588                         // Here I aggregated some tests mainly because of test 
1589                         // performance (iterating all the culture is heavy process).
1590                 
1591                         for (int i = 0; i < 32768; i++) {
1592                                 CultureInfo ci = null;
1593                                 string stage = "init";
1594                                 try {
1595                                         try {
1596                                                 ci = new CultureInfo (i);
1597                                                 // In fact InvatiantCulture is not neutral.
1598                                                 // See bug #59716.
1599                                                 if (ci.IsNeutralCulture && ci != CultureInfo.InvariantCulture)
1600                                                         continue;
1601                                         } catch (Exception) {
1602                                                 continue;
1603                                         }
1604                                         Thread.CurrentThread.CurrentCulture = ci;
1605                                         DateTime dt;
1606
1607                                         switch (ci.LCID) {
1608                                         case 1025: // ar-SA
1609                                         case 2559: // qps-plocm
1610                                                 continue; // fails too many tests below on .NET.
1611                                         }
1612
1613                                         // Common patterns
1614                                         // X509Certificate pattern is _always_ accepted.
1615                                         stage = "1";
1616                                         dt = DateTime.ParseExact ("19960312183847Z", "yyyyMMddHHmmssZ", null);
1617
1618                                         stage = "2";
1619                                         // fails with many cultures on .NET.
1620         //                              if (i != 127)
1621         //                                      dt = DateTime.Parse ("19960312183847Z");
1622
1623                                         stage = "3";
1624                                         dt = DateTime.Parse ("2004-05-26T03:29:01.1234567");
1625                                         stage = "4";
1626                                         dt = DateTime.Parse ("2004-05-26T03:29:01.1234567-07:00");
1627
1628                                         // memo: the least LCID is 127, and then 1025(ar-SA)
1629
1630                                         // "th-TH" locale rejects them since in
1631                                         // ThaiBuddhistCalendar the week of a day is different.
1632                                         // (and also for years).
1633                                         switch (ci.LCID) {
1634                                         case 1054:
1635                                         case 1128: // ha-Latn-NG
1636                                                 break;
1637                                         default:
1638                                                 try {
1639                                                         stage = "5";
1640                                                         dt = DateTime.Parse ("Sat, 29 Oct 1994 12:00:00 GMT", ci);
1641                                                 } catch (FormatException ex) {
1642                                                         Assert.Fail (String.Format ("stage 5.1 RFC1123: culture {0} {1} failed: {2}", i, ci, ex.Message));
1643                                                 }
1644
1645                                                 // bug #47720
1646                                                 if (dt != TimeZone.CurrentTimeZone.ToUniversalTime (dt))
1647                                                         Assert.IsTrue (12 != dt.Hour, String.Format ("bug #47720 on culture {0} {1}", ci.LCID, ci));
1648
1649                                                 // variant of RFC1123
1650                                                 try {
1651                                                         stage = "6";
1652                                                         dt = DateTime.Parse ("Sat, 1 Oct 1994 03:00:00", ci);
1653                                                 } catch (FormatException ex) {
1654                                                         Assert.Fail (String.Format ("stage 6.1 RFC1123 variant: culture {0} {1} failed: {2}", i, ci, ex.Message));
1655                                                 }
1656                                                 stage = "7";
1657                                                 Assert.AreEqual (3, dt.Hour, String.Format ("stage 7.1 RFC1123 variant on culture {0} {1}", ci.LCID, ci));
1658                                                 break;
1659                                         }
1660
1661                                         switch (ci.LCID) {
1662                                         case 1054: // th-TH
1663                                         case 1123: // ps-AF
1664                                         case 1125: // div-MV
1665                                         case 1164: // prs-AF
1666                                                 break;
1667                                         default:
1668                                                 stage = "8";
1669                                                 // 02/25/2002 04:25:13 as is
1670                                                 long tick1 = 631502079130000000;
1671                                                 long tick2 = TimeZone.CurrentTimeZone.ToLocalTime (new DateTime (tick1)).Ticks; // adjusted to local time
1672                                                 dt = DateTime.Parse ("Mon, 25 Feb 2002 04:25:13 GMT", ci);
1673                                                 Assert.AreEqual (tick2, dt.Ticks, String.Format ("GMT variant. culture={0} {1}", i, ci));
1674                                                 break;
1675                                         }
1676
1677
1678                                         // ka-GE rejects these formats under MS.NET. 
1679                                         // I wonder why. Also, those tests fail under .NET 1.0.
1680                                         if (ci.LCID != 1079) {
1681                                                 stage = "9";
1682                                                 dt = DateTime.Parse ("2002-02-25");
1683                                                 stage = "10";
1684                                                 dt = DateTime.Parse ("2002-02-25Z");
1685                                                 stage = "11";
1686                                                 dt = DateTime.Parse ("2002-02-25T19:20:00+09:00");
1687                                                 switch (ci.LCID) {
1688                                                 case 1038: // FIXME: MS passes this culture.
1689                                                 case 1062: // FIXME: MS passes this culture.
1690                                                 case 1078: // MS does not pass this culture. Dunno why.
1691                                                         break;
1692                                                 default:
1693                                                         break;
1694                                                 }
1695                                                 stage = "14";
1696                                                 dt = DateTime.Parse ("2002-02-25 12:01:03");
1697                                                 stage = "17";
1698                                                 if (ci.DateTimeFormat.TimeSeparator != ".")
1699                                                         dt = DateTime.Parse ("2002.02.25 12:01:03");
1700                                                 stage = "18";
1701                                                 dt = DateTime.Parse ("2003/01/23 01:34:56 GMT");
1702                                                 dt = TimeZone.CurrentTimeZone.ToUniversalTime (dt);
1703                                                 Assert.AreEqual (1, dt.Hour, String.Format ("stage 18.1 RFC1123 UTC {0} {1}", i, ci));
1704                                                 stage = "19";
1705                                                 // This test was fixed from 12:34:56 to
1706                                                 // 01:34:56 since 1078 af-ZA failed
1707                                                 // because of hour interpretation
1708                                                 // difference (af-ZA expects 0).
1709                                                 // (IMHO it is MS BUG though.)
1710                                                 dt = DateTime.Parse ("2003/01/23 12:34:56 GMT");
1711                                                 dt = TimeZone.CurrentTimeZone.ToUniversalTime (dt);
1712                                                 if (i != 1078)
1713                                                         Assert.AreEqual (12, dt.Hour, String.Format ("stage 18.1 RFC1123 UTC {0} {1}", i, ci));
1714                                         }
1715                                 } catch (FormatException ex) {
1716                                         Assert.Fail (String.Format ("stage {3}: Culture {0} {1} failed: {2}", i, ci, ex.Message, stage));
1717                                 }
1718                         }
1719                 }
1720
1721                 [Test]
1722                 [ExpectedException (typeof (ArgumentOutOfRangeException))]
1723                 public void ToFileTime_MinValue () 
1724                 {
1725                         DateTime.FromFileTime (Int64.MinValue);
1726                 }
1727
1728                 [Test]
1729                 [ExpectedException (typeof (ArgumentOutOfRangeException))]
1730                 public void ToFileTime_Negative () 
1731                 {
1732                         DateTime.FromFileTime (-1);
1733                 }
1734
1735                 [Test]
1736                 public void ToFileTime () 
1737                 {
1738                         long u = DateTime.FromFileTimeUtc (0).Ticks;
1739                         Assert.AreEqual (504911232000000000, u, "#A1");
1740                         long max = DateTime.MaxValue.Ticks - 504911232000000000; // w32file_epoch
1741                         Assert.AreEqual (3155378975999999999, DateTime.FromFileTimeUtc (max).Ticks, "#A2");
1742
1743                         long t = DateTime.FromFileTime (0).Ticks;
1744                         Assert.IsTrue (t > (u - TimeSpan.TicksPerDay), "#B1");
1745                         Assert.IsTrue (t < (u + TimeSpan.TicksPerDay), "#B2");
1746                 }
1747
1748                 [Test]
1749                 [ExpectedException (typeof (ArgumentOutOfRangeException))]
1750                 public void ToFileTimeUtc_MinValue () 
1751                 {
1752                         DateTime.FromFileTimeUtc (Int64.MinValue);
1753                 }
1754
1755                 [Test]
1756                 [ExpectedException (typeof (ArgumentOutOfRangeException))]
1757                 public void ToFileTimeUtc_Negative () 
1758                 {
1759                         DateTime.FromFileTimeUtc (-1);
1760                 }
1761
1762                 [Test]
1763                 public void ToFileTimeUtc ()
1764                 {
1765                         // Randomly generated time outside DST.
1766                         var utc = new DateTime (1993, 01, 28, 08, 49, 48, DateTimeKind.Utc);
1767                         var local = utc.ToLocalTime ();
1768                         var unspecified = new DateTime (1993, 01, 28, 08, 49, 48);
1769
1770                         Assert.AreEqual (DateTimeKind.Utc, utc.Kind);
1771                         Assert.AreEqual (DateTimeKind.Local, local.Kind);
1772                         Assert.AreEqual (DateTimeKind.Unspecified, unspecified.Kind);
1773
1774                         Assert.AreEqual (628638077880000000, utc.Ticks);
1775                         Console.WriteLine (local.Ticks - utc.Ticks);
1776
1777                         var offset = TimeZone.CurrentTimeZone.GetUtcOffset (local);
1778
1779                         var utcFt = utc.ToFileTime ();
1780                         var localFt = local.ToFileTime ();
1781                         var unspecifiedFt = unspecified.ToFileTime ();
1782
1783                         var utcUft = utc.ToFileTimeUtc ();
1784                         var localUft = local.ToFileTimeUtc ();
1785                         var unspecifiedUft = unspecified.ToFileTimeUtc ();
1786
1787                         Assert.AreEqual (123726845880000000, utcFt);
1788                         Assert.AreEqual (utcFt, localFt);
1789
1790                         Assert.AreEqual (offset.Ticks, utcFt - unspecifiedFt);
1791
1792                         Assert.AreEqual (utcFt, utcUft);
1793                         Assert.AreEqual (utcFt, localUft);
1794                         Assert.AreEqual (utcFt, unspecifiedUft);
1795                 }
1796                 
1797                 [Test]
1798                 public void FromFileTimeUtcTest ()
1799                 {
1800                         DateTime dt = DateTime.FromFileTimeUtc (123456);
1801                         Assert.AreEqual (dt.Kind, DateTimeKind.Utc, "#Kind");
1802                         Assert.AreEqual (dt.Ticks, 504911232000123456, "#Ticks");
1803                 }
1804
1805                 [Test]
1806                 public void Milliseconds ()
1807                 {
1808                         DateTime dt = DateTime.Parse ("2004-05-26T03:29:01.1234567-07:00");
1809                         dt = TimeZone.CurrentTimeZone.ToUniversalTime (dt);
1810                         Assert.AreEqual (632211641411234567, dt.Ticks);
1811                 }
1812
1813                 [Test]
1814                 [ExpectedException (typeof (FormatException))]
1815                 public void ParseNotExact ()
1816                 {
1817                         // The error reported is:
1818                         // String was not recognized as valid DateTime
1819                         DateTime dt = DateTime.Parse ("2004-05-26T03:29:01-07:00 foo");
1820                         dt = TimeZone.CurrentTimeZone.ToUniversalTime (dt);
1821                         Assert.AreEqual (632211641410000000, dt.Ticks);
1822                 }
1823
1824                 [Test]
1825                 public void ParseExact_Bug80094 ()
1826                 {
1827                         // we can safely change the curernt culture, as the original value will
1828                         // be restored on TearDown
1829                         Thread.CurrentThread.CurrentCulture = new CultureInfo ("ja-JP");
1830                         string y = string.Format ("{0}-{1}-{2} {3}", DateTime.Now.Year.ToString (),
1831                                 "11", "29", "06:34");
1832                         DateTime date = DateTime.ParseExact (y, "yyyy-MMM-dd hh:mm", null);
1833                         Assert.AreEqual (DateTime.Now.Year, date.Year, "#1");
1834                         Assert.AreEqual (11, date.Month, "#2");
1835                         Assert.AreEqual (29, date.Day, "#3");
1836                         Assert.AreEqual (6, date.Hour, "#4");
1837                         Assert.AreEqual (34, date.Minute, "#5");
1838                         Assert.AreEqual (0, date.Second, "#6");
1839                         Assert.AreEqual (0, date.Millisecond, "#7");
1840                 }
1841
1842                 [Test]
1843                 public void ParseExact_Bug324845 ()
1844                 {
1845                         DateTime ctime = new DateTime (2007, 7, 23, 19, 19, 45);
1846                         ctime = ctime.ToUniversalTime ();
1847                         string instr = ctime.ToString ("yyyyMMddHHmmss");
1848
1849                         DateTime t = DateTime.ParseExact (instr, "yyyyMMddHHmmss", null, DateTimeStyles.AssumeUniversal);
1850                         Assert.AreEqual (2007, t.Year);
1851                         Assert.AreEqual (7, t.Month);
1852                         Assert.AreEqual (23, t.Day);
1853                         Assert.AreEqual (19, t.Hour);
1854                         Assert.AreEqual (19, t.Minute);
1855                         Assert.AreEqual (45, t.Second);
1856
1857                 }
1858
1859                 [Test]
1860                 [ExpectedException (typeof (FormatException))]
1861                 public void ParseExactIsExact()
1862                 {
1863                         DateTime.ParseExact ("2004-05-26T03:29:01-07:00 foo", "yyyy-MM-ddTHH:mm:sszzz", null);
1864                 }
1865
1866                 [Test]
1867                 [ExpectedException (typeof (FormatException))]
1868                 public void ParseExactDoesNotEatZ ()
1869                 {
1870                         DateTime.ParseExact ("2004-05-26T03:29:01", "yyyy-MM-ddTHH:mm:ssZ", null);
1871                 }
1872
1873                 [Test]
1874                 public void ParseExactMilliseconds ()
1875                 {
1876                         DateTime dt = DateTime.ParseExact ("2004-05-26T03:29:01.1234567-07:00", "yyyy-MM-ddTHH:mm:ss.fffffffzzz", null);
1877                         dt = TimeZone.CurrentTimeZone.ToUniversalTime (dt);
1878                         Assert.AreEqual (632211641411234567, dt.Ticks);
1879                 }
1880
1881                 [Test]
1882                 public void NoColonTimeZone ()
1883                 {
1884                         Assert.IsTrue (DateTime.Parse ("2004-05-26T03:29:01-0700").Ticks
1885                                 != DateTime.Parse ("2004-05-26T03:29:01-0800").Ticks);
1886                 }
1887
1888                 [Test]
1889                 public void WithColonTimeZone ()
1890                 {
1891                         Assert.IsTrue (DateTime.Parse ("2004-05-26T03:29:01-07:00").Ticks
1892                                 != DateTime.Parse ("2004-05-26T03:29:01-08:00").Ticks);
1893                 }
1894
1895                 [Test]
1896                 [ExpectedException (typeof (FormatException))]
1897                 public void EmptyFormatPattern ()
1898                 {
1899                         DateTime.ParseExact (String.Empty, String.Empty, null);
1900                 }
1901
1902                 [Test]
1903                 [ExpectedException (typeof (InvalidCastException))]
1904                 public void IConvertible_ToType_Boolean () 
1905                 {
1906                         ((IConvertible)DateTime.Now).ToType (typeof (bool), null);
1907                 }
1908
1909                 [Test]
1910                 [ExpectedException (typeof (InvalidCastException))]
1911                 public void IConvertible_ToType_Byte () 
1912                 {
1913                         ((IConvertible)DateTime.Now).ToType (typeof (byte), null);
1914                 }
1915
1916                 [Test]
1917                 [ExpectedException (typeof (InvalidCastException))]
1918                 public void IConvertible_ToType_Char () 
1919                 {
1920                         ((IConvertible)DateTime.Now).ToType (typeof (char), null);
1921                 }
1922
1923                 [Test]
1924                 public void IConvertible_ToType_DateTime () 
1925                 {
1926                         DateTime dt = DateTime.Now;
1927                         DateTime dt2 = (DateTime) ((IConvertible)dt).ToType (typeof (DateTime), null);
1928                         Assert.IsTrue (dt.Equals (dt2));
1929                 }
1930
1931                 [Test]
1932                 [ExpectedException (typeof (InvalidCastException))]
1933                 public void IConvertible_ToType_DBNull () 
1934                 {
1935                         ((IConvertible)DateTime.Now).ToType (typeof (DBNull), null);
1936                 }
1937
1938                 [Test]
1939                 [ExpectedException (typeof (InvalidCastException))]
1940                 public void IConvertible_ToType_Decimal () 
1941                 {
1942                         ((IConvertible)DateTime.Now).ToType (typeof (decimal), null);
1943                 }
1944
1945                 [Test]
1946                 [ExpectedException (typeof (InvalidCastException))]
1947                 public void IConvertible_ToType_Double () 
1948                 {
1949                         ((IConvertible)DateTime.Now).ToType (typeof (double), null);
1950                 }
1951
1952                 [Test]
1953                 [ExpectedException (typeof (ArgumentNullException))]
1954                 public void IConvertible_ToType_Empty () 
1955                 {
1956                         ((IConvertible)DateTime.Now).ToType (null, null);
1957                 }
1958
1959                 [Test]
1960                 [ExpectedException (typeof (InvalidCastException))]
1961                 public void IConvertible_ToType_Int16 () 
1962                 {
1963                         ((IConvertible)DateTime.Now).ToType (typeof (short), null);
1964                 }
1965
1966                 [Test]
1967                 [ExpectedException (typeof (InvalidCastException))]
1968                 public void IConvertible_ToType_Int32 () 
1969                 {
1970                         ((IConvertible)DateTime.Now).ToType (typeof (int), null);
1971                 }
1972
1973                 [Test]
1974                 [ExpectedException (typeof (InvalidCastException))]
1975                 public void IConvertible_ToType_Int64 () 
1976                 {
1977                         ((IConvertible)DateTime.Now).ToType (typeof (long), null);
1978                 }
1979
1980                 [Test]
1981                 public void IConvertible_ToType_Object () 
1982                 {
1983                         DateTime dt = DateTime.Now;
1984                         object o = ((IConvertible)dt).ToType (typeof (object), null);
1985                         Assert.IsTrue (dt.Equals (o));
1986                 }
1987
1988                 [Test]
1989                 [ExpectedException (typeof (InvalidCastException))]
1990                 public void IConvertible_ToType_SByte () 
1991                 {
1992                         ((IConvertible)DateTime.Now).ToType (typeof (sbyte), null);
1993                 }
1994
1995                 [Test]
1996                 [ExpectedException (typeof (InvalidCastException))]
1997                 public void IConvertible_ToType_Single () 
1998                 {
1999                         ((IConvertible)DateTime.Now).ToType (typeof (float), null);
2000                 }
2001
2002                 [Test]
2003                 public void IConvertible_ToType_String () 
2004                 {
2005                         DateTime dt = DateTime.Now;
2006                         string s = (string) ((IConvertible)dt).ToType (typeof (string), null);
2007                         Assert.AreEqual (s, dt.ToString ());
2008                 }
2009
2010                 [Test]
2011                 [ExpectedException (typeof (InvalidCastException))]
2012                 public void IConvertible_ToType_UInt16 () 
2013                 {
2014                         ((IConvertible)DateTime.Now).ToType (typeof (ushort), null);
2015                 }
2016
2017                 [Test]
2018                 [ExpectedException (typeof (InvalidCastException))]
2019                 public void IConvertible_ToType_UInt32 () 
2020                 {
2021                         ((IConvertible)DateTime.Now).ToType (typeof (uint), null);
2022                 }
2023
2024                 [Test]
2025                 [ExpectedException (typeof (InvalidCastException))]
2026                 public void IConvertible_ToType_UInt64 () 
2027                 {
2028                         ((IConvertible)DateTime.Now).ToType (typeof (ulong), null);
2029                 }
2030
2031                 [Test]
2032                 public void Bug352210 ()
2033                 {
2034                         DateTime dt = DateTime.ParseExact ("2007-06-15T10:30:10.5", "yyyy-MM-ddTHH:mm:ss.f", null);
2035                         Assert.AreEqual (633175002105000000, dt.Ticks, "#1");
2036                         Assert.AreEqual (DateTimeKind.Unspecified, dt.Kind, "#2");
2037                 }
2038
2039                 [Test]
2040                 public void Bug352210_New ()
2041                 {
2042                         long ticksUTC = 633377759060000000;
2043                         long ticksLocal = ticksUTC + TimeZone.CurrentTimeZone.GetUtcOffset (new DateTime (ticksUTC)).Ticks;
2044                         CultureInfo ci = CultureInfo.InvariantCulture;
2045                         DateTime dt;
2046
2047                         // Should return same time with Unspecified kind
2048                         dt = DateTime.ParseExact ("2008-02-05 02:38:26", "yyyy-MM-dd HH:mm:ss", ci);
2049                         Assert.AreEqual (DateTimeKind.Unspecified, dt.Kind, "A1");
2050                         Assert.AreEqual (ticksUTC, dt.Ticks, "A2");
2051
2052                         // Should return same time with Unspecified kind
2053                         dt = DateTime.ParseExact ("2008-02-05 02:38:26Z", "u", ci);
2054                         Assert.AreEqual (DateTimeKind.Unspecified, dt.Kind, "B1");
2055                         Assert.AreEqual (ticksUTC, dt.Ticks, "B2");
2056
2057                         // Should adjust to local time with Local kind
2058                         dt = DateTime.ParseExact ("2008-02-05 00:38:26-02:00", "yyyy-MM-dd HH:mm:ssK", ci);
2059                         Assert.AreEqual (DateTimeKind.Local, dt.Kind, "C1");
2060                         Assert.AreEqual (ticksLocal, dt.Ticks, "C2");
2061
2062                         // Should ignore AssumeUniversal since a timezone specifier is in the format string
2063                         // and return time adjusted to local time with Local kind
2064                         dt = DateTime.ParseExact ("2008-02-05 00:38:26 -2", "yyyy-MM-dd HH:mm:ss z", ci, DateTimeStyles.AssumeUniversal);
2065                         Assert.AreEqual (DateTimeKind.Local, dt.Kind, "D1");
2066                         Assert.AreEqual (ticksLocal, dt.Ticks, "D2");
2067
2068                         try {
2069                                 // GMT in format string can be used to specify time zone
2070                                 dt = DateTime.ParseExact ("2008-02-05 02:38:26 GMT", "yyyy-MM-dd HH:mm:ss GMT", ci);
2071                                 Assert.AreEqual (DateTimeKind.Local, dt.Kind, "E1");
2072                                 Assert.AreEqual (ticksLocal, dt.Ticks, "E2");
2073                         }
2074                         catch {
2075                                 Assert.Fail ("E3");
2076                         }
2077
2078                         try {
2079                                 // Same as above even when surrounded with other characters
2080                                 dt = DateTime.ParseExact ("2008-02-05 02:38:26 qqGMTqq", "yyyy-MM-dd HH:mm:ss qqGMTqq", ci);
2081                                 Assert.AreEqual (DateTimeKind.Local, dt.Kind, "F1");
2082                                 Assert.AreEqual (ticksLocal, dt.Ticks, "F2");
2083                         }
2084                         catch {
2085                                 Assert.Fail ("F3");
2086                         }
2087
2088                         try {
2089                                 // But single quoted GMT in format string should not specify time zone
2090                                 dt = DateTime.ParseExact ("2008-02-05 02:38:26 GMT", "yyyy-MM-dd HH:mm:ss 'GMT'", ci);
2091                                 Assert.AreEqual (DateTimeKind.Unspecified, dt.Kind, "G1");
2092                                 Assert.AreEqual (ticksUTC, dt.Ticks, "G2");
2093                         }
2094                         catch {
2095                                 Assert.Fail ("G3");
2096                         }
2097
2098                         dt = DateTime.Parse ("GMT 2008-02-05 02:38:26", ci);
2099                         Assert.AreEqual (DateTimeKind.Local, dt.Kind, "H1");
2100                         Assert.AreEqual (ticksLocal, dt.Ticks, "H2");
2101                 }
2102
2103                 [Test]
2104                 public void Bug377042 ()
2105                 {
2106                         string [] f = new string [] {
2107                                 "yyyy-MM-ddTHH:mm:ssZ",
2108                                 "yyyy-MM-ddTHH:mm:sszzzz",
2109                                 "yyyy-MM-dd"
2110                                 };
2111                         DateTimeStyles dts = DateTimeStyles.AdjustToUniversal;
2112                         dts |= DateTimeStyles.AssumeUniversal;
2113                         DateTime result = DateTime.ParseExact ("2005-01-01T01:11:11+8:00", f, new DateTimeFormatInfo (), dts);
2114                 }
2115
2116         [Test]
2117         public void TestParseExactXmlTimeFormats()
2118         {
2119             //Xamarin Bug 16742
2120             string[] xmlTimeFormats = {
2121                 "HH:mm:ss", "HH:mm:ss.FFFFFFF",
2122                 "HH:mm:sszzz", "HH:mm:ss.FFFFFFFzzz",
2123                 "HH:mm:ssZ", "HH:mm:ss.FFFFFFFZ"
2124             };
2125             DateTimeStyles style = DateTimeStyles.AllowLeadingWhite | DateTimeStyles.AllowTrailingWhite;
2126
2127             //time local
2128             Assert.AreEqual(new DateTime(DateTime.Now.Year, DateTime.Now.Month, DateTime.Now.Day, 13, 30, 44, 000).Ticks,
2129                 DateTime.ParseExact("13:30:44", xmlTimeFormats, DateTimeFormatInfo.InvariantInfo, style).Ticks);
2130             Assert.AreEqual(new DateTime(DateTime.Now.Year, DateTime.Now.Month, DateTime.Now.Day, 13, 30, 44, 000).Ticks,
2131                 DateTime.ParseExact("13:30:44.0", xmlTimeFormats, DateTimeFormatInfo.InvariantInfo, style).Ticks);
2132             Assert.AreEqual(new DateTime(DateTime.Now.Year, DateTime.Now.Month, DateTime.Now.Day, 13, 30, 44, 000).Ticks,
2133                 DateTime.ParseExact("13:30:44.00", xmlTimeFormats, DateTimeFormatInfo.InvariantInfo, style).Ticks);
2134             Assert.AreEqual(new DateTime(DateTime.Now.Year, DateTime.Now.Month, DateTime.Now.Day, 13, 30, 44, 000).Ticks,
2135                 DateTime.ParseExact("13:30:44.000", xmlTimeFormats, DateTimeFormatInfo.InvariantInfo, style).Ticks);
2136             Assert.AreEqual(new DateTime(DateTime.Now.Year, DateTime.Now.Month, DateTime.Now.Day, 13, 30, 44, 000).Ticks,
2137                 DateTime.ParseExact("13:30:44.0000", xmlTimeFormats, DateTimeFormatInfo.InvariantInfo, style).Ticks);
2138             Assert.AreEqual(new DateTime(DateTime.Now.Year, DateTime.Now.Month, DateTime.Now.Day, 13, 30, 44, 000).Ticks,
2139                 DateTime.ParseExact("13:30:44.00000", xmlTimeFormats, DateTimeFormatInfo.InvariantInfo, style).Ticks);
2140             Assert.AreEqual(new DateTime(DateTime.Now.Year, DateTime.Now.Month, DateTime.Now.Day, 13, 30, 44, 000).Ticks,
2141                 DateTime.ParseExact("13:30:44.000000", xmlTimeFormats, DateTimeFormatInfo.InvariantInfo, style).Ticks);
2142             Assert.AreEqual(new DateTime(DateTime.Now.Year, DateTime.Now.Month, DateTime.Now.Day, 13, 30, 44, 000).Ticks,
2143                 DateTime.ParseExact("13:30:44.0000000", xmlTimeFormats, DateTimeFormatInfo.InvariantInfo, style).Ticks);
2144             Assert.AreEqual(new DateTime(DateTime.Now.Year, DateTime.Now.Month, DateTime.Now.Day, 13, 30, 44, 123).Ticks,
2145                 DateTime.ParseExact("13:30:44.123", xmlTimeFormats, DateTimeFormatInfo.InvariantInfo, style).Ticks);
2146
2147             //time with zero timezone
2148             Assert.AreEqual(new DateTime(DateTime.Now.Year, DateTime.Now.Month, DateTime.Now.Day, 13, 30, 44, 000, DateTimeKind.Utc).ToLocalTime().Ticks,
2149                 DateTime.ParseExact("13:30:44Z", xmlTimeFormats, DateTimeFormatInfo.InvariantInfo, style).Ticks);
2150             Assert.AreEqual(new DateTime(DateTime.Now.Year, DateTime.Now.Month, DateTime.Now.Day, 13, 30, 44, 000, DateTimeKind.Utc).ToLocalTime().Ticks,
2151                 DateTime.ParseExact("13:30:44.0Z", xmlTimeFormats, DateTimeFormatInfo.InvariantInfo, style).Ticks);
2152             Assert.AreEqual(new DateTime(DateTime.Now.Year, DateTime.Now.Month, DateTime.Now.Day, 13, 30, 44, 000, DateTimeKind.Utc).ToLocalTime().Ticks,
2153                 DateTime.ParseExact("13:30:44.00Z", xmlTimeFormats, DateTimeFormatInfo.InvariantInfo, style).Ticks);
2154             Assert.AreEqual(new DateTime(DateTime.Now.Year, DateTime.Now.Month, DateTime.Now.Day, 13, 30, 44, 000, DateTimeKind.Utc).ToLocalTime().Ticks,
2155                 DateTime.ParseExact("13:30:44.000Z", xmlTimeFormats, DateTimeFormatInfo.InvariantInfo, style).Ticks);
2156             Assert.AreEqual(new DateTime(DateTime.Now.Year, DateTime.Now.Month, DateTime.Now.Day, 13, 30, 44, 000, DateTimeKind.Utc).ToLocalTime().Ticks,
2157                 DateTime.ParseExact("13:30:44.0000Z", xmlTimeFormats, DateTimeFormatInfo.InvariantInfo, style).Ticks);
2158             Assert.AreEqual(new DateTime(DateTime.Now.Year, DateTime.Now.Month, DateTime.Now.Day, 13, 30, 44, 000, DateTimeKind.Utc).ToLocalTime().Ticks,
2159                 DateTime.ParseExact("13:30:44.00000Z", xmlTimeFormats, DateTimeFormatInfo.InvariantInfo, style).Ticks);
2160             Assert.AreEqual(new DateTime(DateTime.Now.Year, DateTime.Now.Month, DateTime.Now.Day, 13, 30, 44, 000, DateTimeKind.Utc).ToLocalTime().Ticks,
2161                 DateTime.ParseExact("13:30:44.000000Z", xmlTimeFormats, DateTimeFormatInfo.InvariantInfo, style).Ticks);
2162             Assert.AreEqual(new DateTime(DateTime.Now.Year, DateTime.Now.Month, DateTime.Now.Day, 13, 30, 44, 000, DateTimeKind.Utc).ToLocalTime().Ticks,
2163                 DateTime.ParseExact("13:30:44.0000000Z", xmlTimeFormats, DateTimeFormatInfo.InvariantInfo, style).Ticks);
2164             Assert.AreEqual(new DateTime(DateTime.Now.Year, DateTime.Now.Month, DateTime.Now.Day, 13, 30, 44, 123, DateTimeKind.Utc).ToLocalTime().Ticks,
2165                 DateTime.ParseExact("13:30:44.123Z", xmlTimeFormats, DateTimeFormatInfo.InvariantInfo, style).Ticks);
2166
2167             //time with timezone
2168             Assert.AreEqual(new DateTimeOffset(DateTime.Now.Year, DateTime.Now.Month, DateTime.Now.Day, 13, 30, 44, 000, new TimeSpan(13, 0, 0)).ToLocalTime().Ticks,
2169                 DateTime.ParseExact("13:30:44+13:00", xmlTimeFormats, DateTimeFormatInfo.InvariantInfo, style).Ticks);
2170             Assert.AreEqual(new DateTimeOffset(DateTime.Now.Year, DateTime.Now.Month, DateTime.Now.Day, 13, 30, 44, 000, new TimeSpan(13, 0, 0)).ToLocalTime().Ticks,
2171                 DateTime.ParseExact("13:30:44.0+13:00", xmlTimeFormats, DateTimeFormatInfo.InvariantInfo, style).Ticks);
2172             Assert.AreEqual(new DateTimeOffset(DateTime.Now.Year, DateTime.Now.Month, DateTime.Now.Day, 13, 30, 44, 000, new TimeSpan(13, 0, 0)).ToLocalTime().Ticks,
2173                 DateTime.ParseExact("13:30:44.00+13:00", xmlTimeFormats, DateTimeFormatInfo.InvariantInfo, style).Ticks);
2174             Assert.AreEqual(new DateTimeOffset(DateTime.Now.Year, DateTime.Now.Month, DateTime.Now.Day, 13, 30, 44, 000, new TimeSpan(13, 0, 0)).ToLocalTime().Ticks,
2175                 DateTime.ParseExact("13:30:44.000+13:00", xmlTimeFormats, DateTimeFormatInfo.InvariantInfo, style).Ticks);
2176             Assert.AreEqual(new DateTimeOffset(DateTime.Now.Year, DateTime.Now.Month, DateTime.Now.Day, 13, 30, 44, 000, new TimeSpan(13, 0, 0)).ToLocalTime().Ticks,
2177                 DateTime.ParseExact("13:30:44.0000+13:00", xmlTimeFormats, DateTimeFormatInfo.InvariantInfo, style).Ticks);
2178             Assert.AreEqual(new DateTimeOffset(DateTime.Now.Year, DateTime.Now.Month, DateTime.Now.Day, 13, 30, 44, 000, new TimeSpan(13, 0, 0)).ToLocalTime().Ticks,
2179                 DateTime.ParseExact("13:30:44.00000+13:00", xmlTimeFormats, DateTimeFormatInfo.InvariantInfo, style).Ticks);
2180             Assert.AreEqual(new DateTimeOffset(DateTime.Now.Year, DateTime.Now.Month, DateTime.Now.Day, 13, 30, 44, 000, new TimeSpan(13, 0, 0)).ToLocalTime().Ticks,
2181                 DateTime.ParseExact("13:30:44.000000+13:00", xmlTimeFormats, DateTimeFormatInfo.InvariantInfo, style).Ticks);
2182             Assert.AreEqual(new DateTimeOffset(DateTime.Now.Year, DateTime.Now.Month, DateTime.Now.Day, 13, 30, 44, 000, new TimeSpan(13, 0, 0)).ToLocalTime().Ticks,
2183                 DateTime.ParseExact("13:30:44.0000000+13:00", xmlTimeFormats, DateTimeFormatInfo.InvariantInfo, style).Ticks);
2184             Assert.AreEqual(new DateTimeOffset(DateTime.Now.Year, DateTime.Now.Month, DateTime.Now.Day, 13, 30, 44, 123, new TimeSpan(13, 0, 0)).ToLocalTime().Ticks,
2185                 DateTime.ParseExact("13:30:44.123+13:00", xmlTimeFormats, DateTimeFormatInfo.InvariantInfo, style).Ticks);
2186         }
2187
2188                 [Test]
2189                 [ExpectedException (typeof (FormatException))]
2190                 public void EmptyString ()
2191                 {
2192                         DateTime.Parse ("");
2193                 }
2194
2195                 [Test]
2196                 public void TryEmptyString ()
2197                 {
2198                         DateTime date;
2199                         Assert.IsFalse (DateTime.TryParse ("", out date));
2200                 }
2201
2202                 [Test]
2203                 public void Kind ()
2204                 {
2205                         if (DateTime.Now == DateTime.UtcNow)
2206                                 Assert.Ignore (); // This test does not make sense.
2207                         if (TimeZone.CurrentTimeZone.GetUtcOffset (DateTime.UtcNow)
2208                                 != TimeZone.CurrentTimeZone.GetUtcOffset (DateTime.Now))
2209                                 Assert.Ignore (); // In this case it does not satisfy the test premises.
2210
2211                         Assert.AreEqual (DateTimeKind.Local, DateTime.Now.Kind, "#A1");
2212                         Assert.AreEqual (DateTimeKind.Local, DateTime.Today.Kind, "#A2");
2213
2214                         DateTime utc = DateTime.UtcNow;
2215                         DateTime now = new DateTime (utc.Ticks + TimeZone.
2216                                 CurrentTimeZone.GetUtcOffset (utc).Ticks, DateTimeKind.Local);
2217                         DateTime utctouniv = utc.ToUniversalTime ();
2218                         DateTime nowtouniv = now.ToUniversalTime ();
2219                         DateTime utctoloc = utc.ToLocalTime ();
2220                         DateTime nowtoloc = now.ToLocalTime ();
2221
2222                         Assert.AreEqual (DateTimeKind.Utc, utc.Kind, "#B1");
2223                         Assert.AreEqual (DateTimeKind.Local, now.Kind, "#B2");
2224                         Assert.AreEqual (DateTimeKind.Utc, utctouniv.Kind, "#B3");
2225                         Assert.AreEqual (DateTimeKind.Utc, nowtouniv.Kind, "#B4");
2226                         Assert.AreEqual (DateTimeKind.Local, utctoloc.Kind, "#B5");
2227                         Assert.AreEqual (DateTimeKind.Local, nowtoloc.Kind, "#B6");
2228                         Assert.AreEqual (utc, utctouniv, "#B7");
2229                         Assert.AreEqual (utc, nowtouniv, "#B8");
2230                         Assert.AreEqual (now, nowtoloc, "#B9");
2231                         Assert.AreEqual (now, utctoloc, "#B10");
2232                 }
2233
2234                 [Test]
2235                 public void InstanceMembersAndKind ()
2236                 {
2237                         Assert.AreEqual (DateTimeKind.Utc, DateTime.UtcNow.Date.Kind, "#1");
2238                         Assert.AreEqual (DateTimeKind.Utc, DateTime.UtcNow.Add (TimeSpan.FromMinutes (1)).Kind, "#2");
2239                         Assert.AreEqual (DateTimeKind.Utc, DateTime.UtcNow.Subtract (TimeSpan.FromMinutes (1)).Kind, "#3");
2240                         Assert.AreEqual (DateTimeKind.Utc, DateTime.UtcNow.AddDays (1).Kind, "#4");
2241                         Assert.AreEqual (DateTimeKind.Utc, DateTime.UtcNow.AddTicks (1).Kind, "#5");
2242                         Assert.AreEqual (DateTimeKind.Utc, DateTime.UtcNow.AddHours (1).Kind, "#6");
2243                         Assert.AreEqual (DateTimeKind.Utc, DateTime.UtcNow.AddMinutes (1).Kind, "#7");
2244                         Assert.AreEqual (DateTimeKind.Utc, DateTime.UtcNow.AddSeconds (1).Kind, "#8");
2245                         Assert.AreEqual (DateTimeKind.Utc, DateTime.UtcNow.AddMilliseconds (1).Kind, "#9");
2246                         Assert.AreEqual (DateTimeKind.Utc, DateTime.UtcNow.AddMonths (1).Kind, "#10");
2247                         Assert.AreEqual (DateTimeKind.Utc, DateTime.UtcNow.AddYears (1).Kind, "#11");
2248                         Assert.AreEqual (DateTimeKind.Utc, (DateTime.UtcNow + TimeSpan.FromMinutes (1)).Kind, "#12");
2249                         Assert.AreEqual (DateTimeKind.Utc, (DateTime.UtcNow - TimeSpan.FromMinutes (1)).Kind, "#13");
2250                 }
2251
2252                 [Test]
2253                 public void CompareTicks ()
2254                 {
2255                         // Only ticks are compared, not kind.
2256                         var d = new DateTime (0, DateTimeKind.Utc);
2257                         var f = new DateTime (0);
2258
2259                         Assert.AreEqual (d == f, true, "#1");
2260                 }
2261                 
2262                 [Test]
2263                 public void FromBinary ()
2264                 {
2265                         DateTime dt_utc = DateTime.FromBinary (0x4000000000000001);
2266                         Assert.AreEqual (DateTimeKind.Utc, dt_utc.Kind, "#1");
2267                         Assert.AreEqual (1, dt_utc.Ticks, "#2");
2268
2269                         DateTime dt_local = DateTime.FromBinary (unchecked ((long) 0x8000000000000001));
2270                         Assert.AreEqual (DateTimeKind.Local, dt_local.Kind, "#3");
2271
2272                         DateTime dt_unspecified = DateTime.FromBinary (0x0000000000000001);
2273                         Assert.AreEqual (DateTimeKind.Unspecified, dt_unspecified.Kind, "#4");
2274                         Assert.AreEqual (1, dt_unspecified.Ticks, "#5");
2275
2276                         DateTime dt_local2 = DateTime.FromBinary (unchecked ((long) 0xC000000000000001));
2277                         Assert.AreEqual (DateTimeKind.Local, dt_local2.Kind, "#6");
2278                         Assert.AreEqual (dt_local.Ticks, dt_local2.Ticks, "#7");
2279                 }
2280
2281                 [Test]
2282                 public void ToBinary ()
2283                 {
2284                         DateTime dt_local = new DateTime (1, DateTimeKind.Local);
2285                         Assert.AreEqual (1, (ulong) dt_local.ToBinary () >> 63, "#1");
2286                         Assert.AreEqual (1, dt_local.Ticks, "#2");
2287
2288                         DateTime dt_utc = new DateTime (1, DateTimeKind.Utc);
2289                         Assert.AreEqual (0x4000000000000001, dt_utc.ToBinary (), "#3");
2290                         Assert.AreEqual (1, dt_utc.Ticks, "#4");
2291
2292                         DateTime dt_unspecified = new DateTime (1, DateTimeKind.Unspecified);
2293                         Assert.AreEqual (1, dt_unspecified.ToBinary (), "#5");
2294                         Assert.AreEqual (1, dt_unspecified.Ticks, "#6");
2295                 }
2296
2297                 [Test]
2298                 public void RoundtripBinary ()
2299                 {
2300                         DateTime dt = DateTime.Now;
2301                         DateTime dt2 = DateTime.SpecifyKind (dt, DateTimeKind.Utc);
2302                         DateTime dt3 = DateTime.SpecifyKind (dt, DateTimeKind.Unspecified);
2303                         Assert.AreEqual (dt, DateTime.FromBinary (dt.ToBinary ()), "#1");
2304                         Assert.AreEqual (dt2, DateTime.FromBinary (dt2.ToBinary ()), "#2");
2305                         Assert.AreEqual (dt3, DateTime.FromBinary (dt3.ToBinary ()), "#3");
2306                         Assert.AreEqual (DateTimeKind.Local, DateTime.FromBinary (dt.ToBinary ()).Kind, "#4");
2307                         Assert.AreEqual (DateTimeKind.Utc, DateTime.FromBinary (dt2.ToBinary ()).Kind, "#5");
2308                         Assert.AreEqual (DateTimeKind.Unspecified, DateTime.FromBinary (dt3.ToBinary ()).Kind, "#6");
2309
2310                         Assert.AreEqual (TimeZone.CurrentTimeZone.GetUtcOffset (dt).Ticks, dt3.ToBinary () - (dt.ToBinary () & 0x7FFFFFFFFFFFFFFF), "#7");
2311                 }
2312
2313                 [Test]
2314                 public void TestMin ()
2315                 {
2316                         // This should never throw.
2317                         DateTime.MinValue.ToLocalTime ();
2318                 }
2319
2320                 [Test]
2321                 public void OmittedSecondsFraction ()
2322                 {
2323                         DateTime today = DateTime.Today;
2324                         Assert.AreEqual ("00:00:00.13579", today.AddTicks (1357900).ToString ("HH:mm:ss.FFFFFFF"), "#1");
2325                         DateTime dt = DateTime.ParseExact ("00:00:00.13579", "HH:mm:ss.FFFFFFF", CultureInfo.InvariantCulture);
2326                         Assert.AreEqual (today, dt.AddTicks (-1357900), "#2");
2327                         // it's more than strange ...
2328                         Assert.AreEqual (String.Empty, today.ToString (".FFFFFFF"), "#3");
2329                         Assert.AreEqual ("$", today.ToString ("$FFFFFFF"), "#4");
2330                 }
2331
2332                 [Test]
2333                 public void KindInPattern ()
2334                 {
2335                         // only 2.0 supports 'K'
2336                         Assert.AreEqual ("00:00:00", new DateTime (2000, 1, 1).ToString ("HH:mm:ssK"), "#1");
2337                         Assert.AreEqual ('Z', DateTime.Today.ToUniversalTime ().ToString ("HH:mm:ssK") [8], "#2");
2338                         Assert.AreEqual ("00:00:00+09:00".Length, DateTime.Today.ToString ("HH:mm:ssK").Length, "#3");
2339                 }
2340
2341                 [Test]
2342                 public void RoundtripPattern ()
2343                 {
2344                         // only 2.0 supports 'o'
2345                         Assert.AreEqual ("2000-01-01T00:00:00.0000000", new DateTime (2000, 1, 1).ToString ("o"), "#1");
2346                         Assert.AreEqual ("2000-01-01T00:00:00.0000000Z", DateTime.SpecifyKind (new DateTime (2000, 1, 1), DateTimeKind.Utc).ToString ("o"), "#2");
2347                         Assert.AreEqual ("2000-01-01T00:00:00.0000000+09:00".Length, DateTime.SpecifyKind (
2348                                 new DateTime (2000, 1, 1), DateTimeKind.Local).ToString ("o").Length, "#3");
2349
2350                         var culture = new CultureInfo ("ps-AF");
2351                         Assert.AreEqual ("1976-06-19T00:00:00.0000000", new DateTime(1976, 6, 19).ToString ("O", culture), "#4");
2352                         Assert.AreEqual ("1976-06-19T00:00:00.0000000", new DateTime(1976, 6, 19).ToString ("o", culture), "#5");
2353                 }
2354
2355                 [Test]
2356                 public void KindPattern ()
2357                 {
2358                         // no matter how the format string contains 'K' and the
2359                         // output string contains kind information, it does not
2360                         // assure that the string is parsed as roundtrip kind.
2361
2362                         // only 2.0 supports 'K'
2363                         string format = "yyyy-MM-dd'T'HH:mm:ss.fffK";
2364                         CultureInfo ci = CultureInfo.CurrentCulture;
2365                         DateTime dt = DateTime.SpecifyKind (new DateTime (2007, 11, 1, 2, 30, 45), DateTimeKind.Utc);
2366                         string s = dt.ToString (format);
2367                         DateTime d1 = DateTime.ParseExact (s, format, ci); // d1 is parsed as a local time.
2368                         Assert.AreEqual (dt.Ticks, d1.ToUniversalTime ().Ticks, "#1");
2369                         // .NET expects Local here, while s ends with 'Z' and should be parsed as UTC.
2370                         Assert.AreEqual (DateTimeKind.Local, d1.Kind, "#2");
2371
2372                         format = "yyyy-MM-dd'T'HH:mm:ssK";
2373                         ci = CultureInfo.CurrentCulture;
2374                         dt = new DateTime (2007, 11, 1, 2, 30, 45);
2375                         s = dt.ToString (format);
2376                         d1 = DateTime.ParseExact (s, format, ci);
2377                         Assert.AreEqual (dt.Ticks, d1.Ticks, "#3");
2378                         Assert.AreEqual (DateTimeKind.Unspecified, d1.Kind, "#4");
2379                 }
2380
2381                 [Test]
2382                 public void TestRoundTrip () {
2383                         DateTime result;
2384                         DateTimeStyles roundTripStyle = DateTimeStyles.RoundtripKind;
2385                         string utcDate = "2008-02-21T11:14:18.2721262Z";
2386                         string localDate = "2008-02-21T11:14:18.2721262+02:00";
2387                         string unspec = "2008-02-21T11:14:18.2721262";
2388                         String [] formats = {"yyyy-MM-ddTHH:mm:ssK", "yyyy-MM-ddTHH:mm:ss.FFFFFFFK"};
2389
2390                         result = DateTime.ParseExact (localDate, formats, CultureInfo.InvariantCulture, roundTripStyle);
2391                         Assert.AreEqual (result.Kind, DateTimeKind.Local);
2392                         Assert.AreEqual (result.ToUniversalTime ().Ticks, 633391820582721262);
2393                         
2394                         result = DateTime.ParseExact (unspec, formats, CultureInfo.InvariantCulture, roundTripStyle);
2395                         Assert.AreEqual (result.Kind, DateTimeKind.Unspecified);
2396                         Assert.AreEqual (result.Ticks, 633391892582721262);
2397                         
2398                         result = DateTime.ParseExact (utcDate, formats, CultureInfo.InvariantCulture, roundTripStyle);
2399                         Assert.AreEqual (result.Kind, DateTimeKind.Utc);
2400                         Assert.AreEqual (result.Ticks, 633391892582721262);
2401
2402                 }               
2403
2404                 [Test]
2405                 public void TestRegularStyle () {
2406                         DateTime result;
2407                         DateTimeStyles style = DateTimeStyles.AllowLeadingWhite | DateTimeStyles.AllowTrailingWhite;
2408                         string utcDate = "2008-02-21T11:14:18.2721262Z";
2409                         string localDate = "2008-02-21T11:14:18.2721262+02:00";
2410                         string unspec = "2008-02-21T11:14:18.2721262";
2411                         String [] formats = {"yyyy-MM-ddTHH:mm:ssK", "yyyy-MM-ddTHH:mm:ss.FFFFFFFK"};
2412
2413                         result = DateTime.ParseExact (localDate, formats, CultureInfo.InvariantCulture, style);
2414                         Assert.AreEqual (result.Kind, DateTimeKind.Local);
2415                         Assert.AreEqual (result.ToUniversalTime ().Ticks, 633391820582721262);
2416
2417                         result = DateTime.ParseExact (unspec, formats, CultureInfo.InvariantCulture, style);
2418                         Assert.AreEqual (result.Kind, DateTimeKind.Unspecified);                        
2419                         Assert.AreEqual (result.Ticks, 633391892582721262);
2420
2421                         result = DateTime.ParseExact (utcDate, formats, CultureInfo.InvariantCulture, style);
2422                         Assert.AreEqual (result.Kind, DateTimeKind.Local);
2423                         Assert.AreEqual (result.ToUniversalTime ().Ticks, 633391892582721262);
2424                 }
2425
2426                 [Test]
2427                 public void TestAssumeLocal () {
2428                         DateTime result;
2429                         DateTimeStyles assumeLocal =  DateTimeStyles.AssumeLocal;
2430                         string utcDate = "2008-02-21T11:14:18.2721262Z";
2431                         string localDate = "2008-02-21T11:14:18.2721262+02:00";
2432                         string unspec = "2008-02-21T11:14:18.2721262";
2433                         String [] formats = {"yyyy-MM-ddTHH:mm:ssK", "yyyy-MM-ddTHH:mm:ss.FFFFFFFK"};
2434
2435                         result = DateTime.ParseExact (localDate, formats, CultureInfo.InvariantCulture, assumeLocal);
2436                         Assert.AreEqual (result.Kind, DateTimeKind.Local);
2437                         Assert.AreEqual (result.ToUniversalTime ().Ticks, 633391820582721262);
2438
2439                         result = DateTime.ParseExact (unspec, formats, CultureInfo.InvariantCulture, assumeLocal);
2440                         Assert.AreEqual (result.Kind, DateTimeKind.Local);
2441                         Assert.AreEqual (result.Ticks, 633391892582721262);
2442
2443                         result = DateTime.ParseExact (utcDate, formats, CultureInfo.InvariantCulture, assumeLocal);
2444                         Assert.AreEqual (result.Kind, DateTimeKind.Local);
2445                         Assert.AreEqual (result.ToUniversalTime ().Ticks, 633391892582721262);
2446                 }
2447
2448                 [Test]
2449                 [ExpectedException (typeof (ArgumentException))]
2450                 public void IllegalStyleCombination1()
2451                 {
2452                         DateTimeStyles illegal = DateTimeStyles.RoundtripKind | DateTimeStyles.AssumeLocal;
2453                         DateTime.ParseExact ("", "", null, illegal);                    
2454                 }
2455
2456                 [Test]
2457                 [ExpectedException (typeof (ArgumentException))]
2458                 public void IllegalStyleCombination2()
2459                 {
2460                         DateTimeStyles illegal = DateTimeStyles.RoundtripKind | DateTimeStyles.AdjustToUniversal;
2461                         DateTime.ParseExact ("", "", null, illegal);
2462                 }
2463
2464                 [Test]
2465                 [ExpectedException (typeof (ArgumentException))]
2466                 public void IllegalStyleCombination3()
2467                 {
2468                         DateTimeStyles illegal = DateTimeStyles.RoundtripKind | DateTimeStyles.AssumeUniversal;
2469                         DateTime.ParseExact ("", "", null, illegal);
2470                 }
2471
2472                 [Test]
2473                 [ExpectedException (typeof (ArgumentException))]
2474                 public void IllegalStyleCombination4()
2475                 {
2476                         DateTimeStyles illegal = DateTimeStyles.AssumeLocal | DateTimeStyles.AssumeUniversal;
2477                         DateTime.ParseExact ("", "", null, illegal);
2478                 }
2479
2480                 [Test]
2481                 [ExpectedException (typeof (FormatException))]
2482                 public void TrailingDot ()
2483                 {
2484                         DateTime.ParseExact ("12:00:00", "HH:mm:ss.", null);
2485                 }
2486
2487                 [Test]
2488                 public void TrailingFlexibleMilliseconds ()
2489                 {
2490                         // bug #444103.
2491                         DateTime.ParseExact ("12:00:00", "HH:mm:ss.FFFFFFF", null);
2492                 }
2493                
2494                 [Test]
2495                 public void TryParseExact_NullString ()
2496                 {
2497                         DateTime dt;
2498                         DateTime.TryParseExact(null, "yyyy'-'MM'-'dd'T'HH':'mm':'ss'Z'", CultureInfo.InvariantCulture,
2499                                                DateTimeStyles.AssumeUniversal | DateTimeStyles.AdjustToUniversal, out dt);
2500                         Assert.AreEqual(default(DateTime), dt);
2501                 }
2502
2503                 [Test]
2504                 public void MSAndZ ()
2505                 {
2506                         CultureInfo cultureInfo = CultureInfo.GetCultureInfo ("en-US");
2507                         DateTime dt;
2508                         if (!DateTime.TryParse ("2009.02.24T13:57:07.000 -0800", cultureInfo.DateTimeFormat,
2509                                                 DateTimeStyles.None, out dt))
2510                                 Assert.Fail ("Failed");
2511                 }
2512
2513                 [Test]
2514                 public void Parse_InvalidShortDate ()
2515                 {
2516                         DateTime expected = new DateTime (2011, 03, 22, 07, 32, 00, DateTimeKind.Utc).ToLocalTime();
2517                         DateTime expected2 = new DateTime (2011, 03, 22, 08, 32, 00, DateTimeKind.Utc);
2518
2519                         string [] cultures = new string [] {"es-ES", "en-US", "en-GB", "de-DE", "fr-FR"
2520 #if NET_4_0
2521                                 ,"es", "en", "de", "fr"
2522 #endif
2523                                 };
2524                         
2525                         foreach (string culture in cultures) {
2526                                 CultureInfo ci = new CultureInfo (culture);
2527                                 ci.DateTimeFormat.ShortDatePattern = "d";
2528
2529                                 Assert.AreEqual (expected,  DateTime.Parse ("2011-03-22 08:32:00+01:00", ci, DateTimeStyles.RoundtripKind), "#a01 - " + culture);
2530                                 Assert.AreEqual (expected,  DateTime.Parse ("2011/03/22 08:32:00+01:00", ci, DateTimeStyles.RoundtripKind), "#a02 - " + culture);
2531                                 Assert.AreEqual (expected2, DateTime.Parse ("2011-03-22T08:32:00", ci, DateTimeStyles.RoundtripKind), "#a03 - " + culture);
2532                                 Assert.AreEqual (expected2, DateTime.Parse ("2011/03/22T08:32:00", ci, DateTimeStyles.RoundtripKind), "#a04 - " + culture);
2533                                 Assert.AreEqual (expected2, DateTime.Parse ("03/2011/22T08:32:00", ci, DateTimeStyles.RoundtripKind), "#a05 - " + culture);
2534                                 Assert.AreEqual (expected2, DateTime.Parse ("03-2011-22T08:32:00", ci, DateTimeStyles.RoundtripKind), "#a06 - " + culture);
2535                                 Assert.AreEqual (expected,  DateTime.Parse ("03/2011/22 08:32:00+01:00", ci, DateTimeStyles.RoundtripKind), "#a07 - " + culture);
2536                                 ci.DateTimeFormat.DateSeparator = "%";
2537                                 Assert.AreEqual (expected,  DateTime.Parse ("2011-03-22 08:32:00+01:00", ci, DateTimeStyles.RoundtripKind), "#b01 - " + culture);
2538                                 Assert.AreEqual (expected,  DateTime.Parse ("2011/03/22 08:32:00+01:00", ci, DateTimeStyles.RoundtripKind), "#b02 - " + culture);
2539                                 Assert.AreEqual (expected2, DateTime.Parse ("2011-03-22T08:32:00", ci, DateTimeStyles.RoundtripKind), "#b03 - " + culture);
2540                                 Assert.AreEqual (expected2, DateTime.Parse ("2011/03/22T08:32:00", ci, DateTimeStyles.RoundtripKind), "#b04 - " + culture);
2541                                 Assert.AreEqual (expected2, DateTime.Parse ("03/2011/22T08:32:00", ci, DateTimeStyles.RoundtripKind), "#b05 - " + culture);
2542                                 Assert.AreEqual (expected2, DateTime.Parse ("03-2011-22T08:32:00", ci, DateTimeStyles.RoundtripKind), "#b06 - " + culture);
2543                                 Assert.AreEqual (expected,  DateTime.Parse ("03/2011/22 08:32:00+01:00", ci, DateTimeStyles.RoundtripKind), "#b07 - " + culture);
2544                         }
2545                 }
2546
2547                 // Bug 3392
2548                 [Test]
2549                 public void Parse_DateWithTimeZone_TimeZoneShouldBeCorrectlyRead ()
2550                 {
2551                         string testDateWithTimeZoneInfo  = "2012-01-14T15:09:42.692875+03:00";
2552                         long expectedUtcTics = 634621397826928750;
2553
2554                         DateTimeOffset result = DateTimeOffset.Parse (testDateWithTimeZoneInfo, null, DateTimeStyles.RoundtripKind);
2555
2556                         Assert.AreEqual (expectedUtcTics, result.UtcTicks);
2557                 }
2558                 
2559                 [Test]
2560                 public void GenitiveMonth ()
2561                 {
2562                         var ci = new CultureInfo ("ru-RU");
2563                         var dt = new DateTime (2012, 9, 15);
2564                         Assert.AreEqual ("15 сентября", dt.ToString ("m", ci));
2565                 }
2566
2567                 [Test]
2568                 public void Parse_ThaiCalendar ()
2569                 {
2570                         var culture = CultureInfo.GetCultureInfo ("th-TH");
2571                         Assert.IsTrue (culture.Calendar is ThaiBuddhistCalendar);
2572                         var dt = DateTime.Now.Date;
2573                         var s = dt.ToString (culture);
2574                         var parsed = DateTime.Parse (s, culture);
2575
2576                         Assert.AreEqual (dt, parsed, "#1");
2577                 }
2578         }
2579 }