mirror of
https://github.com/Gnucash/gnucash.git
synced 2024-12-02 05:29:20 -06:00
1221d7ebc1
Extract functions LDT_from_date_time and LDT_from_date_daypart to avoid duplicate code. Handle date-times in start-of-DST transitions and better handle those in end-of-DST transitions. Test the results.
313 lines
12 KiB
C++
313 lines
12 KiB
C++
/********************************************************************
|
|
* Gtest-gnc-int128.cpp -- unit tests for the GncInt128 class *
|
|
* Copyright (C) 2014 John Ralls <jralls@ceridwen.us> *
|
|
* *
|
|
* This program is free software; you can redistribute it and/or *
|
|
* modify it under the terms of the GNU General Public License as *
|
|
* published by the Free Software Foundation; either version 2 of *
|
|
* the License, or (at your option) any later version. *
|
|
* *
|
|
* This program is distributed in the hope that it will be useful, *
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of *
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the *
|
|
* GNU General Public License for more details. *
|
|
* *
|
|
* You should have received a copy of the GNU General Public License*
|
|
* along with this program; if not, contact: *
|
|
* *
|
|
* Free Software Foundation Voice: +1-617-542-5942 *
|
|
* 51 Franklin Street, Fifth Floor Fax: +1-617-542-2652 *
|
|
* Boston, MA 02110-1301, USA gnu@gnu.org *
|
|
* *
|
|
*******************************************************************/
|
|
|
|
#include <gtest/gtest.h>
|
|
#include <string>
|
|
#include "../gnc-timezone.hpp"
|
|
|
|
TEST(gnc_timezone_constructors, test_default_constructor)
|
|
{
|
|
TimeZoneProvider tzp {};
|
|
EXPECT_NO_THROW (tzp.get(2014));
|
|
TZ_Ptr tz = tzp.get (2014);
|
|
|
|
//Can't really test anything explicit, we don't know what to expect
|
|
//from the default TZ.
|
|
EXPECT_FALSE(tz->std_zone_abbrev().empty());
|
|
}
|
|
|
|
TEST(gnc_timezone_constructors, test_pacific_time_constructor)
|
|
{
|
|
#if PLATFORM(WINDOWS)
|
|
std::string timezone("Pacific Standard Time");
|
|
#else
|
|
std::string timezone("America/Los_Angeles");
|
|
#endif
|
|
TimeZoneProvider tzp (timezone);
|
|
EXPECT_NO_THROW (tzp.get(2012));
|
|
TZ_Ptr tz = tzp.get (2012);
|
|
EXPECT_FALSE(tz->std_zone_abbrev().empty());
|
|
#if PLATFORM(WINDOWS)
|
|
EXPECT_TRUE(tz->std_zone_abbrev() == timezone);
|
|
EXPECT_TRUE(tz->dst_zone_abbrev() == "Pacific Daylight Time");
|
|
#else
|
|
EXPECT_TRUE(tz->std_zone_abbrev() == "PST");
|
|
EXPECT_TRUE(tz->dst_zone_abbrev() == "PDT");
|
|
#endif
|
|
EXPECT_EQ(-8, tz->base_utc_offset().hours());
|
|
auto dst_offset = tz->base_utc_offset() + tz->dst_offset();
|
|
EXPECT_EQ(-7, dst_offset.hours());
|
|
EXPECT_EQ(12, tz->dst_local_start_time (2017).date().day());
|
|
EXPECT_EQ(3, tz->dst_local_start_time (2017).date().month());
|
|
EXPECT_EQ(5, tz->dst_local_end_time (2017).date().day());
|
|
EXPECT_EQ(11, tz->dst_local_end_time (2017).date().month());
|
|
//Check some post-2038 dates to make sure that it works even on macOS.
|
|
EXPECT_EQ(10, tz->dst_local_start_time (2052).date().day());
|
|
EXPECT_EQ(3, tz->dst_local_start_time (2052).date().month());
|
|
EXPECT_EQ(3, tz->dst_local_end_time (2052).date().day());
|
|
EXPECT_EQ(11, tz->dst_local_end_time (2052).date().month());
|
|
}
|
|
|
|
#if !PLATFORM(WINDOWS)
|
|
TEST(gnc_timezone_constructors, test_posix_timezone)
|
|
{
|
|
std::string timezone("FST08FDT07,M4.1.0,M10.31.0");
|
|
TimeZoneProvider tzp(timezone);
|
|
TZ_Ptr tz = tzp.get(2006);
|
|
EXPECT_EQ(tz->std_zone_abbrev(), "FST");
|
|
EXPECT_EQ(tz->dst_zone_abbrev(), "FDT");
|
|
EXPECT_TRUE(tz->has_dst());
|
|
EXPECT_EQ(tz->base_utc_offset().hours(), 8L);
|
|
EXPECT_EQ(tz->dst_offset().hours(), 7L);
|
|
}
|
|
|
|
TEST(gnc_timezone_constructors, test_gmt_timezone)
|
|
{
|
|
std::string timezone("GMT");
|
|
TimeZoneProvider tzp(timezone);
|
|
TZ_Ptr tz = tzp.get(2006);
|
|
EXPECT_EQ(tz->std_zone_abbrev(), "GMT");
|
|
EXPECT_FALSE(tz->has_dst());
|
|
EXPECT_EQ(tz->dst_zone_abbrev(), "");
|
|
EXPECT_EQ(tz->base_utc_offset().hours(), 0L);
|
|
EXPECT_EQ(tz->dst_offset().hours(), 0L);
|
|
}
|
|
|
|
TEST(gnc_timezone_constructors, test_GMT_plus_7_timezone)
|
|
{
|
|
std::string timezone("Etc/GMT+7");
|
|
TimeZoneProvider tzp(timezone);
|
|
TZ_Ptr tz = tzp.get(2006);
|
|
EXPECT_EQ(tz->std_zone_abbrev(), "-07");
|
|
EXPECT_EQ(tz->dst_zone_abbrev(), "");
|
|
EXPECT_FALSE(tz->has_dst());
|
|
EXPECT_EQ(tz->base_utc_offset().hours(), -7);
|
|
EXPECT_EQ(tz->dst_offset().hours(), 0L);
|
|
}
|
|
|
|
TEST(gnc_timezone_constructors, test_IANA_Belize_tz)
|
|
{
|
|
TimeZoneProvider tzp("America/Belize");
|
|
for (int year = 1908; year < 1990; ++year)
|
|
{
|
|
auto tz = tzp.get(year);
|
|
if (year < 1912)
|
|
{
|
|
EXPECT_EQ(tz->std_zone_abbrev(), "LMT");
|
|
EXPECT_FALSE(tz->has_dst());
|
|
EXPECT_EQ(tz->base_utc_offset().total_seconds(), -21168);
|
|
}
|
|
else if (year < 1918)
|
|
{
|
|
EXPECT_EQ(tz->std_zone_abbrev(), "CST");
|
|
EXPECT_FALSE(tz->has_dst());
|
|
EXPECT_EQ(tz->base_utc_offset().total_seconds(), -21600);
|
|
}
|
|
else if (year < 1943)
|
|
{
|
|
EXPECT_EQ(tz->std_zone_abbrev(), "CST");
|
|
EXPECT_TRUE(tz->has_dst());
|
|
EXPECT_EQ(tz->base_utc_offset().total_seconds(), -21600);
|
|
EXPECT_EQ(tz->dst_zone_abbrev(), "-0530");
|
|
EXPECT_EQ(tz->dst_offset().total_seconds(), 1800);
|
|
}
|
|
else if (year == 1973 || year == 1982)
|
|
{
|
|
EXPECT_EQ(tz->std_zone_abbrev(), "CST");
|
|
EXPECT_TRUE(tz->has_dst());
|
|
EXPECT_EQ(tz->base_utc_offset().total_seconds(), -21600);
|
|
EXPECT_EQ(tz->dst_zone_abbrev(), "CDT");
|
|
EXPECT_EQ(tz->dst_offset().total_seconds(), 3600);
|
|
}
|
|
/* An IANA update on 22 Dec 2020 added missing DST transitions
|
|
* for Belize between 1943 and 1967. Ignore those years until
|
|
* the oldest supported OS version release is later than that
|
|
* because distros updating of zoneinfo is spotty.
|
|
*/
|
|
else if (year >= 1943 && year <= 1967)
|
|
continue;
|
|
else
|
|
{
|
|
EXPECT_EQ(tz->std_zone_abbrev(), "CST");
|
|
EXPECT_FALSE(tz->has_dst());
|
|
EXPECT_EQ(tz->base_utc_offset().total_seconds(), -21600);
|
|
}
|
|
}
|
|
}
|
|
|
|
TEST(gnc_timezone_constructors, test_IANA_Perth_tz)
|
|
{
|
|
TimeZoneProvider tzp("Australia/Perth");
|
|
for (int year = 1916; year < 2048; ++year)
|
|
{
|
|
auto tz = tzp.get(year);
|
|
if (year < 1917)
|
|
{
|
|
EXPECT_EQ(tz->std_zone_abbrev(), "AWST");
|
|
EXPECT_TRUE(tz->has_dst());
|
|
EXPECT_EQ(tz->base_utc_offset().total_seconds(), 28800);
|
|
EXPECT_EQ(tz->dst_zone_abbrev(), "AWDT");
|
|
EXPECT_EQ(tz->dst_offset().total_seconds(), 3600);
|
|
}
|
|
else if (year < 1941)
|
|
{
|
|
EXPECT_EQ(tz->std_zone_abbrev(), "AWST");
|
|
EXPECT_FALSE(tz->has_dst());
|
|
EXPECT_EQ(tz->base_utc_offset().total_seconds(), 28800);
|
|
}
|
|
else if (year < 1943)
|
|
{
|
|
EXPECT_EQ(tz->std_zone_abbrev(), "AWST");
|
|
EXPECT_TRUE(tz->has_dst());
|
|
EXPECT_EQ(tz->base_utc_offset().total_seconds(), 28800);
|
|
EXPECT_EQ(tz->dst_zone_abbrev(), "AWDT");
|
|
EXPECT_EQ(tz->dst_offset().total_seconds(), 3600);
|
|
}
|
|
else if (year == 1974 || year == 1983 || year == 1991 ||
|
|
(year > 2005 && year < 2009))
|
|
{
|
|
EXPECT_EQ(tz->std_zone_abbrev(), "AWST");
|
|
EXPECT_TRUE(tz->has_dst());
|
|
EXPECT_EQ(tz->base_utc_offset().total_seconds(), 28800);
|
|
EXPECT_EQ(tz->dst_zone_abbrev(), "AWDT");
|
|
EXPECT_EQ(tz->dst_offset().total_seconds(), 3600);
|
|
}
|
|
else
|
|
{
|
|
EXPECT_EQ(tz->std_zone_abbrev(), "AWST");
|
|
EXPECT_FALSE(tz->has_dst());
|
|
EXPECT_EQ(tz->base_utc_offset().total_seconds(), 28800);
|
|
}
|
|
}
|
|
}
|
|
|
|
TEST(gnc_timezone_constructors, test_IANA_Minsk_tz)
|
|
{
|
|
TimeZoneProvider tzp("Europe/Minsk");
|
|
for (int year = 1916; year < 2020; ++year)
|
|
{
|
|
auto tz = tzp.get(year);
|
|
if (year < 1924)
|
|
{
|
|
EXPECT_EQ(tz->std_zone_abbrev(), "MMT");
|
|
EXPECT_FALSE(tz->has_dst());
|
|
EXPECT_EQ(tz->base_utc_offset().total_seconds(), 6600);
|
|
}
|
|
else if (year < 1930)
|
|
{
|
|
EXPECT_EQ(tz->std_zone_abbrev(), "EET");
|
|
EXPECT_FALSE(tz->has_dst());
|
|
EXPECT_EQ(tz->base_utc_offset().total_seconds(), 7200);
|
|
}
|
|
else if (year < 1941)
|
|
{
|
|
EXPECT_EQ(tz->std_zone_abbrev(), "MSK");
|
|
EXPECT_FALSE(tz->has_dst());
|
|
EXPECT_EQ(tz->base_utc_offset().total_seconds(), 10800);
|
|
}
|
|
/* The TZInfo says Minsk had DST from June 1941 - Nov
|
|
* 1942. Boost::date_time doesn't know how to model that so we
|
|
* just pretend that it was a weird standard time. Note that
|
|
* Minsk was under German occupation and got shifted to Berlin
|
|
* time, sort of.
|
|
*/
|
|
else if (year < 1943)
|
|
{
|
|
EXPECT_EQ(tz->std_zone_abbrev(), "CEST");
|
|
EXPECT_FALSE(tz->has_dst());
|
|
EXPECT_EQ(tz->base_utc_offset().total_seconds(), 7200);
|
|
EXPECT_EQ(tz->dst_zone_abbrev(), "");
|
|
EXPECT_EQ(tz->dst_offset().total_seconds(), 0);
|
|
}
|
|
else if (year == 1943)
|
|
{
|
|
EXPECT_EQ(tz->std_zone_abbrev(), "CET");
|
|
EXPECT_TRUE(tz->has_dst());
|
|
EXPECT_EQ(tz->base_utc_offset().total_seconds(), 3600);
|
|
EXPECT_EQ(tz->dst_zone_abbrev(), "CEST");
|
|
EXPECT_EQ(tz->dst_offset().total_seconds(), 3600);
|
|
}
|
|
/* Minsk was "liberated" by the Soviets 2 Jul 1944 and went
|
|
* back to a more reasonable local time with no DST. Another
|
|
* case that's too hard for boost::timezone to model correctly
|
|
* so we fudge.
|
|
*/
|
|
else if (year == 1944)
|
|
{
|
|
EXPECT_EQ(tz->std_zone_abbrev(), "MSK");
|
|
EXPECT_TRUE(tz->has_dst());
|
|
EXPECT_EQ(tz->base_utc_offset().total_seconds(), 10800);
|
|
EXPECT_EQ(tz->dst_zone_abbrev(), "CEST");
|
|
EXPECT_EQ(tz->dst_offset().total_seconds(), -3600);
|
|
}
|
|
else if (year < 1981)
|
|
{
|
|
EXPECT_EQ(tz->std_zone_abbrev(), "MSK");
|
|
EXPECT_FALSE(tz->has_dst());
|
|
EXPECT_EQ(tz->base_utc_offset().total_seconds(), 10800);
|
|
}
|
|
else if (year < 1989)
|
|
{
|
|
EXPECT_EQ(tz->std_zone_abbrev(), "MSK");
|
|
EXPECT_TRUE(tz->has_dst());
|
|
EXPECT_EQ(tz->base_utc_offset().total_seconds(), 10800);
|
|
EXPECT_EQ(tz->dst_zone_abbrev(), "MSD");
|
|
EXPECT_EQ(tz->dst_offset().total_seconds(), 3600);
|
|
}
|
|
else if (year < 1991)
|
|
{
|
|
EXPECT_EQ(tz->std_zone_abbrev(), "MSK");
|
|
EXPECT_FALSE(tz->has_dst());
|
|
EXPECT_EQ(tz->base_utc_offset().total_seconds(), 10800);
|
|
}
|
|
else if (year < 2011)
|
|
{
|
|
EXPECT_EQ(tz->std_zone_abbrev(), "EET");
|
|
EXPECT_TRUE(tz->has_dst());
|
|
EXPECT_EQ(tz->base_utc_offset().total_seconds(), 7200);
|
|
EXPECT_EQ(tz->dst_zone_abbrev(), "EEST");
|
|
EXPECT_EQ(tz->dst_offset().total_seconds(), 3600);
|
|
}
|
|
else
|
|
{
|
|
EXPECT_EQ(tz->std_zone_abbrev(), "+03");
|
|
EXPECT_FALSE(tz->has_dst());
|
|
EXPECT_EQ(tz->base_utc_offset().total_seconds(), 10800);
|
|
}
|
|
}
|
|
}
|
|
#endif
|
|
|
|
TEST(gnc_timezone_constructors, test_bogus_time_constructor)
|
|
{
|
|
#if PLATFORM(WINDOWS)
|
|
EXPECT_THROW(TimeZoneProvider tzp ("New York Standard Time"),
|
|
std::invalid_argument);
|
|
#else
|
|
TimeZoneProvider tzp ("New York Standard Time");
|
|
TimeZoneProvider machine ("");
|
|
EXPECT_EQ(machine.get(2006)->std_zone_abbrev(),
|
|
tzp.get(2006)->std_zone_abbrev());
|
|
#endif
|
|
}
|