mirror of
https://github.com/SerenityOS/serenity.git
synced 2025-01-23 09:51:57 -05:00
82ea53cf10
We already have a helper to split a StringView by line while considering "\n", "\r", and "\r\n". Add an analagous method to just count the number of lines in the same manner.
220 lines
9 KiB
C++
220 lines
9 KiB
C++
/*
|
|
* Copyright (c) 2018-2020, Andreas Kling <kling@serenityos.org>
|
|
*
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
|
*/
|
|
|
|
#include <LibTest/TestCase.h>
|
|
|
|
#include <AK/ByteString.h>
|
|
#include <AK/Vector.h>
|
|
|
|
TEST_CASE(construct_empty)
|
|
{
|
|
EXPECT(StringView().is_null());
|
|
EXPECT(StringView().is_empty());
|
|
EXPECT(!StringView().characters_without_null_termination());
|
|
EXPECT_EQ(StringView().length(), 0u);
|
|
}
|
|
|
|
TEST_CASE(view_literal)
|
|
{
|
|
char const* truth = "cats rule dogs drool";
|
|
StringView view { truth, strlen(truth) };
|
|
EXPECT_EQ(view.is_null(), false);
|
|
EXPECT_EQ(view.characters_without_null_termination(), truth);
|
|
EXPECT_EQ(view, view);
|
|
EXPECT_EQ(view, truth);
|
|
}
|
|
|
|
TEST_CASE(compare_views)
|
|
{
|
|
ByteString foo1 = "foo";
|
|
ByteString foo2 = "foo";
|
|
auto view1 = foo1.view();
|
|
auto view2 = foo2.view();
|
|
|
|
EXPECT_EQ(view1, view2);
|
|
EXPECT_EQ(view1, foo1);
|
|
EXPECT_EQ(view1, foo2);
|
|
EXPECT_EQ(view1, "foo");
|
|
}
|
|
|
|
TEST_CASE(string_view_literal_operator)
|
|
{
|
|
StringView literal_view = "foo"sv;
|
|
ByteString test_string = "foo";
|
|
|
|
EXPECT_EQ(literal_view.length(), test_string.length());
|
|
EXPECT_EQ(literal_view, test_string);
|
|
}
|
|
|
|
TEST_CASE(starts_with)
|
|
{
|
|
ByteString test_string = "ABCDEF";
|
|
StringView test_string_view = test_string.view();
|
|
EXPECT(test_string_view.starts_with('A'));
|
|
EXPECT(!test_string_view.starts_with('B'));
|
|
EXPECT(test_string_view.starts_with("AB"sv));
|
|
EXPECT(test_string_view.starts_with("ABCDEF"sv));
|
|
EXPECT(!test_string_view.starts_with("DEF"sv));
|
|
EXPECT(test_string_view.starts_with("abc"sv, CaseSensitivity::CaseInsensitive));
|
|
EXPECT(!test_string_view.starts_with("abc"sv, CaseSensitivity::CaseSensitive));
|
|
}
|
|
|
|
TEST_CASE(ends_with)
|
|
{
|
|
ByteString test_string = "ABCDEF";
|
|
StringView test_string_view = test_string.view();
|
|
EXPECT(test_string_view.ends_with("DEF"sv));
|
|
EXPECT(test_string_view.ends_with('F'));
|
|
EXPECT(!test_string_view.ends_with('E'));
|
|
EXPECT(test_string_view.ends_with("ABCDEF"sv));
|
|
EXPECT(!test_string_view.ends_with("ABCDE"sv));
|
|
EXPECT(!test_string_view.ends_with("ABCDEFG"sv));
|
|
EXPECT(test_string_view.ends_with("def"sv, CaseSensitivity::CaseInsensitive));
|
|
EXPECT(!test_string_view.ends_with("def"sv, CaseSensitivity::CaseSensitive));
|
|
}
|
|
|
|
TEST_CASE(lines)
|
|
{
|
|
ByteString test_string = "a\rb\nc\r\nd";
|
|
StringView test_string_view = test_string.view();
|
|
Vector<StringView> test_string_vector = test_string_view.lines();
|
|
EXPECT_EQ(test_string_vector.size(), 4u);
|
|
EXPECT(test_string_vector.at(0) == ByteString("a"));
|
|
EXPECT(test_string_vector.at(1) == ByteString("b"));
|
|
EXPECT(test_string_vector.at(2) == ByteString("c"));
|
|
EXPECT(test_string_vector.at(3) == ByteString("d"));
|
|
|
|
test_string = "```\nHello there\r\nHello there\n```";
|
|
test_string_view = test_string.view();
|
|
test_string_vector = test_string_view.lines();
|
|
EXPECT_EQ(test_string_vector.size(), 4u);
|
|
EXPECT(test_string_vector.at(0) == ByteString("```"));
|
|
EXPECT(test_string_vector.at(1) == ByteString("Hello there"));
|
|
EXPECT(test_string_vector.at(2) == ByteString("Hello there"));
|
|
EXPECT(test_string_vector.at(3) == ByteString("```"));
|
|
|
|
test_string = "\n\n\n";
|
|
test_string_view = test_string.view();
|
|
test_string_vector = test_string_view.lines();
|
|
EXPECT_EQ(test_string_vector.size(), 3u);
|
|
EXPECT_EQ(test_string_vector.at(0).is_empty(), true);
|
|
EXPECT_EQ(test_string_vector.at(1).is_empty(), true);
|
|
EXPECT_EQ(test_string_vector.at(2).is_empty(), true);
|
|
}
|
|
|
|
TEST_CASE(count_lines)
|
|
{
|
|
EXPECT_EQ(""sv.count_lines(), 1u);
|
|
EXPECT_EQ("foo"sv.count_lines(), 1u);
|
|
|
|
EXPECT_EQ("foo\nbar"sv.count_lines(), 2u);
|
|
EXPECT_EQ("foo\rbar"sv.count_lines(), 2u);
|
|
EXPECT_EQ("foo\rbar"sv.count_lines(StringView::ConsiderCarriageReturn::No), 1u);
|
|
EXPECT_EQ("foo\r\nbar"sv.count_lines(), 2u);
|
|
EXPECT_EQ("foo\r\nbar"sv.count_lines(StringView::ConsiderCarriageReturn::No), 2u);
|
|
|
|
EXPECT_EQ("foo\nbar\nbax"sv.count_lines(), 3u);
|
|
EXPECT_EQ("foo\rbar\rbaz"sv.count_lines(), 3u);
|
|
EXPECT_EQ("foo\rbar\rbaz"sv.count_lines(StringView::ConsiderCarriageReturn::No), 1u);
|
|
EXPECT_EQ("foo\r\nbar\r\nbaz"sv.count_lines(), 3u);
|
|
EXPECT_EQ("foo\r\nbar\r\nbaz"sv.count_lines(StringView::ConsiderCarriageReturn::No), 3u);
|
|
}
|
|
|
|
TEST_CASE(find)
|
|
{
|
|
auto test_string_view = "aabbcc_xy_ccbbaa"sv;
|
|
EXPECT_EQ(test_string_view.find('b'), 2U);
|
|
EXPECT_EQ(test_string_view.find('_'), 6U);
|
|
EXPECT_EQ(test_string_view.find('n').has_value(), false);
|
|
}
|
|
|
|
TEST_CASE(find_last)
|
|
{
|
|
auto test_string_view = "aabbcc_xy_ccbbaa"sv;
|
|
EXPECT_EQ(test_string_view.find_last('b'), 13U);
|
|
EXPECT_EQ(test_string_view.find_last('_'), 9U);
|
|
EXPECT_EQ(test_string_view.find_last('3').has_value(), false);
|
|
|
|
test_string_view = "/"sv;
|
|
EXPECT_EQ(test_string_view.find_last('/'), 0U);
|
|
}
|
|
|
|
TEST_CASE(find_any_of)
|
|
{
|
|
auto test_string_view = "aabbcc_xy_ccbbaa"sv;
|
|
EXPECT_EQ(test_string_view.find_any_of("bc"sv, StringView::SearchDirection::Forward), 2U);
|
|
EXPECT_EQ(test_string_view.find_any_of("yx"sv, StringView::SearchDirection::Forward), 7U);
|
|
EXPECT_EQ(test_string_view.find_any_of("defg"sv, StringView::SearchDirection::Forward).has_value(), false);
|
|
EXPECT_EQ(test_string_view.find_any_of("bc"sv, StringView::SearchDirection::Backward), 13U);
|
|
EXPECT_EQ(test_string_view.find_any_of("yx"sv, StringView::SearchDirection::Backward), 8U);
|
|
EXPECT_EQ(test_string_view.find_any_of("fghi"sv, StringView::SearchDirection::Backward).has_value(), false);
|
|
|
|
test_string_view = "/"sv;
|
|
EXPECT_EQ(test_string_view.find_any_of("/"sv, StringView::SearchDirection::Forward), 0U);
|
|
EXPECT_EQ(test_string_view.find_any_of("/"sv, StringView::SearchDirection::Backward), 0U);
|
|
}
|
|
|
|
TEST_CASE(split_view)
|
|
{
|
|
StringView test_string_view = "axxbxcxd"sv;
|
|
EXPECT_EQ(test_string_view.split_view('x'), Vector({ "a"sv, "b"sv, "c"sv, "d"sv }));
|
|
EXPECT_EQ(test_string_view.split_view('x', SplitBehavior::KeepEmpty), Vector({ "a"sv, ""sv, "b"sv, "c"sv, "d"sv }));
|
|
EXPECT_EQ(test_string_view.split_view("x"sv), Vector({ "a"sv, "b"sv, "c"sv, "d"sv }));
|
|
EXPECT_EQ(test_string_view.split_view("x"sv, SplitBehavior::KeepEmpty), Vector({ "a"sv, ""sv, "b"sv, "c"sv, "d"sv }));
|
|
|
|
test_string_view = "axxbx"sv;
|
|
EXPECT_EQ(test_string_view.split_view('x'), Vector({ "a"sv, "b"sv }));
|
|
EXPECT_EQ(test_string_view.split_view('x', SplitBehavior::KeepEmpty), Vector({ "a"sv, ""sv, "b"sv, ""sv }));
|
|
EXPECT_EQ(test_string_view.split_view("x"sv), Vector({ "a"sv, "b"sv }));
|
|
EXPECT_EQ(test_string_view.split_view("x"sv, SplitBehavior::KeepEmpty), Vector({ "a"sv, ""sv, "b"sv, ""sv }));
|
|
|
|
test_string_view = "axxbcxxdxx"sv;
|
|
EXPECT_EQ(test_string_view.split_view("xx"sv), Vector({ "a"sv, "bc"sv, "d"sv }));
|
|
EXPECT_EQ(test_string_view.split_view("xx"sv, SplitBehavior::KeepEmpty), Vector({ "a"sv, "bc"sv, "d"sv, ""sv }));
|
|
|
|
test_string_view = "ax_b_cxd"sv;
|
|
Function<bool(char)> predicate = [](char ch) { return ch == 'x' || ch == '_'; };
|
|
EXPECT_EQ(test_string_view.split_view_if(predicate), Vector({ "a"sv, "b"sv, "c"sv, "d"sv }));
|
|
EXPECT_EQ(test_string_view.split_view_if(predicate, SplitBehavior::KeepEmpty), Vector({ "a"sv, ""sv, "b"sv, "c"sv, "d"sv }));
|
|
EXPECT_EQ(test_string_view.split_view_if(predicate), Vector({ "a"sv, "b"sv, "c"sv, "d"sv }));
|
|
EXPECT_EQ(test_string_view.split_view_if(predicate, SplitBehavior::KeepEmpty), Vector({ "a"sv, ""sv, "b"sv, "c"sv, "d"sv }));
|
|
|
|
test_string_view = "a,,,b"sv;
|
|
EXPECT_EQ(test_string_view.split_view(","sv, SplitBehavior::KeepEmpty), Vector({ "a"sv, ""sv, ""sv, "b"sv }));
|
|
EXPECT_EQ(test_string_view.split_view(","sv, SplitBehavior::KeepTrailingSeparator), Vector({ "a,"sv, "b"sv }));
|
|
EXPECT_EQ(test_string_view.split_view(","sv, SplitBehavior::KeepTrailingSeparator | SplitBehavior::KeepEmpty), Vector({ "a,"sv, ","sv, ","sv, "b"sv }));
|
|
}
|
|
|
|
TEST_CASE(constexpr_stuff)
|
|
{
|
|
#define do_test() \
|
|
static_assert(test_constexpr.length() == 3); \
|
|
static_assert(!test_constexpr.is_empty()); \
|
|
static_assert(test_constexpr.is_one_of("foo", "bar", "baz")); \
|
|
static_assert(test_constexpr.is_one_of("foo"sv, "bar"sv, "baz"sv)); \
|
|
static_assert(test_constexpr != "fob"sv); \
|
|
static_assert(test_constexpr != "fob"); \
|
|
static_assert(test_constexpr.substring_view(1).is_one_of("oo"sv));
|
|
|
|
{
|
|
// Can initialize from ""sv.
|
|
constexpr StringView test_constexpr { "foo"sv };
|
|
do_test();
|
|
}
|
|
#undef do_test
|
|
}
|
|
|
|
TEST_CASE(case_insensitive_hash)
|
|
{
|
|
auto string1 = "abcdef"sv;
|
|
auto string2 = "ABCDEF"sv;
|
|
auto string3 = "aBcDeF"sv;
|
|
auto string4 = "foo"sv;
|
|
|
|
EXPECT_EQ(CaseInsensitiveASCIIStringViewTraits::hash(string1), CaseInsensitiveASCIIStringViewTraits::hash(string2));
|
|
EXPECT_EQ(CaseInsensitiveASCIIStringViewTraits::hash(string1), CaseInsensitiveASCIIStringViewTraits::hash(string3));
|
|
EXPECT_NE(CaseInsensitiveASCIIStringViewTraits::hash(string1), CaseInsensitiveASCIIStringViewTraits::hash(string4));
|
|
}
|