Skip to content

Commit 43e3161

Browse files
committed
Test ambiguous time
1 parent 989826c commit 43e3161

File tree

3 files changed

+24
-9
lines changed

3 files changed

+24
-9
lines changed

.github/workflows/linux.yml

+1-1
Original file line numberDiff line numberDiff line change
@@ -58,7 +58,7 @@ jobs:
5858
- uses: actions/checkout@d632683dd7b4114ad314bca15554477dd762a938 # v4.2.0
5959

6060
- name: Set timezone
61-
run: sudo timedatectl set-timezone 'Europe/Minsk'
61+
run: sudo timedatectl set-timezone 'Europe/Kyiv'
6262

6363
- name: Install GCC 4.9
6464
run: |

include/fmt/chrono.h

+10-5
Original file line numberDiff line numberDiff line change
@@ -2320,15 +2320,20 @@ struct formatter<local_time<Duration>, Char> : formatter<std::tm, Char> {
23202320
template <typename FormatContext>
23212321
auto format(local_time<Duration> val, FormatContext& ctx) const
23222322
-> decltype(ctx.out()) {
2323+
auto time_since_epoch = val.time_since_epoch();
2324+
auto seconds_since_epoch =
2325+
detail::duration_cast<std::chrono::seconds>(time_since_epoch);
2326+
// Use gmtime to prevent time conversion since local_time has an
2327+
// unspecified time zone.
2328+
auto t = gmtime(seconds_since_epoch.count());
23232329
using period = typename Duration::period;
23242330
if (period::num == 1 && period::den == 1 &&
23252331
!std::is_floating_point<typename Duration::rep>::value) {
2326-
return formatter<std::tm, Char>::format(localtime(val), ctx);
2332+
return formatter<std::tm, Char>::format(t, ctx);
23272333
}
2328-
auto epoch = val.time_since_epoch();
2329-
auto subsecs = detail::duration_cast<Duration>(
2330-
epoch - detail::duration_cast<std::chrono::seconds>(epoch));
2331-
return formatter<std::tm, Char>::do_format(localtime(val), ctx, &subsecs);
2334+
auto subsecs =
2335+
detail::duration_cast<Duration>(time_since_epoch - seconds_since_epoch);
2336+
return formatter<std::tm, Char>::do_format(t, ctx, &subsecs);
23322337
}
23332338
};
23342339

test/chrono-test.cc

+13-3
Original file line numberDiff line numberDiff line change
@@ -428,6 +428,18 @@ TEST(chrono_test, local_system_clock_time_point) {
428428

429429
#endif // FMT_USE_LOCAL_TIME
430430

431+
TEST(chrono_test, daylight_savings_time_end) {
432+
// 2024-10-27 03:05 as the number of seconds since epoch in Europe/Kyiv time.
433+
// It is slightly after the DST end and passing it to to_sys will result in
434+
// an ambiguous time error:
435+
// 2024-10-27 03:05:00 is ambiguous. It could be
436+
// 2024-10-27 03:05:00 EEST == 2024-10-27 00:05:00 UTC or
437+
// 2024-10-27 03:05:00 EET == 2024-10-27 01:05:00 UTC
438+
auto t =
439+
fmt::local_time<std::chrono::seconds>(std::chrono::seconds(1729998300));
440+
EXPECT_EQ(fmt::format("{}", t), "2024-10-27 03:05:00");
441+
}
442+
431443
#ifndef FMT_STATIC_THOUSANDS_SEPARATOR
432444

433445
TEST(chrono_test, format_default) {
@@ -1033,7 +1045,7 @@ TEST(chrono_test, glibc_extensions) {
10331045
{
10341046
auto t = std::tm();
10351047
t.tm_year = -5 - 1900;
1036-
EXPECT_EQ(fmt::format( "{:%Y}", t), "-005");
1048+
EXPECT_EQ(fmt::format("{:%Y}", t), "-005");
10371049
EXPECT_EQ(fmt::format("{:%_Y}", t), " -5");
10381050
EXPECT_EQ(fmt::format("{:%-Y}", t), "-5");
10391051
}
@@ -1045,8 +1057,6 @@ TEST(chrono_test, glibc_extensions) {
10451057
EXPECT_EQ(fmt::format("{:%_m}", t), " 7");
10461058
EXPECT_EQ(fmt::format("{:%-m}", t), "7");
10471059
}
1048-
1049-
10501060
}
10511061

10521062
TEST(chrono_test, out_of_range) {

0 commit comments

Comments
 (0)