| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572 | // Formatting library for C++ - module tests//// Copyright (c) 2012 - present, Victor Zverovich// All rights reserved.//// For the license information refer to format.h.//// Copyright (c) 2021 - present, Daniela Engert// All Rights Reserved// {fmt} module.#ifdef _MSC_FULL_VER// hide some implementation bugs in msvc// that are not essential to users of the module.#  define FMT_HIDE_MODULE_BUGS#endif#define FMT_MODULE_TEST#include <bit>#include <chrono>#include <exception>#include <iterator>#include <locale>#include <memory>#include <ostream>#include <string>#include <string_view>#include <system_error>#if (__has_include(<fcntl.h>) || defined(__APPLE__) || \     defined(__linux__)) &&                              \    (!defined(WINAPI_FAMILY) || (WINAPI_FAMILY == WINAPI_FAMILY_DESKTOP_APP))#  include <fcntl.h>#  define FMT_USE_FCNTL 1#else#  define FMT_USE_FCNTL 0#endif#if defined(_WIN32) && !defined(__MINGW32__)#  define FMT_POSIX(call) _##call#else#  define FMT_POSIX(call) call#endif#define FMT_OS_H_  // don't pull in os.h directly or indirectlyimport fmt;// check for macros leaking from BMIstatic bool macro_leaked =#if defined(FMT_CORE_H_) || defined(FMT_FORMAT_H_)    true;#else    false;#endif// Include sources to pick up functions and classes from the module rather than// from the non-modular library which is baked into the 'test-main' library.// This averts linker problems:// - strong ownership model: missing linker symbols// - weak ownership model: duplicate linker symbols#include "gtest-extra.cc"#include "util.cc"// an implicitly exported namespace must be visible [module.interface]/2.2TEST(module_test, namespace) {  using namespace fmt;  using namespace fmt::literals;  ASSERT_TRUE(true);}namespace detail {bool oops_detail_namespace_is_visible;}namespace fmt {bool namespace_detail_invisible() {#if defined(FMT_HIDE_MODULE_BUGS) && defined(_MSC_FULL_VER) && \    ((_MSC_VER == 1929 && _MSC_FULL_VER <= 192930136) ||       \     (_MSC_VER == 1930 && _MSC_FULL_VER <= 193030704))  // bug in msvc up to 16.11.5 / 17.0-pre5:  // the namespace is visible even when it is neither  // implicitly nor explicitly exported  return true;#else  using namespace detail;  // this fails to compile if fmt::detail is visible  return !oops_detail_namespace_is_visible;#endif}}  // namespace fmt// the non-exported namespace 'detail' must be invisible [module.interface]/2TEST(module_test, detail_namespace) {  EXPECT_TRUE(fmt::namespace_detail_invisible());}// macros must not be imported from a *named* module  [cpp.import]/5.1TEST(module_test, macros) {#if defined(FMT_HIDE_MODULE_BUGS) && defined(_MSC_FULL_VER) && \    _MSC_FULL_VER <= 192930130  // bug in msvc up to 16.11-pre2:  // include-guard macros leak from BMI  // and even worse: they cannot be #undef-ined  macro_leaked = false;#endif  EXPECT_FALSE(macro_leaked);}// The following is less about functional testing (that's done elsewhere)// but rather visibility of all client-facing overloads, reachability of// non-exported entities, name lookup and overload resolution within// template instantitions.// Excercise all exported entities of the API at least once.// Instantiate as many code paths as possible.TEST(module_test, to_string) {  EXPECT_EQ("42", fmt::to_string(42));  EXPECT_EQ("42", fmt::to_string(42.0));  EXPECT_EQ(L"42", fmt::to_wstring(42));  EXPECT_EQ(L"42", fmt::to_wstring(42.0));}TEST(module_test, format) {  EXPECT_EQ("42", fmt::format("{:}", 42));  EXPECT_EQ("-42", fmt::format("{0}", -42.0));  EXPECT_EQ(L"42", fmt::format(L"{:}", 42));  EXPECT_EQ(L"-42", fmt::format(L"{0}", -42.0));}TEST(module_test, format_to) {  std::string s;  fmt::format_to(std::back_inserter(s), "{}", 42);  EXPECT_EQ("42", s);  char buffer[4] = {0};  fmt::format_to(buffer, "{}", 42);  EXPECT_EQ("42", std::string_view(buffer));  fmt::memory_buffer mb;  fmt::format_to(mb, "{}", 42);  EXPECT_EQ("42", std::string_view(buffer));  std::wstring w;  fmt::format_to(std::back_inserter(w), L"{}", 42);  EXPECT_EQ(L"42", w);  wchar_t wbuffer[4] = {0};  fmt::format_to(wbuffer, L"{}", 42);  EXPECT_EQ(L"42", std::wstring_view(wbuffer));  fmt::wmemory_buffer wb;  fmt::format_to(wb, L"{}", 42);  EXPECT_EQ(L"42", std::wstring_view(wbuffer));}TEST(module_test, formatted_size) {  EXPECT_EQ(2u, fmt::formatted_size("{}", 42));  EXPECT_EQ(2u, fmt::formatted_size(L"{}", 42));}TEST(module_test, format_to_n) {  std::string s;  auto result = fmt::format_to_n(std::back_inserter(s), 1, "{}", 42);  EXPECT_EQ(2u, result.size);  char buffer[4] = {0};  fmt::format_to_n(buffer, 3, "{}", 12345);  std::wstring w;  auto wresult = fmt::format_to_n(std::back_inserter(w), 1, L"{}", 42);  EXPECT_EQ(2u, wresult.size);  wchar_t wbuffer[4] = {0};  fmt::format_to_n(wbuffer, 3, L"{}", 12345);}TEST(module_test, format_args) {  auto no_args = fmt::format_args();  EXPECT_FALSE(no_args.get(1));  fmt::basic_format_args args = fmt::make_format_args(42);  EXPECT_TRUE(args.max_size() > 0);  auto arg0 = args.get(0);  EXPECT_TRUE(arg0);  decltype(arg0) arg_none;  EXPECT_FALSE(arg_none);  EXPECT_TRUE(arg0.type() != arg_none.type());}TEST(module_test, wformat_args) {  auto no_args = fmt::wformat_args();  EXPECT_FALSE(no_args.get(1));  fmt::basic_format_args args = fmt::make_wformat_args(42);  EXPECT_TRUE(args.get(0));}TEST(module_test, dynamic_format_args) {  fmt::dynamic_format_arg_store<fmt::format_context> dyn_store;  dyn_store.push_back(fmt::arg("a42", 42));  fmt::basic_format_args args = dyn_store;  EXPECT_FALSE(args.get(3));  EXPECT_TRUE(args.get(fmt::string_view("a42")));  fmt::dynamic_format_arg_store<fmt::wformat_context> wdyn_store;  wdyn_store.push_back(fmt::arg(L"a42", 42));  fmt::basic_format_args wargs = wdyn_store;  EXPECT_FALSE(wargs.get(3));  EXPECT_TRUE(wargs.get(fmt::wstring_view(L"a42")));}TEST(module_test, vformat) {  EXPECT_EQ("42", fmt::vformat("{}", fmt::make_format_args(42)));  EXPECT_EQ(L"42", fmt::vformat(fmt::to_string_view(L"{}"),                                fmt::make_wformat_args(42)));}TEST(module_test, vformat_to) {  auto store = fmt::make_format_args(42);  std::string s;  fmt::vformat_to(std::back_inserter(s), "{}", store);  EXPECT_EQ("42", s);  char buffer[4] = {0};  fmt::vformat_to(buffer, "{:}", store);  EXPECT_EQ("42", std::string_view(buffer));  auto wstore = fmt::make_wformat_args(42);  std::wstring w;  fmt::vformat_to(std::back_inserter(w), L"{}", wstore);  EXPECT_EQ(L"42", w);  wchar_t wbuffer[4] = {0};  fmt::vformat_to(wbuffer, L"{:}", wstore);  EXPECT_EQ(L"42", std::wstring_view(wbuffer));}TEST(module_test, vformat_to_n) {  auto store = fmt::make_format_args(12345);  std::string s;  auto result = fmt::vformat_to_n(std::back_inserter(s), 1, "{}", store);  char buffer[4] = {0};  fmt::vformat_to_n(buffer, 3, "{:}", store);  auto wstore = fmt::make_wformat_args(12345);  std::wstring w;  auto wresult = fmt::vformat_to_n(std::back_inserter(w), 1,                                   fmt::to_string_view(L"{}"), wstore);  wchar_t wbuffer[4] = {0};  fmt::vformat_to_n(wbuffer, 3, fmt::to_string_view(L"{:}"), wstore);}std::string as_string(std::wstring_view text) {  return {reinterpret_cast<const char*>(text.data()),          text.size() * sizeof(text[0])};}TEST(module_test, print) {  EXPECT_WRITE(stdout, fmt::print("{}µ", 42), "42µ");  EXPECT_WRITE(stderr, fmt::print(stderr, "{}µ", 4.2), "4.2µ");  if (false) {    EXPECT_WRITE(stdout, fmt::print(L"{}µ", 42), as_string(L"42µ"));    EXPECT_WRITE(stderr, fmt::print(stderr, L"{}µ", 4.2), as_string(L"4.2µ"));  }}TEST(module_test, vprint) {  EXPECT_WRITE(stdout, fmt::vprint("{:}µ", fmt::make_format_args(42)), "42µ");  EXPECT_WRITE(stderr, fmt::vprint(stderr, "{}", fmt::make_format_args(4.2)),               "4.2");  if (false) {    EXPECT_WRITE(stdout, fmt::vprint(L"{:}µ", fmt::make_wformat_args(42)),                 as_string(L"42µ"));    EXPECT_WRITE(stderr, fmt::vprint(stderr, L"{}", fmt::make_wformat_args(42)),                 as_string(L"42"));  }}TEST(module_test, named_args) {  EXPECT_EQ("42", fmt::format("{answer}", fmt::arg("answer", 42)));  EXPECT_EQ(L"42", fmt::format(L"{answer}", fmt::arg(L"answer", 42)));}TEST(module_test, literals) {  using namespace fmt::literals;  EXPECT_EQ("42", fmt::format("{answer}", "answer"_a = 42));  EXPECT_EQ("42", "{}"_format(42));  EXPECT_EQ(L"42", fmt::format(L"{answer}", L"answer"_a = 42));  EXPECT_EQ(L"42", L"{}"_format(42));}TEST(module_test, locale) {  auto store = fmt::make_format_args(4.2);  const auto classic = std::locale::classic();  EXPECT_EQ("4.2", fmt::format(classic, "{:L}", 4.2));  EXPECT_EQ("4.2", fmt::vformat(classic, "{:L}", store));  std::string s;  fmt::vformat_to(std::back_inserter(s), classic, "{:L}", store);  EXPECT_EQ("4.2", s);  EXPECT_EQ("4.2", fmt::format("{:L}", 4.2));  auto wstore = fmt::make_wformat_args(4.2);  EXPECT_EQ(L"4.2", fmt::format(classic, L"{:L}", 4.2));  EXPECT_EQ(L"4.2", fmt::vformat(classic, L"{:L}", wstore));  std::wstring w;  fmt::vformat_to(std::back_inserter(w), classic, L"{:L}", wstore);  EXPECT_EQ(L"4.2", w);  EXPECT_EQ(L"4.2", fmt::format(L"{:L}", 4.2));}TEST(module_test, string_view) {  fmt::string_view nsv("fmt");  EXPECT_EQ("fmt", nsv);  EXPECT_TRUE(fmt::string_view("fmt") == nsv);  fmt::wstring_view wsv(L"fmt");  EXPECT_EQ(L"fmt", wsv);  EXPECT_TRUE(fmt::wstring_view(L"fmt") == wsv);}TEST(module_test, memory_buffer) {  fmt::basic_memory_buffer<char, fmt::inline_buffer_size> buffer;  fmt::format_to(buffer, "{}", "42");  EXPECT_EQ("42", to_string(buffer));  fmt::memory_buffer nbuffer(std::move(buffer));  EXPECT_EQ("42", to_string(nbuffer));  buffer = std::move(nbuffer);  EXPECT_EQ("42", to_string(buffer));  nbuffer.clear();  EXPECT_EQ(0u, to_string(nbuffer).size());  fmt::wmemory_buffer wbuffer;  EXPECT_EQ(0u, to_string(wbuffer).size());}TEST(module_test, is_char) {  EXPECT_TRUE(fmt::is_char<char>());  EXPECT_TRUE(fmt::is_char<wchar_t>());  EXPECT_TRUE(fmt::is_char<char8_t>());  EXPECT_TRUE(fmt::is_char<char16_t>());  EXPECT_TRUE(fmt::is_char<char32_t>());  EXPECT_FALSE(fmt::is_char<signed char>());}TEST(module_test, ptr) {  uintptr_t answer = 42;  auto p = std::bit_cast<int*>(answer);  EXPECT_EQ("0x2a", fmt::to_string(fmt::ptr(p)));  std::unique_ptr<int> up(p);  EXPECT_EQ("0x2a", fmt::to_string(fmt::ptr(up)));  up.release();  auto sp = std::make_shared<int>(0);  p = sp.get();  EXPECT_EQ(fmt::to_string(fmt::ptr(p)), fmt::to_string(fmt::ptr(sp)));}TEST(module_test, errors) {  auto store = fmt::make_format_args(42);  EXPECT_THROW(throw fmt::format_error("oops"), std::exception);  EXPECT_THROW(throw fmt::vsystem_error(0, "{}", store), std::system_error);  EXPECT_THROW(throw fmt::system_error(0, "{}", 42), std::system_error);  fmt::memory_buffer buffer;  fmt::format_system_error(buffer, 0, "oops");  auto oops = to_string(buffer);  EXPECT_TRUE(oops.size() > 0);  EXPECT_WRITE(stderr, fmt::report_system_error(0, "oops"), oops + '\n');#ifdef _WIN32  EXPECT_THROW(throw fmt::vwindows_error(0, "{}", store), std::system_error);  EXPECT_THROW(throw fmt::windows_error(0, "{}", 42), std::system_error);  output_redirect redirect(stderr);  fmt::report_windows_error(0, "oops");  EXPECT_TRUE(redirect.restore_and_read().size() > 0);#endif}TEST(module_test, error_code) {  EXPECT_EQ("generic:42",            fmt::format("{0}", std::error_code(42, std::generic_category())));  EXPECT_EQ("system:42",            fmt::format("{0}", std::error_code(42, fmt::system_category())));  EXPECT_EQ(L"generic:42",            fmt::format(L"{0}", std::error_code(42, std::generic_category())));}TEST(module_test, format_int) {  fmt::format_int sanswer(42);  EXPECT_EQ("42", fmt::string_view(sanswer.data(), sanswer.size()));  fmt::format_int uanswer(42u);  EXPECT_EQ("42", fmt::string_view(uanswer.data(), uanswer.size()));}struct test_formatter : fmt::formatter<char> {  bool check() { return true; }};struct test_dynamic_formatter : fmt::dynamic_formatter<> {  bool check() { return true; }};TEST(module_test, formatter) {  EXPECT_TRUE(test_formatter{}.check());  EXPECT_TRUE(test_dynamic_formatter{}.check());}TEST(module_test, join) {  int arr[3] = {1, 2, 3};  std::vector<double> vec{1.0, 2.0, 3.0};  std::initializer_list<int> il{1, 2, 3};  auto sep = fmt::to_string_view(", ");  EXPECT_EQ("1, 2, 3", to_string(fmt::join(arr + 0, arr + 3, sep)));  EXPECT_EQ("1, 2, 3", to_string(fmt::join(arr, sep)));  EXPECT_EQ("1, 2, 3", to_string(fmt::join(vec.begin(), vec.end(), sep)));  EXPECT_EQ("1, 2, 3", to_string(fmt::join(vec, sep)));  EXPECT_EQ("1, 2, 3", to_string(fmt::join(il, sep)));  auto wsep = fmt::to_string_view(L", ");  EXPECT_EQ(L"1, 2, 3", fmt::format(L"{}", fmt::join(arr + 0, arr + 3, wsep)));  EXPECT_EQ(L"1, 2, 3", fmt::format(L"{}", fmt::join(arr, wsep)));  EXPECT_EQ(L"1, 2, 3", fmt::format(L"{}", fmt::join(il, wsep)));}TEST(module_test, time) {  auto time_now = std::time(nullptr);  EXPECT_TRUE(fmt::localtime(time_now).tm_year > 120);  EXPECT_TRUE(fmt::gmtime(time_now).tm_year > 120);  auto chrono_now = std::chrono::system_clock::now();  EXPECT_TRUE(fmt::localtime(chrono_now).tm_year > 120);  EXPECT_TRUE(fmt::gmtime(chrono_now).tm_year > 120);}TEST(module_test, time_point) {  auto now = std::chrono::system_clock::now();  std::string_view past("2021-05-20 10:30:15");  EXPECT_TRUE(past < fmt::format("{:%Y-%m-%d %H:%M:%S}", now));  std::wstring_view wpast(L"2021-05-20 10:30:15");  EXPECT_TRUE(wpast < fmt::format(L"{:%Y-%m-%d %H:%M:%S}", now));}TEST(module_test, time_duration) {  using us = std::chrono::duration<double, std::micro>;  EXPECT_EQ("42s", fmt::format("{}", std::chrono::seconds{42}));  EXPECT_EQ("4.2µs", fmt::format("{:3.1}", us{4.234}));  EXPECT_EQ("4.2µs", fmt::format(std::locale::classic(), "{:L}", us{4.2}));  EXPECT_EQ(L"42s", fmt::format(L"{}", std::chrono::seconds{42}));  EXPECT_EQ(L"4.2µs", fmt::format(L"{:3.1}", us{4.234}));  EXPECT_EQ(L"4.2µs", fmt::format(std::locale::classic(), L"{:L}", us{4.2}));}TEST(module_test, weekday) {  EXPECT_EQ("Mon", fmt::format(std::locale::classic(), "{}", fmt::weekday(1)));}TEST(module_test, to_string_view) {  using fmt::to_string_view;  fmt::string_view nsv{to_string_view("42")};  EXPECT_EQ("42", nsv);  fmt::wstring_view wsv{to_string_view(L"42")};  EXPECT_EQ(L"42", wsv);}TEST(module_test, printf) {  EXPECT_WRITE(stdout, fmt::printf("%f", 42.123456), "42.123456");  EXPECT_WRITE(stdout, fmt::printf("%d", 42), "42");  if (false) {    EXPECT_WRITE(stdout, fmt::printf(L"%f", 42.123456),                 as_string(L"42.123456"));    EXPECT_WRITE(stdout, fmt::printf(L"%d", 42), as_string(L"42"));  }}TEST(module_test, fprintf) {  EXPECT_WRITE(stderr, fmt::fprintf(stderr, "%d", 42), "42");  std::ostringstream os;  fmt::fprintf(os, "%s", "bla");  EXPECT_EQ("bla", os.str());  EXPECT_WRITE(stderr, fmt::fprintf(stderr, L"%d", 42), as_string(L"42"));  std::wostringstream ws;  fmt::fprintf(ws, L"%s", L"bla");  EXPECT_EQ(L"bla", ws.str());}TEST(module_test, sprintf) {  EXPECT_EQ("42", fmt::sprintf("%d", 42));  EXPECT_EQ(L"42", fmt::sprintf(L"%d", 42));}TEST(module_test, vprintf) {  EXPECT_WRITE(stdout, fmt::vprintf("%d", fmt::make_printf_args(42)), "42");  if (false) {    EXPECT_WRITE(stdout, fmt::vprintf(L"%d", fmt::make_wprintf_args(42)),                 as_string(L"42"));  }}TEST(module_test, vfprintf) {  auto args = fmt::make_printf_args(42);  EXPECT_WRITE(stderr, fmt::vfprintf(stderr, "%d", args), "42");  std::ostringstream os;  fmt::vfprintf(os, "%d", args);  EXPECT_EQ("42", os.str());  auto wargs = fmt::make_wprintf_args(42);  if (false) {    EXPECT_WRITE(stderr, fmt::vfprintf(stderr, L"%d", wargs), as_string(L"42"));  }  std::wostringstream ws;  fmt::vfprintf(ws, L"%d", wargs);  EXPECT_EQ(L"42", ws.str());}TEST(module_test, vsprintf) {  EXPECT_EQ("42", fmt::vsprintf("%d", fmt::make_printf_args(42)));  EXPECT_EQ(L"42", fmt::vsprintf(L"%d", fmt::make_wprintf_args(42)));}TEST(module_test, color) {  auto fg_check = fg(fmt::rgb(255, 200, 30));  auto bg_check = bg(fmt::color::dark_slate_gray) | fmt::emphasis::italic;  auto emphasis_check = fmt::emphasis::underline | fmt::emphasis::bold;  EXPECT_EQ("\x1B[30m42\x1B[0m",            fmt::format(fg(fmt::terminal_color::black), "{}", 42));  EXPECT_EQ(L"\x1B[30m42\x1B[0m",            fmt::format(fg(fmt::terminal_color::black), L"{}", 42));}TEST(module_test, cstring_view) {  auto s = "fmt";  EXPECT_EQ(s, fmt::cstring_view(s).c_str());  auto w = L"fmt";  EXPECT_EQ(w, fmt::wcstring_view(w).c_str());}TEST(module_test, buffered_file) {  EXPECT_TRUE(fmt::buffered_file{}.get() == nullptr);}TEST(module_test, output_file) {  fmt::ostream out = fmt::output_file("module-test", fmt::buffer_size = 1);  out.close();}struct custom_context {  using char_type = char;  using parse_context_type = fmt::format_parse_context;};TEST(module_test, custom_context) {  fmt::basic_format_arg<custom_context> custom_arg;  EXPECT_TRUE(!custom_arg);}struct disabled_formatter {};TEST(module_test, has_formatter) {  EXPECT_FALSE(      (fmt::has_formatter<disabled_formatter, fmt::format_context>::value));}TEST(module_test, is_formattable) {  EXPECT_FALSE(fmt::is_formattable<disabled_formatter>::value);}TEST(module_test, compile_format_string) {  using namespace fmt::literals;  EXPECT_EQ("42", fmt::format("{0:x}"_cf, 0x42));  EXPECT_EQ(L"42", fmt::format(L"{:}"_cf, 42));  EXPECT_EQ("4.2", fmt::format("{arg:3.1f}"_cf, "arg"_a = 4.2));  EXPECT_EQ(L" 42", fmt::format(L"{arg:>3}"_cf, L"arg"_a = L"42"));}
 |