| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521 | // Formatting library for C++ - formatting library tests//// Copyright (c) 2012 - present, Victor Zverovich// All rights reserved.//// For the license information refer to format.h.#include "fmt/xchar.h"#include <complex>#include <cwchar>#include <vector>#include "fmt/chrono.h"#include "fmt/color.h"#include "fmt/ostream.h"#include "fmt/ranges.h"#include "gtest-extra.h"  // Contains#include "util.h"         // get_localeusing fmt::detail::max_value;using testing::Contains;namespace test_ns {template <typename Char> class test_string { private:  std::basic_string<Char> s_; public:  test_string(const Char* s) : s_(s) {}  const Char* data() const { return s_.data(); }  size_t length() const { return s_.size(); }  operator const Char*() const { return s_.c_str(); }};template <typename Char>fmt::basic_string_view<Char> to_string_view(const test_string<Char>& s) {  return {s.data(), s.length()};}struct non_string {};}  // namespace test_nstemplate <typename T> class is_string_test : public testing::Test {};using string_char_types = testing::Types<char, wchar_t, char16_t, char32_t>;TYPED_TEST_SUITE(is_string_test, string_char_types);template <typename Char>struct derived_from_string_view : fmt::basic_string_view<Char> {};TYPED_TEST(is_string_test, is_string) {  EXPECT_TRUE(fmt::detail::is_string<TypeParam*>::value);  EXPECT_TRUE(fmt::detail::is_string<const TypeParam*>::value);  EXPECT_TRUE(fmt::detail::is_string<TypeParam[2]>::value);  EXPECT_TRUE(fmt::detail::is_string<const TypeParam[2]>::value);  EXPECT_TRUE(fmt::detail::is_string<std::basic_string<TypeParam>>::value);  EXPECT_TRUE(fmt::detail::is_string<fmt::basic_string_view<TypeParam>>::value);  EXPECT_TRUE(      fmt::detail::is_string<derived_from_string_view<TypeParam>>::value);  using fmt_string_view = fmt::detail::std_string_view<TypeParam>;  EXPECT_TRUE(std::is_empty<fmt_string_view>::value !=              fmt::detail::is_string<fmt_string_view>::value);  EXPECT_TRUE(fmt::detail::is_string<test_ns::test_string<TypeParam>>::value);  EXPECT_FALSE(fmt::detail::is_string<test_ns::non_string>::value);}// std::is_constructible is broken in MSVC until version 2015.#if !FMT_MSC_VERSION || FMT_MSC_VERSION >= 1900struct explicitly_convertible_to_wstring_view {  explicit operator fmt::wstring_view() const { return L"foo"; }};TEST(xchar_test, format_explicitly_convertible_to_wstring_view) {  // Types explicitly convertible to wstring_view are not formattable by  // default because it may introduce ODR violations.  static_assert(      !fmt::is_formattable<explicitly_convertible_to_wstring_view>::value, "");}#endifTEST(xchar_test, format) {  EXPECT_EQ(L"42", fmt::format(L"{}", 42));  EXPECT_EQ(L"4.2", fmt::format(L"{}", 4.2));  EXPECT_EQ(L"abc", fmt::format(L"{}", L"abc"));  EXPECT_EQ(L"z", fmt::format(L"{}", L'z'));  EXPECT_THROW(fmt::format(fmt::runtime(L"{:*\x343E}"), 42), fmt::format_error);  EXPECT_EQ(L"true", fmt::format(L"{}", true));  EXPECT_EQ(L"a", fmt::format(L"{0}", 'a'));  EXPECT_EQ(L"a", fmt::format(L"{0}", L'a'));  EXPECT_EQ(L"Cyrillic letter \x42e",            fmt::format(L"Cyrillic letter {}", L'\x42e'));  EXPECT_EQ(L"abc1", fmt::format(L"{}c{}", L"ab", 1));}TEST(xchar_test, is_formattable) {  static_assert(!fmt::is_formattable<const wchar_t*>::value, "");}TEST(xchar_test, compile_time_string) {  EXPECT_EQ(fmt::format(fmt::wformat_string<int>(L"{}"), 42), L"42");#if defined(FMT_USE_STRING_VIEW) && FMT_CPLUSPLUS >= 201703L  EXPECT_EQ(fmt::format(FMT_STRING(std::wstring_view(L"{}")), 42), L"42");#endif}#if FMT_CPLUSPLUS > 201103Lstruct custom_char {  int value;  custom_char() = default;  template <typename T>  constexpr custom_char(T val) : value(static_cast<int>(val)) {}  operator int() const { return value; }};int to_ascii(custom_char c) { return c; }FMT_BEGIN_NAMESPACEtemplate <> struct is_char<custom_char> : std::true_type {};FMT_END_NAMESPACETEST(xchar_test, format_custom_char) {  const custom_char format[] = {'{', '}', 0};  auto result = fmt::format(format, custom_char('x'));  EXPECT_EQ(result.size(), 1);  EXPECT_EQ(result[0], custom_char('x'));}#endif// Convert a char8_t string to std::string. Otherwise GTest will insist on// inserting `char8_t` NTBS into a `char` stream which is disabled by P1423.template <typename S> std::string from_u8str(const S& str) {  return std::string(str.begin(), str.end());}TEST(xchar_test, format_utf8_precision) {  using str_type = std::basic_string<fmt::detail::char8_type>;  auto format =      str_type(reinterpret_cast<const fmt::detail::char8_type*>(u8"{:.4}"));  auto str = str_type(reinterpret_cast<const fmt::detail::char8_type*>(      u8"caf\u00e9s"));  // cafés  auto result = fmt::format(format, str);  EXPECT_EQ(fmt::detail::compute_width(result), 4);  EXPECT_EQ(result.size(), 5);  EXPECT_EQ(from_u8str(result), from_u8str(str.substr(0, 5)));}TEST(xchar_test, format_to) {  auto buf = std::vector<wchar_t>();  fmt::format_to(std::back_inserter(buf), L"{}{}", 42, L'\0');  EXPECT_STREQ(buf.data(), L"42");}TEST(xchar_test, vformat_to) {  using wcontext = fmt::wformat_context;  fmt::basic_format_arg<wcontext> warg = fmt::detail::make_arg<wcontext>(42);  auto wargs = fmt::basic_format_args<wcontext>(&warg, 1);  auto w = std::wstring();  fmt::vformat_to(std::back_inserter(w), L"{}", wargs);  EXPECT_EQ(L"42", w);  w.clear();  fmt::vformat_to(std::back_inserter(w), FMT_STRING(L"{}"), wargs);  EXPECT_EQ(L"42", w);}TEST(format_test, wide_format_to_n) {  wchar_t buffer[4];  buffer[3] = L'x';  auto result = fmt::format_to_n(buffer, 3, L"{}", 12345);  EXPECT_EQ(5u, result.size);  EXPECT_EQ(buffer + 3, result.out);  EXPECT_EQ(L"123x", fmt::wstring_view(buffer, 4));  buffer[0] = L'x';  buffer[1] = L'x';  buffer[2] = L'x';  result = fmt::format_to_n(buffer, 3, L"{}", L'A');  EXPECT_EQ(1u, result.size);  EXPECT_EQ(buffer + 1, result.out);  EXPECT_EQ(L"Axxx", fmt::wstring_view(buffer, 4));  result = fmt::format_to_n(buffer, 3, L"{}{} ", L'B', L'C');  EXPECT_EQ(3u, result.size);  EXPECT_EQ(buffer + 3, result.out);  EXPECT_EQ(L"BC x", fmt::wstring_view(buffer, 4));}#if FMT_USE_USER_DEFINED_LITERALSTEST(xchar_test, named_arg_udl) {  using namespace fmt::literals;  auto udl_a =      fmt::format(L"{first}{second}{first}{third}", L"first"_a = L"abra",                  L"second"_a = L"cad", L"third"_a = 99);  EXPECT_EQ(      fmt::format(L"{first}{second}{first}{third}", fmt::arg(L"first", L"abra"),                  fmt::arg(L"second", L"cad"), fmt::arg(L"third", 99)),      udl_a);}#endif  // FMT_USE_USER_DEFINED_LITERALSTEST(xchar_test, print) {  // Check that the wide print overload compiles.  if (fmt::detail::const_check(false)) fmt::print(L"test");}TEST(xchar_test, join) {  int v[3] = {1, 2, 3};  EXPECT_EQ(fmt::format(L"({})", fmt::join(v, v + 3, L", ")), L"(1, 2, 3)");  auto t = std::tuple<wchar_t, int, float>('a', 1, 2.0f);  EXPECT_EQ(fmt::format(L"({})", fmt::join(t, L", ")), L"(a, 1, 2)");}enum streamable_enum {};std::wostream& operator<<(std::wostream& os, streamable_enum) {  return os << L"streamable_enum";}namespace fmt {template <>struct formatter<streamable_enum, wchar_t> : basic_ostream_formatter<wchar_t> {};}  // namespace fmtenum unstreamable_enum {};auto format_as(unstreamable_enum e) -> int { return e; }TEST(xchar_test, enum) {  EXPECT_EQ(L"streamable_enum", fmt::format(L"{}", streamable_enum()));  EXPECT_EQ(L"0", fmt::format(L"{}", unstreamable_enum()));}struct streamable_and_unformattable {};auto operator<<(std::wostream& os, streamable_and_unformattable)    -> std::wostream& {  return os << L"foo";}TEST(xchar_test, streamed) {  EXPECT_FALSE(fmt::is_formattable<streamable_and_unformattable>());  EXPECT_EQ(fmt::format(L"{}", fmt::streamed(streamable_and_unformattable())),            L"foo");}TEST(xchar_test, sign_not_truncated) {  wchar_t format_str[] = {      L'{', L':',      '+' | static_cast<wchar_t>(1 << fmt::detail::num_bits<char>()), L'}', 0};  EXPECT_THROW(fmt::format(fmt::runtime(format_str), 42), fmt::format_error);}TEST(xchar_test, chrono) {  auto tm = std::tm();  tm.tm_year = 116;  tm.tm_mon = 3;  tm.tm_mday = 25;  tm.tm_hour = 11;  tm.tm_min = 22;  tm.tm_sec = 33;  EXPECT_EQ(fmt::format("The date is {:%Y-%m-%d %H:%M:%S}.", tm),            "The date is 2016-04-25 11:22:33.");  EXPECT_EQ(L"42s", fmt::format(L"{}", std::chrono::seconds(42)));  EXPECT_EQ(fmt::format(L"{:%F}", tm), L"2016-04-25");  EXPECT_EQ(fmt::format(L"{:%T}", tm), L"11:22:33");}std::wstring system_wcsftime(const std::wstring& format, const std::tm* timeptr,                             std::locale* locptr = nullptr) {  auto loc = locptr ? *locptr : std::locale::classic();  auto& facet = std::use_facet<std::time_put<wchar_t>>(loc);  std::wostringstream os;  os.imbue(loc);  facet.put(os, os, L' ', timeptr, format.c_str(),            format.c_str() + format.size());#ifdef _WIN32  // Workaround a bug in older versions of Universal CRT.  auto str = os.str();  if (str == L"-0000") str = L"+0000";  return str;#else  return os.str();#endif}TEST(chrono_test_wchar, time_point) {  auto t1 = std::chrono::system_clock::now();  std::vector<std::wstring> spec_list = {      L"%%",  L"%n",  L"%t",  L"%Y",  L"%EY", L"%y",  L"%Oy", L"%Ey", L"%C",      L"%EC", L"%G",  L"%g",  L"%b",  L"%h",  L"%B",  L"%m",  L"%Om", L"%U",      L"%OU", L"%W",  L"%OW", L"%V",  L"%OV", L"%j",  L"%d",  L"%Od", L"%e",      L"%Oe", L"%a",  L"%A",  L"%w",  L"%Ow", L"%u",  L"%Ou", L"%H",  L"%OH",      L"%I",  L"%OI", L"%M",  L"%OM", L"%S",  L"%OS", L"%x",  L"%Ex", L"%X",      L"%EX", L"%D",  L"%F",  L"%R",  L"%T",  L"%p",  L"%z",  L"%Z"};#ifndef _WIN32  // Disabled on Windows, because these formats is not consistent among  // platforms.  spec_list.insert(spec_list.end(), {L"%c", L"%Ec", L"%r"});#elif defined(__MINGW32__) && !defined(_UCRT)  // Only C89 conversion specifiers when using MSVCRT instead of UCRT  spec_list = {L"%%", L"%Y", L"%y", L"%b", L"%B", L"%m", L"%U",               L"%W", L"%j", L"%d", L"%a", L"%A", L"%w", L"%H",               L"%I", L"%M", L"%S", L"%x", L"%X", L"%p", L"%Z"};#endif  spec_list.push_back(L"%Y-%m-%d %H:%M:%S");  for (const auto& spec : spec_list) {    auto t = std::chrono::system_clock::to_time_t(t1);    auto tm = *std::localtime(&t);    auto sys_output = system_wcsftime(spec, &tm);    auto fmt_spec = fmt::format(L"{{:{}}}", spec);    EXPECT_EQ(sys_output, fmt::format(fmt::runtime(fmt_spec), t1));    EXPECT_EQ(sys_output, fmt::format(fmt::runtime(fmt_spec), tm));  }}TEST(xchar_test, color) {  EXPECT_EQ(fmt::format(fg(fmt::rgb(255, 20, 30)), L"rgb(255,20,30) wide"),            L"\x1b[38;2;255;020;030mrgb(255,20,30) wide\x1b[0m");}TEST(xchar_test, ostream) {#if !FMT_GCC_VERSION || FMT_GCC_VERSION >= 409  std::wostringstream wos;  fmt::print(wos, L"Don't {}!", L"panic");  EXPECT_EQ(wos.str(), L"Don't panic!");#endif}TEST(xchar_test, format_map) {  auto m = std::map<std::wstring, int>{{L"one", 1}, {L"t\"wo", 2}};  EXPECT_EQ(fmt::format(L"{}", m), L"{\"one\": 1, \"t\\\"wo\": 2}");}TEST(xchar_test, escape_string) {  using vec = std::vector<std::wstring>;  EXPECT_EQ(fmt::format(L"{}", vec{L"\n\r\t\"\\"}), L"[\"\\n\\r\\t\\\"\\\\\"]");  EXPECT_EQ(fmt::format(L"{}", vec{L"понедельник"}), L"[\"понедельник\"]");}TEST(xchar_test, to_wstring) { EXPECT_EQ(L"42", fmt::to_wstring(42)); }#ifndef FMT_STATIC_THOUSANDS_SEPARATORtemplate <typename Char> struct numpunct : std::numpunct<Char> { protected:  Char do_decimal_point() const override { return '?'; }  std::string do_grouping() const override { return "\03"; }  Char do_thousands_sep() const override { return '~'; }};template <typename Char> struct no_grouping : std::numpunct<Char> { protected:  Char do_decimal_point() const override { return '.'; }  std::string do_grouping() const override { return ""; }  Char do_thousands_sep() const override { return ','; }};template <typename Char> struct special_grouping : std::numpunct<Char> { protected:  Char do_decimal_point() const override { return '.'; }  std::string do_grouping() const override { return "\03\02"; }  Char do_thousands_sep() const override { return ','; }};template <typename Char> struct small_grouping : std::numpunct<Char> { protected:  Char do_decimal_point() const override { return '.'; }  std::string do_grouping() const override { return "\01"; }  Char do_thousands_sep() const override { return ','; }};TEST(locale_test, localized_double) {  auto loc = std::locale(std::locale(), new numpunct<char>());  EXPECT_EQ(fmt::format(loc, "{:L}", 1.23), "1?23");  EXPECT_EQ(fmt::format(loc, "{:Lf}", 1.23), "1?230000");  EXPECT_EQ(fmt::format(loc, "{:L}", 1234.5), "1~234?5");  EXPECT_EQ(fmt::format(loc, "{:L}", 12000.0), "12~000");  EXPECT_EQ(fmt::format(loc, "{:8L}", 1230.0), "   1~230");}TEST(locale_test, format) {  auto loc = std::locale(std::locale(), new numpunct<char>());  EXPECT_EQ("1234567", fmt::format(std::locale(), "{:L}", 1234567));  EXPECT_EQ("1~234~567", fmt::format(loc, "{:L}", 1234567));  EXPECT_EQ("-1~234~567", fmt::format(loc, "{:L}", -1234567));  EXPECT_EQ("-256", fmt::format(loc, "{:L}", -256));  fmt::format_arg_store<fmt::format_context, int> as{1234567};  EXPECT_EQ("1~234~567", fmt::vformat(loc, "{:L}", fmt::format_args(as)));  auto s = std::string();  fmt::format_to(std::back_inserter(s), loc, "{:L}", 1234567);  EXPECT_EQ("1~234~567", s);  auto no_grouping_loc = std::locale(std::locale(), new no_grouping<char>());  EXPECT_EQ("1234567", fmt::format(no_grouping_loc, "{:L}", 1234567));  auto special_grouping_loc =      std::locale(std::locale(), new special_grouping<char>());  EXPECT_EQ("1,23,45,678", fmt::format(special_grouping_loc, "{:L}", 12345678));  EXPECT_EQ("12,345", fmt::format(special_grouping_loc, "{:L}", 12345));  auto small_grouping_loc =      std::locale(std::locale(), new small_grouping<char>());  EXPECT_EQ("4,2,9,4,9,6,7,2,9,5",            fmt::format(small_grouping_loc, "{:L}", max_value<uint32_t>()));}TEST(locale_test, format_detault_align) {  auto loc = std::locale({}, new special_grouping<char>());  EXPECT_EQ("  12,345", fmt::format(loc, "{:8L}", 12345));}TEST(locale_test, format_plus) {  auto loc = std::locale({}, new special_grouping<char>());  EXPECT_EQ("+100", fmt::format(loc, "{:+L}", 100));}TEST(locale_test, wformat) {  auto loc = std::locale(std::locale(), new numpunct<wchar_t>());  EXPECT_EQ(L"1234567", fmt::format(std::locale(), L"{:L}", 1234567));  EXPECT_EQ(L"1~234~567", fmt::format(loc, L"{:L}", 1234567));  using wcontext = fmt::buffer_context<wchar_t>;  fmt::format_arg_store<wcontext, int> as{1234567};  EXPECT_EQ(L"1~234~567",            fmt::vformat(loc, L"{:L}", fmt::basic_format_args<wcontext>(as)));  EXPECT_EQ(L"1234567", fmt::format(std::locale("C"), L"{:L}", 1234567));  auto no_grouping_loc = std::locale(std::locale(), new no_grouping<wchar_t>());  EXPECT_EQ(L"1234567", fmt::format(no_grouping_loc, L"{:L}", 1234567));  auto special_grouping_loc =      std::locale(std::locale(), new special_grouping<wchar_t>());  EXPECT_EQ(L"1,23,45,678",            fmt::format(special_grouping_loc, L"{:L}", 12345678));  auto small_grouping_loc =      std::locale(std::locale(), new small_grouping<wchar_t>());  EXPECT_EQ(L"4,2,9,4,9,6,7,2,9,5",            fmt::format(small_grouping_loc, L"{:L}", max_value<uint32_t>()));}TEST(locale_test, double_formatter) {  auto loc = std::locale(std::locale(), new special_grouping<char>());  auto f = fmt::formatter<int>();  auto parse_ctx = fmt::format_parse_context("L");  f.parse(parse_ctx);  char buf[10] = {};  fmt::basic_format_context<char*, char> format_ctx(      buf, {}, fmt::detail::locale_ref(loc));  *f.format(12345, format_ctx) = 0;  EXPECT_STREQ("12,345", buf);}FMT_BEGIN_NAMESPACEtemplate <class charT> struct formatter<std::complex<double>, charT> { private:  detail::dynamic_format_specs<char> specs_; public:  FMT_CONSTEXPR typename basic_format_parse_context<charT>::iterator parse(      basic_format_parse_context<charT>& ctx) {    using handler_type =        detail::dynamic_specs_handler<basic_format_parse_context<charT>>;    detail::specs_checker<handler_type> handler(handler_type(specs_, ctx),                                                detail::type::string_type);    auto it = parse_format_specs(ctx.begin(), ctx.end(), handler);    detail::parse_float_type_spec(specs_, ctx.error_handler());    return it;  }  template <class FormatContext>  typename FormatContext::iterator format(const std::complex<double>& c,                                          FormatContext& ctx) {    detail::handle_dynamic_spec<detail::precision_checker>(        specs_.precision, specs_.precision_ref, ctx);    auto specs = std::string();    if (specs_.precision > 0) specs = fmt::format(".{}", specs_.precision);    if (specs_.type == presentation_type::fixed_lower) specs += 'f';    auto real = fmt::format(ctx.locale().template get<std::locale>(),                            fmt::runtime("{:" + specs + "}"), c.real());    auto imag = fmt::format(ctx.locale().template get<std::locale>(),                            fmt::runtime("{:" + specs + "}"), c.imag());    auto fill_align_width = std::string();    if (specs_.width > 0) fill_align_width = fmt::format(">{}", specs_.width);    return format_to(ctx.out(), runtime("{:" + fill_align_width + "}"),                     c.real() != 0 ? fmt::format("({}+{}i)", real, imag)                                   : fmt::format("{}i", imag));  }};FMT_END_NAMESPACETEST(locale_test, complex) {  std::string s = fmt::format("{}", std::complex<double>(1, 2));  EXPECT_EQ(s, "(1+2i)");  EXPECT_EQ(fmt::format("{:.2f}", std::complex<double>(1, 2)), "(1.00+2.00i)");  EXPECT_EQ(fmt::format("{:8}", std::complex<double>(1, 2)), "  (1+2i)");}TEST(locale_test, chrono_weekday) {  auto loc = get_locale("ru_RU.UTF-8", "Russian_Russia.1251");  auto loc_old = std::locale::global(loc);  auto mon = fmt::weekday(1);  EXPECT_EQ(fmt::format(L"{}", mon), L"Mon");  if (loc != std::locale::classic()) {    // {L"\x43F\x43D", L"\x41F\x43D", L"\x43F\x43D\x434", L"\x41F\x43D\x434"}    // {L"пн", L"Пн", L"пнд", L"Пнд"}    EXPECT_THAT(        (std::vector<std::wstring>{L"\x43F\x43D", L"\x41F\x43D",                                   L"\x43F\x43D\x434", L"\x41F\x43D\x434"}),        Contains(fmt::format(loc, L"{:L}", mon)));  }  std::locale::global(loc_old);}TEST(locale_test, sign) {  EXPECT_EQ(fmt::format(std::locale(), L"{:L}", -50), L"-50");}#endif  // FMT_STATIC_THOUSANDS_SEPARATOR
 |