Edit report at https://bugs.php.net/bug.php?id=63953&edit=1
ID: 63953 Comment by: chris at modernmedia dot co Reported by: gianluca dot pinna at allbus dot com Summary: DateTime::diff produces a wrong DateInterval on DST change Status: Open Type: Bug Package: Date/time related Operating System: Linux 64 PHP Version: 5.3.20 Block user comment: N Private report: N New Comment: I can more or less confirm that DateInterval has Daylight Savings changes backward, at least for my 2014 test cases. In the springtime in the US we spring forward at 2am on a Sunday morning. 2am and 3am are the same exact time, and share the same exact timestamp. At 3am, 2 actual hours have elapsed since midnight. In the fall we fall backward on a Sunday at 2am. In this case, at 3am 4 actual hours have elapsed since midnight. My code shows that DateTime and DateTimezone handle DST changes correctly -- its DateInterval that is having the problem. Here's my test code: <?php /** Daylight Saving Time (United States) 2014 begins at 2:00 AM on Sunday, March 9 and ends at 2:00 AM on Sunday, November 2 */ $tz = new DateTimeZone( 'America/New_York'); $d1 = new DateTime('2014-03-09 00:00:00', $tz); echo PHP_EOL; echo 'Springing forward at 2am on Sunday, March 9, 2014...' . PHP_EOL; for ($n = 0; $n < 7; $n++){ $d2 = new DateTime(sprintf('2014-03-09 %s:00:00', $n), $tz); $ts_diff = ($d2->getTimestamp() - $d1->getTimestamp())/(60 * 60); $int = $d2->diff($d1); echo 'Comparing ' . $d2->format('g a') . ' to ' . $d1->format('g a') . PHP_EOL; echo 'Timestamp diff: ' . $ts_diff . PHP_EOL; echo 'Interval diff: ' . $int->h . PHP_EOL; } echo PHP_EOL; $d1 = new DateTime('2014-11-02 00:00:00', $tz); echo 'Falling backward at 2am on Sunday, November 2, 2014...' . PHP_EOL; for ($n = 0; $n < 7; $n++){ $d2 = new DateTime(sprintf('2014-11-02 %s:00:00', $n), $tz); $ts_diff = ($d2->getTimestamp() - $d1->getTimestamp())/(60 * 60); $int = $d2->diff($d1); echo 'Comparing ' . $d2->format('g a') . ' to ' . $d1->format('g a') . PHP_EOL; echo 'Timestamp diff: ' . $ts_diff . PHP_EOL; echo 'Interval diff: ' . $int->h . PHP_EOL; } phpinfo(INFO_GENERAL); ?> Output ======== Springing forward at 2am on Sunday, March 9, 2014... Comparing 12 am to 12 am Timestamp diff: 0 Interval diff: 0 Comparing 1 am to 12 am Timestamp diff: 1 Interval diff: 1 Comparing 2 am to 12 am Timestamp diff: 2 Interval diff: 3 Comparing 3 am to 12 am Timestamp diff: 2 Interval diff: 3 Comparing 4 am to 12 am Timestamp diff: 3 Interval diff: 4 Comparing 5 am to 12 am Timestamp diff: 4 Interval diff: 5 Comparing 6 am to 12 am Timestamp diff: 5 Interval diff: 6 Falling backward at 2am on Sunday, November 2, 2014... Comparing 12 am to 12 am Timestamp diff: 0 Interval diff: 0 Comparing 1 am to 12 am Timestamp diff: 1 Interval diff: 1 Comparing 2 am to 12 am Timestamp diff: 3 Interval diff: 2 Comparing 3 am to 12 am Timestamp diff: 4 Interval diff: 3 Comparing 4 am to 12 am Timestamp diff: 5 Interval diff: 4 Comparing 5 am to 12 am Timestamp diff: 6 Interval diff: 5 Comparing 6 am to 12 am Timestamp diff: 7 Interval diff: 6 phpinfo() PHP Version => 5.4.9 System => Darwin Christophers-MacBook-Air.local 12.4.0 Darwin Kernel Version 12.4.0: Wed May 1 17:57:12 PDT 2013; root:xnu-2050.24.15~1/RELEASE_X86_64 x86_64 Build Date => Dec 11 2012 12:36:33 Previous Comments: ------------------------------------------------------------------------ [2013-02-05 17:42:11] kaido at tradenet dot ee test script: $d1 = new DateTime('2012-05-31'); $d2 = new DateTime('2013-03-02'); $diff1 = $d1->diff ($d2); $diff2 = $d2->diff ($d1); echo "case 1: ".$diff1->m." months, case 2: ".$diff2->m." months"; expected result: both $diff1->m and $diff2->m are 9 ------------------------------------------------------------------------ [2013-01-09 16:45:28] gianluca dot pinna at allbus dot com Description: ------------ I have two dates, the second date is after a DST change. I tried to get a diff between the two dates. Computing the difference between the timestamps I get the correct result (33 hours). Using DateTime->diff() I get 34 hours. Is this expected? Test script: --------------- <?php $originDate = new DateTime('2013-03-30 00:00:00', new DateTimeZone('Europe/Rome')); $destinationDate = new DateTime('2013-03-31 10:00:00', new DateTimeZone('Europe/Rome')); $diff = $destinationDate->getTimestamp() - $originDate->getTimestamp(); $dateinterval = $originDate->diff($destinationDate); var_dump($diff/3600); var_dump($dateinterval->d*24 + $dateinterval->h); // int(33) // int(34) Expected result: ---------------- I would expect to always get 33 hours. ------------------------------------------------------------------------ -- Edit this bug report at https://bugs.php.net/bug.php?id=63953&edit=1