#include #define DOCTEST_CONFIG_IMPLEMENT_WITH_MAIN #include "../include/ss/converter.hpp" #include "doctest.h" #include class buffer { constexpr static auto buff_size = 1024; char data_[buff_size]; public: char* operator()(const char* data) { memset(data_, '\0', buff_size); strcpy(data_, data); return data_; } }; buffer buff; TEST_CASE("testing splitter with escaping") { std::vector values{"10", "he\\\"llo", "\\\"", "\\\"a\\,a\\\"", "3.33", "a\\\""}; char buff[128]; // with quote ss::splitter, ss::escape<'\\'>> s; std::string delim = ","; for (size_t i = 0; i < values.size() * values.size(); ++i) { std::string input1; std::string input2; for (size_t j = 0; j < values.size(); ++j) { if (i & (1 << j) && j != 2 && j != 3) { input1.append(values[j]); input2.append(values.at(values.size() - 1 - j)); } else { input1.append("\"" + values[j] + "\""); input2.append("\"" + values.at(values.size() - 1 - j) + "\""); } input1.append(delim); input2.append(delim); } input1.pop_back(); input2.pop_back(); input1.append("\0\""); input2.append("\0\""); memcpy(buff, input1.c_str(), input1.size() + 1); auto tup1 = s.split(buff, delim); CHECK(tup1.size() == 6); memcpy(buff, input2.c_str(), input2.size() + 1); auto tup2 = s.split(buff, delim); CHECK(tup2.size() == 6); } } /* TEST_CASE("testing quoting without escaping") { std::vector values{"10", "hello", ",", "a,a", "3.33", "a"}; // with quote ss::converter c; for (size_t i = 0; i < values.size() * values.size(); ++i) { std::string input1; std::string input2; for (size_t j = 0; j < values.size(); ++j) { if (i & (1 << j) && j != 2 && j != 3) { input1.append(values[j]); input2.append(values.at(values.size() - 1 - j)); } else { input1.append("\"" + values[j] + "\""); input2.append("\"" + values.at(values.size() - 1 - j) + "\""); } input1.append("__"); input1.push_back(','); input1.append("__"); input2.push_back(','); } input1.pop_back(); input1.pop_back(); input1.pop_back(); input2.pop_back(); input1.append("\0\""); input2.append("\0\""); auto tup1 = c.convert(input1.c_str(), ","); if (!c.valid()) { FAIL("invalid: " + input1); } else { auto [a, b, c, d, e, f] = tup1; CHECK(a == 10); CHECK(b == "hello"); CHECK(c == ","); CHECK(d == "a,a"); CHECK(e == 3.33); CHECK(f == 'a'); } auto tup2 = c.convert(input2.c_str(), ","); if (!c.valid()) { FAIL("invalid: " + input2); } else { auto [f, e, d, c, b, a] = tup2; CHECK(a == 10); CHECK(b == "hello"); CHECK(c == ","); CHECK(d == "a,a"); CHECK(e == 3.33); CHECK(f == 'a'); } } } */ TEST_CASE("testing split") { ss::converter c; for (const auto& [s, expected, delim] : // clang-format off {std::tuple{"a,b,c,d", std::vector{"a", "b", "c", "d"}, ","}, {"", {}, " "}, {" x x x x | x ", {" x x x x ", " x "}, "|"}, {"a::b::c::d", {"a", "b", "c", "d"}, "::"}, {"x\t-\ty", {"x", "y"}, "\t-\t"}, {"x", {"x"}, ","}} // clang-format on ) { auto split = c.split(buff(s), delim); CHECK(split.size() == expected.size()); for (size_t i = 0; i < split.size(); ++i) { auto s = std::string(split[i].first, split[i].second); CHECK(s == expected[i]); } } } TEST_CASE("testing valid conversions") { ss::converter c; { auto tup = c.convert(buff("5")); REQUIRE(c.valid()); CHECK(tup == 5); } { auto tup = c.convert(buff("5,junk")); REQUIRE(c.valid()); CHECK(tup == 5); } { auto tup = c.convert(buff("junk,5")); REQUIRE(c.valid()); CHECK(tup == 5); } { auto tup = c.convert(buff("5\njunk\njunk"), "\n"); REQUIRE(c.valid()); CHECK(tup == 5); } { // TODO make \t -> ' ' auto tup = c.convert(buff("junk\t5\tjunk"), "\t"); REQUIRE(c.valid()); CHECK(tup == 5); } { auto tup = c.convert(buff("junk\tjunk\t5"), "\t"); REQUIRE(c.valid()); CHECK(tup == 5); } { auto tup = c.convert>(buff("junk\tjunk\t5"), "\t"); REQUIRE(c.valid()); REQUIRE(tup.has_value()); CHECK(tup == 5); } { auto tup = c.convert(buff("5,6.6,junk")); REQUIRE(c.valid()); CHECK(tup == std::tuple{5, 6.6}); } { auto tup = c.convert(buff("5,junk,6.6")); REQUIRE(c.valid()); CHECK(tup == std::tuple{5, 6.6}); } { auto tup = c.convert(buff("junk;5;6.6"), ";"); REQUIRE(c.valid()); CHECK(tup == std::tuple{5, 6.6}); } { auto tup = c.convert, double>(buff("junk;5;6.6"), ";"); REQUIRE(c.valid()); REQUIRE(std::get<0>(tup).has_value()); CHECK(tup == std::tuple{5, 6.6}); } { auto tup = c.convert, double>(buff("junk;5.4;6.6"), ";"); REQUIRE(c.valid()); REQUIRE(!std::get<0>(tup).has_value()); CHECK(tup == std::tuple{std::nullopt, 6.6}); } { auto tup = c.convert, double>(buff("junk;5;6.6"), ";"); REQUIRE(c.valid()); REQUIRE(std::holds_alternative(std::get<0>(tup))); CHECK(tup == std::tuple{std::variant{5}, 6.6}); } { auto tup = c.convert, double>(buff("junk;5.5;6.6"), ";"); REQUIRE(c.valid()); REQUIRE(std::holds_alternative(std::get<0>(tup))); CHECK(tup == std::tuple{std::variant{5.5}, 6.6}); } } TEST_CASE("testing invalid conversions") { ss::converter c; c.convert(buff("")); REQUIRE(!c.valid()); c.convert(buff("")); REQUIRE(!c.valid()); c.convert(buff(",junk")); REQUIRE(!c.valid()); c.convert(buff("junk,")); REQUIRE(!c.valid()); c.convert(buff("x")); REQUIRE(!c.valid()); c.convert(buff("x")); REQUIRE(!c.valid()); c.convert(buff("x,junk")); REQUIRE(!c.valid()); c.convert(buff("junk,x")); REQUIRE(!c.valid()); c.convert, double>(buff("junk;.5.5;6"), ";"); REQUIRE(!c.valid()); } TEST_CASE("testing ss:ax restriction (all except)") { ss::converter c; c.convert>(buff("0")); REQUIRE(!c.valid()); c.convert>(buff("1")); REQUIRE(!c.valid()); c.convert>(buff("junk,c,1")); REQUIRE(!c.valid()); c.convert, char>(buff("1,c")); REQUIRE(!c.valid()); { int tup = c.convert>(buff("3")); REQUIRE(c.valid()); CHECK(tup == 3); } { std::tuple tup = c.convert>(buff("c,3")); REQUIRE(c.valid()); CHECK(tup == std::tuple{'c', 3}); } { std::tuple tup = c.convert, char>(buff("3,c")); REQUIRE(c.valid()); CHECK(tup == std::tuple{3, 'c'}); } } TEST_CASE("testing ss:nx restriction (none except)") { ss::converter c; c.convert>(buff("3")); REQUIRE(!c.valid()); c.convert>(buff("c,3")); REQUIRE(!c.valid()); c.convert, char>(buff("3,c")); REQUIRE(!c.valid()); { auto tup = c.convert>(buff("3")); REQUIRE(c.valid()); CHECK(tup == 3); } { auto tup = c.convert>(buff("2")); REQUIRE(c.valid()); CHECK(tup == 2); } { auto tup = c.convert>(buff("c,junk,1")); REQUIRE(c.valid()); CHECK(tup == std::tuple{'c', 1}); } { auto tup = c.convert, char>(buff("1,c")); REQUIRE(c.valid()); CHECK(tup == std::tuple{1, 'c'}); } } TEST_CASE("testing ss:ir restriction (in range)") { ss::converter c; c.convert>(buff("3")); REQUIRE(!c.valid()); c.convert>(buff("c,3")); REQUIRE(!c.valid()); c.convert, char>(buff("3,c")); REQUIRE(!c.valid()); { auto tup = c.convert>(buff("3")); REQUIRE(c.valid()); CHECK(tup == 3); } { auto tup = c.convert>(buff("2")); REQUIRE(c.valid()); CHECK(tup == 2); } { auto tup = c.convert>(buff("c,junk,1")); REQUIRE(c.valid()); CHECK(tup == std::tuple{'c', 1}); } { auto tup = c.convert, char>(buff("1,c")); REQUIRE(c.valid()); CHECK(tup == std::tuple{1, 'c'}); } } TEST_CASE("testing ss:oor restriction (out of range)") { ss::converter c; c.convert>(buff("3")); REQUIRE(!c.valid()); c.convert>(buff("2")); REQUIRE(!c.valid()); c.convert, void>(buff("c,1,junk")); REQUIRE(!c.valid()); c.convert, char>(buff("1,c")); REQUIRE(!c.valid()); { auto tup = c.convert>(buff("3")); REQUIRE(c.valid()); CHECK(tup == 3); } { auto tup = c.convert>(buff("c,junk,3")); REQUIRE(c.valid()); CHECK(tup == std::tuple{'c', 3}); } { auto tup = c.convert, char>(buff("3,c")); REQUIRE(c.valid()); CHECK(tup == std::tuple{3, 'c'}); } } const std::vector extracted_vector = {1, 2, 3}; // custom extract template <> inline bool ss::extract(const char* begin, const char* end, std::vector& value) { if (begin == end) { return false; } value = extracted_vector; return true; } TEST_CASE("testing ss:ne restriction (not empty)") { ss::converter c; c.convert>(buff("")); REQUIRE(!c.valid()); c.convert>(buff("3,")); REQUIRE(!c.valid()); c.convert, int>(buff(",3")); REQUIRE(!c.valid()); c.convert, int>(buff("junk,,3")); REQUIRE(!c.valid()); c.convert>>(buff("")); REQUIRE(!c.valid()); { auto tup = c.convert>(buff("s")); REQUIRE(c.valid()); CHECK(tup == "s"); } { auto tup = c.convert, ss::ne>(buff("1,s")); REQUIRE(c.valid()); CHECK(tup == std::tuple{1, "s"}); } { auto tup = c.convert>>(buff("{1 2 3}")); REQUIRE(c.valid()); CHECK(tup == extracted_vector); } } TEST_CASE("testing ss:lt ss::lte ss::gt ss::gte restriction (in range)") { ss::converter c; c.convert>(buff("3")); REQUIRE(!c.valid()); c.convert>(buff("3")); REQUIRE(!c.valid()); c.convert>(buff("3")); REQUIRE(!c.valid()); c.convert>(buff("3")); REQUIRE(!c.valid()); c.convert>(buff("3")); REQUIRE(!c.valid()); c.convert>(buff("3")); REQUIRE(!c.valid()); { auto tup = c.convert>(buff("3")); REQUIRE(c.valid()); CHECK(tup == 3); } { auto tup = c.convert>(buff("3")); REQUIRE(c.valid()); CHECK(tup == 3); } { auto tup = c.convert>(buff("3")); REQUIRE(c.valid()); CHECK(tup == 3); } { auto tup = c.convert>(buff("3")); REQUIRE(c.valid()); CHECK(tup == 3); } { auto tup = c.convert>(buff("3")); REQUIRE(c.valid()); CHECK(tup == 3); } { auto tup = c.convert>(buff("3")); REQUIRE(c.valid()); CHECK(tup == 3); } } TEST_CASE("testing error mode") { ss::converter c; c.convert(buff("junk")); CHECK(!c.valid()); CHECK(c.error_msg().empty()); c.set_error_mode(ss::error_mode::error_string); c.convert(buff("junk")); CHECK(!c.valid()); CHECK(!c.error_msg().empty()); }