// 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 #include #include #include "fmt/os.h" // fmt::system_category #include "fmt/ranges.h" #include "gtest-extra.h" // StartsWith using testing::StartsWith; TEST(std_test, path) { #ifdef __cpp_lib_filesystem EXPECT_EQ(fmt::format("{:8}", std::filesystem::path("foo")), "\"foo\" "); EXPECT_EQ(fmt::format("{}", std::filesystem::path("foo\"bar.txt")), "\"foo\\\"bar.txt\""); EXPECT_EQ(fmt::format("{:?}", std::filesystem::path("foo\"bar.txt")), "\"foo\\\"bar.txt\""); # ifdef _WIN32 // File.txt in Russian. const wchar_t unicode_path[] = {0x424, 0x430, 0x439, 0x43b, 0x2e, 0x74, 0x78, 0x74, 0}; const char unicode_u8path[] = {'"', char(0xd0), char(0xa4), char(0xd0), char(0xb0), char(0xd0), char(0xb9), char(0xd0), char(0xbb), '.', 't', 'x', 't', '"', '\0'}; EXPECT_EQ(fmt::format("{}", std::filesystem::path(unicode_path)), unicode_u8path); # endif #endif } TEST(ranges_std_test, format_vector_path) { // Test ambiguity problem described in #2954. #ifdef __cpp_lib_filesystem auto p = std::filesystem::path("foo/bar.txt"); auto c = std::vector{"abc", "def"}; EXPECT_EQ(fmt::format("path={}, range={}", p, c), "path=\"foo/bar.txt\", range=[\"abc\", \"def\"]"); #endif } TEST(ranges_std_test, format_quote_path) { // Test that path is not escaped twice in the debug mode. #ifdef __cpp_lib_filesystem auto vec = std::vector{"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("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, optional) { #ifdef __cpp_lib_optional EXPECT_EQ(fmt::format("{}", std::optional{}), "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>{{"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::value)); EXPECT_FALSE((fmt::is_formattable>::value)); EXPECT_TRUE((fmt::is_formattable>::value)); #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 : formatter { auto format(const throws_on_move&, format_context& ctx) const -> decltype(ctx.out()) { string_view str(""); return formatter::format(str, ctx); } }; } // namespace fmt TEST(std_test, variant) { #ifdef __cpp_lib_variant EXPECT_EQ(fmt::format("{}", std::monostate{}), "monostate"); using V0 = std::variant; 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::value)); EXPECT_FALSE((fmt::is_formattable>::value)); EXPECT_FALSE((fmt::is_formattable>::value)); EXPECT_FALSE((fmt::is_formattable>::value)); EXPECT_FALSE( (fmt::is_formattable>::value)); EXPECT_TRUE((fmt::is_formattable>::value)); using V1 = std::variant; 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 v6; try { throws_on_move thrower; v6.emplace(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, error_code) { EXPECT_EQ("generic:42", fmt::format(FMT_STRING("{0}"), std::error_code(42, std::generic_category()))); EXPECT_EQ("system:42", fmt::format(FMT_STRING("{0}"), std::error_code(42, fmt::system_category()))); EXPECT_EQ("system:-42", fmt::format(FMT_STRING("{0}"), std::error_code(-42, fmt::system_category()))); } template 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) { exception_test(); exception_test(); 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 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 }