1*5c90c05cSAndroid Build Coastguard Worker // Formatting library for C++ - tests of formatters for standard library types
2*5c90c05cSAndroid Build Coastguard Worker //
3*5c90c05cSAndroid Build Coastguard Worker // Copyright (c) 2012 - present, Victor Zverovich
4*5c90c05cSAndroid Build Coastguard Worker // All rights reserved.
5*5c90c05cSAndroid Build Coastguard Worker //
6*5c90c05cSAndroid Build Coastguard Worker // For the license information refer to format.h.
7*5c90c05cSAndroid Build Coastguard Worker
8*5c90c05cSAndroid Build Coastguard Worker #include "fmt/std.h"
9*5c90c05cSAndroid Build Coastguard Worker
10*5c90c05cSAndroid Build Coastguard Worker #include <bitset>
11*5c90c05cSAndroid Build Coastguard Worker #include <stdexcept>
12*5c90c05cSAndroid Build Coastguard Worker #include <string>
13*5c90c05cSAndroid Build Coastguard Worker #include <vector>
14*5c90c05cSAndroid Build Coastguard Worker
15*5c90c05cSAndroid Build Coastguard Worker #include "fmt/os.h" // fmt::system_category
16*5c90c05cSAndroid Build Coastguard Worker #include "fmt/ranges.h"
17*5c90c05cSAndroid Build Coastguard Worker #include "gtest-extra.h" // StartsWith
18*5c90c05cSAndroid Build Coastguard Worker
19*5c90c05cSAndroid Build Coastguard Worker #ifdef __cpp_lib_filesystem
TEST(std_test,path)20*5c90c05cSAndroid Build Coastguard Worker TEST(std_test, path) {
21*5c90c05cSAndroid Build Coastguard Worker using std::filesystem::path;
22*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(fmt::format("{}", path("/usr/bin")), "/usr/bin");
23*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(fmt::format("{:?}", path("/usr/bin")), "\"/usr/bin\"");
24*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(fmt::format("{:8}", path("foo")), "foo ");
25*5c90c05cSAndroid Build Coastguard Worker
26*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(fmt::format("{}", path("foo\"bar")), "foo\"bar");
27*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(fmt::format("{:?}", path("foo\"bar")), "\"foo\\\"bar\"");
28*5c90c05cSAndroid Build Coastguard Worker
29*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(fmt::format("{:g}", path("/usr/bin")), "/usr/bin");
30*5c90c05cSAndroid Build Coastguard Worker # ifdef _WIN32
31*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(fmt::format("{}", path("C:\\foo")), "C:\\foo");
32*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(fmt::format("{:g}", path("C:\\foo")), "C:/foo");
33*5c90c05cSAndroid Build Coastguard Worker
34*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(fmt::format("{}", path(L"\x0428\x0447\x0443\x0447\x044B\x043D\x0448"
35*5c90c05cSAndroid Build Coastguard Worker L"\x0447\x044B\x043D\x0430")),
36*5c90c05cSAndroid Build Coastguard Worker "Шчучыншчына");
37*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(fmt::format("{}", path(L"\xd800")), "�");
38*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(fmt::format("{}", path(L"HEAD \xd800 TAIL")), "HEAD � TAIL");
39*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(fmt::format("{}", path(L"HEAD \xD83D\xDE00 TAIL")),
40*5c90c05cSAndroid Build Coastguard Worker "HEAD \xF0\x9F\x98\x80 TAIL");
41*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(fmt::format("{}", path(L"HEAD \xD83D\xD83D\xDE00 TAIL")),
42*5c90c05cSAndroid Build Coastguard Worker "HEAD �\xF0\x9F\x98\x80 TAIL");
43*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(fmt::format("{:?}", path(L"\xd800")), "\"\\ud800\"");
44*5c90c05cSAndroid Build Coastguard Worker # endif
45*5c90c05cSAndroid Build Coastguard Worker }
46*5c90c05cSAndroid Build Coastguard Worker
47*5c90c05cSAndroid Build Coastguard Worker // Test ambiguity problem described in #2954.
TEST(ranges_std_test,format_vector_path)48*5c90c05cSAndroid Build Coastguard Worker TEST(ranges_std_test, format_vector_path) {
49*5c90c05cSAndroid Build Coastguard Worker auto p = std::filesystem::path("foo/bar.txt");
50*5c90c05cSAndroid Build Coastguard Worker auto c = std::vector<std::string>{"abc", "def"};
51*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(fmt::format("path={}, range={}", p, c),
52*5c90c05cSAndroid Build Coastguard Worker "path=foo/bar.txt, range=[\"abc\", \"def\"]");
53*5c90c05cSAndroid Build Coastguard Worker }
54*5c90c05cSAndroid Build Coastguard Worker
55*5c90c05cSAndroid Build Coastguard Worker // Test that path is not escaped twice in the debug mode.
TEST(ranges_std_test,format_quote_path)56*5c90c05cSAndroid Build Coastguard Worker TEST(ranges_std_test, format_quote_path) {
57*5c90c05cSAndroid Build Coastguard Worker auto vec =
58*5c90c05cSAndroid Build Coastguard Worker std::vector<std::filesystem::path>{"path1/file1.txt", "path2/file2.txt"};
59*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(fmt::format("{}", vec),
60*5c90c05cSAndroid Build Coastguard Worker "[\"path1/file1.txt\", \"path2/file2.txt\"]");
61*5c90c05cSAndroid Build Coastguard Worker # ifdef __cpp_lib_optional
62*5c90c05cSAndroid Build Coastguard Worker auto o = std::optional<std::filesystem::path>("path/file.txt");
63*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(fmt::format("{}", o), "optional(\"path/file.txt\")");
64*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(fmt::format("{:?}", o), "optional(\"path/file.txt\")");
65*5c90c05cSAndroid Build Coastguard Worker # endif
66*5c90c05cSAndroid Build Coastguard Worker }
67*5c90c05cSAndroid Build Coastguard Worker #endif
68*5c90c05cSAndroid Build Coastguard Worker
TEST(std_test,thread_id)69*5c90c05cSAndroid Build Coastguard Worker TEST(std_test, thread_id) {
70*5c90c05cSAndroid Build Coastguard Worker EXPECT_FALSE(fmt::format("{}", std::this_thread::get_id()).empty());
71*5c90c05cSAndroid Build Coastguard Worker }
72*5c90c05cSAndroid Build Coastguard Worker
TEST(std_test,complex)73*5c90c05cSAndroid Build Coastguard Worker TEST(std_test, complex) {
74*5c90c05cSAndroid Build Coastguard Worker using limits = std::numeric_limits<double>;
75*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(fmt::format("{}", std::complex<double>(1, limits::quiet_NaN())),
76*5c90c05cSAndroid Build Coastguard Worker "(1+nan i)");
77*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(fmt::format("{}", std::complex<double>(1, -limits::infinity())),
78*5c90c05cSAndroid Build Coastguard Worker "(1-inf i)");
79*5c90c05cSAndroid Build Coastguard Worker
80*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(fmt::format("{}", std::complex<int>(1, 2)), "(1+2i)");
81*5c90c05cSAndroid Build Coastguard Worker
82*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(fmt::format("{}", std::complex<double>(1, 2.2)), "(1+2.2i)");
83*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(fmt::format("{}", std::complex<double>(1, -2.2)), "(1-2.2i)");
84*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(fmt::format("{}", std::complex<double>(0, 2.2)), "2.2i");
85*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(fmt::format("{}", std::complex<double>(0, -2.2)), "-2.2i");
86*5c90c05cSAndroid Build Coastguard Worker
87*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(fmt::format("{:+}", std::complex<double>(0, 2.2)), "+2.2i");
88*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(fmt::format("{:+}", std::complex<double>(0, -2.2)), "-2.2i");
89*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(fmt::format("{:+}", std::complex<double>(1, -2.2)), "(+1-2.2i)");
90*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(fmt::format("{:+}", std::complex<double>(1, 2.2)), "(+1+2.2i)");
91*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(fmt::format("{: }", std::complex<double>(1, 2.2)), "( 1+2.2i)");
92*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(fmt::format("{: }", std::complex<double>(1, -2.2)), "( 1-2.2i)");
93*5c90c05cSAndroid Build Coastguard Worker
94*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(fmt::format("{:>20.2f}", std::complex<double>(1, 2.2)),
95*5c90c05cSAndroid Build Coastguard Worker " (1.00+2.20i)");
96*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(fmt::format("{:<20.2f}", std::complex<double>(1, 2.2)),
97*5c90c05cSAndroid Build Coastguard Worker "(1.00+2.20i) ");
98*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(fmt::format("{:<20.2f}", std::complex<double>(1, -2.2)),
99*5c90c05cSAndroid Build Coastguard Worker "(1.00-2.20i) ");
100*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(fmt::format("{:<{}.{}f}", std::complex<double>(1, -2.2), 20, 2),
101*5c90c05cSAndroid Build Coastguard Worker "(1.00-2.20i) ");
102*5c90c05cSAndroid Build Coastguard Worker }
103*5c90c05cSAndroid Build Coastguard Worker
104*5c90c05cSAndroid Build Coastguard Worker #ifdef __cpp_lib_source_location
TEST(std_test,source_location)105*5c90c05cSAndroid Build Coastguard Worker TEST(std_test, source_location) {
106*5c90c05cSAndroid Build Coastguard Worker std::source_location loc = std::source_location::current();
107*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(fmt::format("{}", loc),
108*5c90c05cSAndroid Build Coastguard Worker fmt::format("{}:{}:{}: {}", loc.file_name(), loc.line(),
109*5c90c05cSAndroid Build Coastguard Worker loc.column(), loc.function_name()));
110*5c90c05cSAndroid Build Coastguard Worker }
111*5c90c05cSAndroid Build Coastguard Worker #endif
112*5c90c05cSAndroid Build Coastguard Worker
TEST(std_test,optional)113*5c90c05cSAndroid Build Coastguard Worker TEST(std_test, optional) {
114*5c90c05cSAndroid Build Coastguard Worker #ifdef __cpp_lib_optional
115*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(fmt::format("{}", std::optional<int>{}), "none");
116*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(fmt::format("{}", std::pair{1, "second"}), "(1, \"second\")");
117*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(fmt::format("{}", std::vector{std::optional{1}, std::optional{2},
118*5c90c05cSAndroid Build Coastguard Worker std::optional{3}}),
119*5c90c05cSAndroid Build Coastguard Worker "[optional(1), optional(2), optional(3)]");
120*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(
121*5c90c05cSAndroid Build Coastguard Worker fmt::format("{}", std::optional<std::optional<const char*>>{{"nested"}}),
122*5c90c05cSAndroid Build Coastguard Worker "optional(optional(\"nested\"))");
123*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(
124*5c90c05cSAndroid Build Coastguard Worker fmt::format("{:<{}}", std::optional{std::string{"left aligned"}}, 30),
125*5c90c05cSAndroid Build Coastguard Worker "optional(\"left aligned\" )");
126*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(
127*5c90c05cSAndroid Build Coastguard Worker fmt::format("{::d}", std::optional{std::vector{'h', 'e', 'l', 'l', 'o'}}),
128*5c90c05cSAndroid Build Coastguard Worker "optional([104, 101, 108, 108, 111])");
129*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(fmt::format("{}", std::optional{std::string{"string"}}),
130*5c90c05cSAndroid Build Coastguard Worker "optional(\"string\")");
131*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(fmt::format("{}", std::optional{'C'}), "optional(\'C\')");
132*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(fmt::format("{:.{}f}", std::optional{3.14}, 1), "optional(3.1)");
133*5c90c05cSAndroid Build Coastguard Worker
134*5c90c05cSAndroid Build Coastguard Worker struct unformattable {};
135*5c90c05cSAndroid Build Coastguard Worker EXPECT_FALSE((fmt::is_formattable<unformattable>::value));
136*5c90c05cSAndroid Build Coastguard Worker EXPECT_FALSE((fmt::is_formattable<std::optional<unformattable>>::value));
137*5c90c05cSAndroid Build Coastguard Worker EXPECT_TRUE((fmt::is_formattable<std::optional<int>>::value));
138*5c90c05cSAndroid Build Coastguard Worker #endif
139*5c90c05cSAndroid Build Coastguard Worker }
140*5c90c05cSAndroid Build Coastguard Worker
TEST(std_test,expected)141*5c90c05cSAndroid Build Coastguard Worker TEST(std_test, expected) {
142*5c90c05cSAndroid Build Coastguard Worker #ifdef __cpp_lib_expected
143*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(fmt::format("{}", std::expected<void, int>{}), "expected()");
144*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(fmt::format("{}", std::expected<int, int>{1}), "expected(1)");
145*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(fmt::format("{}", std::expected<int, int>{std::unexpected(1)}),
146*5c90c05cSAndroid Build Coastguard Worker "unexpected(1)");
147*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(fmt::format("{}", std::expected<std::string, int>{"test"}),
148*5c90c05cSAndroid Build Coastguard Worker "expected(\"test\")");
149*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(fmt::format(
150*5c90c05cSAndroid Build Coastguard Worker "{}", std::expected<int, std::string>{std::unexpected("test")}),
151*5c90c05cSAndroid Build Coastguard Worker "unexpected(\"test\")");
152*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(fmt::format("{}", std::expected<char, int>{'a'}), "expected('a')");
153*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(fmt::format("{}", std::expected<int, char>{std::unexpected('a')}),
154*5c90c05cSAndroid Build Coastguard Worker "unexpected('a')");
155*5c90c05cSAndroid Build Coastguard Worker
156*5c90c05cSAndroid Build Coastguard Worker struct unformattable1 {};
157*5c90c05cSAndroid Build Coastguard Worker struct unformattable2 {};
158*5c90c05cSAndroid Build Coastguard Worker EXPECT_FALSE((fmt::is_formattable<unformattable1>::value));
159*5c90c05cSAndroid Build Coastguard Worker EXPECT_FALSE((fmt::is_formattable<unformattable2>::value));
160*5c90c05cSAndroid Build Coastguard Worker EXPECT_FALSE((fmt::is_formattable<
161*5c90c05cSAndroid Build Coastguard Worker std::expected<unformattable1, unformattable2>>::value));
162*5c90c05cSAndroid Build Coastguard Worker EXPECT_FALSE(
163*5c90c05cSAndroid Build Coastguard Worker (fmt::is_formattable<std::expected<unformattable1, int>>::value));
164*5c90c05cSAndroid Build Coastguard Worker EXPECT_FALSE(
165*5c90c05cSAndroid Build Coastguard Worker (fmt::is_formattable<std::expected<int, unformattable2>>::value));
166*5c90c05cSAndroid Build Coastguard Worker EXPECT_TRUE((fmt::is_formattable<std::expected<int, int>>::value));
167*5c90c05cSAndroid Build Coastguard Worker EXPECT_TRUE((fmt::is_formattable<std::expected<void, int>>::value));
168*5c90c05cSAndroid Build Coastguard Worker #endif
169*5c90c05cSAndroid Build Coastguard Worker }
170*5c90c05cSAndroid Build Coastguard Worker
171*5c90c05cSAndroid Build Coastguard Worker namespace my_nso {
172*5c90c05cSAndroid Build Coastguard Worker enum class my_number {
173*5c90c05cSAndroid Build Coastguard Worker one,
174*5c90c05cSAndroid Build Coastguard Worker two,
175*5c90c05cSAndroid Build Coastguard Worker };
format_as(my_number number)176*5c90c05cSAndroid Build Coastguard Worker auto format_as(my_number number) -> fmt::string_view {
177*5c90c05cSAndroid Build Coastguard Worker return number == my_number::one ? "first" : "second";
178*5c90c05cSAndroid Build Coastguard Worker }
179*5c90c05cSAndroid Build Coastguard Worker
180*5c90c05cSAndroid Build Coastguard Worker class my_class {
181*5c90c05cSAndroid Build Coastguard Worker public:
182*5c90c05cSAndroid Build Coastguard Worker int av;
183*5c90c05cSAndroid Build Coastguard Worker
184*5c90c05cSAndroid Build Coastguard Worker private:
format_as(const my_class & elm)185*5c90c05cSAndroid Build Coastguard Worker friend auto format_as(const my_class& elm) -> std::string {
186*5c90c05cSAndroid Build Coastguard Worker return fmt::to_string(elm.av);
187*5c90c05cSAndroid Build Coastguard Worker }
188*5c90c05cSAndroid Build Coastguard Worker };
189*5c90c05cSAndroid Build Coastguard Worker } // namespace my_nso
TEST(std_test,optional_format_as)190*5c90c05cSAndroid Build Coastguard Worker TEST(std_test, optional_format_as) {
191*5c90c05cSAndroid Build Coastguard Worker #ifdef __cpp_lib_optional
192*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(fmt::format("{}", std::optional<my_nso::my_number>{}), "none");
193*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(fmt::format("{}", std::optional{my_nso::my_number::one}),
194*5c90c05cSAndroid Build Coastguard Worker "optional(\"first\")");
195*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(fmt::format("{}", std::optional<my_nso::my_class>{}), "none");
196*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(fmt::format("{}", std::optional{my_nso::my_class{7}}),
197*5c90c05cSAndroid Build Coastguard Worker "optional(\"7\")");
198*5c90c05cSAndroid Build Coastguard Worker #endif
199*5c90c05cSAndroid Build Coastguard Worker }
200*5c90c05cSAndroid Build Coastguard Worker
201*5c90c05cSAndroid Build Coastguard Worker struct throws_on_move {
202*5c90c05cSAndroid Build Coastguard Worker throws_on_move() = default;
203*5c90c05cSAndroid Build Coastguard Worker
throws_on_movethrows_on_move204*5c90c05cSAndroid Build Coastguard Worker [[noreturn]] throws_on_move(throws_on_move&&) {
205*5c90c05cSAndroid Build Coastguard Worker throw std::runtime_error("Thrown by throws_on_move");
206*5c90c05cSAndroid Build Coastguard Worker }
207*5c90c05cSAndroid Build Coastguard Worker
208*5c90c05cSAndroid Build Coastguard Worker throws_on_move(const throws_on_move&) = default;
209*5c90c05cSAndroid Build Coastguard Worker };
210*5c90c05cSAndroid Build Coastguard Worker
211*5c90c05cSAndroid Build Coastguard Worker namespace fmt {
212*5c90c05cSAndroid Build Coastguard Worker template <> struct formatter<throws_on_move> : formatter<string_view> {
formatfmt::formatter213*5c90c05cSAndroid Build Coastguard Worker auto format(const throws_on_move&, format_context& ctx) const
214*5c90c05cSAndroid Build Coastguard Worker -> decltype(ctx.out()) {
215*5c90c05cSAndroid Build Coastguard Worker string_view str("<throws_on_move>");
216*5c90c05cSAndroid Build Coastguard Worker return formatter<string_view>::format(str, ctx);
217*5c90c05cSAndroid Build Coastguard Worker }
218*5c90c05cSAndroid Build Coastguard Worker };
219*5c90c05cSAndroid Build Coastguard Worker } // namespace fmt
220*5c90c05cSAndroid Build Coastguard Worker
TEST(std_test,variant)221*5c90c05cSAndroid Build Coastguard Worker TEST(std_test, variant) {
222*5c90c05cSAndroid Build Coastguard Worker #ifdef __cpp_lib_variant
223*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(fmt::format("{}", std::monostate{}), "monostate");
224*5c90c05cSAndroid Build Coastguard Worker using V0 = std::variant<int, float, std::string, char>;
225*5c90c05cSAndroid Build Coastguard Worker V0 v0(42);
226*5c90c05cSAndroid Build Coastguard Worker V0 v1(1.5f);
227*5c90c05cSAndroid Build Coastguard Worker V0 v2("hello");
228*5c90c05cSAndroid Build Coastguard Worker V0 v3('i');
229*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(fmt::format("{}", v0), "variant(42)");
230*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(fmt::format("{}", v1), "variant(1.5)");
231*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(fmt::format("{}", v2), "variant(\"hello\")");
232*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(fmt::format("{}", v3), "variant('i')");
233*5c90c05cSAndroid Build Coastguard Worker
234*5c90c05cSAndroid Build Coastguard Worker struct unformattable {};
235*5c90c05cSAndroid Build Coastguard Worker EXPECT_FALSE((fmt::is_formattable<unformattable>::value));
236*5c90c05cSAndroid Build Coastguard Worker EXPECT_FALSE((fmt::is_formattable<std::variant<unformattable>>::value));
237*5c90c05cSAndroid Build Coastguard Worker EXPECT_FALSE((fmt::is_formattable<std::variant<unformattable, int>>::value));
238*5c90c05cSAndroid Build Coastguard Worker EXPECT_FALSE((fmt::is_formattable<std::variant<int, unformattable>>::value));
239*5c90c05cSAndroid Build Coastguard Worker EXPECT_FALSE(
240*5c90c05cSAndroid Build Coastguard Worker (fmt::is_formattable<std::variant<unformattable, unformattable>>::value));
241*5c90c05cSAndroid Build Coastguard Worker EXPECT_TRUE((fmt::is_formattable<std::variant<int, float>>::value));
242*5c90c05cSAndroid Build Coastguard Worker
243*5c90c05cSAndroid Build Coastguard Worker using V1 = std::variant<std::monostate, std::string, std::string>;
244*5c90c05cSAndroid Build Coastguard Worker V1 v4{};
245*5c90c05cSAndroid Build Coastguard Worker V1 v5{std::in_place_index<1>, "yes, this is variant"};
246*5c90c05cSAndroid Build Coastguard Worker
247*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(fmt::format("{}", v4), "variant(monostate)");
248*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(fmt::format("{}", v5), "variant(\"yes, this is variant\")");
249*5c90c05cSAndroid Build Coastguard Worker
250*5c90c05cSAndroid Build Coastguard Worker volatile int i = 42; // Test compile error before GCC 11 described in #3068.
251*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(fmt::format("{}", i), "42");
252*5c90c05cSAndroid Build Coastguard Worker
253*5c90c05cSAndroid Build Coastguard Worker std::variant<std::monostate, throws_on_move> v6;
254*5c90c05cSAndroid Build Coastguard Worker
255*5c90c05cSAndroid Build Coastguard Worker try {
256*5c90c05cSAndroid Build Coastguard Worker throws_on_move thrower;
257*5c90c05cSAndroid Build Coastguard Worker v6.emplace<throws_on_move>(std::move(thrower));
258*5c90c05cSAndroid Build Coastguard Worker } catch (const std::runtime_error&) {
259*5c90c05cSAndroid Build Coastguard Worker }
260*5c90c05cSAndroid Build Coastguard Worker // v6 is now valueless by exception
261*5c90c05cSAndroid Build Coastguard Worker
262*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(fmt::format("{}", v6), "variant(valueless by exception)");
263*5c90c05cSAndroid Build Coastguard Worker
264*5c90c05cSAndroid Build Coastguard Worker #endif
265*5c90c05cSAndroid Build Coastguard Worker }
266*5c90c05cSAndroid Build Coastguard Worker
TEST(std_test,error_code)267*5c90c05cSAndroid Build Coastguard Worker TEST(std_test, error_code) {
268*5c90c05cSAndroid Build Coastguard Worker auto& generic = std::generic_category();
269*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ("generic:42",
270*5c90c05cSAndroid Build Coastguard Worker fmt::format(FMT_STRING("{0}"), std::error_code(42, generic)));
271*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(" generic:42",
272*5c90c05cSAndroid Build Coastguard Worker fmt::format(FMT_STRING("{:>12}"), std::error_code(42, generic)));
273*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ("generic:42 ",
274*5c90c05cSAndroid Build Coastguard Worker fmt::format(FMT_STRING("{:12}"), std::error_code(42, generic)));
275*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ("system:42",
276*5c90c05cSAndroid Build Coastguard Worker fmt::format(FMT_STRING("{0}"),
277*5c90c05cSAndroid Build Coastguard Worker std::error_code(42, fmt::system_category())));
278*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ("system:-42",
279*5c90c05cSAndroid Build Coastguard Worker fmt::format(FMT_STRING("{0}"),
280*5c90c05cSAndroid Build Coastguard Worker std::error_code(-42, fmt::system_category())));
281*5c90c05cSAndroid Build Coastguard Worker }
282*5c90c05cSAndroid Build Coastguard Worker
exception_test()283*5c90c05cSAndroid Build Coastguard Worker template <typename Catch> void exception_test() {
284*5c90c05cSAndroid Build Coastguard Worker try {
285*5c90c05cSAndroid Build Coastguard Worker throw std::runtime_error("Test Exception");
286*5c90c05cSAndroid Build Coastguard Worker } catch (const Catch& ex) {
287*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ("Test Exception", fmt::format("{}", ex));
288*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ("std::runtime_error: Test Exception", fmt::format("{:t}", ex));
289*5c90c05cSAndroid Build Coastguard Worker }
290*5c90c05cSAndroid Build Coastguard Worker }
291*5c90c05cSAndroid Build Coastguard Worker
292*5c90c05cSAndroid Build Coastguard Worker namespace my_ns1 {
293*5c90c05cSAndroid Build Coastguard Worker namespace my_ns2 {
294*5c90c05cSAndroid Build Coastguard Worker struct my_exception : public std::exception {
295*5c90c05cSAndroid Build Coastguard Worker private:
296*5c90c05cSAndroid Build Coastguard Worker std::string msg;
297*5c90c05cSAndroid Build Coastguard Worker
298*5c90c05cSAndroid Build Coastguard Worker public:
my_exceptionmy_ns1::my_ns2::my_exception299*5c90c05cSAndroid Build Coastguard Worker my_exception(const std::string& s) : msg(s) {}
300*5c90c05cSAndroid Build Coastguard Worker const char* what() const noexcept override;
301*5c90c05cSAndroid Build Coastguard Worker };
what() const302*5c90c05cSAndroid Build Coastguard Worker const char* my_exception::what() const noexcept { return msg.c_str(); }
303*5c90c05cSAndroid Build Coastguard Worker } // namespace my_ns2
304*5c90c05cSAndroid Build Coastguard Worker } // namespace my_ns1
305*5c90c05cSAndroid Build Coastguard Worker
TEST(std_test,exception)306*5c90c05cSAndroid Build Coastguard Worker TEST(std_test, exception) {
307*5c90c05cSAndroid Build Coastguard Worker using testing::StartsWith;
308*5c90c05cSAndroid Build Coastguard Worker exception_test<std::exception>();
309*5c90c05cSAndroid Build Coastguard Worker exception_test<std::runtime_error>();
310*5c90c05cSAndroid Build Coastguard Worker
311*5c90c05cSAndroid Build Coastguard Worker try {
312*5c90c05cSAndroid Build Coastguard Worker using namespace my_ns1::my_ns2;
313*5c90c05cSAndroid Build Coastguard Worker throw my_exception("My Exception");
314*5c90c05cSAndroid Build Coastguard Worker } catch (const std::exception& ex) {
315*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ("my_ns1::my_ns2::my_exception: My Exception",
316*5c90c05cSAndroid Build Coastguard Worker fmt::format("{:t}", ex));
317*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ("My Exception", fmt::format("{:}", ex));
318*5c90c05cSAndroid Build Coastguard Worker }
319*5c90c05cSAndroid Build Coastguard Worker
320*5c90c05cSAndroid Build Coastguard Worker try {
321*5c90c05cSAndroid Build Coastguard Worker throw std::system_error(std::error_code(), "message");
322*5c90c05cSAndroid Build Coastguard Worker } catch (const std::system_error& ex) {
323*5c90c05cSAndroid Build Coastguard Worker EXPECT_THAT(fmt::format("{:t}", ex), StartsWith("std::system_error: "));
324*5c90c05cSAndroid Build Coastguard Worker }
325*5c90c05cSAndroid Build Coastguard Worker
326*5c90c05cSAndroid Build Coastguard Worker #ifdef __cpp_lib_filesystem
327*5c90c05cSAndroid Build Coastguard Worker // Tests that the inline namespace is stripped out, e.g.
328*5c90c05cSAndroid Build Coastguard Worker // std::filesystem::__cxx11::* -> std::filesystem::*.
329*5c90c05cSAndroid Build Coastguard Worker try {
330*5c90c05cSAndroid Build Coastguard Worker throw std::filesystem::filesystem_error("message", std::error_code());
331*5c90c05cSAndroid Build Coastguard Worker } catch (const std::filesystem::filesystem_error& ex) {
332*5c90c05cSAndroid Build Coastguard Worker EXPECT_THAT(fmt::format("{:t}", ex),
333*5c90c05cSAndroid Build Coastguard Worker StartsWith("std::filesystem::filesystem_error: "));
334*5c90c05cSAndroid Build Coastguard Worker }
335*5c90c05cSAndroid Build Coastguard Worker #endif
336*5c90c05cSAndroid Build Coastguard Worker }
337*5c90c05cSAndroid Build Coastguard Worker
338*5c90c05cSAndroid Build Coastguard Worker #if FMT_USE_RTTI
TEST(std_test,type_info)339*5c90c05cSAndroid Build Coastguard Worker TEST(std_test, type_info) {
340*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(fmt::format("{}", typeid(std::runtime_error)),
341*5c90c05cSAndroid Build Coastguard Worker "std::runtime_error");
342*5c90c05cSAndroid Build Coastguard Worker }
343*5c90c05cSAndroid Build Coastguard Worker #endif
344*5c90c05cSAndroid Build Coastguard Worker
TEST(std_test,format_bit_reference)345*5c90c05cSAndroid Build Coastguard Worker TEST(std_test, format_bit_reference) {
346*5c90c05cSAndroid Build Coastguard Worker std::bitset<2> bs(1);
347*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(fmt::format("{} {}", bs[0], bs[1]), "true false");
348*5c90c05cSAndroid Build Coastguard Worker std::vector<bool> v = {true, false};
349*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(fmt::format("{} {}", v[0], v[1]), "true false");
350*5c90c05cSAndroid Build Coastguard Worker }
351*5c90c05cSAndroid Build Coastguard Worker
TEST(std_test,format_const_bit_reference)352*5c90c05cSAndroid Build Coastguard Worker TEST(std_test, format_const_bit_reference) {
353*5c90c05cSAndroid Build Coastguard Worker const std::bitset<2> bs(1);
354*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(fmt::format("{} {}", bs[0], bs[1]), "true false");
355*5c90c05cSAndroid Build Coastguard Worker const std::vector<bool> v = {true, false};
356*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(fmt::format("{} {}", v[0], v[1]), "true false");
357*5c90c05cSAndroid Build Coastguard Worker }
358*5c90c05cSAndroid Build Coastguard Worker
TEST(std_test,format_bitset)359*5c90c05cSAndroid Build Coastguard Worker TEST(std_test, format_bitset) {
360*5c90c05cSAndroid Build Coastguard Worker auto bs = std::bitset<6>(42);
361*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(fmt::format("{}", bs), "101010");
362*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(fmt::format("{:0>8}", bs), "00101010");
363*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(fmt::format("{:-^12}", bs), "---101010---");
364*5c90c05cSAndroid Build Coastguard Worker }
365*5c90c05cSAndroid Build Coastguard Worker
TEST(std_test,format_atomic)366*5c90c05cSAndroid Build Coastguard Worker TEST(std_test, format_atomic) {
367*5c90c05cSAndroid Build Coastguard Worker std::atomic<bool> b(false);
368*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(fmt::format("{}", b), "false");
369*5c90c05cSAndroid Build Coastguard Worker
370*5c90c05cSAndroid Build Coastguard Worker const std::atomic<bool> cb(true);
371*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(fmt::format("{}", cb), "true");
372*5c90c05cSAndroid Build Coastguard Worker }
373*5c90c05cSAndroid Build Coastguard Worker
374*5c90c05cSAndroid Build Coastguard Worker #ifdef __cpp_lib_atomic_flag_test
TEST(std_test,format_atomic_flag)375*5c90c05cSAndroid Build Coastguard Worker TEST(std_test, format_atomic_flag) {
376*5c90c05cSAndroid Build Coastguard Worker std::atomic_flag f = ATOMIC_FLAG_INIT;
377*5c90c05cSAndroid Build Coastguard Worker (void)f.test_and_set();
378*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(fmt::format("{}", f), "true");
379*5c90c05cSAndroid Build Coastguard Worker
380*5c90c05cSAndroid Build Coastguard Worker const std::atomic_flag cf = ATOMIC_FLAG_INIT;
381*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(fmt::format("{}", cf), "false");
382*5c90c05cSAndroid Build Coastguard Worker }
383*5c90c05cSAndroid Build Coastguard Worker #endif // __cpp_lib_atomic_flag_test
384*5c90c05cSAndroid Build Coastguard Worker
TEST(std_test,format_unique_ptr)385*5c90c05cSAndroid Build Coastguard Worker TEST(std_test, format_unique_ptr) {
386*5c90c05cSAndroid Build Coastguard Worker std::unique_ptr<int> up(new int(1));
387*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(fmt::format("{}", fmt::ptr(up.get())),
388*5c90c05cSAndroid Build Coastguard Worker fmt::format("{}", fmt::ptr(up)));
389*5c90c05cSAndroid Build Coastguard Worker struct custom_deleter {
390*5c90c05cSAndroid Build Coastguard Worker void operator()(int* p) const { delete p; }
391*5c90c05cSAndroid Build Coastguard Worker };
392*5c90c05cSAndroid Build Coastguard Worker std::unique_ptr<int, custom_deleter> upcd(new int(1));
393*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(fmt::format("{}", fmt::ptr(upcd.get())),
394*5c90c05cSAndroid Build Coastguard Worker fmt::format("{}", fmt::ptr(upcd)));
395*5c90c05cSAndroid Build Coastguard Worker }
396*5c90c05cSAndroid Build Coastguard Worker
TEST(std_test,format_shared_ptr)397*5c90c05cSAndroid Build Coastguard Worker TEST(std_test, format_shared_ptr) {
398*5c90c05cSAndroid Build Coastguard Worker std::shared_ptr<int> sp(new int(1));
399*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(fmt::format("{}", fmt::ptr(sp.get())),
400*5c90c05cSAndroid Build Coastguard Worker fmt::format("{}", fmt::ptr(sp)));
401*5c90c05cSAndroid Build Coastguard Worker }
402*5c90c05cSAndroid Build Coastguard Worker
TEST(std_test,format_reference_wrapper)403*5c90c05cSAndroid Build Coastguard Worker TEST(std_test, format_reference_wrapper) {
404*5c90c05cSAndroid Build Coastguard Worker int num = 35;
405*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ("35", fmt::to_string(std::cref(num)));
406*5c90c05cSAndroid Build Coastguard Worker }
407