blob: cfde13b158713f41fe24dcfa1151485b57c10f56 [file] [log] [blame]
// Copyright 2017-2020 The Verible Authors.
//
// 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.
// Tests for LineColumnMap.
#include "common/strings/line_column_map.h"
#include <sstream> // IWYU pragma: keep // for ostringstream
#include <vector>
#include "absl/strings/str_split.h"
#include "absl/strings/string_view.h"
#include "gtest/gtest.h"
namespace verible {
namespace {
struct LineColumnTestData {
LineColumn line_col;
const char* text;
};
struct LineColumnRangeTestData {
LineColumnRange range;
const char* text;
};
struct LineColumnQuery {
int offset;
LineColumn line_col;
};
struct LineColumnMapTestData {
const char* text;
std::vector<int> expected_offsets;
std::vector<LineColumnQuery> queries;
};
// Raw line and column are 0-indexed.
const LineColumnMapTestData map_test_data[] = {
// Also testing beyond the end of the file - it should return last position
{"", {0}, {{0, {0, 0}}, {1, {0, 0}}, {100, {0, 0}}}}, // empty file
{"_", {0}, {{0, {0, 0}}, {1, {0, 1}}}}, // no \n before EOF
{"abc", {0}, {{0, {0, 0}}, {2, {0, 2}}, {3, {0, 3}}}},
{"\n", {0, 1}, {{0, {0, 0}}, {1, {1, 0}}}}, // one empty line
{"\n\n", {0, 1, 2}, {{0, {0, 0}}, {1, {1, 0}}, {2, {2, 0}}}},
{"ab\nc", {0, 3}, {{0, {0, 0}}, {2, {0, 2}}, {3, {1, 0}}, {4, {1, 1}}}},
{"_\n_\n", {0, 2, 4}, {{0, {0, 0}}, {1, {0, 1}}, {2, {1, 0}}, {3, {1, 1}}}},
{"\nxx\n", {0, 1, 4}, {{0, {0, 0}}, {1, {1, 0}}, {2, {1, 1}}, {3, {1, 2}}}},
{"hello\ndarkness\nmy old friend\n",
{0, 6, 15, 29},
{{0, {0, 0}}, {10, {1, 4}}, {15, {2, 0}}, {20, {2, 5}}}},
// Multi-byte characters. Let's use strlen() to count the bytes, the
// column should accurately point to the character.
{"😀😀😀", {0}, {{static_cast<int>(2 * strlen("😀")), {0, 2}}}},
{"Heizölrückstoßabdämpfung",
{0},
{{static_cast<int>(strlen("Heizölrückstoß")), {0, 14}}}},
};
// Test test verifies that line-column offset appear to the user correctly.
TEST(LineColumnTextTest, PrintLineColumn) {
static constexpr LineColumnTestData text_test_data[] = {
{{0, 0}, "1:1"},
{{0, 1}, "1:2"},
{{1, 0}, "2:1"},
{{10, 8}, "11:9"},
};
for (const auto& test_case : text_test_data) {
std::ostringstream oss;
oss << test_case.line_col;
EXPECT_EQ(oss.str(), test_case.text);
}
}
TEST(LineColumnTextTest, PrintLineColumnRange) {
static constexpr LineColumnRangeTestData text_test_data[] = {
{{{0, 0}, {0, 7}}, "1:1-7:"}, // Same line, multiple columns
{{{0, 1}, {0, 3}}, "1:2-3:"},
{{{1, 0}, {2, 14}}, "2:1:3:14:"}, // start/end different lines
{{{10, 8}, {11, 2}}, "11:9:12:2:"},
{{{10, 8}, {10, 9}}, "11:9:"}, // Single character range
{{{10, 8}, {10, 8}}, "11:9:"}, // Empty range.
};
for (const auto& test_case : text_test_data) {
std::ostringstream oss;
oss << test_case.range;
EXPECT_EQ(oss.str(), test_case.text);
}
}
// Test offset lookup values by line number.
TEST(LineColumnMapTest, OffsetAtLine) {
LineColumnMap line_map("hello\n\nworld\n");
EXPECT_EQ(line_map.OffsetAtLine(0), 0);
EXPECT_EQ(line_map.OffsetAtLine(1), 6);
EXPECT_EQ(line_map.OffsetAtLine(2), 7);
EXPECT_EQ(line_map.OffsetAtLine(3), 13); // There is no line[3].
}
// This test verifies the offsets where columns are reset to 0,
// which happens after every newline.
TEST(LineColumnMapTest, Offsets) {
for (const auto& test_case : map_test_data) {
const LineColumnMap line_map(test_case.text);
EXPECT_EQ(line_map.GetBeginningOfLineOffsets(), test_case.expected_offsets)
<< "Text: \"" << test_case.text << "\"";
}
}
// This tests that computing offsets from pre-split lines is consistent
// with the constructor that takes the whole text string.
TEST(LineColumnMapTest, OffsetsFromLines) {
for (const auto& test_case : map_test_data) {
const LineColumnMap line_map(test_case.text);
std::vector<absl::string_view> lines = absl::StrSplit(test_case.text, '\n');
const LineColumnMap alt_line_map(lines);
EXPECT_EQ(line_map.GetBeginningOfLineOffsets(),
alt_line_map.GetBeginningOfLineOffsets())
<< "Text: \"" << test_case.text << "\"";
}
}
TEST(LineColumnMapTest, EndOffsetNoLines) {
const std::vector<absl::string_view> lines;
const LineColumnMap map(lines);
EXPECT_EQ(map.LastLineOffset(), 0);
}
struct EndOffsetTestCase {
absl::string_view text;
int expected_offset;
};
TEST(LineColumnMapTest, EndOffsetVarious) {
const EndOffsetTestCase kTestCases[] = {
{"", 0}, // empty text
{"aaaa", 0}, // missing EOL
{"aaaa\nbbb", 5}, // missing EOL
{"\n", 1}, //
{"aaaa\n", 5}, //
{"aaaa\nbbb\n", 9}, //
{"\n\n", 2},
};
for (const auto& test : kTestCases) {
const LineColumnMap map(test.text);
EXPECT_EQ(map.LastLineOffset(), test.expected_offset) << "text:\n"
<< test.text;
}
}
// This test verifies the translation from byte-offset to line-column.
TEST(LineColumnMapTest, Lookup) {
for (const auto& test_case : map_test_data) {
const LineColumnMap line_map(test_case.text);
for (const auto& q : test_case.queries) {
EXPECT_EQ(q.line_col,
line_map.GetLineColAtOffset(test_case.text, q.offset))
<< "Text: \"" << test_case.text << "\"\n"
<< "Failed testing offset " << q.offset;
}
}
}
TEST(LineColumnTest, LineColumnComparison) {
constexpr LineColumn before_line{.line = 41, .column = 1};
constexpr LineColumn before_col{.line = 42, .column = 1};
constexpr LineColumn center{.line = 42, .column = 8};
constexpr LineColumn after_col{.line = 42, .column = 8};
EXPECT_LT(before_line, center);
EXPECT_LT(before_col, center);
EXPECT_EQ(center, center);
EXPECT_GE(center, center);
EXPECT_GE(after_col, center);
}
TEST(LineColumnTest, LineColumnRangeComparison) {
constexpr LineColumnRange range{{.line = 42, .column = 17},
{.line = 42, .column = 22}};
constexpr LineColumn before{.line = 42, .column = 16};
constexpr LineColumn inside_start{.line = 42, .column = 17};
constexpr LineColumn inside_end{.line = 42, .column = 21};
constexpr LineColumn outside_after_end{.line = 42, .column = 22};
EXPECT_FALSE(range.PositionInRange(before));
EXPECT_TRUE(range.PositionInRange(inside_start));
EXPECT_TRUE(range.PositionInRange(inside_end));
EXPECT_FALSE(range.PositionInRange(outside_after_end));
}
} // namespace
} // namespace verible