| // Copyright 2013 The Chromium Authors |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #include "base/strings/stringprintf.h" |
| |
| #include <errno.h> |
| #include <stddef.h> |
| |
| #include "build/build_config.h" |
| #include "testing/gtest/include/gtest/gtest.h" |
| |
| namespace gurl_base { |
| |
| namespace { |
| |
| // A helper for the StringAppendV test that follows. |
| // |
| // Just forwards its args to StringAppendV. |
| template <class CharT> |
| static void StringAppendVTestHelper(std::basic_string<CharT>* out, |
| const CharT* format, |
| ...) { |
| va_list ap; |
| va_start(ap, format); |
| StringAppendV(out, format, ap); |
| va_end(ap); |
| } |
| |
| } // namespace |
| |
| TEST(StringPrintfTest, StringPrintfEmpty) { |
| EXPECT_EQ("", StringPrintf("%s", "")); |
| } |
| |
| TEST(StringPrintfTest, StringPrintfMisc) { |
| EXPECT_EQ("123hello w", StringPrintf("%3d%2s %1c", 123, "hello", 'w')); |
| #if BUILDFLAG(IS_WIN) |
| EXPECT_EQ(L"123hello w", StringPrintf(L"%3d%2ls %1lc", 123, L"hello", 'w')); |
| EXPECT_EQ(u"123hello w", StringPrintf(u"%3d%2ls %1lc", 123, u"hello", 'w')); |
| #endif |
| } |
| |
| TEST(StringPrintfTest, StringAppendfEmptyString) { |
| std::string value("Hello"); |
| StringAppendF(&value, "%s", ""); |
| EXPECT_EQ("Hello", value); |
| |
| #if BUILDFLAG(IS_WIN) |
| std::wstring valuew(L"Hello"); |
| StringAppendF(&valuew, L"%ls", L""); |
| EXPECT_EQ(L"Hello", valuew); |
| |
| std::u16string value16(u"Hello"); |
| StringAppendF(&value16, u"%ls", u""); |
| EXPECT_EQ(u"Hello", value16); |
| #endif |
| } |
| |
| TEST(StringPrintfTest, StringAppendfString) { |
| std::string value("Hello"); |
| StringAppendF(&value, " %s", "World"); |
| EXPECT_EQ("Hello World", value); |
| |
| #if BUILDFLAG(IS_WIN) |
| std::wstring valuew(L"Hello"); |
| StringAppendF(&valuew, L" %ls", L"World"); |
| EXPECT_EQ(L"Hello World", valuew); |
| |
| std::u16string value16(u"Hello"); |
| StringAppendF(&value16, u" %ls", u"World"); |
| EXPECT_EQ(u"Hello World", value16); |
| #endif |
| } |
| |
| TEST(StringPrintfTest, StringAppendfInt) { |
| std::string value("Hello"); |
| StringAppendF(&value, " %d", 123); |
| EXPECT_EQ("Hello 123", value); |
| |
| #if BUILDFLAG(IS_WIN) |
| std::wstring valuew(L"Hello"); |
| StringAppendF(&valuew, L" %d", 123); |
| EXPECT_EQ(L"Hello 123", valuew); |
| |
| std::u16string value16(u"Hello"); |
| StringAppendF(&value16, u" %d", 123); |
| EXPECT_EQ(u"Hello 123", value16); |
| #endif |
| } |
| |
| // Make sure that lengths exactly around the initial buffer size are handled |
| // correctly. |
| TEST(StringPrintfTest, StringPrintfBounds) { |
| const int kSrcLen = 1026; |
| char src[kSrcLen]; |
| std::fill_n(src, kSrcLen, 'A'); |
| |
| wchar_t srcw[kSrcLen]; |
| std::fill_n(srcw, kSrcLen, 'A'); |
| |
| char16_t src16[kSrcLen]; |
| std::fill_n(src16, kSrcLen, 'A'); |
| |
| for (int i = 1; i < 3; i++) { |
| src[kSrcLen - i] = 0; |
| std::string out; |
| SStringPrintf(&out, "%s", src); |
| EXPECT_STREQ(src, out.c_str()); |
| |
| #if BUILDFLAG(IS_WIN) |
| srcw[kSrcLen - i] = 0; |
| std::wstring outw; |
| SStringPrintf(&outw, L"%ls", srcw); |
| EXPECT_STREQ(srcw, outw.c_str()); |
| |
| src16[kSrcLen - i] = 0; |
| std::u16string out16; |
| SStringPrintf(&out16, u"%ls", src16); |
| // EXPECT_STREQ does not support const char16_t* strings yet. |
| // Dispatch to the const wchar_t* overload instead. |
| EXPECT_STREQ(reinterpret_cast<const wchar_t*>(src16), |
| reinterpret_cast<const wchar_t*>(out16.c_str())); |
| #endif |
| } |
| } |
| |
| // Test very large sprintfs that will cause the buffer to grow. |
| TEST(StringPrintfTest, Grow) { |
| char src[1026]; |
| for (auto& i : src) |
| i = 'A'; |
| src[1025] = 0; |
| |
| const char fmt[] = "%sB%sB%sB%sB%sB%sB%s"; |
| |
| std::string out; |
| SStringPrintf(&out, fmt, src, src, src, src, src, src, src); |
| |
| const int kRefSize = 320000; |
| char* ref = new char[kRefSize]; |
| #if BUILDFLAG(IS_WIN) |
| sprintf_s(ref, kRefSize, fmt, src, src, src, src, src, src, src); |
| #elif BUILDFLAG(IS_POSIX) || BUILDFLAG(IS_FUCHSIA) |
| snprintf(ref, kRefSize, fmt, src, src, src, src, src, src, src); |
| #endif |
| |
| EXPECT_STREQ(ref, out.c_str()); |
| delete[] ref; |
| } |
| |
| TEST(StringPrintfTest, StringAppendV) { |
| std::string out; |
| StringAppendVTestHelper(&out, "%d foo %s", 1, "bar"); |
| EXPECT_EQ("1 foo bar", out); |
| |
| #if BUILDFLAG(IS_WIN) |
| std::wstring outw; |
| StringAppendVTestHelper(&outw, L"%d foo %ls", 1, L"bar"); |
| EXPECT_EQ(L"1 foo bar", outw); |
| |
| std::u16string out16; |
| StringAppendVTestHelper(&out16, u"%d foo %ls", 1, u"bar"); |
| EXPECT_EQ(u"1 foo bar", out16); |
| #endif |
| } |
| |
| // Test the boundary condition for the size of the string_util's |
| // internal buffer. |
| TEST(StringPrintfTest, GrowBoundary) { |
| const int kStringUtilBufLen = 1024; |
| // Our buffer should be one larger than the size of StringAppendVT's stack |
| // buffer. |
| // And need extra one for NULL-terminator. |
| const int kBufLen = kStringUtilBufLen + 1 + 1; |
| char src[kBufLen]; |
| for (int i = 0; i < kBufLen - 1; ++i) |
| src[i] = 'a'; |
| src[kBufLen - 1] = 0; |
| |
| std::string out; |
| SStringPrintf(&out, "%s", src); |
| |
| EXPECT_STREQ(src, out.c_str()); |
| } |
| |
| #if BUILDFLAG(IS_WIN) |
| TEST(StringPrintfTest, Invalid) { |
| wchar_t invalid[2]; |
| invalid[0] = 0xffff; |
| invalid[1] = 0; |
| |
| std::wstring out; |
| SStringPrintf(&out, L"%ls", invalid); |
| EXPECT_STREQ(invalid, out.c_str()); |
| } |
| #endif |
| |
| // Test that StringPrintf and StringAppendV do not change errno. |
| TEST(StringPrintfTest, StringPrintfErrno) { |
| errno = 1; |
| EXPECT_EQ("", StringPrintf("%s", "")); |
| EXPECT_EQ(1, errno); |
| std::string out; |
| StringAppendVTestHelper(&out, "%d foo %s", 1, "bar"); |
| EXPECT_EQ(1, errno); |
| } |
| |
| } // namespace base |