blob: 0138bf16f9d08ac1235e29802c423eb232e74305 [file] [log] [blame] [edit]
/*
* Copyright (C) 2018 The Android Open Source Project
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
#include "perfetto/ext/base/string_utils.h"
#include <optional>
#include "test/gtest_and_gmock.h"
namespace perfetto {
namespace base {
namespace {
template <size_t N>
struct UninitializedBuf {
UninitializedBuf() { memset(data, '?', sizeof(data)); }
operator char*() { return data; }
char data[N];
};
using testing::ElementsAre;
TEST(StringUtilsTest, Lowercase) {
EXPECT_EQ(Lowercase('A'), 'a');
EXPECT_EQ(Lowercase('a'), 'a');
EXPECT_EQ(Lowercase('Z'), 'z');
EXPECT_EQ(Lowercase('z'), 'z');
EXPECT_EQ(Lowercase('!'), '!');
}
TEST(StringUtilsTest, Uppercase) {
EXPECT_EQ(Uppercase('A'), 'A');
EXPECT_EQ(Uppercase('a'), 'A');
EXPECT_EQ(Uppercase('Z'), 'Z');
EXPECT_EQ(Uppercase('z'), 'Z');
EXPECT_EQ(Uppercase('!'), '!');
}
TEST(StringUtilsTest, CStringToUInt32) {
EXPECT_EQ(CStringToUInt32("0"), std::make_optional<uint32_t>(0U));
EXPECT_EQ(CStringToUInt32("1"), std::make_optional<uint32_t>(1U));
EXPECT_EQ(CStringToUInt32("42"), std::make_optional<uint32_t>(42U));
EXPECT_EQ(CStringToUInt32(""), std::nullopt);
EXPECT_EQ(CStringToUInt32("!?"), std::nullopt);
EXPECT_EQ(CStringToUInt32("abc"), std::nullopt);
EXPECT_EQ(CStringToUInt32("123 abc"), std::nullopt);
}
TEST(StringUtilsTest, CStringToInt32) {
EXPECT_EQ(CStringToInt32("0"), std::make_optional<int32_t>(0));
EXPECT_EQ(CStringToInt32("1"), std::make_optional<int32_t>(1));
EXPECT_EQ(CStringToInt32("-42"), std::make_optional<int32_t>(-42));
EXPECT_EQ(CStringToInt32(""), std::nullopt);
EXPECT_EQ(CStringToInt32("!?"), std::nullopt);
EXPECT_EQ(CStringToInt32("abc"), std::nullopt);
EXPECT_EQ(CStringToInt32("123 abc"), std::nullopt);
}
TEST(StringUtilsTest, CStringToDouble) {
EXPECT_DOUBLE_EQ(CStringToDouble("0").value(), 0l);
EXPECT_DOUBLE_EQ(CStringToDouble("1").value(), 1l);
EXPECT_DOUBLE_EQ(CStringToDouble("-42").value(), -42l);
EXPECT_DOUBLE_EQ(CStringToDouble("-42.5").value(), -42.5l);
EXPECT_EQ(CStringToDouble(""), std::nullopt);
EXPECT_EQ(CStringToDouble("!?"), std::nullopt);
EXPECT_EQ(CStringToDouble("abc"), std::nullopt);
EXPECT_EQ(CStringToDouble("123 abc"), std::nullopt);
}
TEST(StringUtilsTest, StringToUInt32) {
EXPECT_EQ(StringToUInt32("0"), std::make_optional<uint32_t>(0U));
EXPECT_EQ(StringToUInt32("1"), std::make_optional<uint32_t>(1U));
EXPECT_EQ(StringToUInt32("42"), std::make_optional<uint32_t>(42U));
EXPECT_EQ(StringToUInt32("a", 16), std::make_optional<uint32_t>(10U));
EXPECT_EQ(StringToUInt32("fffffff0", 16),
std::make_optional<uint32_t>(0xfffffff0));
EXPECT_EQ(StringToUInt32(""), std::nullopt);
EXPECT_EQ(StringToUInt32("!?"), std::nullopt);
EXPECT_EQ(StringToUInt32("abc"), std::nullopt);
EXPECT_EQ(StringToUInt32("123 abc"), std::nullopt);
EXPECT_EQ(StringToUInt32("beefz", 16), std::nullopt);
}
TEST(StringUtilsTest, StringToInt32) {
EXPECT_EQ(StringToInt32("0"), std::make_optional<int32_t>(0));
EXPECT_EQ(StringToInt32("1"), std::make_optional<int32_t>(1));
EXPECT_EQ(StringToInt32("+42"), std::make_optional<int32_t>(42));
EXPECT_EQ(StringToInt32("+0042"), std::make_optional<int32_t>(42));
EXPECT_EQ(StringToInt32("-42"), std::make_optional<int32_t>(-42));
EXPECT_EQ(StringToInt32("42", 16), std::make_optional<int32_t>(0x42));
EXPECT_EQ(StringToInt32("7ffffffe", 16),
std::make_optional<int32_t>(0x7ffffffe));
EXPECT_EQ(StringToInt32(""), std::nullopt);
EXPECT_EQ(StringToInt32("!?"), std::nullopt);
EXPECT_EQ(StringToInt32("abc"), std::nullopt);
EXPECT_EQ(StringToInt32("123 abc"), std::nullopt);
EXPECT_EQ(StringToInt32("beefz", 16), std::nullopt);
}
TEST(StringUtilsTest, StringToUInt64) {
EXPECT_EQ(StringToUInt64("0"), std::make_optional<uint64_t>(0u));
EXPECT_EQ(StringToUInt64("1"), std::make_optional<uint64_t>(1u));
EXPECT_EQ(StringToUInt64("5000000000"),
std::make_optional<uint64_t>(5000000000ULL));
EXPECT_EQ(StringToUInt64("7ffffffffffffffe", 16),
std::make_optional<uint64_t>(0x7ffffffffffffffeULL));
EXPECT_EQ(StringToUInt64("9ffffffffffffffe", 16),
std::make_optional<uint64_t>(0x9ffffffffffffffeULL));
EXPECT_EQ(StringToUInt64(""), std::nullopt);
EXPECT_EQ(StringToUInt64("abc"), std::nullopt);
EXPECT_EQ(StringToUInt64("beefz", 16), std::nullopt);
}
TEST(StringUtilsTest, StringToInt64) {
EXPECT_EQ(StringToInt64("0"), std::make_optional<int64_t>(0));
EXPECT_EQ(StringToInt64("1"), std::make_optional<int64_t>(1));
EXPECT_EQ(StringToInt64("-5000000000"),
std::make_optional<int64_t>(-5000000000LL));
EXPECT_EQ(StringToInt64("5000000000"),
std::make_optional<int64_t>(5000000000LL));
EXPECT_EQ(StringToInt64("7ffffffffffffffe", 16),
std::make_optional<int64_t>(0x7ffffffffffffffeLL));
EXPECT_EQ(StringToInt64("9ffffffe", 16),
std::make_optional<int64_t>(0x9ffffffeLL));
EXPECT_EQ(StringToInt64(""), std::nullopt);
EXPECT_EQ(StringToInt64("abc"), std::nullopt);
EXPECT_EQ(StringToInt64("beefz", 16), std::nullopt);
}
TEST(StringUtilsTest, StringToDouble) {
EXPECT_DOUBLE_EQ(StringToDouble("0").value(), 0l);
EXPECT_DOUBLE_EQ(StringToDouble("1").value(), 1l);
EXPECT_DOUBLE_EQ(StringToDouble("-42").value(), -42l);
EXPECT_DOUBLE_EQ(StringToDouble("-42.5").value(), -42.5l);
EXPECT_DOUBLE_EQ(StringToDouble("0.5").value(), .5l);
EXPECT_DOUBLE_EQ(StringToDouble(".5").value(), .5l);
EXPECT_EQ(StringToDouble(""), std::nullopt);
EXPECT_EQ(StringToDouble("!?"), std::nullopt);
EXPECT_EQ(StringToDouble("abc"), std::nullopt);
EXPECT_EQ(StringToDouble("123 abc"), std::nullopt);
EXPECT_EQ(StringToDouble("124,456"), std::nullopt);
EXPECT_EQ(StringToDouble("4 2"), std::nullopt);
EXPECT_EQ(StringToDouble(" - 42"), std::nullopt);
}
TEST(StringUtilsTest, StartsWith) {
EXPECT_TRUE(StartsWith("", ""));
EXPECT_TRUE(StartsWith("abc", ""));
EXPECT_TRUE(StartsWith("abc", "a"));
EXPECT_TRUE(StartsWith("abc", "ab"));
EXPECT_TRUE(StartsWith("abc", "abc"));
EXPECT_FALSE(StartsWith("abc", "abcd"));
EXPECT_FALSE(StartsWith("aa", "ab"));
EXPECT_FALSE(StartsWith("", "ab"));
}
TEST(StringUtilsTest, StartsWithAny) {
EXPECT_FALSE(StartsWithAny("", {"a", "b"}));
EXPECT_FALSE(StartsWithAny("abcd", {}));
EXPECT_FALSE(StartsWithAny("", {}));
EXPECT_TRUE(StartsWithAny("abcd", {"ac", "ab"}));
EXPECT_FALSE(StartsWithAny("abcd", {"bc", "ac"}));
}
TEST(StringUtilsTest, EndsWith) {
EXPECT_TRUE(EndsWith("", ""));
EXPECT_TRUE(EndsWith("abc", ""));
EXPECT_TRUE(EndsWith("abc", "c"));
EXPECT_TRUE(EndsWith("abc", "bc"));
EXPECT_TRUE(EndsWith("abc", "abc"));
EXPECT_FALSE(EndsWith("bcd", "abcd"));
EXPECT_FALSE(EndsWith("abc", "abd"));
EXPECT_FALSE(EndsWith("", "c"));
}
TEST(StringUtilsTest, ToHex) {
EXPECT_EQ(ToHex(""), "");
EXPECT_EQ(ToHex("abc123"), "616263313233");
}
TEST(StringUtilsTest, IntToHex) {
EXPECT_EQ(IntToHexString(0), "0x00");
EXPECT_EQ(IntToHexString(1), "0x01");
EXPECT_EQ(IntToHexString(16), "0x10");
EXPECT_EQ(IntToHexString(4294967295), "0xffffffff");
}
TEST(StringUtilsTest, Uint64ToHex) {
EXPECT_EQ(Uint64ToHexString(0), "0x0");
EXPECT_EQ(Uint64ToHexString(1), "0x1");
EXPECT_EQ(Uint64ToHexString(16), "0x10");
EXPECT_EQ(Uint64ToHexString(18446744073709551615UL), "0xffffffffffffffff");
}
TEST(StringUtilsTest, Uint64ToHexNoPrefix) {
EXPECT_EQ(Uint64ToHexStringNoPrefix(0), "0");
EXPECT_EQ(Uint64ToHexStringNoPrefix(1), "1");
EXPECT_EQ(Uint64ToHexStringNoPrefix(16), "10");
EXPECT_EQ(Uint64ToHexStringNoPrefix(18446744073709551615UL),
"ffffffffffffffff");
}
TEST(StringUtilsTest, CaseInsensitiveEqual) {
EXPECT_TRUE(CaseInsensitiveEqual("", ""));
EXPECT_TRUE(CaseInsensitiveEqual("abc", "abc"));
EXPECT_TRUE(CaseInsensitiveEqual("ABC", "abc"));
EXPECT_TRUE(CaseInsensitiveEqual("abc", "ABC"));
EXPECT_FALSE(CaseInsensitiveEqual("abc", "AB"));
EXPECT_FALSE(CaseInsensitiveEqual("ab", "ABC"));
}
TEST(StringUtilsTest, SplitString) {
EXPECT_THAT(SplitString("", ":"), ElementsAre());
EXPECT_THAT(SplitString("a:b:c", ":"), ElementsAre("a", "b", "c"));
EXPECT_THAT(SplitString("a::b::c", "::"), ElementsAre("a", "b", "c"));
EXPECT_THAT(SplitString("::::a::b::::c::", "::"), ElementsAre("a", "b", "c"));
EXPECT_THAT(SplitString("abc", ":"), ElementsAre("abc"));
EXPECT_THAT(SplitString("abc", "::"), ElementsAre("abc"));
EXPECT_THAT(SplitString("abc", ":"), ElementsAre("abc"));
EXPECT_THAT(SplitString("abc", "::"), ElementsAre("abc"));
}
TEST(StringUtilsTest, Strip) {
EXPECT_EQ(StripPrefix("abc", ""), "abc");
EXPECT_EQ(StripPrefix("abc", "a"), "bc");
EXPECT_EQ(StripPrefix("abc", "ab"), "c");
EXPECT_EQ(StripPrefix("abc", "abc"), "");
EXPECT_EQ(StripPrefix("abc", "abcd"), "abc");
EXPECT_EQ(StripSuffix("abc", ""), "abc");
EXPECT_EQ(StripSuffix("abc", "c"), "ab");
EXPECT_EQ(StripSuffix("abc", "bc"), "a");
EXPECT_EQ(StripSuffix("abc", "abc"), "");
EXPECT_EQ(StripSuffix("abc", "ebcd"), "abc");
EXPECT_EQ(StripChars("foobar", "", '_'), "foobar");
EXPECT_EQ(StripChars("foobar", "x", '_'), "foobar");
EXPECT_EQ(StripChars("foobar", "f", '_'), "_oobar");
EXPECT_EQ(StripChars("foobar", "o", '_'), "f__bar");
EXPECT_EQ(StripChars("foobar", "oa", '_'), "f__b_r");
EXPECT_EQ(StripChars("foobar", "fbr", '_'), "_oo_a_");
EXPECT_EQ(StripChars("foobar", "froab", '_'), "______");
}
TEST(StringUtilsTest, TrimWhitespace) {
EXPECT_EQ(TrimWhitespace(""), "");
EXPECT_EQ(TrimWhitespace(" "), "");
EXPECT_EQ(TrimWhitespace("\t\n"), "");
EXPECT_EQ(TrimWhitespace("\tx\n\n"), "x");
EXPECT_EQ(TrimWhitespace("\tx\n"), "x");
EXPECT_EQ(TrimWhitespace("\tx\nx\n"), "x\nx");
}
TEST(StringUtilsTest, Contains) {
EXPECT_TRUE(Contains("", ""));
EXPECT_TRUE(Contains("abc", ""));
EXPECT_TRUE(Contains("abc", "a"));
EXPECT_TRUE(Contains("abc", "b"));
EXPECT_TRUE(Contains("abc", "c"));
EXPECT_TRUE(Contains("abc", "ab"));
EXPECT_TRUE(Contains("abc", "bc"));
EXPECT_TRUE(Contains("abc", "abc"));
EXPECT_FALSE(Contains("abc", "d"));
EXPECT_FALSE(Contains("abc", "ac"));
EXPECT_FALSE(Contains("abc", "abcd"));
EXPECT_FALSE(Contains("", "a"));
EXPECT_FALSE(Contains("", "abc"));
}
TEST(StringUtilsTest, Find) {
EXPECT_EQ(Find("", ""), 0u);
EXPECT_EQ(Find("", "abc"), 0u);
EXPECT_EQ(Find("a", "abc"), 0u);
EXPECT_EQ(Find("b", "abc"), 1u);
EXPECT_EQ(Find("c", "abc"), 2u);
EXPECT_EQ(Find("ab", "abc"), 0u);
EXPECT_EQ(Find("bc", "abc"), 1u);
EXPECT_EQ(Find("abc", "abc"), 0u);
EXPECT_EQ(Find("d", "abc"), std::string::npos);
EXPECT_EQ(Find("ac", "abc"), std::string::npos);
EXPECT_EQ(Find("abcd", "abc"), std::string::npos);
EXPECT_EQ(Find("a", ""), std::string::npos);
EXPECT_EQ(Find("abc", ""), std::string::npos);
}
TEST(StringUtilsTest, ReplaceAll) {
EXPECT_EQ(ReplaceAll("", "a", ""), "");
EXPECT_EQ(ReplaceAll("", "a", "b"), "");
EXPECT_EQ(ReplaceAll("a", "a", "b"), "b");
EXPECT_EQ(ReplaceAll("aaaa", "a", "b"), "bbbb");
EXPECT_EQ(ReplaceAll("aaaa", "aa", "b"), "bb");
EXPECT_EQ(ReplaceAll("aa", "aa", "bb"), "bb");
EXPECT_EQ(ReplaceAll("aa", "a", "bb"), "bbbb");
EXPECT_EQ(ReplaceAll("abc", "a", "b"), "bbc");
EXPECT_EQ(ReplaceAll("abc", "c", "b"), "abb");
EXPECT_EQ(ReplaceAll("abc", "c", "bbb"), "abbbb");
}
TEST(StringUtilsTest, StringCopy) {
// Nothing should be written when |dst_size| = 0.
{
char dst[2] = {42, 43};
StringCopy(dst, "12345", 0);
EXPECT_EQ(42, dst[0]);
EXPECT_EQ(43, dst[1]);
}
// Nominal case, len(src) < sizeof(dst).
{
UninitializedBuf<10> dst;
StringCopy(dst, "1234567", sizeof(dst));
EXPECT_STREQ(dst, "1234567");
}
// Edge case where we perfectly fit including the \0.
{
UninitializedBuf<8> dst;
StringCopy(dst, "1234567", sizeof(dst));
EXPECT_STREQ(dst, "1234567");
}
// Edge case where |dst| is smaller by one char.
{
UninitializedBuf<8> dst;
StringCopy(dst, "12345678", sizeof(dst));
EXPECT_STREQ(dst, "1234567");
}
// Case when |dst| is smaller than |src|.
{
UninitializedBuf<3> dst;
StringCopy(dst, "12345678", sizeof(dst));
EXPECT_STREQ(dst, "12");
}
}
TEST(StringUtilsTest, SprintfTrunc) {
{
UninitializedBuf<3> dst;
ASSERT_EQ(0u, SprintfTrunc(dst, sizeof(dst), "%s", ""));
EXPECT_STREQ(dst, "");
}
{
char dst[3]{'O', 'K', '\0'};
ASSERT_EQ(0u, SprintfTrunc(dst, 0, "whatever"));
EXPECT_STREQ(dst, "OK"); // dst_size == 0 shouldn't touch the buffer.
}
{
UninitializedBuf<1> dst;
ASSERT_EQ(0u, SprintfTrunc(dst, sizeof(dst), "whatever"));
EXPECT_STREQ(dst, "");
}
{
UninitializedBuf<3> dst;
ASSERT_EQ(1u, SprintfTrunc(dst, sizeof(dst), "1"));
EXPECT_STREQ(dst, "1");
}
{
UninitializedBuf<3> dst;
ASSERT_EQ(2u, SprintfTrunc(dst, sizeof(dst), "12"));
EXPECT_STREQ(dst, "12");
}
{
UninitializedBuf<3> dst;
ASSERT_EQ(2u, SprintfTrunc(dst, sizeof(dst), "123"));
EXPECT_STREQ(dst, "12");
}
{
UninitializedBuf<3> dst;
ASSERT_EQ(2u, SprintfTrunc(dst, sizeof(dst), "1234"));
EXPECT_STREQ(dst, "12");
}
{
UninitializedBuf<11> dst;
ASSERT_EQ(10u, SprintfTrunc(dst, sizeof(dst), "a %d b %s", 42, "foo"));
EXPECT_STREQ(dst, "a 42 b foo");
}
}
TEST(StringUtilsTest, StackString) {
{
StackString<1> s("123");
EXPECT_EQ(0u, s.len());
EXPECT_STREQ("", s.c_str());
}
{
StackString<4> s("123");
EXPECT_EQ(3u, s.len());
EXPECT_STREQ("123", s.c_str());
EXPECT_EQ(s.ToStdString(), std::string(s.c_str()));
EXPECT_EQ(s.string_view().ToStdString(), s.ToStdString());
}
{
StackString<3> s("123");
EXPECT_EQ(2u, s.len());
EXPECT_STREQ("12", s.c_str());
EXPECT_EQ(s.ToStdString(), std::string(s.c_str()));
EXPECT_EQ(s.string_view().ToStdString(), s.ToStdString());
}
{
StackString<11> s("foo %d %s", 42, "bar!!!OVERFLOW");
EXPECT_EQ(10u, s.len());
EXPECT_STREQ("foo 42 bar", s.c_str());
EXPECT_EQ(s.ToStdString(), std::string(s.c_str()));
EXPECT_EQ(s.string_view().ToStdString(), s.ToStdString());
}
}
TEST(FindLineTest, InvalidOffset1) {
std::string str = "abc\ndef\n\nghi";
uint32_t offset = 3;
auto error = FindLineWithOffset(base::StringView(str), offset);
EXPECT_FALSE(error.has_value());
}
TEST(FindLineTest, InvalidOffset2) {
std::string str = "abc\ndef\n\nghi";
uint32_t offset = 8;
auto error = FindLineWithOffset(base::StringView(str), offset);
EXPECT_FALSE(error.has_value());
}
TEST(FindLineTest, FirstCharacter) {
std::string str = "abc\ndef\n\nghi";
uint32_t offset = 0;
auto error = FindLineWithOffset(base::StringView(str), offset);
EXPECT_TRUE(error.has_value());
ASSERT_EQ(error.value().line_num, 1ul);
ASSERT_EQ(error.value().line_offset, 0ul);
ASSERT_EQ(error.value().line, "abc");
}
TEST(FindLineTest, StandardCheck) {
std::string str = "abc\ndef\n\nghi";
uint32_t offset = 5;
auto error = FindLineWithOffset(base::StringView(str), offset);
EXPECT_TRUE(error.has_value());
ASSERT_EQ(error.value().line_num, 2ul);
ASSERT_EQ(error.value().line_offset, 1ul);
ASSERT_EQ(error.value().line, "def");
}
TEST(FindLineTest, TwoBreakLines) {
std::string str = "abc\ndef\n\nghi";
uint32_t offset = 10;
auto error = FindLineWithOffset(base::StringView(str), offset);
EXPECT_TRUE(error.has_value());
ASSERT_EQ(error.value().line_num, 4ul);
ASSERT_EQ(error.value().line_offset, 1ul);
ASSERT_EQ(error.value().line, "ghi");
}
TEST(FindLineTest, EndsWithBreakLine) {
std::string str = "abc\ndef\n\nghi\n";
uint32_t offset = 10;
auto error = FindLineWithOffset(base::StringView(str), offset);
EXPECT_TRUE(error.has_value());
ASSERT_EQ(error.value().line_num, 4ul);
ASSERT_EQ(error.value().line_offset, 1ul);
ASSERT_EQ(error.value().line, "ghi");
}
} // namespace
} // namespace base
} // namespace perfetto