1566868cdSMark de Wever //===----------------------------------------------------------------------===// 2*6a54dfbfSLouis Dionne // 3566868cdSMark de Wever // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. 4566868cdSMark de Wever // See https://llvm.org/LICENSE.txt for license information. 5566868cdSMark de Wever // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception 6566868cdSMark de Wever // 7566868cdSMark de Wever //===----------------------------------------------------------------------===// 8566868cdSMark de Wever 9566868cdSMark de Wever // UNSUPPORTED: c++03, c++11, c++14, c++17 10566868cdSMark de Wever // UNSUPPORTED: no-localization 11520c7fbbSLouis Dionne // UNSUPPORTED: GCC-ALWAYS_INLINE-FIXME 12566868cdSMark 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 19566868cdSMark de Wever // REQUIRES: locale.fr_FR.UTF-8 20566868cdSMark de Wever // REQUIRES: locale.ja_JP.UTF-8 21566868cdSMark de Wever 22566868cdSMark de Wever // <chrono> 23566868cdSMark de Wever 24566868cdSMark de Wever // template<class charT> struct formatter<chrono::weekday, charT>; 25566868cdSMark de Wever 26566868cdSMark de Wever #include <chrono> 27566868cdSMark de Wever #include <format> 28566868cdSMark de Wever 29566868cdSMark de Wever #include <cassert> 30566868cdSMark de Wever #include <concepts> 31566868cdSMark de Wever #include <locale> 32566868cdSMark de Wever #include <iostream> 33566868cdSMark de Wever #include <type_traits> 34566868cdSMark de Wever 35566868cdSMark de Wever #include "formatter_tests.h" 36566868cdSMark de Wever #include "make_string.h" 37566868cdSMark de Wever #include "platform_support.h" // locale name macros 38566868cdSMark de Wever #include "test_macros.h" 39566868cdSMark de Wever 40566868cdSMark de Wever template <class CharT> 41566868cdSMark de Wever static void test_no_chrono_specs() { 42566868cdSMark de Wever using namespace std::literals::chrono_literals; 43566868cdSMark de Wever 44566868cdSMark de Wever // Valid day 45566868cdSMark de Wever check(SV("Sun"), SV("{}"), std::chrono::weekday(0)); 46566868cdSMark de Wever 47566868cdSMark de Wever // Invalid day 48566868cdSMark de Wever check(SV("8 is not a valid weekday"), SV("{}"), std::chrono::weekday(8)); 49566868cdSMark de Wever } 50566868cdSMark de Wever 51566868cdSMark de Wever template <class CharT> 52566868cdSMark de Wever static void test_valid_values() { 53566868cdSMark de Wever constexpr std::basic_string_view<CharT> fmt = 54566868cdSMark de Wever SV("{:%%u='%u'%t%%Ou='%Ou'%t%%w='%w'%t%%Ow='%Ow'%t%%a='%a'%t%%A='%A'%n}"); 55566868cdSMark de Wever constexpr std::basic_string_view<CharT> lfmt = 56566868cdSMark de Wever SV("{:L%%u='%u'%t%%Ou='%Ou'%t%%w='%w'%t%%Ow='%Ow'%t%%a='%a'%t%%A='%A'%n}"); 57566868cdSMark de Wever 58566868cdSMark de Wever const std::locale loc(LOCALE_ja_JP_UTF_8); 59566868cdSMark de Wever std::locale::global(std::locale(LOCALE_fr_FR_UTF_8)); 60566868cdSMark de Wever 61566868cdSMark de Wever // Non localized output using C-locale 62566868cdSMark de Wever check(SV("%u='7'\t%Ou='7'\t%w='0'\t%Ow='0'\t%a='Sun'\t%A='Sunday'\n"), fmt, std::chrono::weekday(0)); 63566868cdSMark de Wever check(SV("%u='1'\t%Ou='1'\t%w='1'\t%Ow='1'\t%a='Mon'\t%A='Monday'\n"), fmt, std::chrono::weekday(1)); 64566868cdSMark de Wever check(SV("%u='2'\t%Ou='2'\t%w='2'\t%Ow='2'\t%a='Tue'\t%A='Tuesday'\n"), fmt, std::chrono::weekday(2)); 65566868cdSMark de Wever check(SV("%u='3'\t%Ou='3'\t%w='3'\t%Ow='3'\t%a='Wed'\t%A='Wednesday'\n"), fmt, std::chrono::weekday(3)); 66566868cdSMark de Wever check(SV("%u='4'\t%Ou='4'\t%w='4'\t%Ow='4'\t%a='Thu'\t%A='Thursday'\n"), fmt, std::chrono::weekday(4)); 67566868cdSMark de Wever check(SV("%u='5'\t%Ou='5'\t%w='5'\t%Ow='5'\t%a='Fri'\t%A='Friday'\n"), fmt, std::chrono::weekday(5)); 68566868cdSMark de Wever check(SV("%u='6'\t%Ou='6'\t%w='6'\t%Ow='6'\t%a='Sat'\t%A='Saturday'\n"), fmt, std::chrono::weekday(6)); 69566868cdSMark de Wever check(SV("%u='7'\t%Ou='7'\t%w='0'\t%Ow='0'\t%a='Sun'\t%A='Sunday'\n"), fmt, std::chrono::weekday(7)); 70566868cdSMark de Wever 71566868cdSMark de Wever #if defined(__APPLE__) 72566868cdSMark de Wever // Use the global locale (fr_FR) 73566868cdSMark de Wever check(SV("%u='7'\t%Ou='7'\t%w='0'\t%Ow='0'\t%a='Dim'\t%A='Dimanche'\n"), lfmt, std::chrono::weekday(0)); 74566868cdSMark de Wever check(SV("%u='1'\t%Ou='1'\t%w='1'\t%Ow='1'\t%a='Lun'\t%A='Lundi'\n"), lfmt, std::chrono::weekday(1)); 75566868cdSMark de Wever check(SV("%u='2'\t%Ou='2'\t%w='2'\t%Ow='2'\t%a='Mar'\t%A='Mardi'\n"), lfmt, std::chrono::weekday(2)); 76566868cdSMark de Wever check(SV("%u='3'\t%Ou='3'\t%w='3'\t%Ow='3'\t%a='Mer'\t%A='Mercredi'\n"), lfmt, std::chrono::weekday(3)); 77566868cdSMark de Wever check(SV("%u='4'\t%Ou='4'\t%w='4'\t%Ow='4'\t%a='Jeu'\t%A='Jeudi'\n"), lfmt, std::chrono::weekday(4)); 78566868cdSMark de Wever check(SV("%u='5'\t%Ou='5'\t%w='5'\t%Ow='5'\t%a='Ven'\t%A='Vendredi'\n"), lfmt, std::chrono::weekday(5)); 79566868cdSMark de Wever check(SV("%u='6'\t%Ou='6'\t%w='6'\t%Ow='6'\t%a='Sam'\t%A='Samedi'\n"), lfmt, std::chrono::weekday(6)); 80566868cdSMark de Wever check(SV("%u='7'\t%Ou='7'\t%w='0'\t%Ow='0'\t%a='Dim'\t%A='Dimanche'\n"), lfmt, std::chrono::weekday(7)); 81566868cdSMark de Wever #else 82566868cdSMark de Wever // Use the global locale (fr_FR) 83566868cdSMark de Wever check(SV("%u='7'\t%Ou='7'\t%w='0'\t%Ow='0'\t%a='dim.'\t%A='dimanche'\n"), lfmt, std::chrono::weekday(0)); 84566868cdSMark de Wever check(SV("%u='1'\t%Ou='1'\t%w='1'\t%Ow='1'\t%a='lun.'\t%A='lundi'\n"), lfmt, std::chrono::weekday(1)); 85566868cdSMark de Wever check(SV("%u='2'\t%Ou='2'\t%w='2'\t%Ow='2'\t%a='mar.'\t%A='mardi'\n"), lfmt, std::chrono::weekday(2)); 86566868cdSMark de Wever check(SV("%u='3'\t%Ou='3'\t%w='3'\t%Ow='3'\t%a='mer.'\t%A='mercredi'\n"), lfmt, std::chrono::weekday(3)); 87566868cdSMark de Wever check(SV("%u='4'\t%Ou='4'\t%w='4'\t%Ow='4'\t%a='jeu.'\t%A='jeudi'\n"), lfmt, std::chrono::weekday(4)); 88566868cdSMark de Wever check(SV("%u='5'\t%Ou='5'\t%w='5'\t%Ow='5'\t%a='ven.'\t%A='vendredi'\n"), lfmt, std::chrono::weekday(5)); 89566868cdSMark de Wever check(SV("%u='6'\t%Ou='6'\t%w='6'\t%Ow='6'\t%a='sam.'\t%A='samedi'\n"), lfmt, std::chrono::weekday(6)); 90566868cdSMark de Wever check(SV("%u='7'\t%Ou='7'\t%w='0'\t%Ow='0'\t%a='dim.'\t%A='dimanche'\n"), lfmt, std::chrono::weekday(7)); 91566868cdSMark de Wever #endif 92566868cdSMark de Wever 93566868cdSMark de Wever // Use supplied locale (ja_JP). 94566868cdSMark de Wever // This locale has a different alternate, but not on all platforms 958f01029bSMark de Wever #if defined(_WIN32) || defined(__APPLE__) || defined(_AIX) || defined(__FreeBSD__) 96566868cdSMark de Wever check(loc, SV("%u='7'\t%Ou='7'\t%w='0'\t%Ow='0'\t%a='日'\t%A='日曜日'\n"), lfmt, std::chrono::weekday(0)); 97566868cdSMark de Wever check(loc, SV("%u='1'\t%Ou='1'\t%w='1'\t%Ow='1'\t%a='月'\t%A='月曜日'\n"), lfmt, std::chrono::weekday(1)); 98566868cdSMark de Wever check(loc, SV("%u='2'\t%Ou='2'\t%w='2'\t%Ow='2'\t%a='火'\t%A='火曜日'\n"), lfmt, std::chrono::weekday(2)); 99566868cdSMark de Wever check(loc, SV("%u='3'\t%Ou='3'\t%w='3'\t%Ow='3'\t%a='水'\t%A='水曜日'\n"), lfmt, std::chrono::weekday(3)); 100566868cdSMark de Wever check(loc, SV("%u='4'\t%Ou='4'\t%w='4'\t%Ow='4'\t%a='木'\t%A='木曜日'\n"), lfmt, std::chrono::weekday(4)); 101566868cdSMark de Wever check(loc, SV("%u='5'\t%Ou='5'\t%w='5'\t%Ow='5'\t%a='金'\t%A='金曜日'\n"), lfmt, std::chrono::weekday(5)); 102566868cdSMark de Wever check(loc, SV("%u='6'\t%Ou='6'\t%w='6'\t%Ow='6'\t%a='土'\t%A='土曜日'\n"), lfmt, std::chrono::weekday(6)); 103566868cdSMark de Wever check(loc, SV("%u='7'\t%Ou='7'\t%w='0'\t%Ow='0'\t%a='日'\t%A='日曜日'\n"), lfmt, std::chrono::weekday(7)); 104566868cdSMark de Wever #else 105566868cdSMark de Wever check(loc, SV("%u='7'\t%Ou='七'\t%w='0'\t%Ow='〇'\t%a='日'\t%A='日曜日'\n"), lfmt, std::chrono::weekday(0)); 106566868cdSMark de Wever check(loc, SV("%u='1'\t%Ou='一'\t%w='1'\t%Ow='一'\t%a='月'\t%A='月曜日'\n"), lfmt, std::chrono::weekday(1)); 107566868cdSMark de Wever check(loc, SV("%u='2'\t%Ou='二'\t%w='2'\t%Ow='二'\t%a='火'\t%A='火曜日'\n"), lfmt, std::chrono::weekday(2)); 108566868cdSMark de Wever check(loc, SV("%u='3'\t%Ou='三'\t%w='3'\t%Ow='三'\t%a='水'\t%A='水曜日'\n"), lfmt, std::chrono::weekday(3)); 109566868cdSMark de Wever check(loc, SV("%u='4'\t%Ou='四'\t%w='4'\t%Ow='四'\t%a='木'\t%A='木曜日'\n"), lfmt, std::chrono::weekday(4)); 110566868cdSMark de Wever check(loc, SV("%u='5'\t%Ou='五'\t%w='5'\t%Ow='五'\t%a='金'\t%A='金曜日'\n"), lfmt, std::chrono::weekday(5)); 111566868cdSMark de Wever check(loc, SV("%u='6'\t%Ou='六'\t%w='6'\t%Ow='六'\t%a='土'\t%A='土曜日'\n"), lfmt, std::chrono::weekday(6)); 112566868cdSMark de Wever check(loc, SV("%u='7'\t%Ou='七'\t%w='0'\t%Ow='〇'\t%a='日'\t%A='日曜日'\n"), lfmt, std::chrono::weekday(7)); 113566868cdSMark de Wever #endif 114566868cdSMark de Wever 115566868cdSMark de Wever std::locale::global(std::locale::classic()); 116566868cdSMark de Wever } 117566868cdSMark de Wever 118566868cdSMark de Wever template <class CharT> 119566868cdSMark de Wever static void test_invalid_values() { 120566868cdSMark de Wever // Test that %a and %A throw an exception. 121402eb2efSMark de Wever check_exception("Formatting a weekday name needs a valid weekday", SV("{:%a}"), std::chrono::weekday(8)); 122402eb2efSMark de Wever check_exception("Formatting a weekday name needs a valid weekday", SV("{:%a}"), std::chrono::weekday(255)); 123402eb2efSMark de Wever check_exception("Formatting a weekday name needs a valid weekday", SV("{:%A}"), std::chrono::weekday(8)); 124402eb2efSMark de Wever check_exception("Formatting a weekday name needs a valid weekday", SV("{:%A}"), std::chrono::weekday(255)); 125566868cdSMark de Wever 126566868cdSMark de Wever constexpr std::basic_string_view<CharT> fmt = SV("{:%%u='%u'%t%%Ou='%Ou'%t%%w='%w'%t%%Ow='%Ow'%n}"); 127566868cdSMark de Wever constexpr std::basic_string_view<CharT> lfmt = SV("{:L%%u='%u'%t%%Ou='%Ou'%t%%w='%w'%t%%Ow='%Ow'%n}"); 128566868cdSMark de Wever 129566868cdSMark de Wever const std::locale loc(LOCALE_ja_JP_UTF_8); 130566868cdSMark de Wever std::locale::global(std::locale(LOCALE_fr_FR_UTF_8)); 131566868cdSMark de Wever 1328f01029bSMark de Wever #if defined(__APPLE__) || defined(__FreeBSD__) 133566868cdSMark de Wever // Non localized output using C-locale 134566868cdSMark de Wever check(SV("%u='8'\t%Ou='8'\t%w='8'\t%Ow='8'\n"), fmt, std::chrono::weekday(8)); 135566868cdSMark de Wever check(SV("%u='255'\t%Ou='255'\t%w='255'\t%Ow='255'\n"), fmt, std::chrono::weekday(255)); 136566868cdSMark de Wever 137566868cdSMark de Wever // Use the global locale (fr_FR) 138566868cdSMark de Wever check(SV("%u='8'\t%Ou='8'\t%w='8'\t%Ow='8'\n"), lfmt, std::chrono::weekday(8)); 139566868cdSMark de Wever check(SV("%u='255'\t%Ou='255'\t%w='255'\t%Ow='255'\n"), lfmt, std::chrono::weekday(255)); 140566868cdSMark de Wever 141566868cdSMark de Wever // Use supplied locale (ja_JP). This locale has a different alternate. 142566868cdSMark de Wever check(loc, SV("%u='8'\t%Ou='8'\t%w='8'\t%Ow='8'\n"), lfmt, std::chrono::weekday(8)); 143566868cdSMark de Wever check(loc, SV("%u='255'\t%Ou='255'\t%w='255'\t%Ow='255'\n"), lfmt, std::chrono::weekday(255)); 1448f01029bSMark de Wever #elif defined(_AIX) // defined(__APPLE__) || defined(__FreeBSD__) 145566868cdSMark de Wever // Non localized output using C-locale 146566868cdSMark de Wever check(SV("%u='8'\t%Ou='8'\t%w='8'\t%Ow='8'\n"), fmt, std::chrono::weekday(8)); 147566868cdSMark de Wever check(SV("%u='5'\t%Ou='5'\t%w='5'\t%Ow='5'\n"), fmt, std::chrono::weekday(255)); 148566868cdSMark de Wever 149566868cdSMark de Wever // Use the global locale (fr_FR) 150566868cdSMark de Wever check(SV("%u='8'\t%Ou='8'\t%w='8'\t%Ow='8'\n"), lfmt, std::chrono::weekday(8)); 151566868cdSMark de Wever check(SV("%u='5'\t%Ou='5'\t%w='5'\t%Ow='5'\n"), lfmt, std::chrono::weekday(255)); 152566868cdSMark de Wever 153566868cdSMark de Wever // Use supplied locale (ja_JP). This locale has a different alternate. 154566868cdSMark de Wever check(loc, SV("%u='8'\t%Ou='8'\t%w='8'\t%Ow='8'\n"), lfmt, std::chrono::weekday(8)); 155566868cdSMark de Wever check(loc, SV("%u='5'\t%Ou='5'\t%w='5'\t%Ow='5'\n"), lfmt, std::chrono::weekday(255)); 1568f01029bSMark de Wever #elif defined(_WIN32) // defined(__APPLE__) || defined(__FreeBSD__) 1571403080aSMark de Wever // Non localized output using C-locale 1581403080aSMark de Wever check(SV("%u=''\t%Ou=''\t%w=''\t%Ow=''\n"), fmt, std::chrono::weekday(8)); 1591403080aSMark de Wever check(SV("%u=''\t%Ou=''\t%w=''\t%Ow=''\n"), fmt, std::chrono::weekday(255)); 1601403080aSMark de Wever 1611403080aSMark de Wever // Use the global locale (fr_FR) 1621403080aSMark de Wever check(SV("%u=''\t%Ou=''\t%w=''\t%Ow=''\n"), lfmt, std::chrono::weekday(8)); 1631403080aSMark de Wever check(SV("%u=''\t%Ou=''\t%w=''\t%Ow=''\n"), lfmt, std::chrono::weekday(255)); 1641403080aSMark de Wever 1651403080aSMark de Wever // Use supplied locale (ja_JP). This locale has a different alternate. 1661403080aSMark de Wever check(loc, SV("%u=''\t%Ou=''\t%w=''\t%Ow=''\n"), lfmt, std::chrono::weekday(8)); 1671403080aSMark de Wever check(loc, SV("%u=''\t%Ou=''\t%w=''\t%Ow=''\n"), lfmt, std::chrono::weekday(255)); 1688f01029bSMark de Wever #else // defined(__APPLE__) || defined(__FreeBSD__) 169566868cdSMark de Wever // Non localized output using C-locale 170566868cdSMark de Wever check(SV("%u='1'\t%Ou='1'\t%w='8'\t%Ow='8'\n"), fmt, std::chrono::weekday(8)); 171566868cdSMark de Wever check(SV("%u='3'\t%Ou='3'\t%w='255'\t%Ow='255'\n"), fmt, std::chrono::weekday(255)); 172566868cdSMark de Wever 173566868cdSMark de Wever // Use the global locale (fr_FR) 174566868cdSMark de Wever check(SV("%u='1'\t%Ou='1'\t%w='8'\t%Ow='8'\n"), lfmt, std::chrono::weekday(8)); 175566868cdSMark de Wever check(SV("%u='3'\t%Ou='3'\t%w='255'\t%Ow='255'\n"), lfmt, std::chrono::weekday(255)); 176566868cdSMark de Wever 177566868cdSMark de Wever // Use supplied locale (ja_JP). This locale has a different alternate. 178566868cdSMark de Wever check(loc, SV("%u='1'\t%Ou='一'\t%w='8'\t%Ow='八'\n"), lfmt, std::chrono::weekday(8)); 179566868cdSMark de Wever check(loc, SV("%u='3'\t%Ou='三'\t%w='255'\t%Ow='255'\n"), lfmt, std::chrono::weekday(255)); 1808f01029bSMark de Wever #endif // defined(__APPLE__) || defined(__FreeBSD__) 181566868cdSMark de Wever 182566868cdSMark de Wever std::locale::global(std::locale::classic()); 183566868cdSMark de Wever } 184566868cdSMark de Wever 185566868cdSMark de Wever template <class CharT> 186566868cdSMark de Wever static void test() { 187566868cdSMark de Wever test_no_chrono_specs<CharT>(); 188566868cdSMark de Wever test_valid_values<CharT>(); 189566868cdSMark de Wever test_invalid_values<CharT>(); 190566868cdSMark de Wever check_invalid_types<CharT>( 191566868cdSMark de Wever {SV("a"), SV("A"), SV("t"), SV("u"), SV("w"), SV("Ou"), SV("Ow")}, std::chrono::weekday(0)); 192566868cdSMark de Wever 193402eb2efSMark de Wever check_exception("The format specifier expects a '%' or a '}'", SV("{:A"), std::chrono::weekday(0)); 194402eb2efSMark de Wever check_exception("The chrono specifiers contain a '{'", SV("{:%%{"), std::chrono::weekday(0)); 195402eb2efSMark de Wever check_exception("End of input while parsing a conversion specifier", SV("{:%"), std::chrono::weekday(0)); 196566868cdSMark de Wever check_exception("End of input while parsing the modifier E", SV("{:%E"), std::chrono::weekday(0)); 197566868cdSMark de Wever check_exception("End of input while parsing the modifier O", SV("{:%O"), std::chrono::weekday(0)); 198566868cdSMark de Wever 199566868cdSMark de Wever // Precision not allowed 200402eb2efSMark de Wever check_exception("The format specifier expects a '%' or a '}'", SV("{:.3}"), std::chrono::weekday(0)); 201566868cdSMark de Wever } 202566868cdSMark de Wever 203566868cdSMark de Wever int main(int, char**) { 204566868cdSMark de Wever test<char>(); 205566868cdSMark de Wever 206566868cdSMark de Wever #ifndef TEST_HAS_NO_WIDE_CHARACTERS 207566868cdSMark de Wever test<wchar_t>(); 208566868cdSMark de Wever #endif 209566868cdSMark de Wever 210566868cdSMark de Wever return 0; 211566868cdSMark de Wever } 212