libcruft-util/test/string.cpp

231 lines
6.1 KiB
C++
Raw Normal View History

2016-04-05 11:06:01 +10:00
#include "tap.hpp"
#include "string.hpp"
#include "types.hpp"
#include "iterator/zip.hpp"
2017-09-15 15:22:29 +10:00
#include <vector>
2016-03-17 18:13:19 +11:00
2019-11-26 08:10:04 +11:00
///////////////////////////////////////////////////////////////////////////////
void
test_transforms (cruft::TAP::logger &tap)
{
static struct {
std::string initial;
std::string upper;
std::string lower;
char const *message;
} const TESTS[] = {
{ "a", "A", "a", "single lower character" },
{ "A", "A", "a", "single upper character" },
{ "asdf", "ASDF", "asdf", "four lower characters" },
{ "ASDF", "ASDF", "asdf", "four upper characters" },
{ "Don't.", "DON'T.", "don't.", "upper and lower and symbols" },
};
for (auto const &[src, upper, lower, msg]: TESTS) {
2021-04-13 16:05:08 +10:00
tap.expect_eq (cruft::to_upper (src), upper, "to_upper, {:s}", msg);
tap.expect_eq (cruft::to_lower (src), lower, "to_lower, {:s}", msg);
2019-11-26 08:10:04 +11:00
}
}
///////////////////////////////////////////////////////////////////////////////
void
test_position (cruft::TAP::logger &tap)
{
char const *string = "a\nb\nc\n";
struct {
int offset;
int line;
int column;
} TESTS[] = {
{ 0, 0, 0 },
{ 1, 1, -1 },
{ 2, 1, 0 },
{ 3, 2, -1 },
{ 4, 2, 0 },
{ 5, 3, -1 },
};
for (auto const &t: TESTS) {
auto const pos = cruft::character_position ({string, strlen(string) }, string + t.offset);
tap.expect (
pos.line == t.line && pos.column == t.column,
2021-04-13 16:05:08 +10:00
"character_position {}:{}",
t.line,
t.column
);
}
}
///////////////////////////////////////////////////////////////////////////////
void
test_tokeniser (cruft::TAP::logger &tap)
{
// the string_literal prefix is required to (easily) construct a string
// with an internal null character.
using namespace std::literals::string_literals;
const std::string csv = "\0,a,123,,this is a test,"s;
2017-09-15 15:22:29 +10:00
// expected test data must be a std::string so we can check embedded
// nulls (which are ambiguous when using a cstr).
struct foo {
const std::string value;
const char *message;
} TESTS[] = {
2017-11-23 17:24:11 +11:00
{ "\0"s, "null" },
2017-09-15 15:22:29 +10:00
{ "a", "single letter" },
{ "123", "three digits" },
{ "", "empty string" },
{ "this is a test", "string with spaces" },
{ "", "trailing empty" }
2016-03-17 18:13:19 +11:00
};
cruft::view src { csv.c_str (), csv.size () };
for (auto [tok, expected]: cruft::iterator::zip (cruft::tokeniser (src, ','), TESTS)) {
2021-04-13 16:05:08 +10:00
tap.expect (equal (tok, expected.value), "{:s}", expected.message);
2018-03-27 15:49:47 +11:00
}
}
2019-02-20 14:44:33 +11:00
///////////////////////////////////////////////////////////////////////////////
void
test_contains (cruft::TAP::logger &tap)
{
struct {
char const *haystack;
2019-02-20 15:51:54 +11:00
char separator;
2019-02-20 14:44:33 +11:00
char const *needle;
bool expected;
char const *message;
} const TESTS[] = {
2019-02-20 15:51:54 +11:00
{
.haystack = "foo",
.separator = ',',
.needle = "foo",
.expected = true,
.message = "needle is haystack"
},
{
.haystack = "",
.separator = ',',
.needle = "foo",
.expected = false,
.message = "haystack is empty",
},
{
.haystack = "foo",
.separator = ',',
.needle = "",
.expected = false,
.message = "needle is empty",
},
{
.haystack = "a,b,foo",
.separator = ',',
.needle = "foo",
.expected = true,
.message = "needle is last",
},
{
.haystack = "foo,b,c",
.separator = ',',
.needle = "foo",
.expected = true,
.message = "needle is first",
},
{
.haystack = "foo,b,c",
.separator = ';',
.needle = "foo",
.expected = false,
.message = "separator is mismatched",
},
{
.haystack = "a;b;foo",
.separator = ';',
.needle = "foo",
.expected = true,
.message = "separator isn't default"
}
2019-02-20 14:44:33 +11:00
};
2019-02-20 15:51:54 +11:00
for (auto const &t: TESTS) {
cruft::tokeniser tok (t.haystack, t.separator);
auto const found = tok.contains (t.needle);
2021-04-13 16:05:08 +10:00
tap.expect_eq (found, t.expected, "{}", t.message);
2019-02-20 15:51:54 +11:00
}
2019-02-20 14:44:33 +11:00
}
2019-05-01 12:38:55 +10:00
///////////////////////////////////////////////////////////////////////////////
2019-05-02 11:03:22 +10:00
void test_comparator_less (cruft::TAP::logger &tap)
2019-05-01 12:38:55 +10:00
{
cruft::string_less cmp;
struct {
char const *a;
char const *b;
bool less;
char const *message;
} const TESTS[] = {
{ "a", "b", true, "single char less-than" },
{ "b", "a", false, "single char not-less-than" },
{ "a", "aa", true, "single/double char less-than" },
};
for (auto const &t: TESTS)
2021-04-13 16:05:08 +10:00
tap.expect_eq (cmp (t.a, t.b), t.less, "string_less, {}", t.message);
2019-05-01 12:38:55 +10:00
}
2019-05-02 11:03:22 +10:00
///////////////////////////////////////////////////////////////////////////////
void test_comparator_lower (cruft::TAP::logger &tap)
{
struct {
char const *a;
char const *b;
bool equal;
char const *message;
} const TESTS[] = {
{ "a", "a", true, "both single lower" },
{ "a", "A", true, "lower, upper singles" },
{ "A", "a", true, "upper, lower singles" },
{ "AbcDEf", "aBcdEF", true, "mixed upper lower" },
{ "A", "aa", false, "mixed case, different lengths" },
};
2021-11-22 10:42:19 +10:00
cruft::string::equality::lower cmp;
2019-05-02 11:03:22 +10:00
for (auto const &t: TESTS) {
2021-11-22 10:42:19 +10:00
tap.expect_eq (cmp (t.a, t.b), t.equal, "equality::lower, pointers, {}", t.message);
2019-05-02 11:03:22 +10:00
}
}
///////////////////////////////////////////////////////////////////////////////
int
main (int, char**)
{
cruft::TAP::logger tap;
2019-11-26 08:10:04 +11:00
test_transforms (tap);
test_tokeniser (tap);
test_position (tap);
2019-02-20 14:44:33 +11:00
test_contains (tap);
2019-05-02 11:03:22 +10:00
test_comparator_less (tap);
test_comparator_lower (tap);
2016-03-17 18:13:19 +11:00
return tap.status ();
}