xref: /llvm-project/libcxx/test/std/time/time.syn/formatter.day.pass.cpp (revision 6a54dfbfe534276d644d7f9c027f0deeb748dd53)
1e5d2d3eaSMark de Wever //===----------------------------------------------------------------------===//
2*6a54dfbfSLouis Dionne //
3e5d2d3eaSMark de Wever // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4e5d2d3eaSMark de Wever // See https://llvm.org/LICENSE.txt for license information.
5e5d2d3eaSMark de Wever // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6e5d2d3eaSMark de Wever //
7e5d2d3eaSMark de Wever //===----------------------------------------------------------------------===//
8e5d2d3eaSMark de Wever 
9e5d2d3eaSMark de Wever // UNSUPPORTED: c++03, c++11, c++14, c++17
10d529e811SLouis Dionne // UNSUPPORTED: no-localization
11520c7fbbSLouis Dionne // UNSUPPORTED: GCC-ALWAYS_INLINE-FIXME
12e5d2d3eaSMark de Wever 
133d334df5SLouis Dionne // TODO FMT This test should not require std::to_chars(floating-point)
14f0fc8c48SLouis Dionne // XFAIL: availability-fp_to_chars-missing
153d334df5SLouis Dionne 
161403080aSMark de Wever // TODO FMT Investigate Windows issues.
171403080aSMark de Wever // XFAIL: msvc
181403080aSMark de Wever 
19e5d2d3eaSMark de Wever // REQUIRES: locale.fr_FR.UTF-8
20e5d2d3eaSMark de Wever // REQUIRES: locale.ja_JP.UTF-8
21e5d2d3eaSMark de Wever 
22e5d2d3eaSMark de Wever // <chrono>
23e5d2d3eaSMark de Wever 
24e5d2d3eaSMark de Wever // template<class charT> struct formatter<chrono::day, charT>;
25e5d2d3eaSMark de Wever 
26e5d2d3eaSMark de Wever #include <chrono>
27e5d2d3eaSMark de Wever #include <format>
28e5d2d3eaSMark de Wever 
29e5d2d3eaSMark de Wever #include <cassert>
30e5d2d3eaSMark de Wever #include <concepts>
31e5d2d3eaSMark de Wever #include <locale>
32e5d2d3eaSMark de Wever #include <iostream>
33e5d2d3eaSMark de Wever #include <type_traits>
34e5d2d3eaSMark de Wever 
35e5d2d3eaSMark de Wever #include "formatter_tests.h"
36e5d2d3eaSMark de Wever #include "make_string.h"
37e5d2d3eaSMark de Wever #include "platform_support.h" // locale name macros
38e5d2d3eaSMark de Wever #include "string_literal.h"
39e5d2d3eaSMark de Wever #include "test_macros.h"
40e5d2d3eaSMark de Wever 
41e5d2d3eaSMark de Wever template <class CharT>
42e5d2d3eaSMark de Wever static void test_no_chrono_specs() {
43e5d2d3eaSMark de Wever   using namespace std::literals::chrono_literals;
44e5d2d3eaSMark de Wever 
45e5d2d3eaSMark de Wever   // Valid day
46d23f609dSMark de Wever   check(SV("01"), SV("{}"), 1d);
47d23f609dSMark de Wever   check(SV("*01*"), SV("{:*^4}"), 1d);
48d23f609dSMark de Wever   check(SV("*01"), SV("{:*>3}"), 1d);
49e5d2d3eaSMark de Wever 
50e5d2d3eaSMark de Wever   // Invalid day
51d23f609dSMark de Wever   check(SV("00 is not a valid day"), SV("{}"), 0d);
52d23f609dSMark de Wever   check(SV("*00 is not a valid day*"), SV("{:*^23}"), 0d);
53e5d2d3eaSMark de Wever }
54e5d2d3eaSMark de Wever 
55e5d2d3eaSMark de Wever template <class CharT>
56e5d2d3eaSMark de Wever static void test_valid_values() {
57e5d2d3eaSMark de Wever   using namespace std::literals::chrono_literals;
58e5d2d3eaSMark de Wever 
59d23f609dSMark de Wever   constexpr std::basic_string_view<CharT> fmt  = SV("{:%%d='%d'%t%%Od='%Od'%t%%e='%e'%t%%Oe='%Oe'%n}");
60d23f609dSMark de Wever   constexpr std::basic_string_view<CharT> lfmt = SV("{:L%%d='%d'%t%%Od='%Od'%t%%e='%e'%t%%Oe='%Oe'%n}");
61e5d2d3eaSMark de Wever 
62e5d2d3eaSMark de Wever   const std::locale loc(LOCALE_ja_JP_UTF_8);
63e5d2d3eaSMark de Wever   std::locale::global(std::locale(LOCALE_fr_FR_UTF_8));
64e5d2d3eaSMark de Wever 
65e5d2d3eaSMark de Wever   // Non localized output using C-locale
661403080aSMark de Wever #if defined(_WIN32)
671403080aSMark de Wever   check(SV("%d=''\t%Od=''\t%e=''\t%Oe=''\n"), fmt, 0d);
681403080aSMark de Wever #else
69d23f609dSMark de Wever   check(SV("%d='00'\t%Od='00'\t%e=' 0'\t%Oe=' 0'\n"), fmt, 0d);
701403080aSMark de Wever #endif
71d23f609dSMark de Wever   check(SV("%d='01'\t%Od='01'\t%e=' 1'\t%Oe=' 1'\n"), fmt, 1d);
72d23f609dSMark de Wever   check(SV("%d='31'\t%Od='31'\t%e='31'\t%Oe='31'\n"), fmt, 31d);
731403080aSMark de Wever #if defined(_WIN32)
741403080aSMark de Wever   check(SV("%d=''\t%Od=''\t%e=''\t%Oe=''\n"), fmt, 0d);
751403080aSMark de Wever #elif defined(_AIX)
76d23f609dSMark de Wever   check(SV("%d='55'\t%Od='55'\t%e='55'\t%Oe='55'\n"), fmt, 255d);
77e5d2d3eaSMark de Wever #else
78d23f609dSMark de Wever   check(SV("%d='255'\t%Od='255'\t%e='255'\t%Oe='255'\n"), fmt, 255d);
79e5d2d3eaSMark de Wever #endif
80e5d2d3eaSMark de Wever 
81e5d2d3eaSMark de Wever   // Use the global locale (fr_FR)
821403080aSMark de Wever #if defined(_WIN32)
831403080aSMark de Wever   check(SV("%d=''\t%Od=''\t%e=''\t%Oe=''\n"), lfmt, 0d);
841403080aSMark de Wever #else
85d23f609dSMark de Wever   check(SV("%d='00'\t%Od='00'\t%e=' 0'\t%Oe=' 0'\n"), lfmt, 0d);
861403080aSMark de Wever #endif
87d23f609dSMark de Wever   check(SV("%d='01'\t%Od='01'\t%e=' 1'\t%Oe=' 1'\n"), lfmt, 1d);
88d23f609dSMark de Wever   check(SV("%d='31'\t%Od='31'\t%e='31'\t%Oe='31'\n"), lfmt, 31d);
891403080aSMark de Wever #if defined(_WIN32)
901403080aSMark de Wever   check(SV("%d=''\t%Od=''\t%e=''\t%Oe=''\n"), lfmt, 255d);
911403080aSMark de Wever #elif defined(_AIX)
92d23f609dSMark de Wever   check(SV("%d='55'\t%Od='55'\t%e='55'\t%Oe='55'\n"), lfmt, 255d);
93e5d2d3eaSMark de Wever #else
94d23f609dSMark de Wever   check(SV("%d='255'\t%Od='255'\t%e='255'\t%Oe='255'\n"), lfmt, 255d);
95e5d2d3eaSMark de Wever #endif
96e5d2d3eaSMark de Wever 
97e5d2d3eaSMark de Wever   // Use supplied locale (ja_JP). This locale has a different alternate on some platforms.
988f01029bSMark de Wever #if defined(__APPLE__) || defined(_AIX) || defined(_WIN32) || defined(__FreeBSD__)
991403080aSMark de Wever #  if defined(_WIN32)
1001403080aSMark de Wever   check(loc, SV("%d=''\t%Od=''\t%e=''\t%Oe=''\n"), lfmt, 0d);
1011403080aSMark de Wever #  else
102d23f609dSMark de Wever   check(loc, SV("%d='00'\t%Od='00'\t%e=' 0'\t%Oe=' 0'\n"), lfmt, 0d);
1031403080aSMark de Wever #  endif
104d23f609dSMark de Wever   check(loc, SV("%d='01'\t%Od='01'\t%e=' 1'\t%Oe=' 1'\n"), lfmt, 1d);
105d23f609dSMark de Wever   check(loc, SV("%d='31'\t%Od='31'\t%e='31'\t%Oe='31'\n"), lfmt, 31d);
1061403080aSMark de Wever #  if defined(_WIN32)
1071403080aSMark de Wever   check(SV("%d=''\t%Od=''\t%e=''\t%Oe=''\n"), fmt, 255d);
1081403080aSMark de Wever #  elif defined(_AIX)
109d23f609dSMark de Wever   check(SV("%d='55'\t%Od='55'\t%e='55'\t%Oe='55'\n"), fmt, 255d);
110e5d2d3eaSMark de Wever #  else
111d23f609dSMark de Wever   check(SV("%d='255'\t%Od='255'\t%e='255'\t%Oe='255'\n"), fmt, 255d);
112e5d2d3eaSMark de Wever #  endif
1138f01029bSMark de Wever #else  // defined(__APPLE__) || defined(_AIX) || defined(_WIN32) || defined(__FreeBSD__)
114d23f609dSMark de Wever   check(loc, SV("%d='00'\t%Od='〇'\t%e=' 0'\t%Oe='〇'\n"), lfmt, 0d);
115d23f609dSMark de Wever   check(loc, SV("%d='01'\t%Od='一'\t%e=' 1'\t%Oe='一'\n"), lfmt, 1d);
116d23f609dSMark de Wever   check(loc, SV("%d='31'\t%Od='三十一'\t%e='31'\t%Oe='三十一'\n"), lfmt, 31d);
117d23f609dSMark de Wever   check(loc, SV("%d='255'\t%Od='255'\t%e='255'\t%Oe='255'\n"), lfmt, 255d);
1188f01029bSMark de Wever #endif // defined(__APPLE__) || defined(_AIX) || defined(_WIN32) || defined(__FreeBSD__)
119e5d2d3eaSMark de Wever 
120e5d2d3eaSMark de Wever   std::locale::global(std::locale::classic());
121e5d2d3eaSMark de Wever }
122e5d2d3eaSMark de Wever 
123e5d2d3eaSMark de Wever template <class CharT>
124e5d2d3eaSMark de Wever static void test() {
125e5d2d3eaSMark de Wever   using namespace std::literals::chrono_literals;
126e5d2d3eaSMark de Wever 
127e5d2d3eaSMark de Wever   test_no_chrono_specs<CharT>();
128e5d2d3eaSMark de Wever   test_valid_values<CharT>();
129e5d2d3eaSMark de Wever   check_invalid_types<CharT>({SV("d"), SV("e"), SV("Od"), SV("Oe")}, 0d);
130e5d2d3eaSMark de Wever 
131402eb2efSMark de Wever   check_exception("The format specifier expects a '%' or a '}'", SV("{:A"), 0d);
132402eb2efSMark de Wever   check_exception("The chrono specifiers contain a '{'", SV("{:%%{"), 0d);
133402eb2efSMark de Wever   check_exception("End of input while parsing a conversion specifier", SV("{:%"), 0d);
134e5d2d3eaSMark de Wever   check_exception("End of input while parsing the modifier E", SV("{:%E"), 0d);
135e5d2d3eaSMark de Wever   check_exception("End of input while parsing the modifier O", SV("{:%O"), 0d);
136e5d2d3eaSMark de Wever 
137e5d2d3eaSMark de Wever   // Precision not allowed
138402eb2efSMark de Wever   check_exception("The format specifier expects a '%' or a '}'", SV("{:.3}"), 0d);
139e5d2d3eaSMark de Wever }
140e5d2d3eaSMark de Wever 
141e5d2d3eaSMark de Wever int main(int, char**) {
142e5d2d3eaSMark de Wever   test<char>();
143e5d2d3eaSMark de Wever 
144e5d2d3eaSMark de Wever #ifndef TEST_HAS_NO_WIDE_CHARACTERS
145e5d2d3eaSMark de Wever   test<wchar_t>();
146e5d2d3eaSMark de Wever #endif
147e5d2d3eaSMark de Wever 
148e5d2d3eaSMark de Wever   return 0;
149e5d2d3eaSMark de Wever }
150