Mercurial > minori
comparison dep/fmt/test/format-test.cc @ 343:1faa72660932
*: transfer back to cmake from autotools
autotools just made lots of things more complicated than
they should have and many things broke (i.e. translations)
author | Paper <paper@paper.us.eu.org> |
---|---|
date | Thu, 20 Jun 2024 05:56:06 -0400 (7 months ago) |
parents | |
children |
comparison
equal
deleted
inserted
replaced
342:adb79bdde329 | 343:1faa72660932 |
---|---|
1 // Formatting library for C++ - formatting library tests | |
2 // | |
3 // Copyright (c) 2012 - present, Victor Zverovich | |
4 // All rights reserved. | |
5 // | |
6 // For the license information refer to format.h. | |
7 | |
8 // Check if fmt/format.h compiles with windows.h included before it. | |
9 #ifdef _WIN32 | |
10 # include <windows.h> | |
11 #endif | |
12 // clang-format off | |
13 #include "fmt/format.h" | |
14 // clang-format on | |
15 | |
16 #include <stdint.h> // uint32_t | |
17 | |
18 #include <climits> // INT_MAX | |
19 #include <cmath> // std::signbit | |
20 #include <cstring> // std::strlen | |
21 #include <iterator> // std::back_inserter | |
22 #include <list> // std::list | |
23 #include <memory> // std::unique_ptr | |
24 #include <type_traits> // std::is_default_constructible | |
25 | |
26 #include "gtest-extra.h" | |
27 #include "mock-allocator.h" | |
28 #include "util.h" | |
29 | |
30 using fmt::basic_memory_buffer; | |
31 using fmt::format_error; | |
32 using fmt::memory_buffer; | |
33 using fmt::runtime; | |
34 using fmt::string_view; | |
35 using fmt::detail::max_value; | |
36 using fmt::detail::uint128_fallback; | |
37 | |
38 using testing::Return; | |
39 using testing::StrictMock; | |
40 | |
41 enum { buffer_size = 256 }; | |
42 | |
43 TEST(uint128_test, ctor) { | |
44 auto n = uint128_fallback(); | |
45 EXPECT_EQ(n, 0); | |
46 n = uint128_fallback(42); | |
47 EXPECT_EQ(n, 42); | |
48 EXPECT_EQ(static_cast<uint64_t>(n), 42); | |
49 } | |
50 | |
51 TEST(uint128_test, shift) { | |
52 auto n = uint128_fallback(42); | |
53 n = n << 64; | |
54 EXPECT_EQ(static_cast<uint64_t>(n), 0); | |
55 n = n >> 64; | |
56 EXPECT_EQ(static_cast<uint64_t>(n), 42); | |
57 n = n << 62; | |
58 EXPECT_EQ(static_cast<uint64_t>(n >> 64), 0xa); | |
59 EXPECT_EQ(static_cast<uint64_t>(n), 0x8000000000000000); | |
60 n = n >> 62; | |
61 EXPECT_EQ(static_cast<uint64_t>(n), 42); | |
62 EXPECT_EQ(uint128_fallback(1) << 112, uint128_fallback(0x1000000000000, 0)); | |
63 EXPECT_EQ(uint128_fallback(0x1000000000000, 0) >> 112, uint128_fallback(1)); | |
64 } | |
65 | |
66 TEST(uint128_test, minus) { | |
67 auto n = uint128_fallback(42); | |
68 EXPECT_EQ(n - 2, 40); | |
69 } | |
70 | |
71 TEST(uint128_test, plus_assign) { | |
72 auto n = uint128_fallback(32); | |
73 n += uint128_fallback(10); | |
74 EXPECT_EQ(n, 42); | |
75 n = uint128_fallback(max_value<uint64_t>()); | |
76 n += uint128_fallback(1); | |
77 EXPECT_EQ(n, uint128_fallback(1) << 64); | |
78 } | |
79 | |
80 TEST(uint128_test, multiply) { | |
81 auto n = uint128_fallback(2251799813685247); | |
82 n = n * 3611864890; | |
83 EXPECT_EQ(static_cast<uint64_t>(n >> 64), 440901); | |
84 } | |
85 | |
86 template <typename Float> void check_isfinite() { | |
87 using fmt::detail::isfinite; | |
88 EXPECT_TRUE(isfinite(Float(0.0))); | |
89 EXPECT_TRUE(isfinite(Float(42.0))); | |
90 EXPECT_TRUE(isfinite(Float(-42.0))); | |
91 EXPECT_TRUE(isfinite(Float(fmt::detail::max_value<double>()))); | |
92 // Use double because std::numeric_limits is broken for __float128. | |
93 using limits = std::numeric_limits<double>; | |
94 FMT_CONSTEXPR20 auto result = isfinite(Float(limits::infinity())); | |
95 EXPECT_FALSE(result); | |
96 EXPECT_FALSE(isfinite(Float(limits::infinity()))); | |
97 EXPECT_FALSE(isfinite(Float(-limits::infinity()))); | |
98 EXPECT_FALSE(isfinite(Float(limits::quiet_NaN()))); | |
99 EXPECT_FALSE(isfinite(Float(-limits::quiet_NaN()))); | |
100 } | |
101 | |
102 TEST(float_test, isfinite) { | |
103 check_isfinite<double>(); | |
104 #if FMT_USE_FLOAT128 | |
105 check_isfinite<fmt::detail::float128>(); | |
106 #endif | |
107 } | |
108 | |
109 template <typename Float> void check_isnan() { | |
110 using fmt::detail::isnan; | |
111 EXPECT_FALSE(isnan(Float(0.0))); | |
112 EXPECT_FALSE(isnan(Float(42.0))); | |
113 EXPECT_FALSE(isnan(Float(-42.0))); | |
114 EXPECT_FALSE(isnan(Float(fmt::detail::max_value<double>()))); | |
115 // Use double because std::numeric_limits is broken for __float128. | |
116 using limits = std::numeric_limits<double>; | |
117 EXPECT_FALSE(isnan(Float(limits::infinity()))); | |
118 EXPECT_FALSE(isnan(Float(-limits::infinity()))); | |
119 EXPECT_TRUE(isnan(Float(limits::quiet_NaN()))); | |
120 EXPECT_TRUE(isnan(Float(-limits::quiet_NaN()))); | |
121 } | |
122 | |
123 TEST(float_test, isnan) { | |
124 check_isnan<double>(); | |
125 #if FMT_USE_FLOAT128 | |
126 check_isnan<fmt::detail::float128>(); | |
127 #endif | |
128 } | |
129 | |
130 struct uint32_pair { | |
131 uint32_t u[2]; | |
132 }; | |
133 | |
134 TEST(util_test, bit_cast) { | |
135 auto s = fmt::detail::bit_cast<uint32_pair>(uint64_t{42}); | |
136 EXPECT_EQ(fmt::detail::bit_cast<uint64_t>(s), 42ull); | |
137 s = fmt::detail::bit_cast<uint32_pair>(~uint64_t{0}); | |
138 EXPECT_EQ(fmt::detail::bit_cast<uint64_t>(s), ~0ull); | |
139 } | |
140 | |
141 // Increment a number in a string. | |
142 void increment(char* s) { | |
143 for (int i = static_cast<int>(std::strlen(s)) - 1; i >= 0; --i) { | |
144 if (s[i] != '9') { | |
145 ++s[i]; | |
146 break; | |
147 } | |
148 s[i] = '0'; | |
149 } | |
150 } | |
151 | |
152 TEST(util_test, increment) { | |
153 char s[10] = "123"; | |
154 increment(s); | |
155 EXPECT_STREQ("124", s); | |
156 s[2] = '8'; | |
157 increment(s); | |
158 EXPECT_STREQ("129", s); | |
159 increment(s); | |
160 EXPECT_STREQ("130", s); | |
161 s[1] = s[2] = '9'; | |
162 increment(s); | |
163 EXPECT_STREQ("200", s); | |
164 } | |
165 | |
166 TEST(util_test, parse_nonnegative_int) { | |
167 auto s = fmt::string_view("10000000000"); | |
168 auto begin = s.begin(), end = s.end(); | |
169 EXPECT_EQ(fmt::detail::parse_nonnegative_int(begin, end, -1), -1); | |
170 s = "2147483649"; | |
171 begin = s.begin(); | |
172 end = s.end(); | |
173 EXPECT_EQ(fmt::detail::parse_nonnegative_int(begin, end, -1), -1); | |
174 } | |
175 | |
176 TEST(format_impl_test, compute_width) { | |
177 EXPECT_EQ(fmt::detail::compute_width("вожык"), 5); | |
178 } | |
179 | |
180 TEST(util_test, utf8_to_utf16) { | |
181 auto u = fmt::detail::utf8_to_utf16("лошадка"); | |
182 EXPECT_EQ(L"\x043B\x043E\x0448\x0430\x0434\x043A\x0430", u.str()); | |
183 EXPECT_EQ(7, u.size()); | |
184 // U+10437 { DESERET SMALL LETTER YEE } | |
185 EXPECT_EQ(L"\xD801\xDC37", fmt::detail::utf8_to_utf16("𐐷").str()); | |
186 EXPECT_THROW_MSG(fmt::detail::utf8_to_utf16("\xc3\x28"), std::runtime_error, | |
187 "invalid utf8"); | |
188 EXPECT_THROW_MSG(fmt::detail::utf8_to_utf16(fmt::string_view("л", 1)), | |
189 std::runtime_error, "invalid utf8"); | |
190 EXPECT_EQ(L"123456", fmt::detail::utf8_to_utf16("123456").str()); | |
191 } | |
192 | |
193 TEST(util_test, utf8_to_utf16_empty_string) { | |
194 auto s = std::string(); | |
195 auto u = fmt::detail::utf8_to_utf16(s.c_str()); | |
196 EXPECT_EQ(L"", u.str()); | |
197 EXPECT_EQ(s.size(), u.size()); | |
198 } | |
199 | |
200 TEST(util_test, allocator_ref) { | |
201 using test_allocator_ref = allocator_ref<mock_allocator<int>>; | |
202 auto check_forwarding = [](mock_allocator<int>& alloc, | |
203 test_allocator_ref& ref) { | |
204 int mem; | |
205 // Check if value_type is properly defined. | |
206 allocator_ref<mock_allocator<int>>::value_type* ptr = &mem; | |
207 // Check forwarding. | |
208 EXPECT_CALL(alloc, allocate(42)).WillOnce(Return(ptr)); | |
209 ref.allocate(42); | |
210 EXPECT_CALL(alloc, deallocate(ptr, 42)); | |
211 ref.deallocate(ptr, 42); | |
212 }; | |
213 | |
214 StrictMock<mock_allocator<int>> alloc; | |
215 auto ref = test_allocator_ref(&alloc); | |
216 // Check if allocator_ref forwards to the underlying allocator. | |
217 check_forwarding(alloc, ref); | |
218 test_allocator_ref ref2(ref); | |
219 check_forwarding(alloc, ref2); | |
220 test_allocator_ref ref3; | |
221 EXPECT_EQ(nullptr, ref3.get()); | |
222 ref3 = ref; | |
223 check_forwarding(alloc, ref3); | |
224 } | |
225 | |
226 TEST(util_test, format_system_error) { | |
227 fmt::memory_buffer message; | |
228 fmt::format_system_error(message, EDOM, "test"); | |
229 auto ec = std::error_code(EDOM, std::generic_category()); | |
230 EXPECT_EQ(to_string(message), std::system_error(ec, "test").what()); | |
231 message = fmt::memory_buffer(); | |
232 | |
233 // Check if std::allocator throws on allocating max size_t / 2 chars. | |
234 size_t max_size = max_value<size_t>() / 2; | |
235 bool throws_on_alloc = false; | |
236 try { | |
237 auto alloc = std::allocator<char>(); | |
238 alloc.deallocate(alloc.allocate(max_size), max_size); | |
239 } catch (const std::bad_alloc&) { | |
240 throws_on_alloc = true; | |
241 } | |
242 if (!throws_on_alloc) { | |
243 fmt::print("warning: std::allocator allocates {} chars\n", max_size); | |
244 return; | |
245 } | |
246 } | |
247 | |
248 TEST(util_test, system_error) { | |
249 auto test_error = fmt::system_error(EDOM, "test"); | |
250 auto ec = std::error_code(EDOM, std::generic_category()); | |
251 EXPECT_STREQ(test_error.what(), std::system_error(ec, "test").what()); | |
252 EXPECT_EQ(test_error.code(), ec); | |
253 | |
254 auto error = std::system_error(std::error_code()); | |
255 try { | |
256 throw fmt::system_error(EDOM, "test {}", "error"); | |
257 } catch (const std::system_error& e) { | |
258 error = e; | |
259 } | |
260 fmt::memory_buffer message; | |
261 fmt::format_system_error(message, EDOM, "test error"); | |
262 EXPECT_EQ(error.what(), to_string(message)); | |
263 EXPECT_EQ(error.code(), std::error_code(EDOM, std::generic_category())); | |
264 } | |
265 | |
266 TEST(util_test, report_system_error) { | |
267 fmt::memory_buffer out; | |
268 fmt::format_system_error(out, EDOM, "test error"); | |
269 out.push_back('\n'); | |
270 EXPECT_WRITE(stderr, fmt::report_system_error(EDOM, "test error"), | |
271 to_string(out)); | |
272 } | |
273 | |
274 TEST(memory_buffer_test, ctor) { | |
275 basic_memory_buffer<char, 123> buffer; | |
276 EXPECT_EQ(static_cast<size_t>(0), buffer.size()); | |
277 EXPECT_EQ(123u, buffer.capacity()); | |
278 } | |
279 | |
280 using std_allocator = allocator_ref<std::allocator<char>>; | |
281 | |
282 TEST(memory_buffer_test, move_ctor_inline_buffer) { | |
283 auto check_move_buffer = | |
284 [](const char* str, basic_memory_buffer<char, 5, std_allocator>& buffer) { | |
285 std::allocator<char>* alloc = buffer.get_allocator().get(); | |
286 basic_memory_buffer<char, 5, std_allocator> buffer2(std::move(buffer)); | |
287 // Move shouldn't destroy the inline content of the first buffer. | |
288 EXPECT_EQ(str, std::string(&buffer[0], buffer.size())); | |
289 EXPECT_EQ(str, std::string(&buffer2[0], buffer2.size())); | |
290 EXPECT_EQ(5u, buffer2.capacity()); | |
291 // Move should transfer allocator. | |
292 EXPECT_EQ(nullptr, buffer.get_allocator().get()); | |
293 EXPECT_EQ(alloc, buffer2.get_allocator().get()); | |
294 }; | |
295 | |
296 auto alloc = std::allocator<char>(); | |
297 basic_memory_buffer<char, 5, std_allocator> buffer((std_allocator(&alloc))); | |
298 const char test[] = "test"; | |
299 buffer.append(string_view(test, 4)); | |
300 check_move_buffer("test", buffer); | |
301 // Adding one more character fills the inline buffer, but doesn't cause | |
302 // dynamic allocation. | |
303 buffer.push_back('a'); | |
304 check_move_buffer("testa", buffer); | |
305 } | |
306 | |
307 TEST(memory_buffer_test, move_ctor_dynamic_buffer) { | |
308 auto alloc = std::allocator<char>(); | |
309 basic_memory_buffer<char, 4, std_allocator> buffer((std_allocator(&alloc))); | |
310 const char test[] = "test"; | |
311 buffer.append(test, test + 4); | |
312 const char* inline_buffer_ptr = &buffer[0]; | |
313 // Adding one more character causes the content to move from the inline to | |
314 // a dynamically allocated buffer. | |
315 buffer.push_back('a'); | |
316 basic_memory_buffer<char, 4, std_allocator> buffer2(std::move(buffer)); | |
317 // Move should rip the guts of the first buffer. | |
318 EXPECT_EQ(&buffer[0], inline_buffer_ptr); | |
319 EXPECT_EQ(buffer.size(), 0); | |
320 EXPECT_EQ(std::string(&buffer2[0], buffer2.size()), "testa"); | |
321 EXPECT_GT(buffer2.capacity(), 4u); | |
322 } | |
323 | |
324 void check_move_assign_buffer(const char* str, | |
325 basic_memory_buffer<char, 5>& buffer) { | |
326 basic_memory_buffer<char, 5> buffer2; | |
327 buffer2 = std::move(buffer); | |
328 // Move shouldn't destroy the inline content of the first buffer. | |
329 EXPECT_EQ(str, std::string(&buffer[0], buffer.size())); | |
330 EXPECT_EQ(str, std::string(&buffer2[0], buffer2.size())); | |
331 EXPECT_EQ(5u, buffer2.capacity()); | |
332 } | |
333 | |
334 TEST(memory_buffer_test, move_assignment) { | |
335 basic_memory_buffer<char, 5> buffer; | |
336 const char test[] = "test"; | |
337 buffer.append(test, test + 4); | |
338 check_move_assign_buffer("test", buffer); | |
339 // Adding one more character fills the inline buffer, but doesn't cause | |
340 // dynamic allocation. | |
341 buffer.push_back('a'); | |
342 check_move_assign_buffer("testa", buffer); | |
343 const char* inline_buffer_ptr = &buffer[0]; | |
344 // Adding one more character causes the content to move from the inline to | |
345 // a dynamically allocated buffer. | |
346 buffer.push_back('b'); | |
347 basic_memory_buffer<char, 5> buffer2; | |
348 buffer2 = std::move(buffer); | |
349 // Move should rip the guts of the first buffer. | |
350 EXPECT_EQ(inline_buffer_ptr, &buffer[0]); | |
351 EXPECT_EQ("testab", std::string(&buffer2[0], buffer2.size())); | |
352 EXPECT_GT(buffer2.capacity(), 5u); | |
353 } | |
354 | |
355 TEST(memory_buffer_test, grow) { | |
356 using allocator = allocator_ref<mock_allocator<int>>; | |
357 mock_allocator<int> alloc; | |
358 basic_memory_buffer<int, 10, allocator> buffer((allocator(&alloc))); | |
359 buffer.resize(7); | |
360 using fmt::detail::to_unsigned; | |
361 for (int i = 0; i < 7; ++i) buffer[to_unsigned(i)] = i * i; | |
362 EXPECT_EQ(10u, buffer.capacity()); | |
363 int mem[20]; | |
364 mem[7] = 0xdead; | |
365 EXPECT_CALL(alloc, allocate(20)).WillOnce(Return(mem)); | |
366 buffer.try_reserve(20); | |
367 EXPECT_EQ(20u, buffer.capacity()); | |
368 // Check if size elements have been copied | |
369 for (int i = 0; i < 7; ++i) EXPECT_EQ(i * i, buffer[to_unsigned(i)]); | |
370 // and no more than that. | |
371 EXPECT_EQ(0xdead, buffer[7]); | |
372 EXPECT_CALL(alloc, deallocate(mem, 20)); | |
373 } | |
374 | |
375 TEST(memory_buffer_test, allocator) { | |
376 using test_allocator = allocator_ref<mock_allocator<char>>; | |
377 basic_memory_buffer<char, 10, test_allocator> buffer; | |
378 EXPECT_EQ(nullptr, buffer.get_allocator().get()); | |
379 StrictMock<mock_allocator<char>> alloc; | |
380 char mem; | |
381 { | |
382 basic_memory_buffer<char, 10, test_allocator> buffer2( | |
383 (test_allocator(&alloc))); | |
384 EXPECT_EQ(&alloc, buffer2.get_allocator().get()); | |
385 size_t size = 2 * fmt::inline_buffer_size; | |
386 EXPECT_CALL(alloc, allocate(size)).WillOnce(Return(&mem)); | |
387 buffer2.reserve(size); | |
388 EXPECT_CALL(alloc, deallocate(&mem, size)); | |
389 } | |
390 } | |
391 | |
392 TEST(memory_buffer_test, exception_in_deallocate) { | |
393 using test_allocator = allocator_ref<mock_allocator<char>>; | |
394 StrictMock<mock_allocator<char>> alloc; | |
395 basic_memory_buffer<char, 10, test_allocator> buffer( | |
396 (test_allocator(&alloc))); | |
397 size_t size = 2 * fmt::inline_buffer_size; | |
398 auto mem = std::vector<char>(size); | |
399 { | |
400 EXPECT_CALL(alloc, allocate(size)).WillOnce(Return(&mem[0])); | |
401 buffer.resize(size); | |
402 std::fill(&buffer[0], &buffer[0] + size, 'x'); | |
403 } | |
404 auto mem2 = std::vector<char>(2 * size); | |
405 { | |
406 EXPECT_CALL(alloc, allocate(2 * size)).WillOnce(Return(&mem2[0])); | |
407 auto e = std::exception(); | |
408 EXPECT_CALL(alloc, deallocate(&mem[0], size)).WillOnce(testing::Throw(e)); | |
409 EXPECT_THROW(buffer.reserve(2 * size), std::exception); | |
410 EXPECT_EQ(&mem2[0], &buffer[0]); | |
411 // Check that the data has been copied. | |
412 for (size_t i = 0; i < size; ++i) EXPECT_EQ('x', buffer[i]); | |
413 } | |
414 EXPECT_CALL(alloc, deallocate(&mem2[0], 2 * size)); | |
415 } | |
416 | |
417 template <typename Allocator, size_t MaxSize> | |
418 class max_size_allocator : public Allocator { | |
419 public: | |
420 using typename Allocator::value_type; | |
421 size_t max_size() const noexcept { return MaxSize; } | |
422 value_type* allocate(size_t n) { | |
423 if (n > max_size()) { | |
424 throw std::length_error("size > max_size"); | |
425 } | |
426 return std::allocator_traits<Allocator>::allocate( | |
427 *static_cast<Allocator*>(this), n); | |
428 } | |
429 void deallocate(value_type* p, size_t n) { | |
430 std::allocator_traits<Allocator>::deallocate(*static_cast<Allocator*>(this), | |
431 p, n); | |
432 } | |
433 }; | |
434 | |
435 TEST(memory_buffer_test, max_size_allocator) { | |
436 // 160 = 128 + 32 | |
437 using test_allocator = max_size_allocator<std::allocator<char>, 160>; | |
438 basic_memory_buffer<char, 10, test_allocator> buffer; | |
439 buffer.resize(128); | |
440 // new_capacity = 128 + 128/2 = 192 > 160 | |
441 buffer.resize(160); // Shouldn't throw. | |
442 } | |
443 | |
444 TEST(memory_buffer_test, max_size_allocator_overflow) { | |
445 using test_allocator = max_size_allocator<std::allocator<char>, 160>; | |
446 basic_memory_buffer<char, 10, test_allocator> buffer; | |
447 EXPECT_THROW(buffer.resize(161), std::exception); | |
448 } | |
449 | |
450 TEST(format_test, exception_from_lib) { | |
451 EXPECT_THROW_MSG(fmt::throw_format_error("test"), format_error, "test"); | |
452 } | |
453 | |
454 TEST(format_test, escape) { | |
455 EXPECT_EQ(fmt::format("{{"), "{"); | |
456 EXPECT_EQ(fmt::format("before {{"), "before {"); | |
457 EXPECT_EQ(fmt::format("{{ after"), "{ after"); | |
458 EXPECT_EQ(fmt::format("before {{ after"), "before { after"); | |
459 | |
460 EXPECT_EQ(fmt::format("}}"), "}"); | |
461 EXPECT_EQ(fmt::format("before }}"), "before }"); | |
462 EXPECT_EQ(fmt::format("}} after"), "} after"); | |
463 EXPECT_EQ(fmt::format("before }} after"), "before } after"); | |
464 | |
465 EXPECT_EQ(fmt::format("{{}}"), "{}"); | |
466 EXPECT_EQ(fmt::format("{{{0}}}", 42), "{42}"); | |
467 } | |
468 | |
469 TEST(format_test, unmatched_braces) { | |
470 EXPECT_THROW_MSG((void)fmt::format(runtime("{")), format_error, | |
471 "invalid format string"); | |
472 EXPECT_THROW_MSG((void)fmt::format(runtime("}")), format_error, | |
473 "unmatched '}' in format string"); | |
474 EXPECT_THROW_MSG((void)fmt::format(runtime("{0{}")), format_error, | |
475 "invalid format string"); | |
476 } | |
477 | |
478 TEST(format_test, no_args) { EXPECT_EQ(fmt::format("test"), "test"); } | |
479 | |
480 TEST(format_test, args_in_different_positions) { | |
481 EXPECT_EQ(fmt::format("{0}", 42), "42"); | |
482 EXPECT_EQ(fmt::format("before {0}", 42), "before 42"); | |
483 EXPECT_EQ(fmt::format("{0} after", 42), "42 after"); | |
484 EXPECT_EQ(fmt::format("before {0} after", 42), "before 42 after"); | |
485 EXPECT_EQ(fmt::format("{0} = {1}", "answer", 42), "answer = 42"); | |
486 EXPECT_EQ(fmt::format("{1} is the {0}", "answer", 42), "42 is the answer"); | |
487 EXPECT_EQ(fmt::format("{0}{1}{0}", "abra", "cad"), "abracadabra"); | |
488 } | |
489 | |
490 TEST(format_test, arg_errors) { | |
491 EXPECT_THROW_MSG((void)fmt::format(runtime("{")), format_error, | |
492 "invalid format string"); | |
493 EXPECT_THROW_MSG((void)fmt::format(runtime("{?}")), format_error, | |
494 "invalid format string"); | |
495 EXPECT_THROW_MSG((void)fmt::format(runtime("{0")), format_error, | |
496 "invalid format string"); | |
497 EXPECT_THROW_MSG((void)fmt::format(runtime("{0}")), format_error, | |
498 "argument not found"); | |
499 EXPECT_THROW_MSG((void)fmt::format(runtime("{00}"), 42), format_error, | |
500 "invalid format string"); | |
501 | |
502 auto int_max = std::to_string(INT_MAX); | |
503 EXPECT_THROW_MSG((void)fmt::format(runtime("{" + int_max)), format_error, | |
504 "invalid format string"); | |
505 EXPECT_THROW_MSG((void)fmt::format(runtime("{" + int_max + "}")), | |
506 format_error, "argument not found"); | |
507 | |
508 auto int_maxer = std::to_string(INT_MAX + 1u); | |
509 EXPECT_THROW_MSG((void)fmt::format(runtime("{" + int_maxer)), format_error, | |
510 "invalid format string"); | |
511 EXPECT_THROW_MSG((void)fmt::format(runtime("{" + int_maxer + "}")), | |
512 format_error, "argument not found"); | |
513 } | |
514 | |
515 template <int N> struct test_format { | |
516 template <typename... T> | |
517 static auto format(fmt::string_view fmt, const T&... args) -> std::string { | |
518 return test_format<N - 1>::format(fmt, N - 1, args...); | |
519 } | |
520 }; | |
521 | |
522 template <> struct test_format<0> { | |
523 template <typename... T> | |
524 static auto format(fmt::string_view fmt, const T&... args) -> std::string { | |
525 return fmt::format(runtime(fmt), args...); | |
526 } | |
527 }; | |
528 | |
529 TEST(format_test, many_args) { | |
530 EXPECT_EQ("19", test_format<20>::format("{19}")); | |
531 EXPECT_THROW_MSG(test_format<20>::format("{20}"), format_error, | |
532 "argument not found"); | |
533 EXPECT_THROW_MSG(test_format<21>::format("{21}"), format_error, | |
534 "argument not found"); | |
535 using fmt::detail::max_packed_args; | |
536 std::string format_str = fmt::format("{{{}}}", max_packed_args + 1); | |
537 EXPECT_THROW_MSG(test_format<max_packed_args>::format(format_str), | |
538 format_error, "argument not found"); | |
539 } | |
540 | |
541 TEST(format_test, named_arg) { | |
542 EXPECT_EQ("1/a/A", fmt::format("{_1}/{a_}/{A_}", fmt::arg("a_", 'a'), | |
543 fmt::arg("A_", "A"), fmt::arg("_1", 1))); | |
544 EXPECT_EQ(fmt::format("{0:{width}}", -42, fmt::arg("width", 4)), " -42"); | |
545 EXPECT_EQ("st", | |
546 fmt::format("{0:.{precision}}", "str", fmt::arg("precision", 2))); | |
547 EXPECT_EQ(fmt::format("{} {two}", 1, fmt::arg("two", 2)), "1 2"); | |
548 EXPECT_EQ("42", | |
549 fmt::format("{c}", fmt::arg("a", 0), fmt::arg("b", 0), | |
550 fmt::arg("c", 42), fmt::arg("d", 0), fmt::arg("e", 0), | |
551 fmt::arg("f", 0), fmt::arg("g", 0), fmt::arg("h", 0), | |
552 fmt::arg("i", 0), fmt::arg("j", 0), fmt::arg("k", 0), | |
553 fmt::arg("l", 0), fmt::arg("m", 0), fmt::arg("n", 0), | |
554 fmt::arg("o", 0), fmt::arg("p", 0))); | |
555 EXPECT_THROW_MSG((void)fmt::format(runtime("{a}")), format_error, | |
556 "argument not found"); | |
557 EXPECT_THROW_MSG((void)fmt::format(runtime("{a}"), 42), format_error, | |
558 "argument not found"); | |
559 } | |
560 | |
561 TEST(format_test, auto_arg_index) { | |
562 EXPECT_EQ(fmt::format("{}{}{}", 'a', 'b', 'c'), "abc"); | |
563 EXPECT_THROW_MSG((void)fmt::format(runtime("{0}{}"), 'a', 'b'), format_error, | |
564 "cannot switch from manual to automatic argument indexing"); | |
565 EXPECT_THROW_MSG((void)fmt::format(runtime("{}{0}"), 'a', 'b'), format_error, | |
566 "cannot switch from automatic to manual argument indexing"); | |
567 EXPECT_EQ(fmt::format("{:.{}}", 1.2345, 2), "1.2"); | |
568 EXPECT_THROW_MSG((void)fmt::format(runtime("{0}:.{}"), 1.2345, 2), | |
569 format_error, | |
570 "cannot switch from manual to automatic argument indexing"); | |
571 EXPECT_THROW_MSG((void)fmt::format(runtime("{:.{0}}"), 1.2345, 2), | |
572 format_error, | |
573 "cannot switch from automatic to manual argument indexing"); | |
574 EXPECT_THROW_MSG((void)fmt::format(runtime("{}")), format_error, | |
575 "argument not found"); | |
576 } | |
577 | |
578 TEST(format_test, empty_specs) { EXPECT_EQ(fmt::format("{0:}", 42), "42"); } | |
579 | |
580 TEST(format_test, left_align) { | |
581 EXPECT_EQ(fmt::format("{0:<4}", 42), "42 "); | |
582 EXPECT_EQ(fmt::format("{0:<4o}", 042), "42 "); | |
583 EXPECT_EQ(fmt::format("{0:<4x}", 0x42), "42 "); | |
584 EXPECT_EQ(fmt::format("{0:<5}", -42), "-42 "); | |
585 EXPECT_EQ(fmt::format("{0:<5}", 42u), "42 "); | |
586 EXPECT_EQ(fmt::format("{0:<5}", -42l), "-42 "); | |
587 EXPECT_EQ(fmt::format("{0:<5}", 42ul), "42 "); | |
588 EXPECT_EQ(fmt::format("{0:<5}", -42ll), "-42 "); | |
589 EXPECT_EQ(fmt::format("{0:<5}", 42ull), "42 "); | |
590 EXPECT_EQ(fmt::format("{0:<5}", -42.0), "-42 "); | |
591 EXPECT_EQ(fmt::format("{0:<5}", -42.0l), "-42 "); | |
592 EXPECT_EQ(fmt::format("{0:<5}", 'c'), "c "); | |
593 EXPECT_EQ(fmt::format("{0:<5}", "abc"), "abc "); | |
594 EXPECT_EQ(fmt::format("{0:<8}", reinterpret_cast<void*>(0xface)), "0xface "); | |
595 } | |
596 | |
597 TEST(format_test, right_align) { | |
598 EXPECT_EQ(fmt::format("{0:>4}", 42), " 42"); | |
599 EXPECT_EQ(fmt::format("{0:>4o}", 042), " 42"); | |
600 EXPECT_EQ(fmt::format("{0:>4x}", 0x42), " 42"); | |
601 EXPECT_EQ(fmt::format("{0:>5}", -42), " -42"); | |
602 EXPECT_EQ(fmt::format("{0:>5}", 42u), " 42"); | |
603 EXPECT_EQ(fmt::format("{0:>5}", -42l), " -42"); | |
604 EXPECT_EQ(fmt::format("{0:>5}", 42ul), " 42"); | |
605 EXPECT_EQ(fmt::format("{0:>5}", -42ll), " -42"); | |
606 EXPECT_EQ(fmt::format("{0:>5}", 42ull), " 42"); | |
607 EXPECT_EQ(fmt::format("{0:>5}", -42.0), " -42"); | |
608 EXPECT_EQ(fmt::format("{0:>5}", -42.0l), " -42"); | |
609 EXPECT_EQ(fmt::format("{0:>5}", 'c'), " c"); | |
610 EXPECT_EQ(fmt::format("{0:>5}", "abc"), " abc"); | |
611 EXPECT_EQ(fmt::format("{0:>8}", reinterpret_cast<void*>(0xface)), " 0xface"); | |
612 } | |
613 | |
614 TEST(format_test, center_align) { | |
615 EXPECT_EQ(fmt::format("{0:^5}", 42), " 42 "); | |
616 EXPECT_EQ(fmt::format("{0:^5o}", 042), " 42 "); | |
617 EXPECT_EQ(fmt::format("{0:^5x}", 0x42), " 42 "); | |
618 EXPECT_EQ(fmt::format("{0:^5}", -42), " -42 "); | |
619 EXPECT_EQ(fmt::format("{0:^5}", 42u), " 42 "); | |
620 EXPECT_EQ(fmt::format("{0:^5}", -42l), " -42 "); | |
621 EXPECT_EQ(fmt::format("{0:^5}", 42ul), " 42 "); | |
622 EXPECT_EQ(fmt::format("{0:^5}", -42ll), " -42 "); | |
623 EXPECT_EQ(fmt::format("{0:^5}", 42ull), " 42 "); | |
624 EXPECT_EQ(fmt::format("{0:^5}", -42.0), " -42 "); | |
625 EXPECT_EQ(fmt::format("{0:^5}", -42.0l), " -42 "); | |
626 EXPECT_EQ(fmt::format("{0:^5}", 'c'), " c "); | |
627 EXPECT_EQ(fmt::format("{0:^6}", "abc"), " abc "); | |
628 EXPECT_EQ(fmt::format("{0:^8}", reinterpret_cast<void*>(0xface)), " 0xface "); | |
629 } | |
630 | |
631 TEST(format_test, fill) { | |
632 EXPECT_THROW_MSG((void)fmt::format(runtime("{0:{<5}"), 'c'), format_error, | |
633 "invalid fill character '{'"); | |
634 EXPECT_THROW_MSG((void)fmt::format(runtime("{0:{<5}}"), 'c'), format_error, | |
635 "invalid fill character '{'"); | |
636 EXPECT_EQ(fmt::format("{0:*>4}", 42), "**42"); | |
637 EXPECT_EQ(fmt::format("{0:*>5}", -42), "**-42"); | |
638 EXPECT_EQ(fmt::format("{0:*>5}", 42u), "***42"); | |
639 EXPECT_EQ(fmt::format("{0:*>5}", -42l), "**-42"); | |
640 EXPECT_EQ(fmt::format("{0:*>5}", 42ul), "***42"); | |
641 EXPECT_EQ(fmt::format("{0:*>5}", -42ll), "**-42"); | |
642 EXPECT_EQ(fmt::format("{0:*>5}", 42ull), "***42"); | |
643 EXPECT_EQ(fmt::format("{0:*>5}", -42.0), "**-42"); | |
644 EXPECT_EQ(fmt::format("{0:*>5}", -42.0l), "**-42"); | |
645 EXPECT_EQ(fmt::format("{0:*<5}", 'c'), "c****"); | |
646 EXPECT_EQ(fmt::format("{0:*<5}", "abc"), "abc**"); | |
647 EXPECT_EQ("**0xface", | |
648 fmt::format("{0:*>8}", reinterpret_cast<void*>(0xface))); | |
649 EXPECT_EQ(fmt::format("{:}=", "foo"), "foo="); | |
650 EXPECT_EQ(std::string("\0\0\0*", 4), | |
651 fmt::format(string_view("{:\0>4}", 6), '*')); | |
652 EXPECT_EQ(fmt::format("{0:ж>4}", 42), "жж42"); | |
653 EXPECT_THROW_MSG((void)fmt::format(runtime("{:\x80\x80\x80\x80\x80>}"), 0), | |
654 format_error, "invalid format specifier"); | |
655 } | |
656 | |
657 TEST(format_test, plus_sign) { | |
658 EXPECT_EQ(fmt::format("{0:+}", 42), "+42"); | |
659 EXPECT_EQ(fmt::format("{0:+}", -42), "-42"); | |
660 EXPECT_EQ(fmt::format("{0:+}", 42), "+42"); | |
661 EXPECT_THROW_MSG((void)fmt::format(runtime("{0:+}"), 42u), format_error, | |
662 "invalid format specifier"); | |
663 EXPECT_EQ(fmt::format("{0:+}", 42l), "+42"); | |
664 EXPECT_THROW_MSG((void)fmt::format(runtime("{0:+}"), 42ul), format_error, | |
665 "invalid format specifier"); | |
666 EXPECT_EQ(fmt::format("{0:+}", 42ll), "+42"); | |
667 #if FMT_USE_INT128 | |
668 EXPECT_EQ(fmt::format("{0:+}", __int128_t(42)), "+42"); | |
669 #endif | |
670 EXPECT_THROW_MSG((void)fmt::format(runtime("{0:+}"), 42ull), format_error, | |
671 "invalid format specifier"); | |
672 EXPECT_EQ(fmt::format("{0:+}", 42.0), "+42"); | |
673 EXPECT_EQ(fmt::format("{0:+}", 42.0l), "+42"); | |
674 EXPECT_THROW_MSG((void)fmt::format(runtime("{0:+}"), 'c'), format_error, | |
675 "invalid format specifier"); | |
676 EXPECT_THROW_MSG((void)fmt::format(runtime("{0:+}"), "abc"), format_error, | |
677 "invalid format specifier"); | |
678 EXPECT_THROW_MSG( | |
679 (void)fmt::format(runtime("{0:+}"), reinterpret_cast<void*>(0x42)), | |
680 format_error, "invalid format specifier"); | |
681 } | |
682 | |
683 TEST(format_test, minus_sign) { | |
684 EXPECT_EQ(fmt::format("{0:-}", 42), "42"); | |
685 EXPECT_EQ(fmt::format("{0:-}", -42), "-42"); | |
686 EXPECT_EQ(fmt::format("{0:-}", 42), "42"); | |
687 EXPECT_THROW_MSG((void)fmt::format(runtime("{0:-}"), 42u), format_error, | |
688 "invalid format specifier"); | |
689 EXPECT_EQ(fmt::format("{0:-}", 42l), "42"); | |
690 EXPECT_THROW_MSG((void)fmt::format(runtime("{0:-}"), 42ul), format_error, | |
691 "invalid format specifier"); | |
692 EXPECT_EQ(fmt::format("{0:-}", 42ll), "42"); | |
693 EXPECT_THROW_MSG((void)fmt::format(runtime("{0:-}"), 42ull), format_error, | |
694 "invalid format specifier"); | |
695 EXPECT_EQ(fmt::format("{0:-}", 42.0), "42"); | |
696 EXPECT_EQ(fmt::format("{0:-}", 42.0l), "42"); | |
697 EXPECT_THROW_MSG((void)fmt::format(runtime("{0:-}"), 'c'), format_error, | |
698 "invalid format specifier"); | |
699 EXPECT_THROW_MSG((void)fmt::format(runtime("{0:-}"), "abc"), format_error, | |
700 "invalid format specifier"); | |
701 EXPECT_THROW_MSG( | |
702 (void)fmt::format(runtime("{0:-}"), reinterpret_cast<void*>(0x42)), | |
703 format_error, "invalid format specifier"); | |
704 } | |
705 | |
706 TEST(format_test, space_sign) { | |
707 EXPECT_EQ(fmt::format("{0: }", 42), " 42"); | |
708 EXPECT_EQ(fmt::format("{0: }", -42), "-42"); | |
709 EXPECT_EQ(fmt::format("{0: }", 42), " 42"); | |
710 EXPECT_THROW_MSG((void)fmt::format(runtime("{0: }"), 42u), format_error, | |
711 "invalid format specifier"); | |
712 EXPECT_EQ(fmt::format("{0: }", 42l), " 42"); | |
713 EXPECT_THROW_MSG((void)fmt::format(runtime("{0: }"), 42ul), format_error, | |
714 "invalid format specifier"); | |
715 EXPECT_EQ(fmt::format("{0: }", 42ll), " 42"); | |
716 EXPECT_THROW_MSG((void)fmt::format(runtime("{0: }"), 42ull), format_error, | |
717 "invalid format specifier"); | |
718 EXPECT_EQ(fmt::format("{0: }", 42.0), " 42"); | |
719 EXPECT_EQ(fmt::format("{0: }", 42.0l), " 42"); | |
720 EXPECT_THROW_MSG((void)fmt::format(runtime("{0: }"), 'c'), format_error, | |
721 "invalid format specifier"); | |
722 EXPECT_THROW_MSG((void)fmt::format(runtime("{0: }"), "abc"), format_error, | |
723 "invalid format specifier"); | |
724 EXPECT_THROW_MSG( | |
725 (void)fmt::format(runtime("{0: }"), reinterpret_cast<void*>(0x42)), | |
726 format_error, "invalid format specifier"); | |
727 } | |
728 | |
729 TEST(format_test, hash_flag) { | |
730 EXPECT_EQ(fmt::format("{0:#}", 42), "42"); | |
731 EXPECT_EQ(fmt::format("{0:#}", -42), "-42"); | |
732 EXPECT_EQ(fmt::format("{0:#b}", 42), "0b101010"); | |
733 EXPECT_EQ(fmt::format("{0:#B}", 42), "0B101010"); | |
734 EXPECT_EQ(fmt::format("{0:#b}", -42), "-0b101010"); | |
735 EXPECT_EQ(fmt::format("{0:#x}", 0x42), "0x42"); | |
736 EXPECT_EQ(fmt::format("{0:#X}", 0x42), "0X42"); | |
737 EXPECT_EQ(fmt::format("{0:#x}", -0x42), "-0x42"); | |
738 EXPECT_EQ(fmt::format("{0:#o}", 0), "0"); | |
739 EXPECT_EQ(fmt::format("{0:#o}", 042), "042"); | |
740 EXPECT_EQ(fmt::format("{0:#o}", -042), "-042"); | |
741 EXPECT_EQ(fmt::format("{0:#}", 42u), "42"); | |
742 EXPECT_EQ(fmt::format("{0:#x}", 0x42u), "0x42"); | |
743 EXPECT_EQ(fmt::format("{0:#o}", 042u), "042"); | |
744 | |
745 EXPECT_EQ(fmt::format("{0:#}", -42l), "-42"); | |
746 EXPECT_EQ(fmt::format("{0:#x}", 0x42l), "0x42"); | |
747 EXPECT_EQ(fmt::format("{0:#x}", -0x42l), "-0x42"); | |
748 EXPECT_EQ(fmt::format("{0:#o}", 042l), "042"); | |
749 EXPECT_EQ(fmt::format("{0:#o}", -042l), "-042"); | |
750 EXPECT_EQ(fmt::format("{0:#}", 42ul), "42"); | |
751 EXPECT_EQ(fmt::format("{0:#x}", 0x42ul), "0x42"); | |
752 EXPECT_EQ(fmt::format("{0:#o}", 042ul), "042"); | |
753 | |
754 EXPECT_EQ(fmt::format("{0:#}", -42ll), "-42"); | |
755 EXPECT_EQ(fmt::format("{0:#x}", 0x42ll), "0x42"); | |
756 EXPECT_EQ(fmt::format("{0:#x}", -0x42ll), "-0x42"); | |
757 EXPECT_EQ(fmt::format("{0:#o}", 042ll), "042"); | |
758 EXPECT_EQ(fmt::format("{0:#o}", -042ll), "-042"); | |
759 EXPECT_EQ(fmt::format("{0:#}", 42ull), "42"); | |
760 EXPECT_EQ(fmt::format("{0:#x}", 0x42ull), "0x42"); | |
761 EXPECT_EQ(fmt::format("{0:#o}", 042ull), "042"); | |
762 | |
763 EXPECT_EQ(fmt::format("{0:#}", -42.0), "-42."); | |
764 EXPECT_EQ(fmt::format("{0:#}", -42.0l), "-42."); | |
765 EXPECT_EQ(fmt::format("{:#.0e}", 42.0), "4.e+01"); | |
766 EXPECT_EQ(fmt::format("{:#.0f}", 0.01), "0."); | |
767 EXPECT_EQ(fmt::format("{:#.2g}", 0.5), "0.50"); | |
768 EXPECT_EQ(fmt::format("{:#.0f}", 0.5), "0."); | |
769 EXPECT_THROW_MSG((void)fmt::format(runtime("{0:#"), 'c'), format_error, | |
770 "missing '}' in format string"); | |
771 EXPECT_THROW_MSG((void)fmt::format(runtime("{0:#}"), 'c'), format_error, | |
772 "invalid format specifier for char"); | |
773 EXPECT_THROW_MSG((void)fmt::format(runtime("{0:#}"), "abc"), format_error, | |
774 "invalid format specifier"); | |
775 EXPECT_THROW_MSG( | |
776 (void)fmt::format(runtime("{0:#}"), reinterpret_cast<void*>(0x42)), | |
777 format_error, "invalid format specifier"); | |
778 } | |
779 | |
780 TEST(format_test, zero_flag) { | |
781 EXPECT_EQ(fmt::format("{0:0}", 42), "42"); | |
782 EXPECT_EQ(fmt::format("{0:05}", -42), "-0042"); | |
783 EXPECT_EQ(fmt::format("{0:05}", 42u), "00042"); | |
784 EXPECT_EQ(fmt::format("{0:05}", -42l), "-0042"); | |
785 EXPECT_EQ(fmt::format("{0:05}", 42ul), "00042"); | |
786 EXPECT_EQ(fmt::format("{0:05}", -42ll), "-0042"); | |
787 EXPECT_EQ(fmt::format("{0:05}", 42ull), "00042"); | |
788 EXPECT_EQ(fmt::format("{0:07}", -42.0), "-000042"); | |
789 EXPECT_EQ(fmt::format("{0:07}", -42.0l), "-000042"); | |
790 EXPECT_THROW_MSG((void)fmt::format(runtime("{0:0"), 'c'), format_error, | |
791 "missing '}' in format string"); | |
792 EXPECT_THROW_MSG((void)fmt::format(runtime("{0:05}"), 'c'), format_error, | |
793 "invalid format specifier for char"); | |
794 EXPECT_THROW_MSG((void)fmt::format(runtime("{0:05}"), "abc"), format_error, | |
795 "format specifier requires numeric argument"); | |
796 EXPECT_THROW_MSG( | |
797 (void)fmt::format(runtime("{0:05}"), reinterpret_cast<void*>(0x42)), | |
798 format_error, "format specifier requires numeric argument"); | |
799 } | |
800 | |
801 TEST(format_test, zero_flag_and_align) { | |
802 // If the 0 character and an align option both appear, the 0 character is | |
803 // ignored. | |
804 EXPECT_EQ(fmt::format("{:<05}", 42), "42 "); | |
805 EXPECT_EQ(fmt::format("{:<05}", -42), "-42 "); | |
806 EXPECT_EQ(fmt::format("{:^05}", 42), " 42 "); | |
807 EXPECT_EQ(fmt::format("{:^05}", -42), " -42 "); | |
808 EXPECT_EQ(fmt::format("{:>05}", 42), " 42"); | |
809 EXPECT_EQ(fmt::format("{:>05}", -42), " -42"); | |
810 } | |
811 | |
812 TEST(format_test, width) { | |
813 auto int_maxer = std::to_string(INT_MAX + 1u); | |
814 EXPECT_THROW_MSG((void)fmt::format(runtime("{:" + int_maxer), 0), | |
815 format_error, "number is too big"); | |
816 EXPECT_THROW_MSG((void)fmt::format(runtime("{:" + int_maxer + "}"), 0), | |
817 format_error, "number is too big"); | |
818 | |
819 EXPECT_EQ(fmt::format("{:4}", -42), " -42"); | |
820 EXPECT_EQ(fmt::format("{:5}", 42u), " 42"); | |
821 EXPECT_EQ(fmt::format("{:6}", -42l), " -42"); | |
822 EXPECT_EQ(fmt::format("{:7}", 42ul), " 42"); | |
823 EXPECT_EQ(fmt::format("{:6}", -42ll), " -42"); | |
824 EXPECT_EQ(fmt::format("{:7}", 42ull), " 42"); | |
825 EXPECT_EQ(fmt::format("{:8}", -1.23), " -1.23"); | |
826 EXPECT_EQ(fmt::format("{:9}", -1.23l), " -1.23"); | |
827 EXPECT_EQ(fmt::format("{:10}", reinterpret_cast<void*>(0xcafe)), | |
828 " 0xcafe"); | |
829 EXPECT_EQ(fmt::format("{:11}", 'x'), "x "); | |
830 EXPECT_EQ(fmt::format("{:12}", "str"), "str "); | |
831 EXPECT_EQ(fmt::format("{:*^6}", "🤡"), "**🤡**"); | |
832 EXPECT_EQ(fmt::format("{:*^8}", "你好"), "**你好**"); | |
833 EXPECT_EQ(fmt::format("{:#6}", 42.0), " 42."); | |
834 EXPECT_EQ(fmt::format("{:6c}", static_cast<int>('x')), "x "); | |
835 EXPECT_EQ(fmt::format("{:>06.0f}", 0.00884311), " 0"); | |
836 } | |
837 | |
838 TEST(format_test, runtime_width) { | |
839 auto int_maxer = std::to_string(INT_MAX + 1u); | |
840 EXPECT_THROW_MSG((void)fmt::format(runtime("{0:{" + int_maxer), 0), | |
841 format_error, "invalid format string"); | |
842 EXPECT_THROW_MSG((void)fmt::format(runtime("{0:{" + int_maxer + "}"), 0), | |
843 format_error, "argument not found"); | |
844 EXPECT_THROW_MSG((void)fmt::format(runtime("{0:{" + int_maxer + "}}"), 0), | |
845 format_error, "argument not found"); | |
846 | |
847 EXPECT_THROW_MSG((void)fmt::format(runtime("{0:{"), 0), format_error, | |
848 "invalid format string"); | |
849 EXPECT_THROW_MSG((void)fmt::format(runtime("{0:{}"), 0), format_error, | |
850 "cannot switch from manual to automatic argument indexing"); | |
851 EXPECT_THROW_MSG((void)fmt::format(runtime("{0:{?}}"), 0), format_error, | |
852 "invalid format string"); | |
853 EXPECT_THROW_MSG((void)fmt::format(runtime("{0:{1}}"), 0), format_error, | |
854 "argument not found"); | |
855 | |
856 EXPECT_THROW_MSG((void)fmt::format(runtime("{0:{0:}}"), 0), format_error, | |
857 "invalid format string"); | |
858 | |
859 EXPECT_THROW_MSG((void)fmt::format(runtime("{0:{1}}"), 0, -1), format_error, | |
860 "negative width"); | |
861 EXPECT_THROW_MSG((void)fmt::format(runtime("{0:{1}}"), 0, (INT_MAX + 1u)), | |
862 format_error, "number is too big"); | |
863 EXPECT_THROW_MSG((void)fmt::format(runtime("{0:{1}}"), 0, -1l), format_error, | |
864 "negative width"); | |
865 if (fmt::detail::const_check(sizeof(long) > sizeof(int))) { | |
866 long value = INT_MAX; | |
867 EXPECT_THROW_MSG((void)fmt::format(runtime("{0:{1}}"), 0, (value + 1)), | |
868 format_error, "number is too big"); | |
869 } | |
870 EXPECT_THROW_MSG((void)fmt::format(runtime("{0:{1}}"), 0, (INT_MAX + 1ul)), | |
871 format_error, "number is too big"); | |
872 | |
873 EXPECT_THROW_MSG((void)fmt::format(runtime("{0:{1}}"), 0, '0'), format_error, | |
874 "width is not integer"); | |
875 EXPECT_THROW_MSG((void)fmt::format(runtime("{0:{1}}"), 0, 0.0), format_error, | |
876 "width is not integer"); | |
877 | |
878 EXPECT_EQ(fmt::format("{0:{1}}", -42, 4), " -42"); | |
879 EXPECT_EQ(fmt::format("{0:{1}}", 42u, 5), " 42"); | |
880 EXPECT_EQ(fmt::format("{0:{1}}", -42l, 6), " -42"); | |
881 EXPECT_EQ(fmt::format("{0:{1}}", 42ul, 7), " 42"); | |
882 EXPECT_EQ(fmt::format("{0:{1}}", -42ll, 6), " -42"); | |
883 EXPECT_EQ(fmt::format("{0:{1}}", 42ull, 7), " 42"); | |
884 EXPECT_EQ(fmt::format("{0:{1}}", -1.23, 8), " -1.23"); | |
885 EXPECT_EQ(fmt::format("{0:{1}}", -1.23l, 9), " -1.23"); | |
886 EXPECT_EQ(" 0xcafe", | |
887 fmt::format("{0:{1}}", reinterpret_cast<void*>(0xcafe), 10)); | |
888 EXPECT_EQ(fmt::format("{0:{1}}", 'x', 11), "x "); | |
889 EXPECT_EQ(fmt::format("{0:{1}}", "str", 12), "str "); | |
890 EXPECT_EQ(fmt::format("{:{}}", 42, short(4)), " 42"); | |
891 } | |
892 | |
893 TEST(format_test, precision) { | |
894 char format_str[buffer_size]; | |
895 safe_sprintf(format_str, "{0:.%u", UINT_MAX); | |
896 increment(format_str + 4); | |
897 EXPECT_THROW_MSG((void)fmt::format(runtime(format_str), 0.0), format_error, | |
898 "number is too big"); | |
899 size_t size = std::strlen(format_str); | |
900 format_str[size] = '}'; | |
901 format_str[size + 1] = 0; | |
902 EXPECT_THROW_MSG((void)fmt::format(runtime(format_str), 0.0), format_error, | |
903 "number is too big"); | |
904 | |
905 safe_sprintf(format_str, "{0:.%u", INT_MAX + 1u); | |
906 EXPECT_THROW_MSG((void)fmt::format(runtime(format_str), 0.0), format_error, | |
907 "number is too big"); | |
908 safe_sprintf(format_str, "{0:.%u}", INT_MAX + 1u); | |
909 EXPECT_THROW_MSG((void)fmt::format(runtime(format_str), 0.0), format_error, | |
910 "number is too big"); | |
911 | |
912 EXPECT_THROW_MSG((void)fmt::format(runtime("{0:."), 0.0), format_error, | |
913 "invalid precision"); | |
914 EXPECT_THROW_MSG((void)fmt::format(runtime("{0:.}"), 0.0), format_error, | |
915 "invalid precision"); | |
916 | |
917 EXPECT_THROW_MSG((void)fmt::format(runtime("{0:.2"), 0), format_error, | |
918 "invalid format specifier"); | |
919 EXPECT_THROW_MSG((void)fmt::format(runtime("{0:.2}"), 42), format_error, | |
920 "invalid format specifier"); | |
921 EXPECT_THROW_MSG((void)fmt::format(runtime("{0:.2f}"), 42), format_error, | |
922 "invalid format specifier"); | |
923 EXPECT_THROW_MSG((void)fmt::format(runtime("{0:.2}"), 42u), format_error, | |
924 "invalid format specifier"); | |
925 EXPECT_THROW_MSG((void)fmt::format(runtime("{0:.2f}"), 42u), format_error, | |
926 "invalid format specifier"); | |
927 EXPECT_THROW_MSG((void)fmt::format(runtime("{0:.2}"), 42l), format_error, | |
928 "invalid format specifier"); | |
929 EXPECT_THROW_MSG((void)fmt::format(runtime("{0:.2f}"), 42l), format_error, | |
930 "invalid format specifier"); | |
931 EXPECT_THROW_MSG((void)fmt::format(runtime("{0:.2}"), 42ul), format_error, | |
932 "invalid format specifier"); | |
933 EXPECT_THROW_MSG((void)fmt::format(runtime("{0:.2f}"), 42ul), format_error, | |
934 "invalid format specifier"); | |
935 EXPECT_THROW_MSG((void)fmt::format(runtime("{0:.2}"), 42ll), format_error, | |
936 "invalid format specifier"); | |
937 EXPECT_THROW_MSG((void)fmt::format(runtime("{0:.2f}"), 42ll), format_error, | |
938 "invalid format specifier"); | |
939 EXPECT_THROW_MSG((void)fmt::format(runtime("{0:.2}"), 42ull), format_error, | |
940 "invalid format specifier"); | |
941 EXPECT_THROW_MSG((void)fmt::format(runtime("{0:.2f}"), 42ull), format_error, | |
942 "invalid format specifier"); | |
943 EXPECT_THROW_MSG((void)fmt::format(runtime("{0:3.0}"), 'x'), format_error, | |
944 "invalid format specifier"); | |
945 EXPECT_EQ(fmt::format("{0:.2}", 1.2345), "1.2"); | |
946 EXPECT_EQ(fmt::format("{0:.2}", 1.2345l), "1.2"); | |
947 EXPECT_EQ(fmt::format("{:.2}", 1.234e56), "1.2e+56"); | |
948 EXPECT_EQ(fmt::format("{0:.3}", 1.1), "1.1"); | |
949 EXPECT_EQ(fmt::format("{:.0e}", 1.0L), "1e+00"); | |
950 EXPECT_EQ(fmt::format("{:9.1e}", 0.0), " 0.0e+00"); | |
951 EXPECT_EQ( | |
952 fmt::format("{:.494}", 4.9406564584124654E-324), | |
953 "4.9406564584124654417656879286822137236505980261432476442558568250067550" | |
954 "727020875186529983636163599237979656469544571773092665671035593979639877" | |
955 "479601078187812630071319031140452784581716784898210368871863605699873072" | |
956 "305000638740915356498438731247339727316961514003171538539807412623856559" | |
957 "117102665855668676818703956031062493194527159149245532930545654440112748" | |
958 "012970999954193198940908041656332452475714786901472678015935523861155013" | |
959 "480352649347201937902681071074917033322268447533357208324319361e-324"); | |
960 EXPECT_EQ( | |
961 fmt::format("{:.1074f}", 1.1125369292536e-308), | |
962 "0.0000000000000000000000000000000000000000000000000000000000000000000000" | |
963 "000000000000000000000000000000000000000000000000000000000000000000000000" | |
964 "000000000000000000000000000000000000000000000000000000000000000000000000" | |
965 "000000000000000000000000000000000000000000000000000000000000000000000000" | |
966 "000000000000000000000111253692925360019747947051741965785554081512200979" | |
967 "355021686109411883779182127659725163430929750364498219730822952552570601" | |
968 "152163505899912777129583674906301179059298598412303893909188340988729019" | |
969 "014361467448914817838555156840459458527907308695109202499990850735085304" | |
970 "478476991912072201449236975063640913461919914396877093174125167509869762" | |
971 "482369631100360266123742648159508919592746619553246586039571522788247697" | |
972 "156360766271842991667238355464496455107749716934387136380536472531224398" | |
973 "559833794807213172371254492216255558078524900147957309382830827524104234" | |
974 "530961756787819847850302379672357738807808384667004752163416921762619527" | |
975 "462847642037420991432005657440259928195996762610375541867198059294212446" | |
976 "81962777939941034720757232455434770912461317493580281734466552734375"); | |
977 | |
978 std::string outputs[] = { | |
979 "-0X1.41FE3FFE71C9E000000000000000000000000000000000000000000000000000000" | |
980 "000000000000000000000000000000000000000000000000000000000000000000000000" | |
981 "000000000000000000000000000000000000000000000000000000000000000000000000" | |
982 "000000000000000000000000000000000000000000000000000000000000000000000000" | |
983 "000000000000000000000000000000000000000000000000000000000000000000000000" | |
984 "000000000000000000000000000000000000000000000000000000000000000000000000" | |
985 "000000000000000000000000000000000000000000000000000000000000000000000000" | |
986 "000000000000000000000000000000000000000000000000000000000000000000000000" | |
987 "000000000000000000000000000000000000000000000000000000000000000000000000" | |
988 "000000000000000000000000000000000000000000000000000000000000000000000000" | |
989 "000000000000000000000000000000000000000000000000000000000000000000000000" | |
990 "000000000000000000000000000000000000000000000000000P+127", | |
991 "-0XA.0FF1FFF38E4F0000000000000000000000000000000000000000000000000000000" | |
992 "000000000000000000000000000000000000000000000000000000000000000000000000" | |
993 "000000000000000000000000000000000000000000000000000000000000000000000000" | |
994 "000000000000000000000000000000000000000000000000000000000000000000000000" | |
995 "000000000000000000000000000000000000000000000000000000000000000000000000" | |
996 "000000000000000000000000000000000000000000000000000000000000000000000000" | |
997 "000000000000000000000000000000000000000000000000000000000000000000000000" | |
998 "000000000000000000000000000000000000000000000000000000000000000000000000" | |
999 "000000000000000000000000000000000000000000000000000000000000000000000000" | |
1000 "000000000000000000000000000000000000000000000000000000000000000000000000" | |
1001 "000000000000000000000000000000000000000000000000000000000000000000000000" | |
1002 "000000000000000000000000000000000000000000000000000P+124"}; | |
1003 EXPECT_THAT(outputs, | |
1004 testing::Contains(fmt::format("{:.838A}", -2.14001164E+38))); | |
1005 | |
1006 if (std::numeric_limits<long double>::digits == 64) { | |
1007 auto ld = (std::numeric_limits<long double>::min)(); | |
1008 EXPECT_EQ(fmt::format("{:.0}", ld), "3e-4932"); | |
1009 EXPECT_EQ( | |
1010 fmt::format("{:0g}", std::numeric_limits<long double>::denorm_min()), | |
1011 "3.6452e-4951"); | |
1012 } | |
1013 | |
1014 EXPECT_EQ(fmt::format("{:#.0f}", 123.0), "123."); | |
1015 EXPECT_EQ(fmt::format("{:.02f}", 1.234), "1.23"); | |
1016 EXPECT_EQ(fmt::format("{:.1g}", 0.001), "0.001"); | |
1017 EXPECT_EQ(fmt::format("{}", 1019666432.0f), "1019666400"); | |
1018 EXPECT_EQ(fmt::format("{:.0e}", 9.5), "1e+01"); | |
1019 EXPECT_EQ(fmt::format("{:.1e}", 1e-34), "1.0e-34"); | |
1020 | |
1021 EXPECT_THROW_MSG( | |
1022 (void)fmt::format(runtime("{0:.2}"), reinterpret_cast<void*>(0xcafe)), | |
1023 format_error, "invalid format specifier"); | |
1024 EXPECT_THROW_MSG( | |
1025 (void)fmt::format(runtime("{0:.2f}"), reinterpret_cast<void*>(0xcafe)), | |
1026 format_error, "invalid format specifier"); | |
1027 EXPECT_THROW_MSG((void)fmt::format(runtime("{:.{}e}"), 42.0, | |
1028 fmt::detail::max_value<int>()), | |
1029 format_error, "number is too big"); | |
1030 EXPECT_THROW_MSG( | |
1031 (void)fmt::format("{:.2147483646f}", -2.2121295195081227E+304), | |
1032 format_error, "number is too big"); | |
1033 | |
1034 EXPECT_EQ(fmt::format("{0:.2}", "str"), "st"); | |
1035 EXPECT_EQ(fmt::format("{0:.5}", "вожыкі"), "вожык"); | |
1036 EXPECT_EQ(fmt::format("{0:.6}", "123456\xad"), "123456"); | |
1037 } | |
1038 | |
1039 TEST(xchar_test, utf8_precision) { | |
1040 auto result = fmt::format("{:.4}", "caf\u00e9s"); // cafés | |
1041 EXPECT_EQ(fmt::detail::compute_width(result), 4); | |
1042 EXPECT_EQ(result, "caf\u00e9"); | |
1043 } | |
1044 | |
1045 TEST(format_test, runtime_precision) { | |
1046 char format_str[buffer_size]; | |
1047 safe_sprintf(format_str, "{0:.{%u", UINT_MAX); | |
1048 increment(format_str + 5); | |
1049 EXPECT_THROW_MSG((void)fmt::format(runtime(format_str), 0.0), format_error, | |
1050 "invalid format string"); | |
1051 size_t size = std::strlen(format_str); | |
1052 format_str[size] = '}'; | |
1053 format_str[size + 1] = 0; | |
1054 EXPECT_THROW_MSG((void)fmt::format(runtime(format_str), 0.0), format_error, | |
1055 "argument not found"); | |
1056 format_str[size + 1] = '}'; | |
1057 format_str[size + 2] = 0; | |
1058 EXPECT_THROW_MSG((void)fmt::format(runtime(format_str), 0.0), format_error, | |
1059 "argument not found"); | |
1060 | |
1061 EXPECT_THROW_MSG((void)fmt::format(runtime("{0:.{"), 0.0), format_error, | |
1062 "invalid format string"); | |
1063 EXPECT_THROW_MSG((void)fmt::format(runtime("{0:.{}"), 0.0), format_error, | |
1064 "cannot switch from manual to automatic argument indexing"); | |
1065 EXPECT_THROW_MSG((void)fmt::format(runtime("{0:.{?}}"), 0.0), format_error, | |
1066 "invalid format string"); | |
1067 EXPECT_THROW_MSG((void)fmt::format(runtime("{0:.{1}"), 0, 0), format_error, | |
1068 "invalid format specifier"); | |
1069 EXPECT_THROW_MSG((void)fmt::format(runtime("{0:.{1}}"), 0.0), format_error, | |
1070 "argument not found"); | |
1071 | |
1072 EXPECT_THROW_MSG((void)fmt::format(runtime("{0:.{0:}}"), 0.0), format_error, | |
1073 "invalid format string"); | |
1074 | |
1075 EXPECT_THROW_MSG((void)fmt::format(runtime("{0:.{1}}"), 0.0, -1), | |
1076 format_error, "negative precision"); | |
1077 EXPECT_THROW_MSG((void)fmt::format(runtime("{0:.{1}}"), 0.0, (INT_MAX + 1u)), | |
1078 format_error, "number is too big"); | |
1079 EXPECT_THROW_MSG((void)fmt::format(runtime("{0:.{1}}"), 0.0, -1l), | |
1080 format_error, "negative precision"); | |
1081 if (fmt::detail::const_check(sizeof(long) > sizeof(int))) { | |
1082 long value = INT_MAX; | |
1083 EXPECT_THROW_MSG((void)fmt::format(runtime("{0:.{1}}"), 0.0, (value + 1)), | |
1084 format_error, "number is too big"); | |
1085 } | |
1086 EXPECT_THROW_MSG((void)fmt::format(runtime("{0:.{1}}"), 0.0, (INT_MAX + 1ul)), | |
1087 format_error, "number is too big"); | |
1088 | |
1089 EXPECT_THROW_MSG((void)fmt::format(runtime("{0:.{1}}"), 0.0, '0'), | |
1090 format_error, "precision is not integer"); | |
1091 EXPECT_THROW_MSG((void)fmt::format(runtime("{0:.{1}}"), 0.0, 0.0), | |
1092 format_error, "precision is not integer"); | |
1093 | |
1094 EXPECT_THROW_MSG((void)fmt::format(runtime("{0:.{1}}"), 42, 2), format_error, | |
1095 "invalid format specifier"); | |
1096 EXPECT_THROW_MSG((void)fmt::format(runtime("{0:.{1}f}"), 42, 2), format_error, | |
1097 "invalid format specifier"); | |
1098 EXPECT_THROW_MSG((void)fmt::format(runtime("{0:.{1}}"), 42u, 2), format_error, | |
1099 "invalid format specifier"); | |
1100 EXPECT_THROW_MSG((void)fmt::format(runtime("{0:.{1}f}"), 42u, 2), | |
1101 format_error, "invalid format specifier"); | |
1102 EXPECT_THROW_MSG((void)fmt::format(runtime("{0:.{1}}"), 42l, 2), format_error, | |
1103 "invalid format specifier"); | |
1104 EXPECT_THROW_MSG((void)fmt::format(runtime("{0:.{1}f}"), 42l, 2), | |
1105 format_error, "invalid format specifier"); | |
1106 EXPECT_THROW_MSG((void)fmt::format(runtime("{0:.{1}}"), 42ul, 2), | |
1107 format_error, "invalid format specifier"); | |
1108 EXPECT_THROW_MSG((void)fmt::format(runtime("{0:.{1}f}"), 42ul, 2), | |
1109 format_error, "invalid format specifier"); | |
1110 EXPECT_THROW_MSG((void)fmt::format(runtime("{0:.{1}}"), 42ll, 2), | |
1111 format_error, "invalid format specifier"); | |
1112 EXPECT_THROW_MSG((void)fmt::format(runtime("{0:.{1}f}"), 42ll, 2), | |
1113 format_error, "invalid format specifier"); | |
1114 EXPECT_THROW_MSG((void)fmt::format(runtime("{0:.{1}}"), 42ull, 2), | |
1115 format_error, "invalid format specifier"); | |
1116 EXPECT_THROW_MSG((void)fmt::format(runtime("{0:.{1}f}"), 42ull, 2), | |
1117 format_error, "invalid format specifier"); | |
1118 EXPECT_THROW_MSG((void)fmt::format(runtime("{0:3.{1}}"), 'x', 0), | |
1119 format_error, "invalid format specifier"); | |
1120 EXPECT_EQ(fmt::format("{0:.{1}}", 1.2345, 2), "1.2"); | |
1121 EXPECT_EQ(fmt::format("{1:.{0}}", 2, 1.2345l), "1.2"); | |
1122 | |
1123 EXPECT_THROW_MSG((void)fmt::format(runtime("{0:.{1}}"), | |
1124 reinterpret_cast<void*>(0xcafe), 2), | |
1125 format_error, "invalid format specifier"); | |
1126 EXPECT_THROW_MSG((void)fmt::format(runtime("{0:.{1}f}"), | |
1127 reinterpret_cast<void*>(0xcafe), 2), | |
1128 format_error, "invalid format specifier"); | |
1129 | |
1130 EXPECT_EQ(fmt::format("{0:.{1}}", "str", 2), "st"); | |
1131 } | |
1132 | |
1133 TEST(format_test, format_bool) { | |
1134 EXPECT_EQ(fmt::format("{}", true), "true"); | |
1135 EXPECT_EQ(fmt::format("{}", false), "false"); | |
1136 EXPECT_EQ(fmt::format("{:d}", true), "1"); | |
1137 EXPECT_EQ(fmt::format("{:5}", true), "true "); | |
1138 EXPECT_EQ(fmt::format("{:s}", true), "true"); | |
1139 EXPECT_EQ(fmt::format("{:s}", false), "false"); | |
1140 EXPECT_EQ(fmt::format("{:6s}", false), "false "); | |
1141 EXPECT_THROW_MSG((void)fmt::format(runtime("{:c}"), false), format_error, | |
1142 "invalid format specifier"); | |
1143 } | |
1144 | |
1145 TEST(format_test, format_short) { | |
1146 short s = 42; | |
1147 EXPECT_EQ(fmt::format("{0:d}", s), "42"); | |
1148 unsigned short us = 42; | |
1149 EXPECT_EQ(fmt::format("{0:d}", us), "42"); | |
1150 } | |
1151 | |
1152 template <typename T> | |
1153 void check_unknown_types(const T& value, const char* types, const char*) { | |
1154 char format_str[buffer_size]; | |
1155 const char* special = ".0123456789L?}"; | |
1156 for (int i = CHAR_MIN; i <= CHAR_MAX; ++i) { | |
1157 char c = static_cast<char>(i); | |
1158 if (std::strchr(types, c) || std::strchr(special, c) || !c) continue; | |
1159 safe_sprintf(format_str, "{0:10%c}", c); | |
1160 const char* message = "invalid format specifier"; | |
1161 EXPECT_THROW_MSG((void)fmt::format(runtime(format_str), value), | |
1162 format_error, message) | |
1163 << format_str << " " << message; | |
1164 } | |
1165 } | |
1166 | |
1167 TEST(format_test, format_int) { | |
1168 EXPECT_THROW_MSG((void)fmt::format(runtime("{0:v"), 42), format_error, | |
1169 "invalid format specifier"); | |
1170 check_unknown_types(42, "bBdoxXnLc", "integer"); | |
1171 EXPECT_EQ(fmt::format("{:c}", static_cast<int>('x')), "x"); | |
1172 } | |
1173 | |
1174 TEST(format_test, format_bin) { | |
1175 EXPECT_EQ(fmt::format("{0:b}", 0), "0"); | |
1176 EXPECT_EQ(fmt::format("{0:b}", 42), "101010"); | |
1177 EXPECT_EQ(fmt::format("{0:b}", 42u), "101010"); | |
1178 EXPECT_EQ(fmt::format("{0:b}", -42), "-101010"); | |
1179 EXPECT_EQ(fmt::format("{0:b}", 12345), "11000000111001"); | |
1180 EXPECT_EQ(fmt::format("{0:b}", 0x12345678), "10010001101000101011001111000"); | |
1181 EXPECT_EQ("10010000101010111100110111101111", | |
1182 fmt::format("{0:b}", 0x90ABCDEF)); | |
1183 EXPECT_EQ("11111111111111111111111111111111", | |
1184 fmt::format("{0:b}", max_value<uint32_t>())); | |
1185 } | |
1186 | |
1187 #if FMT_USE_INT128 | |
1188 constexpr auto int128_max = static_cast<__int128_t>( | |
1189 (static_cast<__uint128_t>(1) << ((__SIZEOF_INT128__ * CHAR_BIT) - 1)) - 1); | |
1190 constexpr auto int128_min = -int128_max - 1; | |
1191 | |
1192 constexpr auto uint128_max = ~static_cast<__uint128_t>(0); | |
1193 #endif | |
1194 | |
1195 TEST(format_test, format_dec) { | |
1196 EXPECT_EQ(fmt::format("{0}", 0), "0"); | |
1197 EXPECT_EQ(fmt::format("{0}", 42), "42"); | |
1198 EXPECT_EQ(fmt::format("{:}>", 42), "42>"); | |
1199 EXPECT_EQ(fmt::format("{0:d}", 42), "42"); | |
1200 EXPECT_EQ(fmt::format("{0}", 42u), "42"); | |
1201 EXPECT_EQ(fmt::format("{0}", -42), "-42"); | |
1202 EXPECT_EQ(fmt::format("{0}", 12345), "12345"); | |
1203 EXPECT_EQ(fmt::format("{0}", 67890), "67890"); | |
1204 #if FMT_USE_INT128 | |
1205 EXPECT_EQ(fmt::format("{0}", static_cast<__int128_t>(0)), "0"); | |
1206 EXPECT_EQ(fmt::format("{0}", static_cast<__uint128_t>(0)), "0"); | |
1207 EXPECT_EQ("9223372036854775808", | |
1208 fmt::format("{0}", static_cast<__int128_t>(INT64_MAX) + 1)); | |
1209 EXPECT_EQ("-9223372036854775809", | |
1210 fmt::format("{0}", static_cast<__int128_t>(INT64_MIN) - 1)); | |
1211 EXPECT_EQ("18446744073709551616", | |
1212 fmt::format("{0}", static_cast<__int128_t>(UINT64_MAX) + 1)); | |
1213 EXPECT_EQ("170141183460469231731687303715884105727", | |
1214 fmt::format("{0}", int128_max)); | |
1215 EXPECT_EQ("-170141183460469231731687303715884105728", | |
1216 fmt::format("{0}", int128_min)); | |
1217 EXPECT_EQ("340282366920938463463374607431768211455", | |
1218 fmt::format("{0}", uint128_max)); | |
1219 #endif | |
1220 | |
1221 char buffer[buffer_size]; | |
1222 safe_sprintf(buffer, "%d", INT_MIN); | |
1223 EXPECT_EQ(buffer, fmt::format("{0}", INT_MIN)); | |
1224 safe_sprintf(buffer, "%d", INT_MAX); | |
1225 EXPECT_EQ(buffer, fmt::format("{0}", INT_MAX)); | |
1226 safe_sprintf(buffer, "%u", UINT_MAX); | |
1227 EXPECT_EQ(buffer, fmt::format("{0}", UINT_MAX)); | |
1228 safe_sprintf(buffer, "%ld", 0 - static_cast<unsigned long>(LONG_MIN)); | |
1229 EXPECT_EQ(buffer, fmt::format("{0}", LONG_MIN)); | |
1230 safe_sprintf(buffer, "%ld", LONG_MAX); | |
1231 EXPECT_EQ(buffer, fmt::format("{0}", LONG_MAX)); | |
1232 safe_sprintf(buffer, "%lu", ULONG_MAX); | |
1233 EXPECT_EQ(buffer, fmt::format("{0}", ULONG_MAX)); | |
1234 } | |
1235 | |
1236 TEST(format_test, format_hex) { | |
1237 EXPECT_EQ(fmt::format("{0:x}", 0), "0"); | |
1238 EXPECT_EQ(fmt::format("{0:x}", 0x42), "42"); | |
1239 EXPECT_EQ(fmt::format("{0:x}", 0x42u), "42"); | |
1240 EXPECT_EQ(fmt::format("{0:x}", -0x42), "-42"); | |
1241 EXPECT_EQ(fmt::format("{0:x}", 0x12345678), "12345678"); | |
1242 EXPECT_EQ(fmt::format("{0:x}", 0x90abcdef), "90abcdef"); | |
1243 EXPECT_EQ(fmt::format("{0:X}", 0x12345678), "12345678"); | |
1244 EXPECT_EQ(fmt::format("{0:X}", 0x90ABCDEF), "90ABCDEF"); | |
1245 #if FMT_USE_INT128 | |
1246 EXPECT_EQ(fmt::format("{0:x}", static_cast<__int128_t>(0)), "0"); | |
1247 EXPECT_EQ(fmt::format("{0:x}", static_cast<__uint128_t>(0)), "0"); | |
1248 EXPECT_EQ("8000000000000000", | |
1249 fmt::format("{0:x}", static_cast<__int128_t>(INT64_MAX) + 1)); | |
1250 EXPECT_EQ("-8000000000000001", | |
1251 fmt::format("{0:x}", static_cast<__int128_t>(INT64_MIN) - 1)); | |
1252 EXPECT_EQ("10000000000000000", | |
1253 fmt::format("{0:x}", static_cast<__int128_t>(UINT64_MAX) + 1)); | |
1254 EXPECT_EQ("7fffffffffffffffffffffffffffffff", | |
1255 fmt::format("{0:x}", int128_max)); | |
1256 EXPECT_EQ("-80000000000000000000000000000000", | |
1257 fmt::format("{0:x}", int128_min)); | |
1258 EXPECT_EQ("ffffffffffffffffffffffffffffffff", | |
1259 fmt::format("{0:x}", uint128_max)); | |
1260 #endif | |
1261 | |
1262 char buffer[buffer_size]; | |
1263 safe_sprintf(buffer, "-%x", 0 - static_cast<unsigned>(INT_MIN)); | |
1264 EXPECT_EQ(buffer, fmt::format("{0:x}", INT_MIN)); | |
1265 safe_sprintf(buffer, "%x", INT_MAX); | |
1266 EXPECT_EQ(buffer, fmt::format("{0:x}", INT_MAX)); | |
1267 safe_sprintf(buffer, "%x", UINT_MAX); | |
1268 EXPECT_EQ(buffer, fmt::format("{0:x}", UINT_MAX)); | |
1269 safe_sprintf(buffer, "-%lx", 0 - static_cast<unsigned long>(LONG_MIN)); | |
1270 EXPECT_EQ(buffer, fmt::format("{0:x}", LONG_MIN)); | |
1271 safe_sprintf(buffer, "%lx", LONG_MAX); | |
1272 EXPECT_EQ(buffer, fmt::format("{0:x}", LONG_MAX)); | |
1273 safe_sprintf(buffer, "%lx", ULONG_MAX); | |
1274 EXPECT_EQ(buffer, fmt::format("{0:x}", ULONG_MAX)); | |
1275 } | |
1276 | |
1277 TEST(format_test, format_oct) { | |
1278 EXPECT_EQ(fmt::format("{0:o}", 0), "0"); | |
1279 EXPECT_EQ(fmt::format("{0:o}", 042), "42"); | |
1280 EXPECT_EQ(fmt::format("{0:o}", 042u), "42"); | |
1281 EXPECT_EQ(fmt::format("{0:o}", -042), "-42"); | |
1282 EXPECT_EQ(fmt::format("{0:o}", 012345670), "12345670"); | |
1283 #if FMT_USE_INT128 | |
1284 EXPECT_EQ(fmt::format("{0:o}", static_cast<__int128_t>(0)), "0"); | |
1285 EXPECT_EQ(fmt::format("{0:o}", static_cast<__uint128_t>(0)), "0"); | |
1286 EXPECT_EQ("1000000000000000000000", | |
1287 fmt::format("{0:o}", static_cast<__int128_t>(INT64_MAX) + 1)); | |
1288 EXPECT_EQ("-1000000000000000000001", | |
1289 fmt::format("{0:o}", static_cast<__int128_t>(INT64_MIN) - 1)); | |
1290 EXPECT_EQ("2000000000000000000000", | |
1291 fmt::format("{0:o}", static_cast<__int128_t>(UINT64_MAX) + 1)); | |
1292 EXPECT_EQ("1777777777777777777777777777777777777777777", | |
1293 fmt::format("{0:o}", int128_max)); | |
1294 EXPECT_EQ("-2000000000000000000000000000000000000000000", | |
1295 fmt::format("{0:o}", int128_min)); | |
1296 EXPECT_EQ("3777777777777777777777777777777777777777777", | |
1297 fmt::format("{0:o}", uint128_max)); | |
1298 #endif | |
1299 | |
1300 char buffer[buffer_size]; | |
1301 safe_sprintf(buffer, "-%o", 0 - static_cast<unsigned>(INT_MIN)); | |
1302 EXPECT_EQ(buffer, fmt::format("{0:o}", INT_MIN)); | |
1303 safe_sprintf(buffer, "%o", INT_MAX); | |
1304 EXPECT_EQ(buffer, fmt::format("{0:o}", INT_MAX)); | |
1305 safe_sprintf(buffer, "%o", UINT_MAX); | |
1306 EXPECT_EQ(buffer, fmt::format("{0:o}", UINT_MAX)); | |
1307 safe_sprintf(buffer, "-%lo", 0 - static_cast<unsigned long>(LONG_MIN)); | |
1308 EXPECT_EQ(buffer, fmt::format("{0:o}", LONG_MIN)); | |
1309 safe_sprintf(buffer, "%lo", LONG_MAX); | |
1310 EXPECT_EQ(buffer, fmt::format("{0:o}", LONG_MAX)); | |
1311 safe_sprintf(buffer, "%lo", ULONG_MAX); | |
1312 EXPECT_EQ(buffer, fmt::format("{0:o}", ULONG_MAX)); | |
1313 } | |
1314 | |
1315 TEST(format_test, format_int_locale) { | |
1316 EXPECT_EQ(fmt::format("{:L}", 1234), "1234"); | |
1317 } | |
1318 | |
1319 TEST(format_test, format_float) { | |
1320 EXPECT_EQ(fmt::format("{}", 0.0f), "0"); | |
1321 EXPECT_EQ(fmt::format("{0:f}", 392.5f), "392.500000"); | |
1322 } | |
1323 | |
1324 TEST(format_test, format_double) { | |
1325 EXPECT_EQ(fmt::format("{}", 0.0), "0"); | |
1326 check_unknown_types(1.2, "eEfFgGaAnL%", "double"); | |
1327 EXPECT_EQ(fmt::format("{:}", 0.0), "0"); | |
1328 EXPECT_EQ(fmt::format("{:f}", 0.0), "0.000000"); | |
1329 EXPECT_EQ(fmt::format("{:g}", 0.0), "0"); | |
1330 EXPECT_EQ(fmt::format("{:}", 392.65), "392.65"); | |
1331 EXPECT_EQ(fmt::format("{:g}", 392.65), "392.65"); | |
1332 EXPECT_EQ(fmt::format("{:G}", 392.65), "392.65"); | |
1333 EXPECT_EQ(fmt::format("{:g}", 4.9014e6), "4.9014e+06"); | |
1334 EXPECT_EQ(fmt::format("{:f}", 392.65), "392.650000"); | |
1335 EXPECT_EQ(fmt::format("{:F}", 392.65), "392.650000"); | |
1336 EXPECT_EQ(fmt::format("{:L}", 42.0), "42"); | |
1337 EXPECT_EQ(fmt::format("{:24a}", 4.2f), " 0x1.0cccccp+2"); | |
1338 EXPECT_EQ(fmt::format("{:24a}", 4.2), " 0x1.0cccccccccccdp+2"); | |
1339 EXPECT_EQ(fmt::format("{:<24a}", 4.2), "0x1.0cccccccccccdp+2 "); | |
1340 EXPECT_EQ(fmt::format("{0:e}", 392.65), "3.926500e+02"); | |
1341 EXPECT_EQ(fmt::format("{0:E}", 392.65), "3.926500E+02"); | |
1342 EXPECT_EQ(fmt::format("{0:+010.4g}", 392.65), "+0000392.6"); | |
1343 | |
1344 #if FMT_CPLUSPLUS >= 201703L | |
1345 double xd = 0x1.ffffffffffp+2; | |
1346 EXPECT_EQ(fmt::format("{:.10a}", xd), "0x1.ffffffffffp+2"); | |
1347 EXPECT_EQ(fmt::format("{:.9a}", xd), "0x2.000000000p+2"); | |
1348 | |
1349 if (std::numeric_limits<long double>::digits == 64) { | |
1350 auto ld = 0xf.ffffffffffp-3l; | |
1351 EXPECT_EQ(fmt::format("{:a}", ld), "0xf.ffffffffffp-3"); | |
1352 EXPECT_EQ(fmt::format("{:.10a}", ld), "0xf.ffffffffffp-3"); | |
1353 EXPECT_EQ(fmt::format("{:.9a}", ld), "0x1.000000000p+1"); | |
1354 } | |
1355 #endif | |
1356 | |
1357 if (fmt::detail::const_check(std::numeric_limits<double>::is_iec559)) { | |
1358 double d = (std::numeric_limits<double>::min)(); | |
1359 EXPECT_EQ(fmt::format("{:a}", d), "0x1p-1022"); | |
1360 EXPECT_EQ(fmt::format("{:#a}", d), "0x1.p-1022"); | |
1361 | |
1362 d = (std::numeric_limits<double>::max)(); | |
1363 EXPECT_EQ(fmt::format("{:a}", d), "0x1.fffffffffffffp+1023"); | |
1364 | |
1365 d = std::numeric_limits<double>::denorm_min(); | |
1366 EXPECT_EQ(fmt::format("{:a}", d), "0x0.0000000000001p-1022"); | |
1367 } | |
1368 | |
1369 if (std::numeric_limits<long double>::digits == 64) { | |
1370 auto ld = (std::numeric_limits<long double>::min)(); | |
1371 EXPECT_EQ(fmt::format("{:a}", ld), "0x8p-16385"); | |
1372 | |
1373 ld = (std::numeric_limits<long double>::max)(); | |
1374 EXPECT_EQ(fmt::format("{:a}", ld), "0xf.fffffffffffffffp+16380"); | |
1375 | |
1376 ld = std::numeric_limits<long double>::denorm_min(); | |
1377 EXPECT_EQ(fmt::format("{:a}", ld), "0x0.000000000000001p-16382"); | |
1378 } | |
1379 | |
1380 EXPECT_EQ(fmt::format("{:.10a}", 4.2), "0x1.0ccccccccdp+2"); | |
1381 | |
1382 EXPECT_EQ(fmt::format("{:a}", -42.0), "-0x1.5p+5"); | |
1383 EXPECT_EQ(fmt::format("{:A}", -42.0), "-0X1.5P+5"); | |
1384 | |
1385 EXPECT_EQ(fmt::format("{:f}", 9223372036854775807.0), | |
1386 "9223372036854775808.000000"); | |
1387 } | |
1388 | |
1389 TEST(format_test, precision_rounding) { | |
1390 EXPECT_EQ(fmt::format("{:.0f}", 0.0), "0"); | |
1391 EXPECT_EQ(fmt::format("{:.0f}", 0.01), "0"); | |
1392 EXPECT_EQ(fmt::format("{:.0f}", 0.1), "0"); | |
1393 EXPECT_EQ(fmt::format("{:.3f}", 0.00049), "0.000"); | |
1394 EXPECT_EQ(fmt::format("{:.3f}", 0.0005), "0.001"); | |
1395 EXPECT_EQ(fmt::format("{:.3f}", 0.00149), "0.001"); | |
1396 EXPECT_EQ(fmt::format("{:.3f}", 0.0015), "0.002"); | |
1397 EXPECT_EQ(fmt::format("{:.3f}", 0.9999), "1.000"); | |
1398 EXPECT_EQ(fmt::format("{:.3}", 0.00123), "0.00123"); | |
1399 EXPECT_EQ(fmt::format("{:.16g}", 0.1), "0.1"); | |
1400 EXPECT_EQ(fmt::format("{:.0}", 1.0), "1"); | |
1401 EXPECT_EQ("225.51575035152063720", | |
1402 fmt::format("{:.17f}", 225.51575035152064)); | |
1403 EXPECT_EQ(fmt::format("{:.1f}", -761519619559038.2), "-761519619559038.2"); | |
1404 EXPECT_EQ("1.9156918820264798e-56", | |
1405 fmt::format("{}", 1.9156918820264798e-56)); | |
1406 EXPECT_EQ(fmt::format("{:.4f}", 7.2809479766055470e-15), "0.0000"); | |
1407 } | |
1408 | |
1409 TEST(format_test, prettify_float) { | |
1410 EXPECT_EQ(fmt::format("{}", 1e-4), "0.0001"); | |
1411 EXPECT_EQ(fmt::format("{}", 1e-5), "1e-05"); | |
1412 EXPECT_EQ(fmt::format("{}", 1e15), "1000000000000000"); | |
1413 EXPECT_EQ(fmt::format("{}", 1e16), "1e+16"); | |
1414 EXPECT_EQ(fmt::format("{}", 9.999e-5), "9.999e-05"); | |
1415 EXPECT_EQ(fmt::format("{}", 1e10), "10000000000"); | |
1416 EXPECT_EQ(fmt::format("{}", 1e11), "100000000000"); | |
1417 EXPECT_EQ(fmt::format("{}", 1234e7), "12340000000"); | |
1418 EXPECT_EQ(fmt::format("{}", 1234e-2), "12.34"); | |
1419 EXPECT_EQ(fmt::format("{}", 1234e-6), "0.001234"); | |
1420 EXPECT_EQ(fmt::format("{}", 0.1f), "0.1"); | |
1421 EXPECT_EQ(fmt::format("{}", 1.35631564e-19f), "1.3563156e-19"); | |
1422 } | |
1423 | |
1424 TEST(format_test, format_nan) { | |
1425 double nan = std::numeric_limits<double>::quiet_NaN(); | |
1426 EXPECT_EQ(fmt::format("{}", nan), "nan"); | |
1427 EXPECT_EQ(fmt::format("{:+}", nan), "+nan"); | |
1428 EXPECT_EQ(fmt::format("{:+06}", nan), " +nan"); | |
1429 EXPECT_EQ(fmt::format("{:<+06}", nan), "+nan "); | |
1430 EXPECT_EQ(fmt::format("{:^+06}", nan), " +nan "); | |
1431 EXPECT_EQ(fmt::format("{:>+06}", nan), " +nan"); | |
1432 if (std::signbit(-nan)) { | |
1433 EXPECT_EQ(fmt::format("{}", -nan), "-nan"); | |
1434 EXPECT_EQ(fmt::format("{:+06}", -nan), " -nan"); | |
1435 } else { | |
1436 fmt::print("Warning: compiler doesn't handle negative NaN correctly"); | |
1437 } | |
1438 EXPECT_EQ(fmt::format("{: }", nan), " nan"); | |
1439 EXPECT_EQ(fmt::format("{:F}", nan), "NAN"); | |
1440 EXPECT_EQ(fmt::format("{:<7}", nan), "nan "); | |
1441 EXPECT_EQ(fmt::format("{:^7}", nan), " nan "); | |
1442 EXPECT_EQ(fmt::format("{:>7}", nan), " nan"); | |
1443 } | |
1444 | |
1445 TEST(format_test, format_infinity) { | |
1446 double inf = std::numeric_limits<double>::infinity(); | |
1447 EXPECT_EQ(fmt::format("{}", inf), "inf"); | |
1448 EXPECT_EQ(fmt::format("{:+}", inf), "+inf"); | |
1449 EXPECT_EQ(fmt::format("{}", -inf), "-inf"); | |
1450 EXPECT_EQ(fmt::format("{:+06}", inf), " +inf"); | |
1451 EXPECT_EQ(fmt::format("{:+06}", -inf), " -inf"); | |
1452 EXPECT_EQ(fmt::format("{:<+06}", inf), "+inf "); | |
1453 EXPECT_EQ(fmt::format("{:^+06}", inf), " +inf "); | |
1454 EXPECT_EQ(fmt::format("{:>+06}", inf), " +inf"); | |
1455 EXPECT_EQ(fmt::format("{: }", inf), " inf"); | |
1456 EXPECT_EQ(fmt::format("{:F}", inf), "INF"); | |
1457 EXPECT_EQ(fmt::format("{:<7}", inf), "inf "); | |
1458 EXPECT_EQ(fmt::format("{:^7}", inf), " inf "); | |
1459 EXPECT_EQ(fmt::format("{:>7}", inf), " inf"); | |
1460 } | |
1461 | |
1462 TEST(format_test, format_long_double) { | |
1463 EXPECT_EQ(fmt::format("{0:}", 0.0l), "0"); | |
1464 EXPECT_EQ(fmt::format("{0:f}", 0.0l), "0.000000"); | |
1465 EXPECT_EQ(fmt::format("{:.1f}", 0.000000001l), "0.0"); | |
1466 EXPECT_EQ(fmt::format("{:.2f}", 0.099l), "0.10"); | |
1467 EXPECT_EQ(fmt::format("{0:}", 392.65l), "392.65"); | |
1468 EXPECT_EQ(fmt::format("{0:g}", 392.65l), "392.65"); | |
1469 EXPECT_EQ(fmt::format("{0:G}", 392.65l), "392.65"); | |
1470 EXPECT_EQ(fmt::format("{0:f}", 392.65l), "392.650000"); | |
1471 EXPECT_EQ(fmt::format("{0:F}", 392.65l), "392.650000"); | |
1472 char buffer[buffer_size]; | |
1473 safe_sprintf(buffer, "%Le", 392.65l); | |
1474 EXPECT_EQ(buffer, fmt::format("{0:e}", 392.65l)); | |
1475 EXPECT_EQ(fmt::format("{0:+010.4g}", 392.64l), "+0000392.6"); | |
1476 | |
1477 auto ld = 3.31l; | |
1478 if (fmt::detail::is_double_double<decltype(ld)>::value) { | |
1479 safe_sprintf(buffer, "%a", static_cast<double>(ld)); | |
1480 EXPECT_EQ(buffer, fmt::format("{:a}", ld)); | |
1481 } else if (std::numeric_limits<long double>::digits == 64) { | |
1482 EXPECT_EQ(fmt::format("{:a}", ld), "0xd.3d70a3d70a3d70ap-2"); | |
1483 } | |
1484 } | |
1485 | |
1486 TEST(format_test, format_char) { | |
1487 const char types[] = "cbBdoxX"; | |
1488 check_unknown_types('a', types, "char"); | |
1489 EXPECT_EQ(fmt::format("{0}", 'a'), "a"); | |
1490 EXPECT_EQ(fmt::format("{0:c}", 'z'), "z"); | |
1491 int n = 'x'; | |
1492 for (const char* type = types + 1; *type; ++type) { | |
1493 std::string format_str = fmt::format("{{:{}}}", *type); | |
1494 EXPECT_EQ(fmt::format(runtime(format_str), n), | |
1495 fmt::format(runtime(format_str), 'x')) | |
1496 << format_str; | |
1497 } | |
1498 EXPECT_EQ(fmt::format("{:02X}", n), fmt::format("{:02X}", 'x')); | |
1499 | |
1500 EXPECT_EQ(fmt::format("{}", '\n'), "\n"); | |
1501 EXPECT_EQ(fmt::format("{:?}", '\n'), "'\\n'"); | |
1502 EXPECT_EQ(fmt::format("{:x}", '\xff'), "ff"); | |
1503 } | |
1504 | |
1505 TEST(format_test, format_volatile_char) { | |
1506 volatile char c = 'x'; | |
1507 EXPECT_EQ(fmt::format("{}", c), "x"); | |
1508 } | |
1509 | |
1510 TEST(format_test, format_unsigned_char) { | |
1511 EXPECT_EQ(fmt::format("{}", static_cast<unsigned char>(42)), "42"); | |
1512 EXPECT_EQ(fmt::format("{}", static_cast<uint8_t>(42)), "42"); | |
1513 } | |
1514 | |
1515 TEST(format_test, format_cstring) { | |
1516 check_unknown_types("test", "sp", "string"); | |
1517 EXPECT_EQ(fmt::format("{0}", "test"), "test"); | |
1518 EXPECT_EQ(fmt::format("{0:s}", "test"), "test"); | |
1519 char nonconst[] = "nonconst"; | |
1520 EXPECT_EQ(fmt::format("{0}", nonconst), "nonconst"); | |
1521 auto nullstr = static_cast<const char*>(nullptr); | |
1522 EXPECT_THROW_MSG((void)fmt::format("{}", nullstr), format_error, | |
1523 "string pointer is null"); | |
1524 EXPECT_THROW_MSG((void)fmt::format("{:s}", nullstr), format_error, | |
1525 "string pointer is null"); | |
1526 } | |
1527 | |
1528 void function_pointer_test(int, double, std::string) {} | |
1529 | |
1530 TEST(format_test, format_pointer) { | |
1531 check_unknown_types(reinterpret_cast<void*>(0x1234), "p", "pointer"); | |
1532 EXPECT_EQ(fmt::format("{0}", static_cast<void*>(nullptr)), "0x0"); | |
1533 EXPECT_EQ(fmt::format("{0}", reinterpret_cast<void*>(0x1234)), "0x1234"); | |
1534 EXPECT_EQ(fmt::format("{0:p}", reinterpret_cast<void*>(0x1234)), "0x1234"); | |
1535 // On CHERI (or other fat-pointer) systems, the size of a pointer is greater | |
1536 // than the size an integer that can hold a virtual address. There is no | |
1537 // portable address-as-an-integer type (yet) in C++, so we use `size_t` as | |
1538 // the closest equivalent for now. | |
1539 EXPECT_EQ("0x" + std::string(sizeof(size_t) * CHAR_BIT / 4, 'f'), | |
1540 fmt::format("{0}", reinterpret_cast<void*>(~uintptr_t()))); | |
1541 EXPECT_EQ("0x1234", | |
1542 fmt::format("{}", fmt::ptr(reinterpret_cast<int*>(0x1234)))); | |
1543 std::unique_ptr<int> up(new int(1)); | |
1544 EXPECT_EQ(fmt::format("{}", fmt::ptr(up.get())), | |
1545 fmt::format("{}", fmt::ptr(up))); | |
1546 struct custom_deleter { | |
1547 void operator()(int* p) const { delete p; } | |
1548 }; | |
1549 std::unique_ptr<int, custom_deleter> upcd(new int(1)); | |
1550 EXPECT_EQ(fmt::format("{}", fmt::ptr(upcd.get())), | |
1551 fmt::format("{}", fmt::ptr(upcd))); | |
1552 std::shared_ptr<int> sp(new int(1)); | |
1553 EXPECT_EQ(fmt::format("{}", fmt::ptr(sp.get())), | |
1554 fmt::format("{}", fmt::ptr(sp))); | |
1555 EXPECT_EQ(fmt::format("{}", fmt::detail::bit_cast<const void*>( | |
1556 &function_pointer_test)), | |
1557 fmt::format("{}", fmt::ptr(function_pointer_test))); | |
1558 EXPECT_EQ(fmt::format("{}", nullptr), "0x0"); | |
1559 } | |
1560 | |
1561 TEST(format_test, write_uintptr_fallback) { | |
1562 // Test that formatting a pointer by converting it to uint128_fallback works. | |
1563 // This is needed to support systems without uintptr_t. | |
1564 auto s = std::string(); | |
1565 fmt::detail::write_ptr<char>( | |
1566 std::back_inserter(s), | |
1567 fmt::detail::bit_cast<fmt::detail::uint128_fallback>( | |
1568 reinterpret_cast<void*>(0xface)), | |
1569 nullptr); | |
1570 EXPECT_EQ(s, "0xface"); | |
1571 } | |
1572 | |
1573 enum class color { red, green, blue }; | |
1574 | |
1575 namespace test_ns { | |
1576 enum class color { red, green, blue }; | |
1577 using fmt::enums::format_as; | |
1578 } // namespace test_ns | |
1579 | |
1580 TEST(format_test, format_enum_class) { | |
1581 EXPECT_EQ(fmt::format("{}", fmt::underlying(color::red)), "0"); | |
1582 EXPECT_EQ(fmt::format("{}", test_ns::color::red), "0"); | |
1583 } | |
1584 | |
1585 TEST(format_test, format_string) { | |
1586 EXPECT_EQ(fmt::format("{0}", std::string("test")), "test"); | |
1587 EXPECT_EQ(fmt::format("{0}", std::string("test")), "test"); | |
1588 EXPECT_EQ(fmt::format("{:?}", std::string("test")), "\"test\""); | |
1589 EXPECT_EQ(fmt::format("{:*^10?}", std::string("test")), "**\"test\"**"); | |
1590 EXPECT_EQ(fmt::format("{:?}", std::string("\test")), "\"\\test\""); | |
1591 EXPECT_THROW((void)fmt::format(fmt::runtime("{:x}"), std::string("test")), | |
1592 fmt::format_error); | |
1593 } | |
1594 | |
1595 TEST(format_test, format_string_view) { | |
1596 EXPECT_EQ(fmt::format("{}", string_view("test")), "test"); | |
1597 EXPECT_EQ(fmt::format("{:?}", string_view("t\nst")), "\"t\\nst\""); | |
1598 EXPECT_EQ(fmt::format("{}", string_view()), ""); | |
1599 } | |
1600 | |
1601 #ifdef FMT_USE_STRING_VIEW | |
1602 struct string_viewable {}; | |
1603 | |
1604 FMT_BEGIN_NAMESPACE | |
1605 template <> struct formatter<string_viewable> : formatter<std::string_view> { | |
1606 auto format(string_viewable, format_context& ctx) -> decltype(ctx.out()) { | |
1607 return formatter<std::string_view>::format("foo", ctx); | |
1608 } | |
1609 }; | |
1610 FMT_END_NAMESPACE | |
1611 | |
1612 TEST(format_test, format_std_string_view) { | |
1613 EXPECT_EQ(fmt::format("{}", std::string_view("test")), "test"); | |
1614 EXPECT_EQ(fmt::format("{}", string_viewable()), "foo"); | |
1615 } | |
1616 | |
1617 struct explicitly_convertible_to_std_string_view { | |
1618 explicit operator std::string_view() const { return "foo"; } | |
1619 }; | |
1620 | |
1621 template <> | |
1622 struct fmt::formatter<explicitly_convertible_to_std_string_view> | |
1623 : formatter<std::string_view> { | |
1624 auto format(explicitly_convertible_to_std_string_view v, format_context& ctx) | |
1625 -> decltype(ctx.out()) { | |
1626 return fmt::format_to(ctx.out(), "'{}'", std::string_view(v)); | |
1627 } | |
1628 }; | |
1629 | |
1630 TEST(format_test, format_explicitly_convertible_to_std_string_view) { | |
1631 EXPECT_EQ("'foo'", | |
1632 fmt::format("{}", explicitly_convertible_to_std_string_view())); | |
1633 } | |
1634 #endif | |
1635 | |
1636 class Answer {}; | |
1637 | |
1638 FMT_BEGIN_NAMESPACE | |
1639 template <> struct formatter<date> { | |
1640 template <typename ParseContext> | |
1641 FMT_CONSTEXPR auto parse(ParseContext& ctx) -> decltype(ctx.begin()) { | |
1642 auto it = ctx.begin(); | |
1643 if (it != ctx.end() && *it == 'd') ++it; | |
1644 return it; | |
1645 } | |
1646 | |
1647 auto format(const date& d, format_context& ctx) -> decltype(ctx.out()) { | |
1648 // Namespace-qualify to avoid ambiguity with std::format_to. | |
1649 fmt::format_to(ctx.out(), "{}-{}-{}", d.year(), d.month(), d.day()); | |
1650 return ctx.out(); | |
1651 } | |
1652 }; | |
1653 | |
1654 template <> struct formatter<Answer> : formatter<int> { | |
1655 template <typename FormatContext> | |
1656 auto format(Answer, FormatContext& ctx) -> decltype(ctx.out()) { | |
1657 return formatter<int>::format(42, ctx); | |
1658 } | |
1659 }; | |
1660 FMT_END_NAMESPACE | |
1661 | |
1662 TEST(format_test, format_custom) { | |
1663 EXPECT_THROW_MSG((void)fmt::format(runtime("{:s}"), date(2012, 12, 9)), | |
1664 format_error, "unknown format specifier"); | |
1665 EXPECT_EQ(fmt::format("{0}", Answer()), "42"); | |
1666 EXPECT_EQ(fmt::format("{:04}", Answer()), "0042"); | |
1667 } | |
1668 | |
1669 TEST(format_test, format_to_custom) { | |
1670 char buf[10] = {}; | |
1671 auto end = fmt::format_to(buf, "{}", Answer()); | |
1672 EXPECT_EQ(end, buf + 2); | |
1673 EXPECT_STREQ(buf, "42"); | |
1674 } | |
1675 | |
1676 TEST(format_test, format_string_from_speed_test) { | |
1677 EXPECT_EQ("1.2340000000:0042:+3.13:str:0x3e8:X:%", | |
1678 fmt::format("{0:0.10f}:{1:04}:{2:+g}:{3}:{4}:{5}:%", 1.234, 42, | |
1679 3.13, "str", reinterpret_cast<void*>(1000), 'X')); | |
1680 } | |
1681 | |
1682 TEST(format_test, format_examples) { | |
1683 std::string message = fmt::format("The answer is {}", 42); | |
1684 EXPECT_EQ("The answer is 42", message); | |
1685 | |
1686 EXPECT_EQ(fmt::format("{}", 42), "42"); | |
1687 | |
1688 memory_buffer out; | |
1689 fmt::format_to(std::back_inserter(out), "The answer is {}.", 42); | |
1690 EXPECT_EQ("The answer is 42.", to_string(out)); | |
1691 | |
1692 const char* filename = "nonexistent"; | |
1693 FILE* ftest = safe_fopen(filename, "r"); | |
1694 if (ftest) fclose(ftest); | |
1695 int error_code = errno; | |
1696 EXPECT_TRUE(ftest == nullptr); | |
1697 EXPECT_SYSTEM_ERROR( | |
1698 { | |
1699 FILE* f = safe_fopen(filename, "r"); | |
1700 if (!f) | |
1701 throw fmt::system_error(errno, "Cannot open file '{}'", filename); | |
1702 fclose(f); | |
1703 }, | |
1704 error_code, "Cannot open file 'nonexistent'"); | |
1705 | |
1706 EXPECT_EQ("First, thou shalt count to three", | |
1707 fmt::format("First, thou shalt count to {0}", "three")); | |
1708 EXPECT_EQ(fmt::format("Bring me a {}", "shrubbery"), "Bring me a shrubbery"); | |
1709 EXPECT_EQ(fmt::format("From {} to {}", 1, 3), "From 1 to 3"); | |
1710 | |
1711 char buffer[buffer_size]; | |
1712 safe_sprintf(buffer, "%03.2f", -1.2); | |
1713 EXPECT_EQ(buffer, fmt::format("{:03.2f}", -1.2)); | |
1714 | |
1715 EXPECT_EQ(fmt::format("{0}, {1}, {2}", 'a', 'b', 'c'), "a, b, c"); | |
1716 EXPECT_EQ(fmt::format("{}, {}, {}", 'a', 'b', 'c'), "a, b, c"); | |
1717 EXPECT_EQ(fmt::format("{2}, {1}, {0}", 'a', 'b', 'c'), "c, b, a"); | |
1718 EXPECT_EQ(fmt::format("{0}{1}{0}", "abra", "cad"), "abracadabra"); | |
1719 | |
1720 EXPECT_EQ("left aligned ", | |
1721 fmt::format("{:<30}", "left aligned")); | |
1722 EXPECT_EQ(" right aligned", | |
1723 fmt::format("{:>30}", "right aligned")); | |
1724 EXPECT_EQ(" centered ", | |
1725 fmt::format("{:^30}", "centered")); | |
1726 EXPECT_EQ("***********centered***********", | |
1727 fmt::format("{:*^30}", "centered")); | |
1728 | |
1729 EXPECT_EQ(fmt::format("{:+f}; {:+f}", 3.14, -3.14), "+3.140000; -3.140000"); | |
1730 EXPECT_EQ(fmt::format("{: f}; {: f}", 3.14, -3.14), " 3.140000; -3.140000"); | |
1731 EXPECT_EQ(fmt::format("{:-f}; {:-f}", 3.14, -3.14), "3.140000; -3.140000"); | |
1732 | |
1733 EXPECT_EQ("int: 42; hex: 2a; oct: 52", | |
1734 fmt::format("int: {0:d}; hex: {0:x}; oct: {0:o}", 42)); | |
1735 EXPECT_EQ("int: 42; hex: 0x2a; oct: 052", | |
1736 fmt::format("int: {0:d}; hex: {0:#x}; oct: {0:#o}", 42)); | |
1737 | |
1738 EXPECT_EQ(fmt::format("The answer is {}", 42), "The answer is 42"); | |
1739 EXPECT_THROW_MSG( | |
1740 (void)fmt::format(runtime("The answer is {:d}"), "forty-two"), | |
1741 format_error, "invalid format specifier"); | |
1742 | |
1743 EXPECT_WRITE( | |
1744 stdout, fmt::print("{}", std::numeric_limits<double>::infinity()), "inf"); | |
1745 } | |
1746 | |
1747 TEST(format_test, print) { | |
1748 EXPECT_WRITE(stdout, fmt::print("Don't {}!", "panic"), "Don't panic!"); | |
1749 EXPECT_WRITE(stderr, fmt::print(stderr, "Don't {}!", "panic"), | |
1750 "Don't panic!"); | |
1751 EXPECT_WRITE(stdout, fmt::println("Don't {}!", "panic"), "Don't panic!\n"); | |
1752 EXPECT_WRITE(stderr, fmt::println(stderr, "Don't {}!", "panic"), | |
1753 "Don't panic!\n"); | |
1754 } | |
1755 | |
1756 TEST(format_test, variadic) { | |
1757 EXPECT_EQ(fmt::format("{}c{}", "ab", 1), "abc1"); | |
1758 } | |
1759 | |
1760 TEST(format_test, bytes) { | |
1761 auto s = fmt::format("{:10}", fmt::bytes("ёжик")); | |
1762 EXPECT_EQ("ёжик ", s); | |
1763 EXPECT_EQ(10, s.size()); | |
1764 } | |
1765 | |
1766 TEST(format_test, group_digits_view) { | |
1767 EXPECT_EQ(fmt::format("{}", fmt::group_digits(10000000)), "10,000,000"); | |
1768 EXPECT_EQ(fmt::format("{:8}", fmt::group_digits(1000)), " 1,000"); | |
1769 } | |
1770 | |
1771 #ifdef __cpp_generic_lambdas | |
1772 struct point { | |
1773 double x, y; | |
1774 }; | |
1775 | |
1776 FMT_BEGIN_NAMESPACE | |
1777 template <> struct formatter<point> : nested_formatter<double> { | |
1778 auto format(point p, format_context& ctx) const -> decltype(ctx.out()) { | |
1779 return write_padded(ctx, [this, p](auto out) -> decltype(out) { | |
1780 return fmt::format_to(out, "({}, {})", nested(p.x), nested(p.y)); | |
1781 }); | |
1782 } | |
1783 }; | |
1784 FMT_END_NAMESPACE | |
1785 | |
1786 TEST(format_test, nested_formatter) { | |
1787 EXPECT_EQ(fmt::format("{:>16.2f}", point{1, 2}), " (1.00, 2.00)"); | |
1788 } | |
1789 #endif // __cpp_generic_lambdas | |
1790 | |
1791 enum test_enum { foo, bar }; | |
1792 auto format_as(test_enum e) -> int { return e; } | |
1793 | |
1794 TEST(format_test, join) { | |
1795 using fmt::join; | |
1796 int v1[3] = {1, 2, 3}; | |
1797 auto v2 = std::vector<float>(); | |
1798 v2.push_back(1.2f); | |
1799 v2.push_back(3.4f); | |
1800 void* v3[2] = {&v1[0], &v1[1]}; | |
1801 | |
1802 EXPECT_EQ(fmt::format("({})", join(v1, v1 + 3, ", ")), "(1, 2, 3)"); | |
1803 EXPECT_EQ(fmt::format("({})", join(v1, v1 + 1, ", ")), "(1)"); | |
1804 EXPECT_EQ(fmt::format("({})", join(v1, v1, ", ")), "()"); | |
1805 EXPECT_EQ(fmt::format("({:03})", join(v1, v1 + 3, ", ")), "(001, 002, 003)"); | |
1806 EXPECT_EQ("(+01.20, +03.40)", | |
1807 fmt::format("({:+06.2f})", join(v2.begin(), v2.end(), ", "))); | |
1808 | |
1809 EXPECT_EQ(fmt::format("{0:{1}}", join(v1, v1 + 3, ", "), 1), "1, 2, 3"); | |
1810 | |
1811 EXPECT_EQ(fmt::format("{}, {}", v3[0], v3[1]), | |
1812 fmt::format("{}", join(v3, v3 + 2, ", "))); | |
1813 | |
1814 EXPECT_EQ(fmt::format("({})", join(v1, ", ")), "(1, 2, 3)"); | |
1815 EXPECT_EQ(fmt::format("({:+06.2f})", join(v2, ", ")), "(+01.20, +03.40)"); | |
1816 | |
1817 auto v4 = std::vector<test_enum>{foo, bar, foo}; | |
1818 EXPECT_EQ(fmt::format("{}", join(v4, " ")), "0 1 0"); | |
1819 } | |
1820 | |
1821 #ifdef __cpp_lib_byte | |
1822 TEST(format_test, join_bytes) { | |
1823 auto v = std::vector<std::byte>{std::byte(1), std::byte(2), std::byte(3)}; | |
1824 EXPECT_EQ(fmt::format("{}", fmt::join(v, ", ")), "1, 2, 3"); | |
1825 } | |
1826 #endif | |
1827 | |
1828 std::string vformat_message(int id, const char* format, fmt::format_args args) { | |
1829 auto buffer = fmt::memory_buffer(); | |
1830 fmt::format_to(fmt::appender(buffer), "[{}] ", id); | |
1831 vformat_to(fmt::appender(buffer), format, args); | |
1832 return to_string(buffer); | |
1833 } | |
1834 | |
1835 template <typename... Args> | |
1836 std::string format_message(int id, const char* format, const Args&... args) { | |
1837 auto va = fmt::make_format_args(args...); | |
1838 return vformat_message(id, format, va); | |
1839 } | |
1840 | |
1841 TEST(format_test, format_message_example) { | |
1842 EXPECT_EQ("[42] something happened", | |
1843 format_message(42, "{} happened", "something")); | |
1844 } | |
1845 | |
1846 template <typename... Args> | |
1847 void print_error(const char* file, int line, const char* format, | |
1848 const Args&... args) { | |
1849 fmt::print("{}: {}: ", file, line); | |
1850 fmt::print(format, args...); | |
1851 } | |
1852 | |
1853 TEST(format_test, unpacked_args) { | |
1854 EXPECT_EQ("0123456789abcdefg", | |
1855 fmt::format("{}{}{}{}{}{}{}{}{}{}{}{}{}{}{}{}{}", 0, 1, 2, 3, 4, 5, | |
1856 6, 7, 8, 9, 'a', 'b', 'c', 'd', 'e', 'f', 'g')); | |
1857 } | |
1858 | |
1859 constexpr char with_null[3] = {'{', '}', '\0'}; | |
1860 constexpr char no_null[2] = {'{', '}'}; | |
1861 static constexpr const char static_with_null[3] = {'{', '}', '\0'}; | |
1862 static constexpr const char static_no_null[2] = {'{', '}'}; | |
1863 | |
1864 TEST(format_test, compile_time_string) { | |
1865 EXPECT_EQ(fmt::format(FMT_STRING("foo")), "foo"); | |
1866 EXPECT_EQ(fmt::format(FMT_STRING("{}"), 42), "42"); | |
1867 | |
1868 #if FMT_USE_NONTYPE_TEMPLATE_ARGS | |
1869 using namespace fmt::literals; | |
1870 EXPECT_EQ("foobar", fmt::format(FMT_STRING("{foo}{bar}"), "bar"_a = "bar", | |
1871 "foo"_a = "foo")); | |
1872 EXPECT_EQ(fmt::format(FMT_STRING("")), ""); | |
1873 EXPECT_EQ(fmt::format(FMT_STRING(""), "arg"_a = 42), ""); | |
1874 EXPECT_EQ(fmt::format(FMT_STRING("{answer}"), "answer"_a = Answer()), "42"); | |
1875 EXPECT_EQ(fmt::format(FMT_STRING("{} {two}"), 1, "two"_a = 2), "1 2"); | |
1876 #endif | |
1877 | |
1878 (void)static_with_null; | |
1879 (void)static_no_null; | |
1880 #ifndef _MSC_VER | |
1881 EXPECT_EQ(fmt::format(FMT_STRING(static_with_null), 42), "42"); | |
1882 EXPECT_EQ(fmt::format(FMT_STRING(static_no_null), 42), "42"); | |
1883 #endif | |
1884 | |
1885 (void)with_null; | |
1886 (void)no_null; | |
1887 #if FMT_CPLUSPLUS >= 201703L | |
1888 EXPECT_EQ(fmt::format(FMT_STRING(with_null), 42), "42"); | |
1889 EXPECT_EQ(fmt::format(FMT_STRING(no_null), 42), "42"); | |
1890 #endif | |
1891 #if defined(FMT_USE_STRING_VIEW) && FMT_CPLUSPLUS >= 201703L | |
1892 EXPECT_EQ(fmt::format(FMT_STRING(std::string_view("{}")), 42), "42"); | |
1893 #endif | |
1894 } | |
1895 | |
1896 TEST(format_test, custom_format_compile_time_string) { | |
1897 EXPECT_EQ(fmt::format(FMT_STRING("{}"), Answer()), "42"); | |
1898 auto answer = Answer(); | |
1899 EXPECT_EQ(fmt::format(FMT_STRING("{}"), answer), "42"); | |
1900 char buf[10] = {}; | |
1901 fmt::format_to(buf, FMT_STRING("{}"), answer); | |
1902 const Answer const_answer = Answer(); | |
1903 EXPECT_EQ(fmt::format(FMT_STRING("{}"), const_answer), "42"); | |
1904 } | |
1905 | |
1906 #if FMT_USE_USER_DEFINED_LITERALS | |
1907 TEST(format_test, named_arg_udl) { | |
1908 using namespace fmt::literals; | |
1909 auto udl_a = fmt::format("{first}{second}{first}{third}", "first"_a = "abra", | |
1910 "second"_a = "cad", "third"_a = 99); | |
1911 EXPECT_EQ( | |
1912 fmt::format("{first}{second}{first}{third}", fmt::arg("first", "abra"), | |
1913 fmt::arg("second", "cad"), fmt::arg("third", 99)), | |
1914 udl_a); | |
1915 | |
1916 EXPECT_EQ(fmt::format("{answer}", "answer"_a = Answer()), "42"); | |
1917 } | |
1918 #endif // FMT_USE_USER_DEFINED_LITERALS | |
1919 | |
1920 TEST(format_test, enum) { EXPECT_EQ(fmt::format("{}", foo), "0"); } | |
1921 | |
1922 TEST(format_test, formatter_not_specialized) { | |
1923 static_assert(!fmt::has_formatter<fmt::formatter<test_enum>, | |
1924 fmt::format_context>::value, | |
1925 ""); | |
1926 } | |
1927 | |
1928 #if FMT_HAS_FEATURE(cxx_strong_enums) | |
1929 enum big_enum : unsigned long long { big_enum_value = 5000000000ULL }; | |
1930 auto format_as(big_enum e) -> unsigned long long { return e; } | |
1931 | |
1932 TEST(format_test, strong_enum) { | |
1933 EXPECT_EQ(fmt::format("{}", big_enum_value), "5000000000"); | |
1934 } | |
1935 #endif | |
1936 | |
1937 TEST(format_test, non_null_terminated_format_string) { | |
1938 EXPECT_EQ(fmt::format(string_view("{}foo", 2), 42), "42"); | |
1939 } | |
1940 | |
1941 namespace adl_test { | |
1942 namespace fmt { | |
1943 namespace detail { | |
1944 struct foo {}; | |
1945 template <typename, typename OutputIt> void write(OutputIt, foo) = delete; | |
1946 } // namespace detail | |
1947 } // namespace fmt | |
1948 } // namespace adl_test | |
1949 | |
1950 FMT_BEGIN_NAMESPACE | |
1951 template <> | |
1952 struct formatter<adl_test::fmt::detail::foo> : formatter<std::string> { | |
1953 auto format(adl_test::fmt::detail::foo, format_context& ctx) | |
1954 -> decltype(ctx.out()) { | |
1955 return formatter<std::string>::format("foo", ctx); | |
1956 } | |
1957 }; | |
1958 FMT_END_NAMESPACE | |
1959 | |
1960 TEST(format_test, to_string) { | |
1961 EXPECT_EQ(fmt::to_string(42), "42"); | |
1962 EXPECT_EQ(fmt::to_string(reinterpret_cast<void*>(0x1234)), "0x1234"); | |
1963 EXPECT_EQ(fmt::to_string(adl_test::fmt::detail::foo()), "foo"); | |
1964 EXPECT_EQ(fmt::to_string(foo), "0"); | |
1965 | |
1966 #if FMT_USE_FLOAT128 | |
1967 EXPECT_EQ(fmt::to_string(__float128(0.5)), "0.5"); | |
1968 #endif | |
1969 | |
1970 #if defined(FMT_USE_STRING_VIEW) && FMT_CPLUSPLUS >= 201703L | |
1971 EXPECT_EQ(fmt::to_string(std::string_view()), ""); | |
1972 #endif | |
1973 } | |
1974 | |
1975 TEST(format_test, output_iterators) { | |
1976 std::list<char> out; | |
1977 fmt::format_to(std::back_inserter(out), "{}", 42); | |
1978 EXPECT_EQ("42", std::string(out.begin(), out.end())); | |
1979 std::stringstream s; | |
1980 fmt::format_to(std::ostream_iterator<char>(s), "{}", 42); | |
1981 EXPECT_EQ("42", s.str()); | |
1982 } | |
1983 | |
1984 TEST(format_test, formatted_size) { | |
1985 EXPECT_EQ(2u, fmt::formatted_size("{}", 42)); | |
1986 EXPECT_EQ(2u, fmt::formatted_size(std::locale(), "{}", 42)); | |
1987 } | |
1988 | |
1989 TEST(format_test, format_to_no_args) { | |
1990 std::string s; | |
1991 fmt::format_to(std::back_inserter(s), "test"); | |
1992 EXPECT_EQ("test", s); | |
1993 } | |
1994 | |
1995 TEST(format_test, format_to) { | |
1996 std::string s; | |
1997 fmt::format_to(std::back_inserter(s), "part{0}", 1); | |
1998 EXPECT_EQ("part1", s); | |
1999 fmt::format_to(std::back_inserter(s), "part{0}", 2); | |
2000 EXPECT_EQ("part1part2", s); | |
2001 } | |
2002 | |
2003 TEST(format_test, format_to_memory_buffer) { | |
2004 auto buf = fmt::basic_memory_buffer<char, 100>(); | |
2005 fmt::format_to(fmt::appender(buf), "{}", "foo"); | |
2006 EXPECT_EQ("foo", to_string(buf)); | |
2007 } | |
2008 | |
2009 TEST(format_test, format_to_vector) { | |
2010 std::vector<char> v; | |
2011 fmt::format_to(std::back_inserter(v), "{}", "foo"); | |
2012 EXPECT_EQ(string_view(v.data(), v.size()), "foo"); | |
2013 } | |
2014 | |
2015 struct nongrowing_container { | |
2016 using value_type = char; | |
2017 void push_back(char) { throw std::runtime_error("can't take it any more"); } | |
2018 }; | |
2019 | |
2020 TEST(format_test, format_to_propagates_exceptions) { | |
2021 auto c = nongrowing_container(); | |
2022 EXPECT_THROW(fmt::format_to(std::back_inserter(c), "{}", 42), | |
2023 std::runtime_error); | |
2024 } | |
2025 | |
2026 TEST(format_test, format_to_n) { | |
2027 char buffer[4]; | |
2028 buffer[3] = 'x'; | |
2029 auto result = fmt::format_to_n(buffer, 3, "{}", 12345); | |
2030 EXPECT_EQ(5u, result.size); | |
2031 EXPECT_EQ(buffer + 3, result.out); | |
2032 EXPECT_EQ("123x", fmt::string_view(buffer, 4)); | |
2033 | |
2034 result = fmt::format_to_n(buffer, 3, "{:s}", "foobar"); | |
2035 EXPECT_EQ(6u, result.size); | |
2036 EXPECT_EQ(buffer + 3, result.out); | |
2037 EXPECT_EQ("foox", fmt::string_view(buffer, 4)); | |
2038 | |
2039 buffer[0] = 'x'; | |
2040 buffer[1] = 'x'; | |
2041 buffer[2] = 'x'; | |
2042 result = fmt::format_to_n(buffer, 3, "{}", 'A'); | |
2043 EXPECT_EQ(1u, result.size); | |
2044 EXPECT_EQ(buffer + 1, result.out); | |
2045 EXPECT_EQ("Axxx", fmt::string_view(buffer, 4)); | |
2046 | |
2047 result = fmt::format_to_n(buffer, 3, "{}{} ", 'B', 'C'); | |
2048 EXPECT_EQ(3u, result.size); | |
2049 EXPECT_EQ(buffer + 3, result.out); | |
2050 EXPECT_EQ("BC x", fmt::string_view(buffer, 4)); | |
2051 | |
2052 result = fmt::format_to_n(buffer, 4, "{}", "ABCDE"); | |
2053 EXPECT_EQ(5u, result.size); | |
2054 EXPECT_EQ("ABCD", fmt::string_view(buffer, 4)); | |
2055 | |
2056 buffer[3] = 'x'; | |
2057 result = fmt::format_to_n(buffer, 3, "{}", std::string(1000, '*')); | |
2058 EXPECT_EQ(1000u, result.size); | |
2059 EXPECT_EQ("***x", fmt::string_view(buffer, 4)); | |
2060 } | |
2061 | |
2062 struct test_output_iterator { | |
2063 char* data; | |
2064 | |
2065 using iterator_category = std::output_iterator_tag; | |
2066 using value_type = void; | |
2067 using difference_type = void; | |
2068 using pointer = void; | |
2069 using reference = void; | |
2070 | |
2071 auto operator++() -> test_output_iterator& { | |
2072 ++data; | |
2073 return *this; | |
2074 } | |
2075 auto operator++(int) -> test_output_iterator { | |
2076 auto tmp = *this; | |
2077 ++data; | |
2078 return tmp; | |
2079 } | |
2080 auto operator*() -> char& { return *data; } | |
2081 }; | |
2082 | |
2083 TEST(format_test, format_to_n_output_iterator) { | |
2084 char buf[10] = {}; | |
2085 fmt::format_to_n(test_output_iterator{buf}, 10, "{}", 42); | |
2086 EXPECT_STREQ(buf, "42"); | |
2087 } | |
2088 | |
2089 TEST(format_test, vformat_to) { | |
2090 using context = fmt::format_context; | |
2091 int n = 42; | |
2092 auto args = fmt::make_format_args<context>(n); | |
2093 auto s = std::string(); | |
2094 fmt::vformat_to(std::back_inserter(s), "{}", args); | |
2095 EXPECT_EQ(s, "42"); | |
2096 s.clear(); | |
2097 fmt::vformat_to(std::back_inserter(s), FMT_STRING("{}"), args); | |
2098 EXPECT_EQ(s, "42"); | |
2099 } | |
2100 | |
2101 TEST(format_test, char_traits_not_ambiguous) { | |
2102 // Test that we don't inject detail names into the std namespace. | |
2103 using namespace std; | |
2104 auto c = char_traits<char>::char_type(); | |
2105 (void)c; | |
2106 #if FMT_CPLUSPLUS >= 201103L | |
2107 auto s = std::string(); | |
2108 auto lval = begin(s); | |
2109 (void)lval; | |
2110 #endif | |
2111 } | |
2112 | |
2113 struct check_back_appender {}; | |
2114 | |
2115 FMT_BEGIN_NAMESPACE | |
2116 template <> struct formatter<check_back_appender> { | |
2117 FMT_CONSTEXPR auto parse(format_parse_context& ctx) -> decltype(ctx.begin()) { | |
2118 return ctx.begin(); | |
2119 } | |
2120 | |
2121 template <typename Context> | |
2122 auto format(check_back_appender, Context& ctx) -> decltype(ctx.out()) { | |
2123 auto out = ctx.out(); | |
2124 static_assert(std::is_same<decltype(++out), decltype(out)&>::value, | |
2125 "needs to satisfy weakly_incrementable"); | |
2126 *out = 'y'; | |
2127 return ++out; | |
2128 } | |
2129 }; | |
2130 FMT_END_NAMESPACE | |
2131 | |
2132 TEST(format_test, back_insert_slicing) { | |
2133 EXPECT_EQ(fmt::format("{}", check_back_appender{}), "y"); | |
2134 } | |
2135 | |
2136 namespace test { | |
2137 enum class scoped_enum_as_int {}; | |
2138 auto format_as(scoped_enum_as_int) -> int { return 42; } | |
2139 | |
2140 enum class scoped_enum_as_string_view {}; | |
2141 auto format_as(scoped_enum_as_string_view) -> fmt::string_view { return "foo"; } | |
2142 | |
2143 enum class scoped_enum_as_string {}; | |
2144 auto format_as(scoped_enum_as_string) -> std::string { return "foo"; } | |
2145 | |
2146 struct struct_as_int {}; | |
2147 auto format_as(struct_as_int) -> int { return 42; } | |
2148 | |
2149 struct struct_as_const_reference { | |
2150 const std::string name = "foo"; | |
2151 }; | |
2152 auto format_as(const struct_as_const_reference& s) -> const std::string& { | |
2153 return s.name; | |
2154 } | |
2155 } // namespace test | |
2156 | |
2157 TEST(format_test, format_as) { | |
2158 EXPECT_EQ(fmt::format("{}", test::scoped_enum_as_int()), "42"); | |
2159 EXPECT_EQ(fmt::format("{}", test::scoped_enum_as_string_view()), "foo"); | |
2160 EXPECT_EQ(fmt::format("{}", test::scoped_enum_as_string()), "foo"); | |
2161 EXPECT_EQ(fmt::format("{}", test::struct_as_int()), "42"); | |
2162 EXPECT_EQ(fmt::format("{}", test::struct_as_const_reference()), "foo"); | |
2163 } | |
2164 | |
2165 TEST(format_test, format_as_to_string) { | |
2166 EXPECT_EQ(fmt::to_string(test::scoped_enum_as_int()), "42"); | |
2167 EXPECT_EQ(fmt::to_string(test::scoped_enum_as_string_view()), "foo"); | |
2168 EXPECT_EQ(fmt::to_string(test::scoped_enum_as_string()), "foo"); | |
2169 EXPECT_EQ(fmt::to_string(test::struct_as_int()), "42"); | |
2170 } | |
2171 | |
2172 template <typename Char, typename T> auto check_enabled_formatter() -> bool { | |
2173 static_assert(std::is_default_constructible<fmt::formatter<T, Char>>::value, | |
2174 ""); | |
2175 return true; | |
2176 } | |
2177 | |
2178 template <typename Char, typename... T> void check_enabled_formatters() { | |
2179 auto dummy = {check_enabled_formatter<Char, T>()...}; | |
2180 (void)dummy; | |
2181 } | |
2182 | |
2183 TEST(format_test, test_formatters_enabled) { | |
2184 check_enabled_formatters<char, bool, char, signed char, unsigned char, short, | |
2185 unsigned short, int, unsigned, long, unsigned long, | |
2186 long long, unsigned long long, float, double, | |
2187 long double, void*, const void*, char*, const char*, | |
2188 std::string, std::nullptr_t>(); | |
2189 check_enabled_formatters<wchar_t, bool, wchar_t, signed char, unsigned char, | |
2190 short, unsigned short, int, unsigned, long, | |
2191 unsigned long, long long, unsigned long long, float, | |
2192 double, long double, void*, const void*, wchar_t*, | |
2193 const wchar_t*, std::wstring, std::nullptr_t>(); | |
2194 } | |
2195 | |
2196 TEST(format_int_test, data) { | |
2197 fmt::format_int format_int(42); | |
2198 EXPECT_EQ(std::string(format_int.data(), format_int.size()), "42"); | |
2199 } | |
2200 | |
2201 TEST(format_int_test, format_int) { | |
2202 EXPECT_EQ(fmt::format_int(42).str(), "42"); | |
2203 EXPECT_EQ(fmt::format_int(42).size(), 2u); | |
2204 EXPECT_EQ(fmt::format_int(-42).str(), "-42"); | |
2205 EXPECT_EQ(fmt::format_int(-42).size(), 3u); | |
2206 EXPECT_EQ(fmt::format_int(42ul).str(), "42"); | |
2207 EXPECT_EQ(fmt::format_int(-42l).str(), "-42"); | |
2208 EXPECT_EQ(fmt::format_int(42ull).str(), "42"); | |
2209 EXPECT_EQ(fmt::format_int(-42ll).str(), "-42");\ | |
2210 EXPECT_EQ(fmt::format_int(max_value<int64_t>()).str(), | |
2211 std::to_string(max_value<int64_t>())); | |
2212 } | |
2213 | |
2214 #ifndef FMT_STATIC_THOUSANDS_SEPARATOR | |
2215 | |
2216 # include <locale> | |
2217 | |
2218 class format_facet : public fmt::format_facet<std::locale> { | |
2219 protected: | |
2220 struct int_formatter { | |
2221 fmt::appender out; | |
2222 | |
2223 template <typename T, FMT_ENABLE_IF(fmt::detail::is_integer<T>::value)> | |
2224 auto operator()(T value) -> bool { | |
2225 fmt::format_to(out, "[{}]", value); | |
2226 return true; | |
2227 } | |
2228 | |
2229 template <typename T, FMT_ENABLE_IF(!fmt::detail::is_integer<T>::value)> | |
2230 auto operator()(T) -> bool { | |
2231 return false; | |
2232 } | |
2233 }; | |
2234 | |
2235 auto do_put(fmt::appender out, fmt::loc_value val, | |
2236 const fmt::format_specs<>&) const -> bool override; | |
2237 }; | |
2238 | |
2239 auto format_facet::do_put(fmt::appender out, fmt::loc_value val, | |
2240 const fmt::format_specs<>&) const -> bool { | |
2241 return val.visit(int_formatter{out}); | |
2242 } | |
2243 | |
2244 TEST(format_test, format_facet) { | |
2245 auto loc = std::locale(std::locale(), new format_facet()); | |
2246 EXPECT_EQ(fmt::format(loc, "{:L}", 42), "[42]"); | |
2247 EXPECT_EQ(fmt::format(loc, "{:L}", -42), "[-42]"); | |
2248 } | |
2249 | |
2250 TEST(format_test, format_facet_separator) { | |
2251 // U+2019 RIGHT SINGLE QUOTATION MARK is a digit separator in the de_CH | |
2252 // locale. | |
2253 auto loc = | |
2254 std::locale({}, new fmt::format_facet<std::locale>("\xe2\x80\x99")); | |
2255 EXPECT_EQ(fmt::format(loc, "{:L}", 1000), | |
2256 "1\xe2\x80\x99" | |
2257 "000"); | |
2258 } | |
2259 | |
2260 TEST(format_test, format_facet_grouping) { | |
2261 auto loc = | |
2262 std::locale({}, new fmt::format_facet<std::locale>(",", {1, 2, 3})); | |
2263 EXPECT_EQ(fmt::format(loc, "{:L}", 1234567890), "1,234,567,89,0"); | |
2264 } | |
2265 | |
2266 TEST(format_test, format_named_arg_with_locale) { | |
2267 EXPECT_EQ(fmt::format(std::locale(), "{answer}", fmt::arg("answer", 42)), | |
2268 "42"); | |
2269 } | |
2270 | |
2271 TEST(format_test, format_locale) { | |
2272 auto loc = std::locale({}, new fmt::format_facet<std::locale>(",")); | |
2273 EXPECT_EQ(fmt::format(loc, "{:Lx}", 123456789), "7,5bc,d15"); | |
2274 EXPECT_EQ(fmt::format(loc, "{:#Lb}", -123456789), | |
2275 "-0b111,010,110,111,100,110,100,010,101"); | |
2276 EXPECT_EQ(fmt::format(loc, "{:10Lo}", 12345), " 30,071"); | |
2277 } | |
2278 | |
2279 #endif // FMT_STATIC_THOUSANDS_SEPARATOR | |
2280 | |
2281 struct convertible_to_nonconst_cstring { | |
2282 operator char*() const { | |
2283 static char c[] = "bar"; | |
2284 return c; | |
2285 } | |
2286 }; | |
2287 | |
2288 FMT_BEGIN_NAMESPACE | |
2289 template <> | |
2290 struct formatter<convertible_to_nonconst_cstring> : formatter<char*> {}; | |
2291 FMT_END_NAMESPACE | |
2292 | |
2293 TEST(format_test, formatter_nonconst_char) { | |
2294 EXPECT_EQ(fmt::format("{}", convertible_to_nonconst_cstring()), "bar"); | |
2295 } |