/* * Copyright (c) 2018-2020, Andreas Kling * * SPDX-License-Identifier: BSD-2-Clause */ #include #include #include 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 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(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 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)); }