Além da resposta muito útil de @fyrye esta é uma solução normal para o bug mencionado ( este ), que DatePeriod subtrai uma hora ao entrar no verão, mas não adiciona uma hora ao sair do verão (e, portanto, Europa / Berlim tem seu correto 743 horas, mas outubro tem 744 em vez de 745 horas):
Contando as horas de um mês (ou qualquer intervalo de tempo), considerando as transições do horário de verão em ambas as direções
function getMonthHours(string $year, string $month, \DateTimeZone $timezone): int
{
// or whatever start and end \DateTimeInterface objects you like
$start = new \DateTimeImmutable($year . '-' . $month . '-01 00:00:00', $timezone);
$end = new \DateTimeImmutable((new \DateTimeImmutable($year . '-' . $month . '-01 23:59:59', $timezone))->format('Y-m-t H:i:s'), $timezone);
// count the hours just utilizing \DatePeriod, \DateInterval and iterator_count, hell yeah!
$hours = iterator_count(new \DatePeriod($start, new \DateInterval('PT1H'), $end));
// find transitions and check, if there is one that leads to a positive offset
// that isn't added by \DatePeriod
// this is the workaround for https://bugs.php.net/bug.php?id=75685
$transitions = $timezone->getTransitions((int)$start->format('U'), (int)$end->format('U'));
if (2 === count($transitions) && $transitions[0]['offset'] - $transitions[1]['offset'] > 0) {
$hours += (round(($transitions[0]['offset'] - $transitions[1]['offset'])/3600));
}
return $hours;
}
$myTimezoneWithDST = new \DateTimeZone('Europe/Berlin');
var_dump(getMonthHours('2020', '01', $myTimezoneWithDST)); // 744
var_dump(getMonthHours('2020', '03', $myTimezoneWithDST)); // 743
var_dump(getMonthHours('2020', '10', $myTimezoneWithDST)); // 745, finally!
$myTimezoneWithoutDST = new \DateTimeZone('UTC');
var_dump(getMonthHours('2020', '01', $myTimezoneWithoutDST)); // 744
var_dump(getMonthHours('2020', '03', $myTimezoneWithoutDST)); // 744
var_dump(getMonthHours('2020', '10', $myTimezoneWithoutDST)); // 744
PS Se você marcar um intervalo de tempo (mais longo), o que leva a mais do que essas duas transições, minha solução alternativa não vai tocar nas horas contadas para reduzir o potencial de efeitos colaterais engraçados. Nesses casos, uma solução mais complicada deve ser implementada. Pode-se iterar sobre todas as transições encontradas e comparar a corrente com a última e verificar se é uma com DST verdadeiro-> falso.
strftime()
e dividir a diferença por 3600, mas isso sempre funcionará? Maldito seja, horário de verão!