blob: 28415f49b6968e752a0a2ec2cb435bd2f05f0b23 [file] [log] [blame]
// Copyright 2019 The Fuchsia Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include <assert.h>
#include <lib/fit/string_view.h>
#include <cstring>
#include <limits>
#include <sstream>
#include <zxtest/zxtest.h>
#if !defined(NDEBUG) && defined(__Fuchsia__)
#define DEATH_TESTS 1
#else
#define DEATH_TESTS 0
#endif
namespace fit {
namespace {
TEST(StringViewTest, CreateFromStdString) {
std::string str = "1";
fit::string_view v_str(str);
EXPECT_FALSE(v_str.empty());
EXPECT_EQ(str.data(), v_str.data());
EXPECT_EQ(str.length(), v_str.length());
}
TEST(StringViewTest, CreateFromCArray) {
constexpr char kStr[] = "1";
fit::string_view v_str(kStr);
EXPECT_FALSE(v_str.empty());
EXPECT_EQ(kStr, v_str.data());
EXPECT_EQ(std::strlen(kStr), v_str.length());
}
TEST(StringViewTest, CreateFromConstChar) {
const char* kStr = "1";
fit::string_view v_str(kStr);
EXPECT_FALSE(v_str.empty());
EXPECT_EQ(kStr, v_str.data());
EXPECT_EQ(std::strlen(kStr), v_str.length());
}
TEST(StringViewTest, CreateFromStringView) {
fit::string_view str_view("12345");
fit::string_view v_str(str_view);
EXPECT_FALSE(v_str.empty());
EXPECT_EQ(str_view.data(), v_str.data());
EXPECT_EQ(str_view.length(), v_str.length());
}
TEST(StringViewTest, CreateFromConstexprStringView) {
constexpr fit::string_view kLiteral = "12345";
fit::string_view v_str(kLiteral);
EXPECT_EQ(kLiteral.data(), v_str.data());
EXPECT_EQ(kLiteral.length(), v_str.length());
}
TEST(StringViewTest, CreateFromConstexprStringViewConstructor) {
constexpr fit::string_view kLiteral("12345");
EXPECT_EQ(5, kLiteral.size());
EXPECT_EQ(5, kLiteral.length());
}
TEST(StringViewTest, CreateFromStringViewLiteral) {
constexpr fit::string_view kLiteral = "12345"_sv;
EXPECT_EQ(5, kLiteral.size());
EXPECT_EQ(5, kLiteral.length());
}
TEST(StringViewTest, SizeIsSameAsLength) {
constexpr fit::string_view kLiteral = "12345";
EXPECT_EQ(5, kLiteral.size());
EXPECT_EQ(5, kLiteral.length());
}
TEST(StringViewTest, ArrayAccessOperator) {
// Need static duration to enforce subobject constexpr, otherwise it is not allowed.
constexpr static char const kLiteral[] = "12345";
constexpr fit::string_view kFitLiteral(kLiteral);
for (std::size_t i = 0; i < kFitLiteral.size(); ++i) {
EXPECT_EQ(kLiteral[i], kFitLiteral[i], "Array access returned wrong value.");
EXPECT_EQ(&kLiteral[i], &kFitLiteral[i], "Array access returned value at different address.");
}
}
TEST(StringViewTest, BeginPointsToFirstElement) {
constexpr fit::string_view kLiteral = "12345";
EXPECT_EQ(&kLiteral[0], &(*kLiteral.begin()));
EXPECT_EQ(&kLiteral[4], &(*kLiteral.rbegin()));
}
TEST(StringViewTest, EndPointsOnePastLastElement) {
constexpr fit::string_view kLiteral = "12345";
EXPECT_EQ(&kLiteral[4], &(*(kLiteral.end() - 1)));
EXPECT_EQ(&kLiteral[0], &(*(kLiteral.rend() - 1)));
}
TEST(StringViewTest, EndPointsPastLastElement) {
constexpr fit::string_view kLiteral = "12345";
EXPECT_EQ(kLiteral.begin() + 5, kLiteral.end());
EXPECT_EQ(kLiteral.rbegin() + 5, kLiteral.rend());
}
TEST(StringViewTest, WhenEmptyBeginIsSameAsEnd) {
constexpr fit::string_view kLiteral = "";
EXPECT_EQ(kLiteral.begin(), kLiteral.end());
EXPECT_EQ(kLiteral.rbegin(), kLiteral.rend());
}
TEST(StringViewTest, FrontReturnsRefToFirstElement) {
constexpr fit::string_view kLiteral = "12345";
EXPECT_EQ(&(*kLiteral.begin()), &kLiteral.front());
}
TEST(StringViewTest, BackReturnsRefToLastElement) {
constexpr fit::string_view kLiteral = "12345";
EXPECT_EQ(&(*(kLiteral.begin() + 4)), &kLiteral.back());
}
TEST(StringViewTest, EmptyIsTrueForEmptyString) {
constexpr fit::string_view kStr;
ASSERT_TRUE(kStr.empty());
ASSERT_EQ(0, kStr.size());
ASSERT_EQ(0, kStr.length());
}
TEST(StringViewTest, AtReturnsElementAtIndex) {
// Need static duration to enforce subobject constexpr, otherwise it is not allowed.
constexpr static char const kLiteral[] = "12345";
constexpr fit::string_view kFitLiteral(kLiteral);
for (std::size_t i = 0; i < kFitLiteral.size(); ++i) {
EXPECT_EQ(kLiteral[i], kFitLiteral.at(i), "Array access returned wrong value.");
EXPECT_EQ(&kLiteral[i], &kFitLiteral.at(i),
"Array access returned value at different address.");
}
}
#if DEATH_TESTS
TEST(StringViewTest, AtThrowsExceptionWhenIndexIsOOR) {
ASSERT_DEATH([] {
constexpr fit::string_view kFitLiteral("12345");
kFitLiteral.at(5);
});
}
#endif
// Even though we use a custom compare implementation, because we lack a constexpr compare
// function, we use this test to verify that the expectations are equivalent.
TEST(StringViewTest, CompareVerification) {
constexpr fit::string_view kStr1 = "1234";
// Same string
{
constexpr fit::string_view kStr2 = "1234";
constexpr fit::string_view kStr3 = "01234";
EXPECT_EQ(0, fit::string_view::traits_type::compare(kStr1.data(), kStr2.data(), 4));
EXPECT_EQ(0, kStr1.compare(kStr2));
EXPECT_EQ(0, kStr3.compare(1, kStr3.length() - 1, kStr2));
EXPECT_EQ(0, kStr1.compare(1, kStr1.length() - 2, kStr2, 1, kStr2.length() - 2));
EXPECT_EQ(0, kStr1.compare("1234"));
EXPECT_EQ(0, kStr1.compare(1, kStr1.length() - 1, "234"));
EXPECT_EQ(0, kStr1.compare(2, kStr1.length() - 2, "234", 1, 2));
}
// Same Length higher character
{
constexpr fit::string_view kStr2 = "1235";
EXPECT_LT(fit::string_view::traits_type::compare(kStr1.data(), kStr2.data(), 4), 0);
EXPECT_LT(kStr1.compare(kStr2), 0);
EXPECT_LT(kStr1.compare(0, kStr1.length(), kStr2), 0);
EXPECT_LT(kStr1.compare(1, kStr1.length() - 2, kStr2, 1, kStr2.length() - 1), 0);
EXPECT_LT(kStr1.compare("1235"), 0);
EXPECT_LT(kStr1.compare(1, kStr1.length() - 1, "235"), 0);
EXPECT_LT(kStr1.compare(1, kStr1.length() - 2, "1235", 1, 3), 0);
}
// Same Length lower character
{
constexpr fit::string_view kStr2 = "1232";
EXPECT_GT(fit::string_view::traits_type::compare(kStr1.data(), kStr2.data(), 4), 0);
EXPECT_GT(kStr1.compare(kStr2), 0);
EXPECT_GT(kStr2.compare(1, kStr2.length() - 1, kStr1), 0);
EXPECT_GT(kStr1.compare(1, kStr1.length() - 1, kStr2, 1, kStr2.length() - 1), 0);
EXPECT_GT(kStr1.compare("1232"), 0);
EXPECT_GT(kStr1.compare(1, kStr1.length() - 1, "232"), 0);
EXPECT_GT(kStr1.compare(1, kStr1.length() - 2, "22", 1, kStr2.length() - 2), 0);
}
// Greater Length
{
constexpr fit::string_view kStr2 = "12345";
constexpr fit::string_view kStr3 = "2345";
EXPECT_LT(kStr1.compare(kStr2), 0);
EXPECT_LT(kStr1.compare(1, kStr1.length() - 1, kStr3), 0);
EXPECT_LT(kStr1.compare(1, kStr1.length() - 1, kStr2, 1, kStr2.length() - 1), 0);
EXPECT_LT(kStr1.compare(kStr2.data()), 0);
EXPECT_LT(kStr1.compare(1, kStr1.length() - 1, kStr3.data()), 0);
EXPECT_LT(kStr1.compare(1, kStr1.length() - 1, kStr2.data(), 1, kStr2.length() - 1), 0);
}
// Shorter Length
{
constexpr fit::string_view kStr2 = "123";
constexpr fit::string_view kStr3 = "23";
EXPECT_GT(kStr1.compare(kStr2), 0);
EXPECT_GT(kStr1.compare(1, kStr1.length() - 1, kStr3), 0);
EXPECT_GT(kStr1.compare(1, kStr1.length() - 1, kStr2, 1, kStr2.length() - 1), 0);
EXPECT_GT(kStr1.compare(kStr2.data()), 0);
EXPECT_GT(kStr1.compare(1, kStr1.length() - 1, kStr3.data()), 0);
EXPECT_GT(kStr1.compare(1, kStr1.length() - 1, kStr2.data(), 1, kStr2.length() - 1), 0);
}
}
// Check that they calls are equivalent to what the standard expects.
TEST(StringViewTest, CompareOverloadCheck) {
constexpr fit::string_view kString1 = "123";
constexpr fit::string_view kString2 = "1234";
{
fit::string_view expected = kString1.substr(1, 2);
EXPECT_EQ(kString1.substr(1, 2).compare(expected), kString1.compare(1, 2, expected));
}
{
EXPECT_EQ(kString1.substr(1, 2).compare(kString2.substr(1, 2)),
kString1.compare(1, 2, kString2, 1, 2));
}
{ EXPECT_EQ(kString1.compare(fit::string_view("123")), kString1.compare("123")); }
{
EXPECT_EQ(kString1.substr(1, 2).compare(fit::string_view("123")),
kString1.compare(1, 2, "123"));
}
{
EXPECT_EQ(kString1.substr(1, 2).compare(fit::string_view("123")),
kString1.compare(1, 2, "123"));
}
}
TEST(StringViewTest, OperatorEq) {
constexpr fit::string_view kStrView = "Self1234";
EXPECT_EQ(kStrView, kStrView);
EXPECT_EQ(kStrView, fit::string_view("Self1234"));
EXPECT_EQ(kStrView, fit::string_view("Self12345").substr(0, kStrView.length()));
EXPECT_EQ(kStrView, "Self1234");
EXPECT_EQ("Self1234", kStrView);
}
TEST(StringViewTest, OperatorNe) {
constexpr fit::string_view kStrView = "Self1234";
EXPECT_NE(kStrView, fit::string_view());
EXPECT_NE(kStrView, fit::string_view("Self12345"));
EXPECT_NE(kStrView, "Self12345");
EXPECT_NE("Self12345", kStrView);
}
TEST(StringViewTest, OperatorLess) {
constexpr fit::string_view kStrView = "Self1234";
EXPECT_TRUE(kStrView < "Self12345");
EXPECT_TRUE("Self123" < kStrView);
EXPECT_TRUE(kStrView < fit::string_view("Self12345"));
}
TEST(StringViewTest, OperatorLessOrEq) {
constexpr fit::string_view kStrView = "Self1234";
EXPECT_TRUE(kStrView <= "Self12345");
EXPECT_TRUE("Self123" <= kStrView);
EXPECT_TRUE(kStrView <= fit::string_view("Self12345"));
EXPECT_TRUE(kStrView <= fit::string_view("Self1234"));
}
TEST(StringViewTest, OperatorGreater) {
constexpr fit::string_view kStrView = "Self1234";
EXPECT_TRUE(kStrView > "Self123");
EXPECT_TRUE("Self12345" > kStrView);
EXPECT_TRUE(kStrView > fit::string_view("Self123"));
}
TEST(StringViewTest, OperatorGreaterOrEq) {
constexpr fit::string_view kStrView = "Self1234";
EXPECT_TRUE(kStrView >= "Self123");
EXPECT_TRUE("Self12345" >= kStrView);
EXPECT_TRUE(kStrView >= fit::string_view("Self123"));
EXPECT_TRUE(kStrView >= fit::string_view("Self1234"));
}
TEST(StringViewTest, RemovePrefix) {
constexpr fit::string_view kPrefixWithSuffix = "PrefixSuffix";
fit::string_view str_view = kPrefixWithSuffix;
str_view.remove_prefix(6);
EXPECT_EQ(kPrefixWithSuffix.length() - 6, str_view.length());
auto no_suffix = kPrefixWithSuffix.substr(6, kPrefixWithSuffix.length() - 6);
EXPECT_EQ(no_suffix, str_view);
EXPECT_EQ("Suffix", str_view);
}
TEST(StringViewTest, RemoveSuffix) {
constexpr fit::string_view kPrefixWithSuffix = "PrefixSuffix";
fit::string_view str_view = kPrefixWithSuffix;
str_view.remove_suffix(6);
EXPECT_EQ(kPrefixWithSuffix.length() - 6, str_view.length());
auto no_suffix = kPrefixWithSuffix.substr(0, kPrefixWithSuffix.length() - 6);
EXPECT_EQ(no_suffix, str_view);
EXPECT_EQ("Prefix", str_view);
}
TEST(StringViewTest, SubstrNoArgsAreEqual) {
constexpr fit::string_view kLiteral = "12345";
EXPECT_EQ(kLiteral, kLiteral.substr());
}
TEST(StringViewTest, SubstrWithPosIsMatchesSubstring) {
constexpr fit::string_view kLiteral = "12345";
constexpr fit::string_view kExpectedLiteral = "345";
EXPECT_EQ(kExpectedLiteral, kLiteral.substr(2));
}
TEST(StringViewTest, SubstrWithPosAndCountIsMatchesSubstring) {
constexpr fit::string_view kLiteral = "12345";
constexpr fit::string_view kExpectedLiteral = "34";
EXPECT_EQ(kExpectedLiteral, kLiteral.substr(2, 2));
}
TEST(StringViewTest, Swap) {
fit::string_view str_1 = "12345";
fit::string_view str_2 = "34";
str_1.swap(str_2);
EXPECT_EQ("34", str_1);
EXPECT_EQ("12345", str_2);
}
TEST(StringViewTest, Copy) {
constexpr fit::string_view kBase = "Base";
constexpr fit::string_view::size_type kSize = 2;
fit::string_view::value_type dest[kSize + 1] = {'\0'};
fit::string_view::value_type dest_traits[kSize + 1] = {'\0'};
EXPECT_EQ(kSize, kBase.copy(dest, kSize, 0));
EXPECT_EQ(dest_traits, fit::string_view::traits_type::copy(dest_traits, kBase.data(), kSize));
EXPECT_STR_EQ(dest_traits, dest);
}
#if DEATH_TESTS
TEST(StringViewTest, CopyThrowsExceptionOnOOR) {
ASSERT_DEATH([]() {
constexpr fit::string_view v_str = "Base";
fit::string_view::value_type dest[v_str.length() + 2] = {};
memset(dest, '\0', v_str.length() + 1);
v_str.copy(dest, v_str.length(), v_str.length());
});
}
#endif
TEST(StringViewTest, MaxSizeIsMaxAddressableSize) {
fit::string_view str_view("12345");
EXPECT_EQ(std::numeric_limits<fit::string_view::size_type>::max(), str_view.max_size());
}
TEST(StringViewTest, FindReturnsFirstCharTypeMatch) {
constexpr fit::string_view kString = "12345678901234567890";
EXPECT_EQ(0, kString.find('1'));
EXPECT_EQ(1, kString.find('2'));
EXPECT_EQ(2, kString.find('3'));
EXPECT_EQ(3, kString.find('4'));
EXPECT_EQ(4, kString.find('5'));
EXPECT_EQ(5, kString.find('6'));
EXPECT_EQ(6, kString.find('7'));
EXPECT_EQ(7, kString.find('8'));
EXPECT_EQ(8, kString.find('9'));
EXPECT_EQ(9, kString.find('0'));
}
TEST(StringViewTest, FindWithPosReturnsFirstCharTypeMatch) {
constexpr fit::string_view kString = "12345678901234567890";
EXPECT_EQ(10, kString.find('1', 10));
EXPECT_EQ(11, kString.find('2', 10));
EXPECT_EQ(12, kString.find('3', 10));
EXPECT_EQ(13, kString.find('4', 10));
EXPECT_EQ(14, kString.find('5', 10));
EXPECT_EQ(15, kString.find('6', 10));
EXPECT_EQ(16, kString.find('7', 10));
EXPECT_EQ(17, kString.find('8', 10));
EXPECT_EQ(18, kString.find('9', 10));
EXPECT_EQ(19, kString.find('0', 10));
}
TEST(StringViewTest, FindReturnsNposWhenNoCharTypeMatch) {
constexpr fit::string_view kString = "123456789123456789";
EXPECT_EQ(fit::string_view::npos, kString.find('0'));
}
TEST(StringViewTest, FindReturnsFirstMatch) {
constexpr fit::string_view kString = "12345678901234567890";
EXPECT_EQ(0, kString.find(""));
EXPECT_EQ(0, kString.find("12"));
EXPECT_EQ(1, kString.find("23"));
EXPECT_EQ(2, kString.find("34"));
EXPECT_EQ(3, kString.find("45"));
EXPECT_EQ(4, kString.find("56"));
EXPECT_EQ(5, kString.find("67"));
EXPECT_EQ(6, kString.find("78"));
EXPECT_EQ(7, kString.find("89"));
EXPECT_EQ(8, kString.find("90"));
EXPECT_EQ(9, kString.find("01"));
EXPECT_EQ(9, kString.find("01234"));
}
TEST(StringViewTest, FindWithPosReturnsFirstMatch) {
constexpr fit::string_view kString = "12345678901234567890";
EXPECT_EQ(10, kString.find("", 10));
EXPECT_EQ(10, kString.find("1", 10));
EXPECT_EQ(11, kString.find("2", 10));
EXPECT_EQ(12, kString.find("3", 10));
EXPECT_EQ(13, kString.find("4", 10));
EXPECT_EQ(14, kString.find("5", 10));
EXPECT_EQ(15, kString.find("6", 10));
EXPECT_EQ(16, kString.find("7", 10));
EXPECT_EQ(17, kString.find("8", 10));
EXPECT_EQ(18, kString.find("9", 10));
EXPECT_EQ(19, kString.find("0", 10));
// String of size > 1.
EXPECT_EQ(13, kString.find("456", 10));
}
TEST(StringViewTest, FindReturnsNposWhenNoMatch) {
constexpr fit::string_view kString = "12345678901234567890";
// String of size > 1.
EXPECT_EQ(fit::string_view::npos, kString.find("A"));
EXPECT_EQ(fit::string_view::npos, kString.find("02"));
EXPECT_EQ(fit::string_view::npos, kString.find("42321"));
}
TEST(StringViewTest, FindReturnsNposWhenNeedleIsBiggerThanHaystack) {
constexpr fit::string_view kString = "123";
// String of size > 1.
EXPECT_EQ(fit::string_view::npos, kString.find("1234"));
}
TEST(StringViewTest, RrfindReturnsFirstCharTypeMatch) {
constexpr fit::string_view kString = "12345678901234567890";
EXPECT_EQ(10, kString.rfind('1'));
EXPECT_EQ(11, kString.rfind('2'));
EXPECT_EQ(12, kString.rfind('3'));
EXPECT_EQ(13, kString.rfind('4'));
EXPECT_EQ(14, kString.rfind('5'));
EXPECT_EQ(15, kString.rfind('6'));
EXPECT_EQ(16, kString.rfind('7'));
EXPECT_EQ(17, kString.rfind('8'));
EXPECT_EQ(18, kString.rfind('9'));
EXPECT_EQ(19, kString.rfind('0'));
}
TEST(StringViewTest, RrfindWithPosReturnsFirstCharTypeMatch) {
constexpr fit::string_view kString = "12345678901234567890";
EXPECT_EQ(10, kString.rfind('1', 10));
EXPECT_EQ(11, kString.rfind('2', 10));
EXPECT_EQ(12, kString.rfind('3', 10));
EXPECT_EQ(13, kString.rfind('4', 10));
EXPECT_EQ(14, kString.rfind('5', 10));
EXPECT_EQ(15, kString.rfind('6', 10));
EXPECT_EQ(16, kString.rfind('7', 10));
EXPECT_EQ(17, kString.rfind('8', 10));
EXPECT_EQ(18, kString.rfind('9', 10));
EXPECT_EQ(19, kString.rfind('0', 10));
}
TEST(StringViewTest, RrfindReturnsNposWhenNoCharTypeMatch) {
constexpr fit::string_view kString = "123456789123456789";
EXPECT_EQ(fit::string_view::npos, kString.rfind('0'));
}
TEST(StringViewTest, RrfindReturnsFirstMatch) {
constexpr fit::string_view kString = "12345678901234567890";
EXPECT_EQ(19, kString.rfind(""));
EXPECT_EQ(10, kString.rfind("12"));
EXPECT_EQ(11, kString.rfind("23"));
EXPECT_EQ(12, kString.rfind("34"));
EXPECT_EQ(13, kString.rfind("45"));
EXPECT_EQ(14, kString.rfind("56"));
EXPECT_EQ(15, kString.rfind("67"));
EXPECT_EQ(16, kString.rfind("78"));
EXPECT_EQ(17, kString.rfind("89"));
EXPECT_EQ(18, kString.rfind("90"));
EXPECT_EQ(9, kString.rfind("01"));
EXPECT_EQ(9, kString.rfind("01234"));
}
TEST(StringViewTest, RrfindWithPosReturnsFirstMatch) {
constexpr fit::string_view kString = "12345678901234567890";
EXPECT_EQ(19, kString.rfind("", 10));
EXPECT_EQ(10, kString.rfind("1", 10));
EXPECT_EQ(11, kString.rfind("2", 10));
EXPECT_EQ(12, kString.rfind("3", 10));
EXPECT_EQ(13, kString.rfind("4", 10));
EXPECT_EQ(14, kString.rfind("5", 10));
EXPECT_EQ(15, kString.rfind("6", 10));
EXPECT_EQ(16, kString.rfind("7", 10));
EXPECT_EQ(17, kString.rfind("8", 10));
EXPECT_EQ(18, kString.rfind("9", 10));
EXPECT_EQ(19, kString.rfind("0", 10));
// String of size > 1.
EXPECT_EQ(13, kString.rfind("456", 10));
}
TEST(StringViewTest, RrfindReturnsNposWhenNoMatch) {
constexpr fit::string_view kString = "12345678901234567890";
EXPECT_EQ(fit::string_view::npos, kString.rfind("A"));
EXPECT_EQ(fit::string_view::npos, kString.rfind("02"));
EXPECT_EQ(fit::string_view::npos, kString.rfind("42321"));
EXPECT_EQ(fit::string_view::npos, kString.rfind('A'));
}
TEST(StringViewTest, RfindReturnsNposWhenNeedleIsBiggerThanHaystack) {
constexpr fit::string_view kString = "123";
// String of size > 1.
EXPECT_EQ(fit::string_view::npos, kString.rfind("1234"));
EXPECT_EQ(fit::string_view::npos, fit::string_view().find('1'));
}
TEST(StringViewTest, FindFirstOfReturnsFirstMatch) {
constexpr fit::string_view kString = "ABCDE1234ABCDE1234";
constexpr fit::string_view kMatchers = "123";
// Verify that order of chartacters in |s| does not change first match.
EXPECT_EQ(5, kString.find_first_of("321"));
EXPECT_EQ(5, kString.find_first_of("123"));
EXPECT_EQ(5, kString.find_first_of("231"));
EXPECT_EQ(5, kString.find_first_of("213"));
EXPECT_EQ(5, kString.find_first_of(kMatchers));
EXPECT_EQ(6, kString.find_first_of('2'));
}
TEST(StringViewTest, FindFirstOfWithPosReturnsFirstMatch) {
constexpr fit::string_view kString = "ABCDE1234ABCDE1234";
constexpr fit::string_view kMatchers = "123";
// Verify that order of chartacters in |s| does not change first match.
EXPECT_EQ(14, kString.find_first_of("321", 9));
EXPECT_EQ(14, kString.find_first_of("123", 9));
EXPECT_EQ(14, kString.find_first_of("231", 9));
EXPECT_EQ(14, kString.find_first_of("213", 9));
EXPECT_EQ(14, kString.find_first_of(kMatchers, 9));
EXPECT_EQ(5, kString.find_first_of('1'));
}
TEST(StringViewTest, FindFirstOfWithPosAndCountReturnsFirstMatch) {
constexpr fit::string_view kString = "ABCDE1234ABCDE1234";
// Verify that order of chartacters in |s| does not change first match.
EXPECT_EQ(14, kString.find_first_of("123", 9, 1));
EXPECT_EQ(15, kString.find_first_of("231", 9, 1));
EXPECT_EQ(15, kString.find_first_of("213", 9, 1));
EXPECT_EQ(16, kString.find_first_of("321", 9, 1));
}
TEST(StringViewTest, FindFirstOfReturnsNposWhenNoMatch) {
constexpr fit::string_view kString = "ABCDE1234ABCDE1234";
// Verify that order of chartacters in |s| does not change first match.
EXPECT_EQ(fit::string_view::npos, kString.find_first_of("GHIJK"));
EXPECT_EQ(fit::string_view::npos, kString.find_first_of("G"));
EXPECT_EQ(fit::string_view::npos, kString.find_first_of('G'));
}
TEST(StringViewTest, FindLastOfReturnsLastMatch) {
constexpr fit::string_view kString = "ABCDE1234ABCDE1234";
constexpr fit::string_view kMatchers = "123";
// Verify that order of chartacters in |s| does not change last match.
EXPECT_EQ(16, kString.find_last_of("321"));
EXPECT_EQ(16, kString.find_last_of("123"));
EXPECT_EQ(16, kString.find_last_of("231"));
EXPECT_EQ(16, kString.find_last_of("213"));
EXPECT_EQ(16, kString.find_last_of(kMatchers));
EXPECT_EQ(15, kString.find_last_of('2'));
}
TEST(StringViewTest, FindLastOfWithPosReturnsLastMatch) {
constexpr fit::string_view kString = "ABCDE1234ABCDE1234";
constexpr fit::string_view kMatchers = "123";
// Verify that order of chartacters in |s| does not change last match.
EXPECT_EQ(7, kString.find_last_of("321", 9));
EXPECT_EQ(7, kString.find_last_of("123", 9));
EXPECT_EQ(7, kString.find_last_of("231", 9));
EXPECT_EQ(7, kString.find_last_of("213", 9));
EXPECT_EQ(7, kString.find_last_of(kMatchers, 9));
EXPECT_EQ(5, kString.find_last_of('1', 9));
}
TEST(StringViewTest, FindLastOfWithPosAndCountReturnsLastMatch) {
constexpr fit::string_view kString = "ABCDE1234ABCDE1234";
// Verify that order of chartacters in |s| does not change last match.
EXPECT_EQ(5, kString.find_last_of("123", 9, 1));
EXPECT_EQ(6, kString.find_last_of("231", 9, 1));
EXPECT_EQ(6, kString.find_last_of("213", 9, 1));
EXPECT_EQ(7, kString.find_last_of("321", 9, 1));
}
TEST(StringViewTest, FindLastOfReturnsNposWhenNoMatch) {
constexpr fit::string_view kString = "ABCDE1234ABCDE1234";
// Verify that order of chartacters in |s| does not change last match.
EXPECT_EQ(fit::string_view::npos, kString.find_last_of("GHIJK"));
EXPECT_EQ(fit::string_view::npos, kString.find_last_of("G"));
EXPECT_EQ(fit::string_view::npos, kString.find_last_of('G'));
}
TEST(StringViewTest, FindFirstNofOfReturnsFirstNonMatch) {
constexpr fit::string_view kString = "123ABC123";
constexpr fit::string_view kMatchers = "123";
// Verify that order of chartacters in |s| does not change first match.
EXPECT_EQ(0, kString.find_first_not_of(""));
EXPECT_EQ(3, kString.find_first_not_of("321"));
EXPECT_EQ(3, kString.find_first_not_of("123"));
EXPECT_EQ(3, kString.find_first_not_of("231"));
EXPECT_EQ(3, kString.find_first_not_of("213"));
EXPECT_EQ(3, kString.find_first_not_of(kMatchers));
EXPECT_EQ(1, kString.find_first_not_of('1'));
}
TEST(StringViewTest, FindFirstNofOfWithPosReturnsFirstNonMatch) {
constexpr fit::string_view kString = "123ABC123A";
constexpr fit::string_view kMatchers = "123";
// Verify that order of chartacters in |s| does not change first match.
EXPECT_EQ(6, kString.find_first_not_of("", 6));
EXPECT_EQ(9, kString.find_first_not_of("321", 6));
EXPECT_EQ(9, kString.find_first_not_of("123", 6));
EXPECT_EQ(9, kString.find_first_not_of("231", 6));
EXPECT_EQ(9, kString.find_first_not_of("213", 6));
EXPECT_EQ(9, kString.find_first_not_of(kMatchers, 9));
EXPECT_EQ(7, kString.find_first_not_of('1', 6));
}
TEST(StringViewTest, FindFirstNofOfWithPosAndCountReturnsFirstNofMatch) {
constexpr fit::string_view kString = "123ABC123A";
// Verify that order of chartacters in |s| does not change first match.
EXPECT_EQ(7, kString.find_first_not_of("123", 6, 1));
EXPECT_EQ(6, kString.find_first_not_of("231", 6, 1));
EXPECT_EQ(6, kString.find_first_not_of("213", 6, 1));
EXPECT_EQ(6, kString.find_first_not_of("321", 6, 1));
}
TEST(StringViewTest, FindFirstNofOfReturnsNposWhenNoMatch) {
constexpr fit::string_view kString = "GGGGGGGGGGGGG";
// Verify that order of chartacters in |s| does not change first match.
EXPECT_EQ(fit::string_view::npos, kString.find_first_not_of("ABCG"));
EXPECT_EQ(fit::string_view::npos, kString.find_first_not_of("G"));
EXPECT_EQ(fit::string_view::npos, kString.find_first_not_of('G'));
}
TEST(StringViewTest, FindLastNotOfReturnsLastMatch) {
constexpr fit::string_view kString = "ABCDE1234ABCDE1234";
constexpr fit::string_view kMatchers = "1234";
// Verify that order of chartacters in |s| does not change last_not match.
EXPECT_EQ(13, kString.find_last_not_of("3214"));
EXPECT_EQ(13, kString.find_last_not_of("1234"));
EXPECT_EQ(13, kString.find_last_not_of("2314"));
EXPECT_EQ(13, kString.find_last_not_of("2134"));
EXPECT_EQ(13, kString.find_last_not_of(kMatchers));
EXPECT_EQ(16, kString.find_last_not_of('4'));
}
TEST(StringViewTest, FindLastNotOfWithPosReturnsLastMatch) {
constexpr fit::string_view kString = "ABCDE1234ABCDE1234";
constexpr fit::string_view kMatchers = "1234";
// Verify that order of chartacters in |s| does not change last_not match.
EXPECT_EQ(4, kString.find_last_not_of("3214", 8));
EXPECT_EQ(4, kString.find_last_not_of("1234", 8));
EXPECT_EQ(4, kString.find_last_not_of("2314", 8));
EXPECT_EQ(4, kString.find_last_not_of("2134", 8));
EXPECT_EQ(4, kString.find_last_not_of(kMatchers, 8));
EXPECT_EQ(7, kString.find_last_not_of('4', 8));
}
TEST(StringViewTest, FindLastNotOfWithPosAndCountReturnsLastMatch) {
constexpr fit::string_view kString = "ABCDE1234ABCDE1234";
// Verify that order of chartacters in |s| does not change last_not match.
EXPECT_EQ(8, kString.find_last_not_of("1234", 8, 1));
EXPECT_EQ(8, kString.find_last_not_of("2314", 8, 2));
EXPECT_EQ(5, kString.find_last_not_of("4321", 8, 3));
EXPECT_EQ(4, kString.find_last_not_of("3214", 8, 4));
}
TEST(StringViewTest, FindLastNotOfReturnsNposWhenNoMatch) {
constexpr fit::string_view kString = "GGGGGGG";
// Verify that order of chartacters in |s| does not change last_not match.
EXPECT_EQ(fit::string_view::npos, kString.find_last_not_of("GHIJK"));
EXPECT_EQ(fit::string_view::npos, kString.find_last_not_of("G"));
EXPECT_EQ(fit::string_view::npos, kString.find_last_not_of('G'));
}
TEST(StringViewTest, HashVaidation) {
constexpr fit::string_view kStringView1 = "1234";
constexpr fit::string_view kStringView2 = "123456";
const std::string kString = "123";
// Hash of the full view matches the hash of an equivalent string.
EXPECT_EQ(std::hash<std::string>()(std::string(kStringView1.data())),
std::hash<fit::string_view>()(kStringView1));
EXPECT_EQ(std::hash<std::string>()(std::string(kStringView2.data())),
std::hash<fit::string_view>()(kStringView2));
// Hash uses the view, not the full string.
EXPECT_EQ(std::hash<fit::string_view>()(kStringView1),
std::hash<fit::string_view>()(kStringView2.substr(0, kStringView1.length())));
// Hash matches the hash for a string with the same content.
EXPECT_EQ(std::hash<std::string>()(kString),
std::hash<fit::string_view>()(kStringView2.substr(0, kString.length())));
EXPECT_EQ(std::hash<std::string>()(kString),
std::hash<fit::string_view>()(kStringView1.substr(0, kString.length())));
// If the hash of the contents are different in the default hash of the string, they should be
// different in the specialized hash. We should make no assumptions on whether the values will hit
// the same bucket or not.
EXPECT_EQ(
std::hash<std::string>()(std::string(kStringView1.data())) !=
std::hash<std::string>()(std::string(kStringView2.data())),
std::hash<fit::string_view>()(kStringView1) != std::hash<fit::string_view>()(kStringView2));
}
TEST(StringViewTest, OutputStreamOperatorFitsWithinWidth) {
constexpr fit::string_view kStringView = "1234";
std::ostringstream oss;
oss.width(kStringView.length());
oss << kStringView;
EXPECT_EQ(oss.str(), kStringView);
}
TEST(StringViewTest, OutputStreamOperatorDoesNotFitInWidth) {
constexpr fit::string_view kStringView = "1234";
std::ostringstream oss;
oss.width(kStringView.length() - 1);
oss << kStringView;
EXPECT_EQ(oss.str(), kStringView.substr(0, kStringView.length() - 1));
}
TEST(StringViewTest, OutputStreamOperatorFillsExtraSpaceToTheRight) {
constexpr fit::string_view kStringView = "1234";
constexpr fit::string_view kExpected = "000001234";
std::ostringstream oss;
oss.fill('0');
oss.width(kExpected.length());
oss << kStringView;
EXPECT_EQ(oss.str(), kExpected);
}
TEST(StringViewTest, OutputStreamOperatorFillsExtraSpaceToTheLeft) {
constexpr fit::string_view kStringView = "1234";
constexpr fit::string_view kExpected = "123400000";
std::ostringstream oss;
oss.fill('0');
oss.setf(std::ios_base::left);
oss.width(kExpected.length());
oss << kStringView;
EXPECT_EQ(oss.str(), kExpected);
}
TEST(StringViewTest, OutputStreamOperatorResetsWidthToZero) {
constexpr fit::string_view kStringView = "1234";
std::ostringstream oss;
oss.fill('0');
oss.width(10);
oss << kStringView;
EXPECT_EQ(0, oss.width());
}
} // namespace
} // namespace fit