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 |
| 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 } |
