diff options
Diffstat (limited to 'thirdparty/fmt/test/std-test.cc')
| -rw-r--r-- | thirdparty/fmt/test/std-test.cc | 495 |
1 files changed, 0 insertions, 495 deletions
diff --git a/thirdparty/fmt/test/std-test.cc b/thirdparty/fmt/test/std-test.cc deleted file mode 100644 index 18f6bd3fc..000000000 --- a/thirdparty/fmt/test/std-test.cc +++ /dev/null @@ -1,495 +0,0 @@ -// Formatting library for C++ - tests of formatters for standard library types -// -// Copyright (c) 2012 - present, Victor Zverovich -// All rights reserved. -// -// For the license information refer to format.h. - -#include "fmt/std.h" - -#include <bitset> -#include <stdexcept> -#include <string> -#include <vector> - -#include "fmt/os.h" // fmt::system_category -#include "fmt/ranges.h" -#include "gtest-extra.h" // StartsWith - -#ifdef __cpp_lib_filesystem -TEST(std_test, path) { - using std::filesystem::path; - EXPECT_EQ(fmt::format("{}", path("/usr/bin")), "/usr/bin"); - - // see #4303 - const path p = "/usr/bin"; - EXPECT_EQ(fmt::format("{}", p), "/usr/bin"); - - EXPECT_EQ(fmt::format("{:?}", path("/usr/bin")), "\"/usr/bin\""); - EXPECT_EQ(fmt::format("{:8}", path("foo")), "foo "); - - EXPECT_EQ(fmt::format("{}", path("foo\"bar")), "foo\"bar"); - EXPECT_EQ(fmt::format("{:?}", path("foo\"bar")), "\"foo\\\"bar\""); - - EXPECT_EQ(fmt::format("{:g}", path("/usr/bin")), "/usr/bin"); -# ifdef _WIN32 - EXPECT_EQ(fmt::format("{}", path("C:\\foo")), "C:\\foo"); - EXPECT_EQ(fmt::format("{:g}", path("C:\\foo")), "C:/foo"); - - EXPECT_EQ(fmt::format("{}", path(L"\x0428\x0447\x0443\x0447\x044B\x043D\x0448" - L"\x0447\x044B\x043D\x0430")), - "Шчучыншчына"); - EXPECT_EQ(fmt::format("{}", path(L"\xd800")), "�"); - EXPECT_EQ(fmt::format("{}", path(L"HEAD \xd800 TAIL")), "HEAD � TAIL"); - EXPECT_EQ(fmt::format("{}", path(L"HEAD \xD83D\xDE00 TAIL")), - "HEAD \xF0\x9F\x98\x80 TAIL"); - EXPECT_EQ(fmt::format("{}", path(L"HEAD \xD83D\xD83D\xDE00 TAIL")), - "HEAD �\xF0\x9F\x98\x80 TAIL"); - EXPECT_EQ(fmt::format("{:?}", path(L"\xd800")), "\"\\ud800\""); -# endif -} - -// Intentionally delayed include to test #4303 -# include "fmt/ranges.h" - -// Test ambiguity problem described in #2954. -TEST(ranges_std_test, format_vector_path) { - auto p = std::filesystem::path("foo/bar.txt"); - auto c = std::vector<std::string>{"abc", "def"}; - EXPECT_EQ(fmt::format("path={}, range={}", p, c), - "path=foo/bar.txt, range=[\"abc\", \"def\"]"); -} - -// Test that path is not escaped twice in the debug mode. -TEST(ranges_std_test, format_quote_path) { - auto vec = - std::vector<std::filesystem::path>{"path1/file1.txt", "path2/file2.txt"}; - EXPECT_EQ(fmt::format("{}", vec), - "[\"path1/file1.txt\", \"path2/file2.txt\"]"); -# ifdef __cpp_lib_optional - auto o = std::optional<std::filesystem::path>("path/file.txt"); - EXPECT_EQ(fmt::format("{}", o), "optional(\"path/file.txt\")"); - EXPECT_EQ(fmt::format("{:?}", o), "optional(\"path/file.txt\")"); -# endif -} -#endif - -TEST(std_test, thread_id) { - EXPECT_FALSE(fmt::format("{}", std::this_thread::get_id()).empty()); -} - -TEST(std_test, complex) { - using limits = std::numeric_limits<double>; - EXPECT_EQ(fmt::format("{}", std::complex<double>(1, limits::quiet_NaN())), - "(1+nan i)"); - EXPECT_EQ(fmt::format("{}", std::complex<double>(1, -limits::infinity())), - "(1-inf i)"); - - EXPECT_EQ(fmt::format("{}", std::complex<int>(1, 2)), "(1+2i)"); - - EXPECT_EQ(fmt::format("{}", std::complex<double>(1, 2.2)), "(1+2.2i)"); - EXPECT_EQ(fmt::format("{}", std::complex<double>(1, -2.2)), "(1-2.2i)"); - EXPECT_EQ(fmt::format("{}", std::complex<double>(0, 2.2)), "2.2i"); - EXPECT_EQ(fmt::format("{}", std::complex<double>(0, -2.2)), "-2.2i"); - - EXPECT_EQ(fmt::format("{:+}", std::complex<double>(0, 2.2)), "+2.2i"); - EXPECT_EQ(fmt::format("{:+}", std::complex<double>(0, -2.2)), "-2.2i"); - EXPECT_EQ(fmt::format("{:+}", std::complex<double>(1, -2.2)), "(+1-2.2i)"); - EXPECT_EQ(fmt::format("{:+}", std::complex<double>(1, 2.2)), "(+1+2.2i)"); - EXPECT_EQ(fmt::format("{: }", std::complex<double>(1, 2.2)), "( 1+2.2i)"); - EXPECT_EQ(fmt::format("{: }", std::complex<double>(1, -2.2)), "( 1-2.2i)"); - - EXPECT_EQ(fmt::format("{:8}", std::complex<double>(1, 2)), "(1+2i) "); - EXPECT_EQ(fmt::format("{:-<8}", std::complex<double>(1, 2)), "(1+2i)--"); - - EXPECT_EQ(fmt::format("{:>20.2f}", std::complex<double>(1, 2.2)), - " (1.00+2.20i)"); - EXPECT_EQ(fmt::format("{:<20.2f}", std::complex<double>(1, 2.2)), - "(1.00+2.20i) "); - EXPECT_EQ(fmt::format("{:<20.2f}", std::complex<double>(1, -2.2)), - "(1.00-2.20i) "); - EXPECT_EQ(fmt::format("{:<{}.{}f}", std::complex<double>(1, -2.2), 20, 2), - "(1.00-2.20i) "); -} - -#ifdef __cpp_lib_source_location -TEST(std_test, source_location) { - std::source_location loc = std::source_location::current(); - EXPECT_EQ(fmt::format("{}", loc), - fmt::format("{}:{}:{}: {}", loc.file_name(), loc.line(), - loc.column(), loc.function_name())); -} -#endif - -TEST(std_test, optional) { -#ifdef __cpp_lib_optional - EXPECT_EQ(fmt::format("{}", std::optional<int>{}), "none"); - EXPECT_EQ(fmt::format("{}", std::pair{1, "second"}), "(1, \"second\")"); - EXPECT_EQ(fmt::format("{}", std::vector{std::optional{1}, std::optional{2}, - std::optional{3}}), - "[optional(1), optional(2), optional(3)]"); - EXPECT_EQ( - fmt::format("{}", std::optional<std::optional<const char*>>{{"nested"}}), - "optional(optional(\"nested\"))"); - EXPECT_EQ( - fmt::format("{:<{}}", std::optional{std::string{"left aligned"}}, 30), - "optional(\"left aligned\" )"); - EXPECT_EQ( - fmt::format("{::d}", std::optional{std::vector{'h', 'e', 'l', 'l', 'o'}}), - "optional([104, 101, 108, 108, 111])"); - EXPECT_EQ(fmt::format("{}", std::optional{std::string{"string"}}), - "optional(\"string\")"); - EXPECT_EQ(fmt::format("{}", std::optional{'C'}), "optional(\'C\')"); - EXPECT_EQ(fmt::format("{:.{}f}", std::optional{3.14}, 1), "optional(3.1)"); - - struct unformattable {}; - EXPECT_FALSE((fmt::is_formattable<unformattable>::value)); - EXPECT_FALSE((fmt::is_formattable<std::optional<unformattable>>::value)); - EXPECT_TRUE((fmt::is_formattable<std::optional<int>>::value)); - EXPECT_TRUE((fmt::is_formattable<std::optional<const int>>::value)); -#endif -} - -TEST(std_test, expected) { -#ifdef __cpp_lib_expected - EXPECT_EQ(fmt::format("{}", std::expected<void, int>{}), "expected()"); - EXPECT_EQ(fmt::format("{}", std::expected<int, int>{1}), "expected(1)"); - EXPECT_EQ(fmt::format("{}", std::expected<int, int>{std::unexpected(1)}), - "unexpected(1)"); - EXPECT_EQ(fmt::format("{}", std::expected<std::string, int>{"test"}), - "expected(\"test\")"); - EXPECT_EQ(fmt::format( - "{}", std::expected<int, std::string>{std::unexpected("test")}), - "unexpected(\"test\")"); - EXPECT_EQ(fmt::format("{}", std::expected<char, int>{'a'}), "expected('a')"); - EXPECT_EQ(fmt::format("{}", std::expected<int, char>{std::unexpected('a')}), - "unexpected('a')"); - - struct unformattable1 {}; - struct unformattable2 {}; - EXPECT_FALSE((fmt::is_formattable<unformattable1>::value)); - EXPECT_FALSE((fmt::is_formattable<unformattable2>::value)); - EXPECT_FALSE((fmt::is_formattable< - std::expected<unformattable1, unformattable2>>::value)); - EXPECT_FALSE( - (fmt::is_formattable<std::expected<unformattable1, int>>::value)); - EXPECT_FALSE( - (fmt::is_formattable<std::expected<int, unformattable2>>::value)); - EXPECT_TRUE((fmt::is_formattable<std::expected<int, int>>::value)); - EXPECT_TRUE((fmt::is_formattable<std::expected<void, int>>::value)); -#endif -} - -namespace my_nso { -enum class my_number { - one, - two, -}; -auto format_as(my_number number) -> fmt::string_view { - return number == my_number::one ? "first" : "second"; -} - -class my_class { - public: - int av; - - private: - friend auto format_as(const my_class& elm) -> std::string { - return fmt::to_string(elm.av); - } -}; - -class my_class_int { - public: - int av; - - private: - friend auto format_as(const my_class_int& elm) -> int { return elm.av; } -}; -} // namespace my_nso - -TEST(std_test, expected_format_as) { -#ifdef __cpp_lib_expected - EXPECT_EQ( - fmt::format( - "{}", std::expected<my_nso::my_number, int>{my_nso::my_number::one}), - "expected(\"first\")"); - EXPECT_EQ( - fmt::format("{}", - std::expected<my_nso::my_class, int>{my_nso::my_class{7}}), - "expected(\"7\")"); - EXPECT_EQ(fmt::format("{}", - std::expected<my_nso::my_class_int, int>{ - my_nso::my_class_int{8}}), - "expected(8)"); -#endif -} - -TEST(std_test, optional_format_as) { -#ifdef __cpp_lib_optional - EXPECT_EQ(fmt::format("{}", std::optional<my_nso::my_number>{}), "none"); - EXPECT_EQ(fmt::format("{}", std::optional{my_nso::my_number::one}), - "optional(\"first\")"); - EXPECT_EQ(fmt::format("{}", std::optional<my_nso::my_class>{}), "none"); - EXPECT_EQ(fmt::format("{}", std::optional{my_nso::my_class{7}}), - "optional(\"7\")"); - EXPECT_EQ(fmt::format("{}", std::optional{my_nso::my_class_int{8}}), - "optional(8)"); -#endif -} - -struct throws_on_move { - throws_on_move() = default; - - [[noreturn]] throws_on_move(throws_on_move&&) { - throw std::runtime_error("Thrown by throws_on_move"); - } - - throws_on_move(const throws_on_move&) = default; -}; - -namespace fmt { -template <> struct formatter<throws_on_move> : formatter<string_view> { - auto format(const throws_on_move&, format_context& ctx) const - -> decltype(ctx.out()) { - string_view str("<throws_on_move>"); - return formatter<string_view>::format(str, ctx); - } -}; -} // namespace fmt - -TEST(std_test, variant) { -#ifdef __cpp_lib_variant - EXPECT_EQ(fmt::format("{}", std::monostate{}), "monostate"); - using V0 = std::variant<int, float, std::string, char>; - V0 v0(42); - V0 v1(1.5f); - V0 v2("hello"); - V0 v3('i'); - EXPECT_EQ(fmt::format("{}", v0), "variant(42)"); - EXPECT_EQ(fmt::format("{}", v1), "variant(1.5)"); - EXPECT_EQ(fmt::format("{}", v2), "variant(\"hello\")"); - EXPECT_EQ(fmt::format("{}", v3), "variant('i')"); - - struct unformattable {}; - EXPECT_FALSE((fmt::is_formattable<unformattable>::value)); - EXPECT_FALSE((fmt::is_formattable<std::variant<unformattable>>::value)); - EXPECT_FALSE((fmt::is_formattable<std::variant<unformattable, int>>::value)); - EXPECT_FALSE((fmt::is_formattable<std::variant<int, unformattable>>::value)); - EXPECT_FALSE( - (fmt::is_formattable<std::variant<unformattable, unformattable>>::value)); - EXPECT_TRUE((fmt::is_formattable<std::variant<int, float>>::value)); - - using V1 = std::variant<std::monostate, std::string, std::string>; - V1 v4{}; - V1 v5{std::in_place_index<1>, "yes, this is variant"}; - - EXPECT_EQ(fmt::format("{}", v4), "variant(monostate)"); - EXPECT_EQ(fmt::format("{}", v5), "variant(\"yes, this is variant\")"); - - volatile int i = 42; // Test compile error before GCC 11 described in #3068. - EXPECT_EQ(fmt::format("{}", i), "42"); - - std::variant<std::monostate, throws_on_move> v6; - - try { - throws_on_move thrower; - v6.emplace<throws_on_move>(std::move(thrower)); - } catch (const std::runtime_error&) { - } - // v6 is now valueless by exception - - EXPECT_EQ(fmt::format("{}", v6), "variant(valueless by exception)"); - -#endif -} - -TEST(std_test, variant_format_as) { -#ifdef __cpp_lib_variant - - EXPECT_EQ(fmt::format("{}", std::variant<my_nso::my_number>{}), - "variant(\"first\")"); - EXPECT_EQ(fmt::format( - "{}", std::variant<my_nso::my_number>{my_nso::my_number::one}), - "variant(\"first\")"); - EXPECT_EQ( - fmt::format("{}", std::variant<my_nso::my_class>{my_nso::my_class{7}}), - "variant(\"7\")"); - EXPECT_EQ( - fmt::format("{}", - std::variant<my_nso::my_class_int>{my_nso::my_class_int{8}}), - "variant(8)"); -#endif -} - -TEST(std_test, error_code) { - auto& generic = std::generic_category(); - EXPECT_EQ(fmt::format("{}", std::error_code(42, generic)), "generic:42"); - EXPECT_EQ(fmt::format("{:>12}", std::error_code(42, generic)), - " generic:42"); - EXPECT_EQ(fmt::format("{:12}", std::error_code(42, generic)), "generic:42 "); - EXPECT_EQ(fmt::format("{}", std::error_code(42, fmt::system_category())), - "system:42"); - EXPECT_EQ(fmt::format("{}", std::error_code(-42, fmt::system_category())), - "system:-42"); - auto ec = std::make_error_code(std::errc::value_too_large); - EXPECT_EQ(fmt::format("{:s}", ec), ec.message()); - EXPECT_EQ(fmt::format("{:?}", std::error_code(42, generic)), - "\"generic:42\""); - EXPECT_EQ(fmt::format("{}", - std::map<std::error_code, int>{ - {std::error_code(42, generic), 0}}), - "{\"generic:42\": 0}"); -} - -template <typename Catch> void exception_test() { - try { - throw std::runtime_error("Test Exception"); - } catch (const Catch& ex) { - EXPECT_EQ("Test Exception", fmt::format("{}", ex)); - EXPECT_EQ("std::runtime_error: Test Exception", fmt::format("{:t}", ex)); - } -} - -namespace my_ns1 { -namespace my_ns2 { -struct my_exception : public std::exception { - private: - std::string msg; - - public: - my_exception(const std::string& s) : msg(s) {} - const char* what() const noexcept override; -}; -const char* my_exception::what() const noexcept { return msg.c_str(); } -} // namespace my_ns2 -} // namespace my_ns1 - -TEST(std_test, exception) { - using testing::StartsWith; - exception_test<std::exception>(); - exception_test<std::runtime_error>(); - - try { - using namespace my_ns1::my_ns2; - throw my_exception("My Exception"); - } catch (const std::exception& ex) { - EXPECT_EQ("my_ns1::my_ns2::my_exception: My Exception", - fmt::format("{:t}", ex)); - EXPECT_EQ("My Exception", fmt::format("{:}", ex)); - } - - try { - throw std::system_error(std::error_code(), "message"); - } catch (const std::system_error& ex) { - EXPECT_THAT(fmt::format("{:t}", ex), StartsWith("std::system_error: ")); - } - -#ifdef __cpp_lib_filesystem - // Tests that the inline namespace is stripped out, e.g. - // std::filesystem::__cxx11::* -> std::filesystem::*. - try { - throw std::filesystem::filesystem_error("message", std::error_code()); - } catch (const std::filesystem::filesystem_error& ex) { - EXPECT_THAT(fmt::format("{:t}", ex), - StartsWith("std::filesystem::filesystem_error: ")); - } -#endif -} - -#if FMT_USE_RTTI -TEST(std_test, type_info) { - EXPECT_EQ(fmt::format("{}", typeid(std::runtime_error)), - "std::runtime_error"); -} -#endif - -TEST(std_test, format_bit_reference) { - std::bitset<2> bs(1); - EXPECT_EQ(fmt::format("{} {}", bs[0], bs[1]), "true false"); - std::vector<bool> v = {true, false}; - EXPECT_EQ(fmt::format("{} {}", v[0], v[1]), "true false"); -} - -TEST(std_test, format_const_bit_reference) { - const std::bitset<2> bs(1); - EXPECT_EQ(fmt::format("{} {}", bs[0], bs[1]), "true false"); - const std::vector<bool> v = {true, false}; - EXPECT_EQ(fmt::format("{} {}", v[0], v[1]), "true false"); -} - -TEST(std_test, format_bitset) { - auto bs = std::bitset<6>(42); - EXPECT_EQ(fmt::format("{}", bs), "101010"); - EXPECT_EQ(fmt::format("{:0>8}", bs), "00101010"); - EXPECT_EQ(fmt::format("{:-^12}", bs), "---101010---"); -} - -TEST(std_test, format_atomic) { - std::atomic<bool> b(false); - EXPECT_EQ(fmt::format("{}", b), "false"); - - const std::atomic<bool> cb(true); - EXPECT_EQ(fmt::format("{}", cb), "true"); -} - -#ifdef __cpp_lib_atomic_flag_test -TEST(std_test, format_atomic_flag) { - std::atomic_flag f; - (void)f.test_and_set(); - EXPECT_EQ(fmt::format("{}", f), "true"); - - f.clear(); - const std::atomic_flag& cf = f; - EXPECT_EQ(fmt::format("{}", cf), "false"); -} -#endif // __cpp_lib_atomic_flag_test - -TEST(std_test, format_unique_ptr) { - std::unique_ptr<int> up(new int(1)); - EXPECT_EQ(fmt::format("{}", fmt::ptr(up.get())), - fmt::format("{}", fmt::ptr(up))); - struct custom_deleter { - void operator()(int* p) const { delete p; } - }; - std::unique_ptr<int, custom_deleter> upcd(new int(1)); - EXPECT_EQ(fmt::format("{}", fmt::ptr(upcd.get())), - fmt::format("{}", fmt::ptr(upcd))); -} - -TEST(std_test, format_shared_ptr) { - std::shared_ptr<int> sp(new int(1)); - EXPECT_EQ(fmt::format("{}", fmt::ptr(sp.get())), - fmt::format("{}", fmt::ptr(sp))); -} - -TEST(std_test, format_reference_wrapper) { - int num = 35; - EXPECT_EQ(fmt::to_string(std::cref(num)), "35"); - EXPECT_EQ(fmt::to_string(std::ref(num)), "35"); - EXPECT_EQ(fmt::format("{}", std::cref(num)), "35"); - EXPECT_EQ(fmt::format("{}", std::ref(num)), "35"); -} - -// Regression test for https://github.com/fmtlib/fmt/issues/4424. -struct type_with_format_as {}; -int format_as(type_with_format_as) { return 20; } - -TEST(std_test, format_reference_wrapper_with_format_as) { - type_with_format_as t; - EXPECT_EQ(fmt::to_string(std::cref(t)), "20"); - EXPECT_EQ(fmt::to_string(std::ref(t)), "20"); - EXPECT_EQ(fmt::format("{}", std::cref(t)), "20"); - EXPECT_EQ(fmt::format("{}", std::ref(t)), "20"); -} - -struct type_with_format_as_string {}; -std::string format_as(type_with_format_as_string) { return "foo"; } - -TEST(std_test, format_reference_wrapper_with_format_as_string) { - type_with_format_as_string t; - EXPECT_EQ(fmt::to_string(std::cref(t)), "foo"); - EXPECT_EQ(fmt::to_string(std::ref(t)), "foo"); - EXPECT_EQ(fmt::format("{}", std::cref(t)), "foo"); - EXPECT_EQ(fmt::format("{}", std::ref(t)), "foo"); -} |