1 // Copyright 2020 The Abseil Authors.
2 //
3 // Licensed under the Apache License, Version 2.0 (the "License");
4 // you may not use this file except in compliance with the License.
5 // You may obtain a copy of the License at
6 //
7 // https://www.apache.org/licenses/LICENSE-2.0
8 //
9 // Unless required by applicable law or agreed to in writing, software
10 // distributed under the License is distributed on an "AS IS" BASIS,
11 // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12 // See the License for the specific language governing permissions and
13 // limitations under the License.
14
15 #include "absl/strings/str_format.h"
16
17 #include <cstdarg>
18 #include <cstdint>
19 #include <cstdio>
20 #include <string>
21
22 #include "gmock/gmock.h"
23 #include "gtest/gtest.h"
24 #include "absl/strings/cord.h"
25 #include "absl/strings/str_cat.h"
26 #include "absl/strings/string_view.h"
27
28 namespace absl {
29 ABSL_NAMESPACE_BEGIN
30 namespace {
31 using str_format_internal::FormatArgImpl;
32
33 using FormatEntryPointTest = ::testing::Test;
34
TEST_F(FormatEntryPointTest,Format)35 TEST_F(FormatEntryPointTest, Format) {
36 std::string sink;
37 EXPECT_TRUE(Format(&sink, "A format %d", 123));
38 EXPECT_EQ("A format 123", sink);
39 sink.clear();
40
41 ParsedFormat<'d'> pc("A format %d");
42 EXPECT_TRUE(Format(&sink, pc, 123));
43 EXPECT_EQ("A format 123", sink);
44 }
TEST_F(FormatEntryPointTest,UntypedFormat)45 TEST_F(FormatEntryPointTest, UntypedFormat) {
46 constexpr const char* formats[] = {
47 "",
48 "a",
49 "%80d",
50 #if !defined(_MSC_VER) && !defined(__ANDROID__) && !defined(__native_client__)
51 // MSVC, NaCL and Android don't support positional syntax.
52 "complicated multipart %% %1$d format %1$0999d",
53 #endif // _MSC_VER
54 };
55 for (const char* fmt : formats) {
56 std::string actual;
57 int i = 123;
58 FormatArgImpl arg_123(i);
59 absl::Span<const FormatArgImpl> args(&arg_123, 1);
60 UntypedFormatSpec format(fmt);
61
62 EXPECT_TRUE(FormatUntyped(&actual, format, args));
63 char buf[4096]{};
64 snprintf(buf, sizeof(buf), fmt, 123);
65 EXPECT_EQ(
66 str_format_internal::FormatPack(
67 str_format_internal::UntypedFormatSpecImpl::Extract(format), args),
68 buf);
69 EXPECT_EQ(actual, buf);
70 }
71 // The internal version works with a preparsed format.
72 ParsedFormat<'d'> pc("A format %d");
73 int i = 345;
74 FormatArg arg(i);
75 std::string out;
76 EXPECT_TRUE(str_format_internal::FormatUntyped(
77 &out, str_format_internal::UntypedFormatSpecImpl(&pc), {&arg, 1}));
78 EXPECT_EQ("A format 345", out);
79 }
80
TEST_F(FormatEntryPointTest,StringFormat)81 TEST_F(FormatEntryPointTest, StringFormat) {
82 EXPECT_EQ("123", StrFormat("%d", 123));
83 constexpr absl::string_view view("=%d=", 4);
84 EXPECT_EQ("=123=", StrFormat(view, 123));
85 }
86
TEST_F(FormatEntryPointTest,AppendFormat)87 TEST_F(FormatEntryPointTest, AppendFormat) {
88 std::string s;
89 std::string& r = StrAppendFormat(&s, "%d", 123);
90 EXPECT_EQ(&s, &r); // should be same object
91 EXPECT_EQ("123", r);
92 }
93
TEST_F(FormatEntryPointTest,AppendFormatFail)94 TEST_F(FormatEntryPointTest, AppendFormatFail) {
95 std::string s = "orig";
96
97 UntypedFormatSpec format(" more %d");
98 FormatArgImpl arg("not an int");
99
100 EXPECT_EQ("orig",
101 str_format_internal::AppendPack(
102 &s, str_format_internal::UntypedFormatSpecImpl::Extract(format),
103 {&arg, 1}));
104 }
105
106
TEST_F(FormatEntryPointTest,ManyArgs)107 TEST_F(FormatEntryPointTest, ManyArgs) {
108 EXPECT_EQ("24", StrFormat("%24$d", 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13,
109 14, 15, 16, 17, 18, 19, 20, 21, 22, 23, 24));
110 EXPECT_EQ("60", StrFormat("%60$d", 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13,
111 14, 15, 16, 17, 18, 19, 20, 21, 22, 23, 24, 25, 26,
112 27, 28, 29, 30, 31, 32, 33, 34, 35, 36, 37, 38, 39,
113 40, 41, 42, 43, 44, 45, 46, 47, 48, 49, 50, 51, 52,
114 53, 54, 55, 56, 57, 58, 59, 60));
115 }
116
TEST_F(FormatEntryPointTest,Preparsed)117 TEST_F(FormatEntryPointTest, Preparsed) {
118 ParsedFormat<'d'> pc("%d");
119 EXPECT_EQ("123", StrFormat(pc, 123));
120 // rvalue ok?
121 EXPECT_EQ("123", StrFormat(ParsedFormat<'d'>("%d"), 123));
122 constexpr absl::string_view view("=%d=", 4);
123 EXPECT_EQ("=123=", StrFormat(ParsedFormat<'d'>(view), 123));
124 }
125
TEST_F(FormatEntryPointTest,FormatCountCapture)126 TEST_F(FormatEntryPointTest, FormatCountCapture) {
127 int n = 0;
128 EXPECT_EQ("", StrFormat("%n", FormatCountCapture(&n)));
129 EXPECT_EQ(0, n);
130 EXPECT_EQ("123", StrFormat("%d%n", 123, FormatCountCapture(&n)));
131 EXPECT_EQ(3, n);
132 }
133
TEST_F(FormatEntryPointTest,FormatCountCaptureWrongType)134 TEST_F(FormatEntryPointTest, FormatCountCaptureWrongType) {
135 // Should reject int*.
136 int n = 0;
137 UntypedFormatSpec format("%d%n");
138 int i = 123, *ip = &n;
139 FormatArgImpl args[2] = {FormatArgImpl(i), FormatArgImpl(ip)};
140
141 EXPECT_EQ("", str_format_internal::FormatPack(
142 str_format_internal::UntypedFormatSpecImpl::Extract(format),
143 absl::MakeSpan(args)));
144 }
145
TEST_F(FormatEntryPointTest,FormatCountCaptureMultiple)146 TEST_F(FormatEntryPointTest, FormatCountCaptureMultiple) {
147 int n1 = 0;
148 int n2 = 0;
149 EXPECT_EQ(" 1 2",
150 StrFormat("%5d%n%10d%n", 1, FormatCountCapture(&n1), 2,
151 FormatCountCapture(&n2)));
152 EXPECT_EQ(5, n1);
153 EXPECT_EQ(15, n2);
154 }
155
TEST_F(FormatEntryPointTest,FormatCountCaptureExample)156 TEST_F(FormatEntryPointTest, FormatCountCaptureExample) {
157 int n;
158 std::string s;
159 StrAppendFormat(&s, "%s: %n%s\n", "(1,1)", FormatCountCapture(&n), "(1,2)");
160 StrAppendFormat(&s, "%*s%s\n", n, "", "(2,2)");
161 EXPECT_EQ(7, n);
162 EXPECT_EQ(
163 "(1,1): (1,2)\n"
164 " (2,2)\n",
165 s);
166 }
167
TEST_F(FormatEntryPointTest,Stream)168 TEST_F(FormatEntryPointTest, Stream) {
169 const std::string formats[] = {
170 "",
171 "a",
172 "%80d",
173 "%d %u %c %s %f %g",
174 #if !defined(_MSC_VER) && !defined(__ANDROID__) && !defined(__native_client__)
175 // MSVC, NaCL and Android don't support positional syntax.
176 "complicated multipart %% %1$d format %1$080d",
177 #endif // _MSC_VER
178 };
179 std::string buf(4096, '\0');
180 for (const auto& fmt : formats) {
181 const auto parsed =
182 ParsedFormat<'d', 'u', 'c', 's', 'f', 'g'>::NewAllowIgnored(fmt);
183 std::ostringstream oss;
184 oss << StreamFormat(*parsed, 123, 3, 49, "multistreaming!!!", 1.01, 1.01);
185 int fmt_result = snprintf(&*buf.begin(), buf.size(), fmt.c_str(), //
186 123, 3, 49, "multistreaming!!!", 1.01, 1.01);
187 ASSERT_TRUE(oss) << fmt;
188 ASSERT_TRUE(fmt_result >= 0 && static_cast<size_t>(fmt_result) < buf.size())
189 << fmt_result;
190 EXPECT_EQ(buf.c_str(), oss.str());
191 }
192 }
193
TEST_F(FormatEntryPointTest,StreamOk)194 TEST_F(FormatEntryPointTest, StreamOk) {
195 std::ostringstream oss;
196 oss << StreamFormat("hello %d", 123);
197 EXPECT_EQ("hello 123", oss.str());
198 EXPECT_TRUE(oss.good());
199 }
200
TEST_F(FormatEntryPointTest,StreamFail)201 TEST_F(FormatEntryPointTest, StreamFail) {
202 std::ostringstream oss;
203 UntypedFormatSpec format("hello %d");
204 FormatArgImpl arg("non-numeric");
205 oss << str_format_internal::Streamable(
206 str_format_internal::UntypedFormatSpecImpl::Extract(format), {&arg, 1});
207 EXPECT_EQ("hello ", oss.str()); // partial write
208 EXPECT_TRUE(oss.fail());
209 }
210
WithSnprintf(const char * fmt,...)211 std::string WithSnprintf(const char* fmt, ...) {
212 std::string buf;
213 buf.resize(128);
214 va_list va;
215 va_start(va, fmt);
216 int r = vsnprintf(&*buf.begin(), buf.size(), fmt, va);
217 va_end(va);
218 EXPECT_GE(r, 0);
219 EXPECT_LT(r, buf.size());
220 buf.resize(r);
221 return buf;
222 }
223
TEST_F(FormatEntryPointTest,FloatPrecisionArg)224 TEST_F(FormatEntryPointTest, FloatPrecisionArg) {
225 // Test that positional parameters for width and precision
226 // are indexed to precede the value.
227 // Also sanity check the same formats against snprintf.
228 EXPECT_EQ("0.1", StrFormat("%.1f", 0.1));
229 EXPECT_EQ("0.1", WithSnprintf("%.1f", 0.1));
230 EXPECT_EQ(" 0.1", StrFormat("%*.1f", 5, 0.1));
231 EXPECT_EQ(" 0.1", WithSnprintf("%*.1f", 5, 0.1));
232 EXPECT_EQ("0.1", StrFormat("%.*f", 1, 0.1));
233 EXPECT_EQ("0.1", WithSnprintf("%.*f", 1, 0.1));
234 EXPECT_EQ(" 0.1", StrFormat("%*.*f", 5, 1, 0.1));
235 EXPECT_EQ(" 0.1", WithSnprintf("%*.*f", 5, 1, 0.1));
236 }
237 namespace streamed_test {
238 struct X {};
operator <<(std::ostream & os,const X &)239 std::ostream& operator<<(std::ostream& os, const X&) {
240 return os << "X";
241 }
242 } // streamed_test
243
TEST_F(FormatEntryPointTest,FormatStreamed)244 TEST_F(FormatEntryPointTest, FormatStreamed) {
245 EXPECT_EQ("123", StrFormat("%s", FormatStreamed(123)));
246 EXPECT_EQ(" 123", StrFormat("%5s", FormatStreamed(123)));
247 EXPECT_EQ("123 ", StrFormat("%-5s", FormatStreamed(123)));
248 EXPECT_EQ("X", StrFormat("%s", FormatStreamed(streamed_test::X())));
249 EXPECT_EQ("123", StrFormat("%s", FormatStreamed(StreamFormat("%d", 123))));
250 }
251
252 // Helper class that creates a temporary file and exposes a FILE* to it.
253 // It will close the file on destruction.
254 class TempFile {
255 public:
TempFile()256 TempFile() : file_(std::tmpfile()) {}
~TempFile()257 ~TempFile() { std::fclose(file_); }
258
file() const259 std::FILE* file() const { return file_; }
260
261 // Read the file into a string.
ReadFile()262 std::string ReadFile() {
263 std::fseek(file_, 0, SEEK_END);
264 int size = std::ftell(file_);
265 EXPECT_GT(size, 0);
266 std::rewind(file_);
267 std::string str(2 * size, ' ');
268 int read_bytes = std::fread(&str[0], 1, str.size(), file_);
269 EXPECT_EQ(read_bytes, size);
270 str.resize(read_bytes);
271 EXPECT_TRUE(std::feof(file_));
272 return str;
273 }
274
275 private:
276 std::FILE* file_;
277 };
278
TEST_F(FormatEntryPointTest,FPrintF)279 TEST_F(FormatEntryPointTest, FPrintF) {
280 TempFile tmp;
281 int result =
282 FPrintF(tmp.file(), "STRING: %s NUMBER: %010d", std::string("ABC"), -19);
283 EXPECT_EQ(result, 30);
284 EXPECT_EQ(tmp.ReadFile(), "STRING: ABC NUMBER: -000000019");
285 }
286
TEST_F(FormatEntryPointTest,FPrintFError)287 TEST_F(FormatEntryPointTest, FPrintFError) {
288 errno = 0;
289 int result = FPrintF(stdin, "ABC");
290 EXPECT_LT(result, 0);
291 EXPECT_EQ(errno, EBADF);
292 }
293
294 #ifdef __GLIBC__
TEST_F(FormatEntryPointTest,FprintfTooLarge)295 TEST_F(FormatEntryPointTest, FprintfTooLarge) {
296 std::FILE* f = std::fopen("/dev/null", "w");
297 int width = 2000000000;
298 errno = 0;
299 int result = FPrintF(f, "%*d %*d", width, 0, width, 0);
300 EXPECT_LT(result, 0);
301 EXPECT_EQ(errno, EFBIG);
302 std::fclose(f);
303 }
304
TEST_F(FormatEntryPointTest,PrintF)305 TEST_F(FormatEntryPointTest, PrintF) {
306 int stdout_tmp = dup(STDOUT_FILENO);
307
308 TempFile tmp;
309 std::fflush(stdout);
310 dup2(fileno(tmp.file()), STDOUT_FILENO);
311
312 int result = PrintF("STRING: %s NUMBER: %010d", std::string("ABC"), -19);
313
314 std::fflush(stdout);
315 dup2(stdout_tmp, STDOUT_FILENO);
316 close(stdout_tmp);
317
318 EXPECT_EQ(result, 30);
319 EXPECT_EQ(tmp.ReadFile(), "STRING: ABC NUMBER: -000000019");
320 }
321 #endif // __GLIBC__
322
TEST_F(FormatEntryPointTest,SNPrintF)323 TEST_F(FormatEntryPointTest, SNPrintF) {
324 char buffer[16];
325 int result =
326 SNPrintF(buffer, sizeof(buffer), "STRING: %s", std::string("ABC"));
327 EXPECT_EQ(result, 11);
328 EXPECT_EQ(std::string(buffer), "STRING: ABC");
329
330 result = SNPrintF(buffer, sizeof(buffer), "NUMBER: %d", 123456);
331 EXPECT_EQ(result, 14);
332 EXPECT_EQ(std::string(buffer), "NUMBER: 123456");
333
334 result = SNPrintF(buffer, sizeof(buffer), "NUMBER: %d", 1234567);
335 EXPECT_EQ(result, 15);
336 EXPECT_EQ(std::string(buffer), "NUMBER: 1234567");
337
338 result = SNPrintF(buffer, sizeof(buffer), "NUMBER: %d", 12345678);
339 EXPECT_EQ(result, 16);
340 EXPECT_EQ(std::string(buffer), "NUMBER: 1234567");
341
342 result = SNPrintF(buffer, sizeof(buffer), "NUMBER: %d", 123456789);
343 EXPECT_EQ(result, 17);
344 EXPECT_EQ(std::string(buffer), "NUMBER: 1234567");
345
346 result = SNPrintF(nullptr, 0, "Just checking the %s of the output.", "size");
347 EXPECT_EQ(result, 37);
348 }
349
TEST(StrFormat,BehavesAsDocumented)350 TEST(StrFormat, BehavesAsDocumented) {
351 std::string s = absl::StrFormat("%s, %d!", "Hello", 123);
352 EXPECT_EQ("Hello, 123!", s);
353 // The format of a replacement is
354 // '%'[position][flags][width['.'precision]][length_modifier][format]
355 EXPECT_EQ(absl::StrFormat("%1$+3.2Lf", 1.1), "+1.10");
356 // Text conversion:
357 // "c" - Character. Eg: 'a' -> "A", 20 -> " "
358 EXPECT_EQ(StrFormat("%c", 'a'), "a");
359 EXPECT_EQ(StrFormat("%c", 0x20), " ");
360 // Formats char and integral types: int, long, uint64_t, etc.
361 EXPECT_EQ(StrFormat("%c", int{'a'}), "a");
362 EXPECT_EQ(StrFormat("%c", long{'a'}), "a"); // NOLINT
363 EXPECT_EQ(StrFormat("%c", uint64_t{'a'}), "a");
364 // "s" - string Eg: "C" -> "C", std::string("C++") -> "C++"
365 // Formats std::string, char*, string_view, and Cord.
366 EXPECT_EQ(StrFormat("%s", "C"), "C");
367 EXPECT_EQ(StrFormat("%s", std::string("C++")), "C++");
368 EXPECT_EQ(StrFormat("%s", string_view("view")), "view");
369 EXPECT_EQ(StrFormat("%s", absl::Cord("cord")), "cord");
370 // Integral Conversion
371 // These format integral types: char, int, long, uint64_t, etc.
372 EXPECT_EQ(StrFormat("%d", char{10}), "10");
373 EXPECT_EQ(StrFormat("%d", int{10}), "10");
374 EXPECT_EQ(StrFormat("%d", long{10}), "10"); // NOLINT
375 EXPECT_EQ(StrFormat("%d", uint64_t{10}), "10");
376 // d,i - signed decimal Eg: -10 -> "-10"
377 EXPECT_EQ(StrFormat("%d", -10), "-10");
378 EXPECT_EQ(StrFormat("%i", -10), "-10");
379 // o - octal Eg: 10 -> "12"
380 EXPECT_EQ(StrFormat("%o", 10), "12");
381 // u - unsigned decimal Eg: 10 -> "10"
382 EXPECT_EQ(StrFormat("%u", 10), "10");
383 // x/X - lower,upper case hex Eg: 10 -> "a"/"A"
384 EXPECT_EQ(StrFormat("%x", 10), "a");
385 EXPECT_EQ(StrFormat("%X", 10), "A");
386 // Floating-point, with upper/lower-case output.
387 // These format floating points types: float, double, long double, etc.
388 EXPECT_EQ(StrFormat("%.1f", float{1}), "1.0");
389 EXPECT_EQ(StrFormat("%.1f", double{1}), "1.0");
390 const long double long_double = 1.0;
391 EXPECT_EQ(StrFormat("%.1f", long_double), "1.0");
392 // These also format integral types: char, int, long, uint64_t, etc.:
393 EXPECT_EQ(StrFormat("%.1f", char{1}), "1.0");
394 EXPECT_EQ(StrFormat("%.1f", int{1}), "1.0");
395 EXPECT_EQ(StrFormat("%.1f", long{1}), "1.0"); // NOLINT
396 EXPECT_EQ(StrFormat("%.1f", uint64_t{1}), "1.0");
397 // f/F - decimal. Eg: 123456789 -> "123456789.000000"
398 EXPECT_EQ(StrFormat("%f", 123456789), "123456789.000000");
399 EXPECT_EQ(StrFormat("%F", 123456789), "123456789.000000");
400 // e/E - exponentiated Eg: .01 -> "1.00000e-2"/"1.00000E-2"
401 EXPECT_EQ(StrFormat("%e", .01), "1.000000e-02");
402 EXPECT_EQ(StrFormat("%E", .01), "1.000000E-02");
403 // g/G - exponentiate to fit Eg: .01 -> "0.01", 1e10 ->"1e+10"/"1E+10"
404 EXPECT_EQ(StrFormat("%g", .01), "0.01");
405 EXPECT_EQ(StrFormat("%g", 1e10), "1e+10");
406 EXPECT_EQ(StrFormat("%G", 1e10), "1E+10");
407 // a/A - lower,upper case hex Eg: -3.0 -> "-0x1.8p+1"/"-0X1.8P+1"
408
409 // On Android platform <=21, there is a regression in hexfloat formatting.
410 #if !defined(__ANDROID_API__) || __ANDROID_API__ > 21
411 EXPECT_EQ(StrFormat("%.1a", -3.0), "-0x1.8p+1"); // .1 to fix MSVC output
412 EXPECT_EQ(StrFormat("%.1A", -3.0), "-0X1.8P+1"); // .1 to fix MSVC output
413 #endif
414
415 // Other conversion
416 int64_t value = 0x7ffdeb4;
417 auto ptr_value = static_cast<uintptr_t>(value);
418 const int& something = *reinterpret_cast<const int*>(ptr_value);
419 EXPECT_EQ(StrFormat("%p", &something), StrFormat("0x%x", ptr_value));
420
421 // Output widths are supported, with optional flags.
422 EXPECT_EQ(StrFormat("%3d", 1), " 1");
423 EXPECT_EQ(StrFormat("%3d", 123456), "123456");
424 EXPECT_EQ(StrFormat("%06.2f", 1.234), "001.23");
425 EXPECT_EQ(StrFormat("%+d", 1), "+1");
426 EXPECT_EQ(StrFormat("% d", 1), " 1");
427 EXPECT_EQ(StrFormat("%-4d", -1), "-1 ");
428 EXPECT_EQ(StrFormat("%#o", 10), "012");
429 EXPECT_EQ(StrFormat("%#x", 15), "0xf");
430 EXPECT_EQ(StrFormat("%04d", 8), "0008");
431 // Posix positional substitution.
432 EXPECT_EQ(absl::StrFormat("%2$s, %3$s, %1$s!", "vici", "veni", "vidi"),
433 "veni, vidi, vici!");
434 // Length modifiers are ignored.
435 EXPECT_EQ(StrFormat("%hhd", int{1}), "1");
436 EXPECT_EQ(StrFormat("%hd", int{1}), "1");
437 EXPECT_EQ(StrFormat("%ld", int{1}), "1");
438 EXPECT_EQ(StrFormat("%lld", int{1}), "1");
439 EXPECT_EQ(StrFormat("%Ld", int{1}), "1");
440 EXPECT_EQ(StrFormat("%jd", int{1}), "1");
441 EXPECT_EQ(StrFormat("%zd", int{1}), "1");
442 EXPECT_EQ(StrFormat("%td", int{1}), "1");
443 EXPECT_EQ(StrFormat("%qd", int{1}), "1");
444 }
445
446 using str_format_internal::ExtendedParsedFormat;
447 using str_format_internal::ParsedFormatBase;
448
449 struct SummarizeConsumer {
450 std::string* out;
SummarizeConsumerabsl::__anon3bc13e630111::SummarizeConsumer451 explicit SummarizeConsumer(std::string* out) : out(out) {}
452
Appendabsl::__anon3bc13e630111::SummarizeConsumer453 bool Append(string_view s) {
454 *out += "[" + std::string(s) + "]";
455 return true;
456 }
457
ConvertOneabsl::__anon3bc13e630111::SummarizeConsumer458 bool ConvertOne(const str_format_internal::UnboundConversion& conv,
459 string_view s) {
460 *out += "{";
461 *out += std::string(s);
462 *out += ":";
463 *out += std::to_string(conv.arg_position) + "$";
464 if (conv.width.is_from_arg()) {
465 *out += std::to_string(conv.width.get_from_arg()) + "$*";
466 }
467 if (conv.precision.is_from_arg()) {
468 *out += "." + std::to_string(conv.precision.get_from_arg()) + "$*";
469 }
470 *out += str_format_internal::FormatConversionCharToChar(conv.conv);
471 *out += "}";
472 return true;
473 }
474 };
475
SummarizeParsedFormat(const ParsedFormatBase & pc)476 std::string SummarizeParsedFormat(const ParsedFormatBase& pc) {
477 std::string out;
478 if (!pc.ProcessFormat(SummarizeConsumer(&out))) out += "!";
479 return out;
480 }
481
482 using ParsedFormatTest = ::testing::Test;
483
TEST_F(ParsedFormatTest,SimpleChecked)484 TEST_F(ParsedFormatTest, SimpleChecked) {
485 EXPECT_EQ("[ABC]{d:1$d}[DEF]",
486 SummarizeParsedFormat(ParsedFormat<'d'>("ABC%dDEF")));
487 EXPECT_EQ("{s:1$s}[FFF]{d:2$d}[ZZZ]{f:3$f}",
488 SummarizeParsedFormat(ParsedFormat<'s', 'd', 'f'>("%sFFF%dZZZ%f")));
489 EXPECT_EQ("{s:1$s}[ ]{.*d:3$.2$*d}",
490 SummarizeParsedFormat(ParsedFormat<'s', '*', 'd'>("%s %.*d")));
491 }
492
TEST_F(ParsedFormatTest,SimpleUncheckedCorrect)493 TEST_F(ParsedFormatTest, SimpleUncheckedCorrect) {
494 auto f = ParsedFormat<'d'>::New("ABC%dDEF");
495 ASSERT_TRUE(f);
496 EXPECT_EQ("[ABC]{d:1$d}[DEF]", SummarizeParsedFormat(*f));
497
498 std::string format = "%sFFF%dZZZ%f";
499 auto f2 = ParsedFormat<'s', 'd', 'f'>::New(format);
500
501 ASSERT_TRUE(f2);
502 EXPECT_EQ("{s:1$s}[FFF]{d:2$d}[ZZZ]{f:3$f}", SummarizeParsedFormat(*f2));
503
504 f2 = ParsedFormat<'s', 'd', 'f'>::New("%s %d %f");
505
506 ASSERT_TRUE(f2);
507 EXPECT_EQ("{s:1$s}[ ]{d:2$d}[ ]{f:3$f}", SummarizeParsedFormat(*f2));
508
509 auto star = ParsedFormat<'*', 'd'>::New("%*d");
510 ASSERT_TRUE(star);
511 EXPECT_EQ("{*d:2$1$*d}", SummarizeParsedFormat(*star));
512
513 auto dollar = ParsedFormat<'d', 's'>::New("%2$s %1$d");
514 ASSERT_TRUE(dollar);
515 EXPECT_EQ("{2$s:2$s}[ ]{1$d:1$d}", SummarizeParsedFormat(*dollar));
516 // with reuse
517 dollar = ParsedFormat<'d', 's'>::New("%2$s %1$d %1$d");
518 ASSERT_TRUE(dollar);
519 EXPECT_EQ("{2$s:2$s}[ ]{1$d:1$d}[ ]{1$d:1$d}",
520 SummarizeParsedFormat(*dollar));
521 }
522
TEST_F(ParsedFormatTest,SimpleUncheckedIgnoredArgs)523 TEST_F(ParsedFormatTest, SimpleUncheckedIgnoredArgs) {
524 EXPECT_FALSE((ParsedFormat<'d', 's'>::New("ABC")));
525 EXPECT_FALSE((ParsedFormat<'d', 's'>::New("%dABC")));
526 EXPECT_FALSE((ParsedFormat<'d', 's'>::New("ABC%2$s")));
527 auto f = ParsedFormat<'d', 's'>::NewAllowIgnored("ABC");
528 ASSERT_TRUE(f);
529 EXPECT_EQ("[ABC]", SummarizeParsedFormat(*f));
530 f = ParsedFormat<'d', 's'>::NewAllowIgnored("%dABC");
531 ASSERT_TRUE(f);
532 EXPECT_EQ("{d:1$d}[ABC]", SummarizeParsedFormat(*f));
533 f = ParsedFormat<'d', 's'>::NewAllowIgnored("ABC%2$s");
534 ASSERT_TRUE(f);
535 EXPECT_EQ("[ABC]{2$s:2$s}", SummarizeParsedFormat(*f));
536 }
537
TEST_F(ParsedFormatTest,SimpleUncheckedUnsupported)538 TEST_F(ParsedFormatTest, SimpleUncheckedUnsupported) {
539 EXPECT_FALSE(ParsedFormat<'d'>::New("%1$d %1$x"));
540 EXPECT_FALSE(ParsedFormat<'x'>::New("%1$d %1$x"));
541 }
542
TEST_F(ParsedFormatTest,SimpleUncheckedIncorrect)543 TEST_F(ParsedFormatTest, SimpleUncheckedIncorrect) {
544 EXPECT_FALSE(ParsedFormat<'d'>::New(""));
545
546 EXPECT_FALSE(ParsedFormat<'d'>::New("ABC%dDEF%d"));
547
548 std::string format = "%sFFF%dZZZ%f";
549 EXPECT_FALSE((ParsedFormat<'s', 'd', 'g'>::New(format)));
550 }
551
552 #if defined(__cpp_nontype_template_parameter_auto)
553
554 template <auto T>
555 std::true_type IsValidParsedFormatArgTest(ParsedFormat<T>*);
556
557 template <auto T>
558 std::false_type IsValidParsedFormatArgTest(...);
559
560 template <auto T>
561 using IsValidParsedFormatArg = decltype(IsValidParsedFormatArgTest<T>(nullptr));
562
TEST_F(ParsedFormatTest,OnlyValidTypesAllowed)563 TEST_F(ParsedFormatTest, OnlyValidTypesAllowed) {
564 ASSERT_TRUE(IsValidParsedFormatArg<'c'>::value);
565
566 ASSERT_TRUE(IsValidParsedFormatArg<FormatConversionCharSet::d>::value);
567
568 ASSERT_TRUE(IsValidParsedFormatArg<absl::FormatConversionCharSet::d |
569 absl::FormatConversionCharSet::x>::value);
570 ASSERT_TRUE(
571 IsValidParsedFormatArg<absl::FormatConversionCharSet::kIntegral>::value);
572
573 // This is an easy mistake to make, however, this will reduce to an integer
574 // which has no meaning, so we need to ensure it doesn't compile.
575 ASSERT_FALSE(IsValidParsedFormatArg<'x' | 'd'>::value);
576
577 // For now, we disallow construction based on ConversionChar (rather than
578 // CharSet)
579 ASSERT_FALSE(IsValidParsedFormatArg<absl::FormatConversionChar::d>::value);
580 }
581
TEST_F(ParsedFormatTest,ExtendedTyping)582 TEST_F(ParsedFormatTest, ExtendedTyping) {
583 EXPECT_FALSE(ParsedFormat<FormatConversionCharSet::d>::New(""));
584 ASSERT_TRUE(ParsedFormat<absl::FormatConversionCharSet::d>::New("%d"));
585 auto v1 = ParsedFormat<'d', absl::FormatConversionCharSet::s>::New("%d%s");
586 ASSERT_TRUE(v1);
587 auto v2 = ParsedFormat<absl::FormatConversionCharSet::d, 's'>::New("%d%s");
588 ASSERT_TRUE(v2);
589 auto v3 = ParsedFormat<absl::FormatConversionCharSet::d |
590 absl::FormatConversionCharSet::s,
591 's'>::New("%d%s");
592 ASSERT_TRUE(v3);
593 auto v4 = ParsedFormat<absl::FormatConversionCharSet::d |
594 absl::FormatConversionCharSet::s,
595 's'>::New("%s%s");
596 ASSERT_TRUE(v4);
597 }
598 #endif
599
TEST_F(ParsedFormatTest,UncheckedCorrect)600 TEST_F(ParsedFormatTest, UncheckedCorrect) {
601 auto f =
602 ExtendedParsedFormat<absl::FormatConversionCharSet::d>::New("ABC%dDEF");
603 ASSERT_TRUE(f);
604 EXPECT_EQ("[ABC]{d:1$d}[DEF]", SummarizeParsedFormat(*f));
605
606 std::string format = "%sFFF%dZZZ%f";
607 auto f2 = ExtendedParsedFormat<
608 absl::FormatConversionCharSet::kString, absl::FormatConversionCharSet::d,
609 absl::FormatConversionCharSet::kFloating>::New(format);
610
611 ASSERT_TRUE(f2);
612 EXPECT_EQ("{s:1$s}[FFF]{d:2$d}[ZZZ]{f:3$f}", SummarizeParsedFormat(*f2));
613
614 f2 = ExtendedParsedFormat<
615 absl::FormatConversionCharSet::kString, absl::FormatConversionCharSet::d,
616 absl::FormatConversionCharSet::kFloating>::New("%s %d %f");
617
618 ASSERT_TRUE(f2);
619 EXPECT_EQ("{s:1$s}[ ]{d:2$d}[ ]{f:3$f}", SummarizeParsedFormat(*f2));
620
621 auto star =
622 ExtendedParsedFormat<absl::FormatConversionCharSet::kStar,
623 absl::FormatConversionCharSet::d>::New("%*d");
624 ASSERT_TRUE(star);
625 EXPECT_EQ("{*d:2$1$*d}", SummarizeParsedFormat(*star));
626
627 auto dollar =
628 ExtendedParsedFormat<absl::FormatConversionCharSet::d,
629 absl::FormatConversionCharSet::s>::New("%2$s %1$d");
630 ASSERT_TRUE(dollar);
631 EXPECT_EQ("{2$s:2$s}[ ]{1$d:1$d}", SummarizeParsedFormat(*dollar));
632 // with reuse
633 dollar = ExtendedParsedFormat<
634 absl::FormatConversionCharSet::d,
635 absl::FormatConversionCharSet::s>::New("%2$s %1$d %1$d");
636 ASSERT_TRUE(dollar);
637 EXPECT_EQ("{2$s:2$s}[ ]{1$d:1$d}[ ]{1$d:1$d}",
638 SummarizeParsedFormat(*dollar));
639 }
640
TEST_F(ParsedFormatTest,UncheckedIgnoredArgs)641 TEST_F(ParsedFormatTest, UncheckedIgnoredArgs) {
642 EXPECT_FALSE(
643 (ExtendedParsedFormat<absl::FormatConversionCharSet::d,
644 absl::FormatConversionCharSet::s>::New("ABC")));
645 EXPECT_FALSE(
646 (ExtendedParsedFormat<absl::FormatConversionCharSet::d,
647 absl::FormatConversionCharSet::s>::New("%dABC")));
648 EXPECT_FALSE(
649 (ExtendedParsedFormat<absl::FormatConversionCharSet::d,
650 absl::FormatConversionCharSet::s>::New("ABC%2$s")));
651 auto f = ExtendedParsedFormat<
652 absl::FormatConversionCharSet::d,
653 absl::FormatConversionCharSet::s>::NewAllowIgnored("ABC");
654 ASSERT_TRUE(f);
655 EXPECT_EQ("[ABC]", SummarizeParsedFormat(*f));
656 f = ExtendedParsedFormat<
657 absl::FormatConversionCharSet::d,
658 absl::FormatConversionCharSet::s>::NewAllowIgnored("%dABC");
659 ASSERT_TRUE(f);
660 EXPECT_EQ("{d:1$d}[ABC]", SummarizeParsedFormat(*f));
661 f = ExtendedParsedFormat<
662 absl::FormatConversionCharSet::d,
663 absl::FormatConversionCharSet::s>::NewAllowIgnored("ABC%2$s");
664 ASSERT_TRUE(f);
665 EXPECT_EQ("[ABC]{2$s:2$s}", SummarizeParsedFormat(*f));
666 }
667
TEST_F(ParsedFormatTest,UncheckedMultipleTypes)668 TEST_F(ParsedFormatTest, UncheckedMultipleTypes) {
669 auto dx =
670 ExtendedParsedFormat<absl::FormatConversionCharSet::d |
671 absl::FormatConversionCharSet::x>::New("%1$d %1$x");
672 EXPECT_TRUE(dx);
673 EXPECT_EQ("{1$d:1$d}[ ]{1$x:1$x}", SummarizeParsedFormat(*dx));
674
675 dx = ExtendedParsedFormat<absl::FormatConversionCharSet::d |
676 absl::FormatConversionCharSet::x>::New("%1$d");
677 EXPECT_TRUE(dx);
678 EXPECT_EQ("{1$d:1$d}", SummarizeParsedFormat(*dx));
679 }
680
TEST_F(ParsedFormatTest,UncheckedIncorrect)681 TEST_F(ParsedFormatTest, UncheckedIncorrect) {
682 EXPECT_FALSE(ExtendedParsedFormat<absl::FormatConversionCharSet::d>::New(""));
683
684 EXPECT_FALSE(ExtendedParsedFormat<absl::FormatConversionCharSet::d>::New(
685 "ABC%dDEF%d"));
686
687 std::string format = "%sFFF%dZZZ%f";
688 EXPECT_FALSE(
689 (ExtendedParsedFormat<absl::FormatConversionCharSet::s,
690 absl::FormatConversionCharSet::d,
691 absl::FormatConversionCharSet::g>::New(format)));
692 }
693
TEST_F(ParsedFormatTest,RegressionMixPositional)694 TEST_F(ParsedFormatTest, RegressionMixPositional) {
695 EXPECT_FALSE(
696 (ExtendedParsedFormat<absl::FormatConversionCharSet::d,
697 absl::FormatConversionCharSet::o>::New("%1$d %o")));
698 }
699
700 using FormatWrapperTest = ::testing::Test;
701
702 // Plain wrapper for StrFormat.
703 template <typename... Args>
WrappedFormat(const absl::FormatSpec<Args...> & format,const Args &...args)704 std::string WrappedFormat(const absl::FormatSpec<Args...>& format,
705 const Args&... args) {
706 return StrFormat(format, args...);
707 }
708
TEST_F(FormatWrapperTest,ConstexprStringFormat)709 TEST_F(FormatWrapperTest, ConstexprStringFormat) {
710 EXPECT_EQ(WrappedFormat("%s there", "hello"), "hello there");
711 }
712
TEST_F(FormatWrapperTest,ParsedFormat)713 TEST_F(FormatWrapperTest, ParsedFormat) {
714 ParsedFormat<'s'> format("%s there");
715 EXPECT_EQ(WrappedFormat(format, "hello"), "hello there");
716 }
717
718 } // namespace
719 ABSL_NAMESPACE_END
720 } // namespace absl
721
722 namespace {
723 using FormatExtensionTest = ::testing::Test;
724
725 struct Point {
726 friend absl::FormatConvertResult<absl::FormatConversionCharSet::kString |
727 absl::FormatConversionCharSet::kIntegral>
AbslFormatConvert(const Point & p,const absl::FormatConversionSpec & spec,absl::FormatSink * s)728 AbslFormatConvert(const Point& p, const absl::FormatConversionSpec& spec,
729 absl::FormatSink* s) {
730 if (spec.conversion_char() == absl::FormatConversionChar::s) {
731 s->Append(absl::StrCat("x=", p.x, " y=", p.y));
732 } else {
733 s->Append(absl::StrCat(p.x, ",", p.y));
734 }
735 return {true};
736 }
737
738 int x = 10;
739 int y = 20;
740 };
741
TEST_F(FormatExtensionTest,AbslFormatConvertExample)742 TEST_F(FormatExtensionTest, AbslFormatConvertExample) {
743 Point p;
744 EXPECT_EQ(absl::StrFormat("a %s z", p), "a x=10 y=20 z");
745 EXPECT_EQ(absl::StrFormat("a %d z", p), "a 10,20 z");
746
747 // Typed formatting will fail to compile an invalid format.
748 // StrFormat("%f", p); // Does not compile.
749 std::string actual;
750 absl::UntypedFormatSpec f1("%f");
751 // FormatUntyped will return false for bad character.
752 EXPECT_FALSE(absl::FormatUntyped(&actual, f1, {absl::FormatArg(p)}));
753 }
754 } // namespace
755
756 // Some codegen thunks that we can use to easily dump the generated assembly for
757 // different StrFormat calls.
758
CodegenAbslStrFormatInt(int i)759 std::string CodegenAbslStrFormatInt(int i) { // NOLINT
760 return absl::StrFormat("%d", i);
761 }
762
CodegenAbslStrFormatIntStringInt64(int i,const std::string & s,int64_t i64)763 std::string CodegenAbslStrFormatIntStringInt64(int i, const std::string& s,
764 int64_t i64) { // NOLINT
765 return absl::StrFormat("%d %s %d", i, s, i64);
766 }
767
CodegenAbslStrAppendFormatInt(std::string * out,int i)768 void CodegenAbslStrAppendFormatInt(std::string* out, int i) { // NOLINT
769 absl::StrAppendFormat(out, "%d", i);
770 }
771
CodegenAbslStrAppendFormatIntStringInt64(std::string * out,int i,const std::string & s,int64_t i64)772 void CodegenAbslStrAppendFormatIntStringInt64(std::string* out, int i,
773 const std::string& s,
774 int64_t i64) { // NOLINT
775 absl::StrAppendFormat(out, "%d %s %d", i, s, i64);
776 }
777