#include "test_helpers.hpp"
#include <algorithm>
#include <ss/converter.hpp>

TEST_CASE("converter test split") {
    ss::converter c;
    for (const auto& [s, expected, delim] :
         // clang-format off
                {std::make_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(s, delim);
        CHECK_EQ(split.size(), expected.size());
        for (size_t i = 0; i < split.size(); ++i) {
            auto s = std::string(split[i].first, split[i].second);
            CHECK_EQ(s, expected[i]);
        }
    }
}

TEST_CASE("converter test split with exceptions") {
    ss::converter<ss::throw_on_error> c;
    try {
        for (const auto& [s, expected, delim] :
             // clang-format off
                {std::make_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(s, delim);
            CHECK_EQ(split.size(), expected.size());
            for (size_t i = 0; i < split.size(); ++i) {
                auto s = std::string(split[i].first, split[i].second);
                CHECK_EQ(s, expected[i]);
            }
        }
    } catch (ss::exception& e) {
        FAIL(std::string{e.what()});
    }
}

TEST_CASE("converter test valid conversions") {
    ss::converter c;

    {
        auto tup = c.convert<int>("5");
        REQUIRE(c.valid());
        CHECK_EQ(tup, 5);
    }
    {
        auto tup = c.convert<int, void>("5,junk");
        REQUIRE(c.valid());
        CHECK_EQ(tup, 5);
    }
    {
        auto tup = c.convert<void, int>("junk,5");
        REQUIRE(c.valid());
        CHECK_EQ(tup, 5);
    }
    {
        auto tup = c.convert<int, void, void>("5\njunk\njunk", "\n");
        REQUIRE(c.valid());
        CHECK_EQ(tup, 5);
    }
    {
        auto tup = c.convert<void, int, void>("junk 5 junk", " ");
        REQUIRE(c.valid());
        CHECK_EQ(tup, 5);
    }
    {
        auto tup = c.convert<void, void, int>("junk\tjunk\t5", "\t");
        REQUIRE(c.valid());
        CHECK_EQ(tup, 5);
    }
    {
        auto tup =
            c.convert<void, void, std::optional<int>>("junk\tjunk\t5", "\t");
        REQUIRE(c.valid());
        REQUIRE(tup.has_value());
        CHECK_EQ(tup, 5);
    }
    {
        auto tup = c.convert<int, double, void>("5,6.6,junk");
        REQUIRE(c.valid());
        CHECK_EQ(tup, std::make_tuple(5, 6.6));
    }
    {
        auto tup = c.convert<int, void, double>("5,junk,6.6");
        REQUIRE(c.valid());
        CHECK_EQ(tup, std::make_tuple(5, 6.6));
    }
    {
        auto tup = c.convert<void, int, double>("junk;5;6.6", ";");
        REQUIRE(c.valid());
        CHECK_EQ(tup, std::make_tuple(5, 6.6));
    }
    {
        auto tup =
            c.convert<void, std::optional<int>, double>("junk;5;6.6", ";");
        REQUIRE(c.valid());
        REQUIRE(std::get<0>(tup).has_value());
        CHECK_EQ(tup, std::make_tuple(5, 6.6));
    }
    {
        auto tup =
            c.convert<void, std::optional<int>, double>("junk;5.4;6.6", ";");
        REQUIRE(c.valid());
        REQUIRE_FALSE(std::get<0>(tup).has_value());
        CHECK_EQ(tup, std::make_tuple(std::optional<int>{}, 6.6));
    }
    {
        auto tup =
            c.convert<void, std::variant<int, double>, double>("junk;5;6.6",
                                                               ";");
        REQUIRE(c.valid());
        REQUIRE(std::holds_alternative<int>(std::get<0>(tup)));
        CHECK_EQ(tup, std::make_tuple(std::variant<int, double>{5}, 6.6));
    }
    {
        auto tup =
            c.convert<void, std::variant<int, double>, double>("junk;5.5;6.6",
                                                               ";");
        REQUIRE(c.valid());
        REQUIRE(std::holds_alternative<double>(std::get<0>(tup)));
        CHECK_EQ(tup, std::make_tuple(std::variant<int, double>{5.5}, 6.6));
    }
    {
        auto tup = c.convert<void, std::string_view, double,
                             std::string_view>("junk;s1;6.6;s2", ";");
        REQUIRE(c.valid());
        CHECK_EQ(tup, std::make_tuple(std::string_view{"s1"}, 6.6,
                                      std::string_view{"s2"}));
    }
}

TEST_CASE("converter test valid conversions with exceptions") {
    ss::converter<ss::throw_on_error> c;

    try {
        auto tup = c.convert<int>("5");
        REQUIRE(c.valid());
        CHECK_EQ(tup, 5);
    } catch (ss::exception& e) {
        FAIL(std::string{e.what()});
    }

    try {
        auto tup = c.convert<int, void>("5,junk");
        REQUIRE(c.valid());
        CHECK_EQ(tup, 5);
    } catch (ss::exception& e) {
        FAIL(std::string{e.what()});
    }

    try {
        auto tup = c.convert<void, int>("junk,5");
        REQUIRE(c.valid());
        CHECK_EQ(tup, 5);
    } catch (ss::exception& e) {
        FAIL(std::string{e.what()});
    }

    try {
        auto tup = c.convert<int, void, void>("5\njunk\njunk", "\n");
        REQUIRE(c.valid());
        CHECK_EQ(tup, 5);
    } catch (ss::exception& e) {
        FAIL(std::string{e.what()});
    }

    try {
        auto tup = c.convert<void, int, void>("junk 5 junk", " ");
        REQUIRE(c.valid());
        CHECK_EQ(tup, 5);
    } catch (ss::exception& e) {
        FAIL(std::string{e.what()});
    }

    try {
        auto tup = c.convert<void, void, int>("junk\tjunk\t5", "\t");
        REQUIRE(c.valid());
        CHECK_EQ(tup, 5);
    } catch (ss::exception& e) {
        FAIL(std::string{e.what()});
    }

    try {
        auto tup =
            c.convert<void, void, std::optional<int>>("junk\tjunk\t5", "\t");
        REQUIRE(c.valid());
        REQUIRE(tup.has_value());
        CHECK_EQ(tup, 5);
    } catch (ss::exception& e) {
        FAIL(std::string{e.what()});
    }

    try {
        auto tup = c.convert<int, double, void>("5,6.6,junk");
        REQUIRE(c.valid());
        CHECK_EQ(tup, std::make_tuple(5, 6.6));
    } catch (ss::exception& e) {
        FAIL(std::string{e.what()});
    }

    try {
        auto tup = c.convert<int, void, double>("5,junk,6.6");
        REQUIRE(c.valid());
        CHECK_EQ(tup, std::make_tuple(5, 6.6));
    } catch (ss::exception& e) {
        FAIL(std::string{e.what()});
    }

    try {
        auto tup = c.convert<void, int, double>("junk;5;6.6", ";");
        REQUIRE(c.valid());
        CHECK_EQ(tup, std::make_tuple(5, 6.6));
    } catch (ss::exception& e) {
        FAIL(std::string{e.what()});
    }

    try {
        auto tup =
            c.convert<void, std::optional<int>, double>("junk;5;6.6", ";");
        REQUIRE(c.valid());
        REQUIRE(std::get<0>(tup).has_value());
        CHECK_EQ(tup, std::make_tuple(5, 6.6));
    } catch (ss::exception& e) {
        FAIL(std::string{e.what()});
    }

    try {
        auto tup =
            c.convert<void, std::optional<int>, double>("junk;5.4;6.6", ";");
        REQUIRE(c.valid());
        REQUIRE_FALSE(std::get<0>(tup).has_value());
        CHECK_EQ(tup, std::make_tuple(std::optional<int>{}, 6.6));
    } catch (ss::exception& e) {
        FAIL(std::string{e.what()});
    }

    try {
        auto tup =
            c.convert<void, std::variant<int, double>, double>("junk;5;6.6",
                                                               ";");
        REQUIRE(c.valid());
        REQUIRE(std::holds_alternative<int>(std::get<0>(tup)));
        CHECK_EQ(tup, std::make_tuple(std::variant<int, double>{5}, 6.6));
    } catch (ss::exception& e) {
        FAIL(std::string{e.what()});
    }

    try {
        auto tup =
            c.convert<void, std::variant<int, double>, double>("junk;5.5;6.6",
                                                               ";");
        REQUIRE(c.valid());
        REQUIRE(std::holds_alternative<double>(std::get<0>(tup)));
        CHECK_EQ(tup, std::make_tuple(std::variant<int, double>{5.5}, 6.6));
    } catch (ss::exception& e) {
        FAIL(std::string{e.what()});
    }

    try {
        auto tup = c.convert<void, std::string_view, double,
                             std::string_view>("junk;s1;6.6;s2", ";");
        REQUIRE(c.valid());
        CHECK_EQ(tup, std::make_tuple(std::string_view{"s1"}, 6.6,
                                      std::string_view{"s2"}));
    } catch (ss::exception& e) {
        FAIL(std::string{e.what()});
    }
}

TEST_CASE("converter test invalid conversions") {
    ss::converter c;

    c.convert<int>("");
    REQUIRE_FALSE(c.valid());

    c.convert<int>("1", "");
    REQUIRE_FALSE(c.valid());

    c.convert<int>("10", "");
    REQUIRE_FALSE(c.valid());

    c.convert<int, void>("");
    REQUIRE_FALSE(c.valid());

    c.convert<int, void>(",junk");
    REQUIRE_FALSE(c.valid());

    c.convert<void, int>("junk,");
    REQUIRE_FALSE(c.valid());

    c.convert<int>("x");
    REQUIRE_FALSE(c.valid());

    c.convert<int, void>("x");
    REQUIRE_FALSE(c.valid());

    c.convert<int, void>("x,junk");
    REQUIRE_FALSE(c.valid());

    c.convert<void, int>("junk,x");
    REQUIRE_FALSE(c.valid());

    c.convert<void, std::variant<int, double>, double>("junk;.5.5;6", ";");
    REQUIRE_FALSE(c.valid());
}

TEST_CASE("converter test invalid conversions with exceptions") {
    ss::converter<ss::throw_on_error> c;

    REQUIRE_EXCEPTION(c.convert<int>(""));
    REQUIRE_EXCEPTION(c.convert<int>("1", ""));
    REQUIRE_EXCEPTION(c.convert<int>("10", ""));
    REQUIRE_EXCEPTION(c.convert<int, void>(""));
    REQUIRE_EXCEPTION(c.convert<int, void>(",junk"));
    REQUIRE_EXCEPTION(c.convert<void, int>("junk,"));
    REQUIRE_EXCEPTION(c.convert<int>("x"));
    REQUIRE_EXCEPTION(c.convert<int, void>("x"));
    REQUIRE_EXCEPTION(c.convert<int, void>("x,junk"));
    REQUIRE_EXCEPTION(c.convert<void, int>("junk,x"));
    REQUIRE_EXCEPTION(
        c.convert<void, std::variant<int, double>, double>("junk;.5.5;6", ";"));
}

TEST_CASE("converter test ss:ax restriction (all except)") {
    ss::converter c;

    c.convert<ss::ax<int, 0>>("0");
    REQUIRE_FALSE(c.valid());

    c.convert<ss::ax<int, 0, 1, 2>>("1");
    REQUIRE_FALSE(c.valid());

    c.convert<void, char, ss::ax<int, 0, 1, 2>>("junk,c,1");
    REQUIRE_FALSE(c.valid());

    c.convert<ss::ax<int, 1>, char>("1,c");
    REQUIRE_FALSE(c.valid());
    {
        int tup = c.convert<ss::ax<int, 1>>("3");
        REQUIRE(c.valid());
        CHECK_EQ(tup, 3);
    }
    {
        std::tuple<char, int> tup = c.convert<char, ss::ax<int, 1>>("c,3");
        REQUIRE(c.valid());
        CHECK_EQ(tup, std::make_tuple('c', 3));
    }
    {
        std::tuple<int, char> tup = c.convert<ss::ax<int, 1>, char>("3,c");
        REQUIRE(c.valid());
        CHECK_EQ(tup, std::make_tuple(3, 'c'));
    }
}

TEST_CASE("converter test ss:ax restriction (all except) with exceptions") {
    ss::converter<ss::throw_on_error> c;

    REQUIRE_EXCEPTION(c.convert<ss::ax<int, 0>>("0"));
    REQUIRE_EXCEPTION(c.convert<ss::ax<int, 0, 1, 2>>("1"));
    REQUIRE_EXCEPTION(c.convert<void, char, ss::ax<int, 0, 1, 2>>("junk,c,1"));
    REQUIRE_EXCEPTION(c.convert<ss::ax<int, 1>, char>("1,c"));

    try {
        {
            int tup = c.convert<ss::ax<int, 1>>("3");
            CHECK_EQ(tup, 3);
        }
        {
            std::tuple<char, int> tup = c.convert<char, ss::ax<int, 1>>("c,3");
            CHECK_EQ(tup, std::make_tuple('c', 3));
        }
        {
            std::tuple<int, char> tup = c.convert<ss::ax<int, 1>, char>("3,c");
            CHECK_EQ(tup, std::make_tuple(3, 'c'));
        }
    } catch (ss::exception& e) {
        FAIL(std::string{e.what()});
    }
}

TEST_CASE("converter test ss:nx restriction (none except)") {
    ss::converter c;

    c.convert<ss::nx<int, 1>>("3");
    REQUIRE_FALSE(c.valid());

    c.convert<char, ss::nx<int, 1, 2, 69>>("c,3");
    REQUIRE_FALSE(c.valid());

    c.convert<ss::nx<int, 1>, char>("3,c");
    REQUIRE_FALSE(c.valid());

    {
        auto tup = c.convert<ss::nx<int, 3>>("3");
        REQUIRE(c.valid());
        CHECK_EQ(tup, 3);
    }
    {
        auto tup = c.convert<ss::nx<int, 0, 1, 2>>("2");
        REQUIRE(c.valid());
        CHECK_EQ(tup, 2);
    }
    {
        auto tup = c.convert<char, void, ss::nx<int, 0, 1, 2>>("c,junk,1");
        REQUIRE(c.valid());
        CHECK_EQ(tup, std::make_tuple('c', 1));
    }
    {
        auto tup = c.convert<ss::nx<int, 1>, char>("1,c");
        REQUIRE(c.valid());
        CHECK_EQ(tup, std::make_tuple(1, 'c'));
    }
}

TEST_CASE("converter test ss:nx restriction (none except) with exceptions") {
    ss::converter<ss::throw_on_error> c;

    REQUIRE_EXCEPTION(c.convert<ss::nx<int, 1>>("3"));
    REQUIRE_EXCEPTION(c.convert<char, ss::nx<int, 1, 2, 69>>("c,3"));
    REQUIRE_EXCEPTION(c.convert<ss::nx<int, 1>, char>("3,c"));

    try {
        {
            auto tup = c.convert<ss::nx<int, 3>>("3");
            REQUIRE(c.valid());
            CHECK_EQ(tup, 3);
        }
        {
            auto tup = c.convert<ss::nx<int, 0, 1, 2>>("2");
            REQUIRE(c.valid());
            CHECK_EQ(tup, 2);
        }
        {
            auto tup = c.convert<char, void, ss::nx<int, 0, 1, 2>>("c,junk,1");
            REQUIRE(c.valid());
            CHECK_EQ(tup, std::make_tuple('c', 1));
        }
        {
            auto tup = c.convert<ss::nx<int, 1>, char>("1,c");
            REQUIRE(c.valid());
            CHECK_EQ(tup, std::make_tuple(1, 'c'));
        }
    } catch (ss::exception& e) {
        FAIL(std::string{e.what()});
    }
}

TEST_CASE("converter test ss:ir restriction (in range)") {
    ss::converter c;

    c.convert<ss::ir<int, 0, 2>>("3");
    REQUIRE_FALSE(c.valid());

    c.convert<char, ss::ir<int, 4, 69>>("c,3");
    REQUIRE_FALSE(c.valid());

    c.convert<ss::ir<int, 1, 2>, char>("3,c");
    REQUIRE_FALSE(c.valid());

    {
        auto tup = c.convert<ss::ir<int, 1, 5>>("3");
        REQUIRE(c.valid());
        CHECK_EQ(tup, 3);
    }
    {
        auto tup = c.convert<ss::ir<int, 0, 2>>("2");
        REQUIRE(c.valid());
        CHECK_EQ(tup, 2);
    }
    {
        auto tup = c.convert<char, void, ss::ir<int, 0, 1>>("c,junk,1");
        REQUIRE(c.valid());
        CHECK_EQ(tup, std::make_tuple('c', 1));
    }
    {
        auto tup = c.convert<ss::ir<int, 1, 20>, char>("1,c");
        REQUIRE(c.valid());
        CHECK_EQ(tup, std::make_tuple(1, 'c'));
    }
}

TEST_CASE("converter test ss:ir restriction (in range) with exceptions") {
    ss::converter<ss::throw_on_error> c;

    REQUIRE_EXCEPTION(c.convert<ss::ir<int, 0, 2>>("3"));
    REQUIRE_EXCEPTION(c.convert<char, ss::ir<int, 4, 69>>("c,3"));
    REQUIRE_EXCEPTION(c.convert<ss::ir<int, 1, 2>, char>("3,c"));

    try {
        {
            auto tup = c.convert<ss::ir<int, 1, 5>>("3");
            REQUIRE(c.valid());
            CHECK_EQ(tup, 3);
        }
        {
            auto tup = c.convert<ss::ir<int, 0, 2>>("2");
            REQUIRE(c.valid());
            CHECK_EQ(tup, 2);
        }
        {
            auto tup = c.convert<char, void, ss::ir<int, 0, 1>>("c,junk,1");
            REQUIRE(c.valid());
            CHECK_EQ(tup, std::make_tuple('c', 1));
        }
        {
            auto tup = c.convert<ss::ir<int, 1, 20>, char>("1,c");
            REQUIRE(c.valid());
            CHECK_EQ(tup, std::make_tuple(1, 'c'));
        }
    } catch (ss::exception& e) {
        FAIL(std::string{e.what()});
    }
}

TEST_CASE("converter test ss:oor restriction (out of range)") {
    ss::converter c;

    c.convert<ss::oor<int, 1, 5>>("3");
    REQUIRE_FALSE(c.valid());

    c.convert<ss::oor<int, 0, 2>>("2");
    REQUIRE_FALSE(c.valid());

    c.convert<char, ss::oor<int, 0, 1>, void>("c,1,junk");
    REQUIRE_FALSE(c.valid());

    c.convert<ss::oor<int, 1, 20>, char>("1,c");
    REQUIRE_FALSE(c.valid());

    {
        auto tup = c.convert<ss::oor<int, 0, 2>>("3");
        REQUIRE(c.valid());
        CHECK_EQ(tup, 3);
    }

    {
        auto tup = c.convert<char, void, ss::oor<int, 4, 69>>("c,junk,3");
        REQUIRE(c.valid());
        CHECK_EQ(tup, std::make_tuple('c', 3));
    }

    {
        auto tup = c.convert<ss::oor<int, 1, 2>, char>("3,c");
        REQUIRE(c.valid());
        CHECK_EQ(tup, std::make_tuple(3, 'c'));
    }
}

TEST_CASE("converter test ss:oor restriction (out of range) with exceptions") {
    ss::converter<ss::throw_on_error> c;

    REQUIRE_EXCEPTION(c.convert<ss::oor<int, 1, 5>>("3"));
    REQUIRE_EXCEPTION(c.convert<ss::oor<int, 0, 2>>("2"));
    REQUIRE_EXCEPTION(c.convert<char, ss::oor<int, 0, 1>, void>("c,1,junk"));
    REQUIRE_EXCEPTION(c.convert<ss::oor<int, 1, 20>, char>("1,c"));

    try {
        {
            auto tup = c.convert<ss::oor<int, 0, 2>>("3");
            REQUIRE(c.valid());
            CHECK_EQ(tup, 3);
        }

        {
            auto tup = c.convert<char, void, ss::oor<int, 4, 69>>("c,junk,3");
            REQUIRE(c.valid());
            CHECK_EQ(tup, std::make_tuple('c', 3));
        }

        {
            auto tup = c.convert<ss::oor<int, 1, 2>, char>("3,c");
            REQUIRE(c.valid());
            CHECK_EQ(tup, std::make_tuple(3, 'c'));
        }
    } catch (ss::exception& e) {
        FAIL(std::string{e.what()});
    }
}

const inline std::vector<int> extracted_vector = {1, 2, 3};

// custom extract
template <>
inline bool ss::extract(const char* begin, const char* end,
                        std::vector<int>& value) {
    if (begin == end) {
        return false;
    }
    value = extracted_vector;
    return true;
}

TEST_CASE("converter test ss:ne restriction (not empty)") {
    ss::converter c;

    c.convert<ss::ne<std::string>>("");
    REQUIRE_FALSE(c.valid());

    c.convert<int, ss::ne<std::string>>("3,");
    REQUIRE_FALSE(c.valid());

    c.convert<ss::ne<std::string>, int>(",3");
    REQUIRE_FALSE(c.valid());

    c.convert<void, ss::ne<std::string>, int>("junk,,3");
    REQUIRE_FALSE(c.valid());

    c.convert<ss::ne<std::vector<int>>>("");
    REQUIRE_FALSE(c.valid());

    {
        auto tup = c.convert<ss::ne<std::string>>("s");
        REQUIRE(c.valid());
        CHECK_EQ(tup, "s");
    }
    {
        auto tup = c.convert<std::optional<int>, ss::ne<std::string>>("1,s");
        REQUIRE(c.valid());
        CHECK_EQ(tup, std::make_tuple(1, "s"));
    }
    {
        auto tup = c.convert<ss::ne<std::vector<int>>>("{1 2 3}");
        REQUIRE(c.valid());
        CHECK_EQ(tup, extracted_vector);
    }
}

TEST_CASE("converter test ss:ne restriction (not empty) with exceptions") {
    ss::converter<ss::throw_on_error> c;

    REQUIRE_EXCEPTION(c.convert<ss::ne<std::string>>(""));
    REQUIRE_EXCEPTION(c.convert<int, ss::ne<std::string>>("3,"));
    REQUIRE_EXCEPTION(c.convert<ss::ne<std::string>, int>(",3"));
    REQUIRE_EXCEPTION(c.convert<void, ss::ne<std::string>, int>("junk,,3"));
    REQUIRE_EXCEPTION(c.convert<ss::ne<std::vector<int>>>(""));

    try {
        {
            auto tup = c.convert<ss::ne<std::string>>("s");
            REQUIRE(c.valid());
            CHECK_EQ(tup, "s");
        }
        {
            auto tup =
                c.convert<std::optional<int>, ss::ne<std::string>>("1,s");
            REQUIRE(c.valid());
            CHECK_EQ(tup, std::make_tuple(1, "s"));
        }
        {
            auto tup = c.convert<ss::ne<std::vector<int>>>("{1 2 3}");
            REQUIRE(c.valid());
            CHECK_EQ(tup, extracted_vector);
        }
    } catch (ss::exception& e) {
        FAIL(std::string{e.what()});
    }
}

TEST_CASE(
    "converter test ss:lt ss::lte ss::gt ss::gte restriction (in range)") {
    ss::converter c;

    c.convert<ss::lt<int, 3>>("3");
    REQUIRE_FALSE(c.valid());

    c.convert<ss::lt<int, 2>>("3");
    REQUIRE_FALSE(c.valid());

    c.convert<ss::gt<int, 3>>("3");
    REQUIRE_FALSE(c.valid());

    c.convert<ss::gt<int, 4>>("3");
    REQUIRE_FALSE(c.valid());

    c.convert<ss::lte<int, 2>>("3");
    REQUIRE_FALSE(c.valid());

    c.convert<ss::gte<int, 4>>("3");
    REQUIRE_FALSE(c.valid());

    {
        auto tup = c.convert<ss::lt<int, 4>>("3");
        REQUIRE(c.valid());
        CHECK_EQ(tup, 3);
    }

    {
        auto tup = c.convert<ss::gt<int, 2>>("3");
        REQUIRE(c.valid());
        CHECK_EQ(tup, 3);
    }

    {
        auto tup = c.convert<ss::lte<int, 4>>("3");
        REQUIRE(c.valid());
        CHECK_EQ(tup, 3);
    }

    {
        auto tup = c.convert<ss::lte<int, 3>>("3");
        REQUIRE(c.valid());
        CHECK_EQ(tup, 3);
    }

    {
        auto tup = c.convert<ss::gte<int, 2>>("3");
        REQUIRE(c.valid());
        CHECK_EQ(tup, 3);
    }

    {
        auto tup = c.convert<ss::gte<int, 3>>("3");
        REQUIRE(c.valid());
        CHECK_EQ(tup, 3);
    }
}

TEST_CASE("converter test ss:lt ss::lte ss::gt ss::gte restriction (in range) "
          "with exception") {
    ss::converter<ss::throw_on_error> c;

    REQUIRE_EXCEPTION(c.convert<ss::lt<int, 3>>("3"));
    REQUIRE_EXCEPTION(c.convert<ss::lt<int, 2>>("3"));
    REQUIRE_EXCEPTION(c.convert<ss::gt<int, 3>>("3"));
    REQUIRE_EXCEPTION(c.convert<ss::gt<int, 4>>("3"));
    REQUIRE_EXCEPTION(c.convert<ss::lte<int, 2>>("3"));
    REQUIRE_EXCEPTION(c.convert<ss::gte<int, 4>>("3"));

    try {
        {
            auto tup = c.convert<ss::lt<int, 4>>("3");
            REQUIRE(c.valid());
            CHECK_EQ(tup, 3);
        }

        {
            auto tup = c.convert<ss::gt<int, 2>>("3");
            REQUIRE(c.valid());
            CHECK_EQ(tup, 3);
        }

        {
            auto tup = c.convert<ss::lte<int, 4>>("3");
            REQUIRE(c.valid());
            CHECK_EQ(tup, 3);
        }

        {
            auto tup = c.convert<ss::lte<int, 3>>("3");
            REQUIRE(c.valid());
            CHECK_EQ(tup, 3);
        }

        {
            auto tup = c.convert<ss::gte<int, 2>>("3");
            REQUIRE(c.valid());
            CHECK_EQ(tup, 3);
        }

        {
            auto tup = c.convert<ss::gte<int, 3>>("3");
            REQUIRE(c.valid());
            CHECK_EQ(tup, 3);
        }
    } catch (ss::exception& e) {
        FAIL(std::string{e.what()});
    }
}

TEST_CASE("converter test error mode") {
    ss::converter<ss::string_error> c;
    c.convert<int>("junk");
    CHECK_FALSE(c.valid());
    CHECK_FALSE(c.error_msg().empty());
}

TEST_CASE("converter test throw on error mode") {
    ss::converter<ss::throw_on_error> c;
    REQUIRE_EXCEPTION(c.convert<int>("junk"));
}

TEST_CASE("converter test converter with quotes spacing and escaping") {
    {
        ss::converter c;

        auto tup = c.convert<std::string, std::string, std::string>(
            R"("just","some","strings")");
        REQUIRE(c.valid());
        CHECK_EQ(tup, std::make_tuple("\"just\"", "\"some\"", "\"strings\""));
    }

    {
        ss::converter<ss::quote<'"'>> c;

        auto tup = c.convert<std::string, std::string, double, char>(
            buff(R"("just",some,"12.3","a")"));
        REQUIRE(c.valid());
        CHECK_EQ(tup, std::make_tuple("just", "some", 12.3, 'a'));
    }

    {
        ss::converter<ss::trim<' '>> c;

        auto tup = c.convert<std::string, std::string, double, char>(
            buff(R"(    just  ,  some   ,  12.3 ,a     )"));
        REQUIRE(c.valid());
        CHECK_EQ(tup, std::make_tuple("just", "some", 12.3, 'a'));
    }

    {
        ss::converter<ss::escape<'\\'>> c;

        auto tup =
            c.convert<std::string, std::string>(buff(R"(ju\,st,strings)"));
        REQUIRE(c.valid());
        CHECK_EQ(tup, std::make_tuple("ju,st", "strings"));
    }

    {
        ss::converter<ss::escape<'\\'>, ss::trim<' '>, ss::quote<'"'>> c;

        auto tup = c.convert<std::string, std::string, double, std::string>(
            buff(R"(  ju\,st  ,  "so,me"  ,   12.34     ,   "str""ings")"));
        REQUIRE(c.valid());
        CHECK_EQ(tup, std::make_tuple("ju,st", "so,me", 12.34, "str\"ings"));
    }
}

TEST_CASE("converter test converter with quotes spacing and escaping with "
          "exceptions") {
    try {
        ss::converter<ss::throw_on_error> c;

        auto tup = c.convert<std::string, std::string, std::string>(
            R"("just","some","strings")");
        REQUIRE(c.valid());
        CHECK_EQ(tup, std::make_tuple("\"just\"", "\"some\"", "\"strings\""));
    } catch (ss::exception& e) {
        FAIL(std::string{e.what()});
    }

    try {
        ss::converter<ss::quote<'"'>> c;

        auto tup = c.convert<std::string, std::string, double, char>(
            buff(R"("just",some,"12.3","a")"));
        REQUIRE(c.valid());
        CHECK_EQ(tup, std::make_tuple("just", "some", 12.3, 'a'));
    } catch (ss::exception& e) {
        FAIL(std::string{e.what()});
    }

    try {
        ss::converter<ss::throw_on_error, ss::trim<' '>> c;

        auto tup = c.convert<std::string, std::string, double, char>(
            buff(R"(    just  ,  some   ,  12.3 ,a     )"));
        REQUIRE(c.valid());
        CHECK_EQ(tup, std::make_tuple("just", "some", 12.3, 'a'));
    } catch (ss::exception& e) {
        FAIL(std::string{e.what()});
    }

    try {
        ss::converter<ss::throw_on_error, ss::escape<'\\'>> c;

        auto tup =
            c.convert<std::string, std::string>(buff(R"(ju\,st,strings)"));
        REQUIRE(c.valid());
        CHECK_EQ(tup, std::make_tuple("ju,st", "strings"));
    } catch (ss::exception& e) {
        FAIL(std::string{e.what()});
    }

    try {
        ss::converter<ss::throw_on_error, ss::escape<'\\'>, ss::trim<' '>,
                      ss::quote<'"'>>
            c;

        auto tup = c.convert<std::string, std::string, double, std::string>(
            buff(R"(  ju\,st  ,  "so,me"  ,   12.34     ,   "str""ings")"));
        REQUIRE(c.valid());
        CHECK_EQ(tup, std::make_tuple("ju,st", "so,me", 12.34, "str\"ings"));
    } catch (ss::exception& e) {
        FAIL(std::string{e.what()});
    }
}

TEST_CASE("converter test invalid split conversions") {
    ss::converter<ss::string_error, ss::escape<'\\'>, ss::trim<' '>,
                  ss::quote<'"'>>
        c;

    {
        // mismatched quote
        c.convert<std::string, std::string, double, char>(
            buff(R"(  "just  , some ,   "12.3","a"  )"));
        CHECK_FALSE(c.valid());
        CHECK_FALSE(c.unterminated_quote());
        CHECK_FALSE(c.error_msg().empty());
    }

    {
        // unterminated quote
        c.convert<std::string, std::string, double, std::string>(
            buff(R"(  ju\,st  ,  "so,me"  ,   12.34     ,   "str""ings)"));
        CHECK_FALSE(c.valid());
        CHECK(c.unterminated_quote());
        CHECK_FALSE(c.error_msg().empty());
    }

    {
        // unterminated escape
        c.convert<std::string, std::string, double, std::string>(
            buff(R"(just,some,2,strings\)"));
        CHECK_FALSE(c.valid());
        CHECK_FALSE(c.unterminated_quote());
        CHECK_FALSE(c.error_msg().empty());
    }

    {
        // unterminated escape while quoting
        c.convert<std::string, std::string, double, std::string>(
            buff(R"(just,some,2,"strings\)"));
        CHECK_FALSE(c.valid());
        CHECK_FALSE(c.unterminated_quote());
        CHECK_FALSE(c.error_msg().empty());
    }

    {
        // unterminated escaped quote
        c.convert<std::string, std::string, double, std::string>(
            buff(R"(just,some,2,"strings\")"));
        CHECK_FALSE(c.valid());
        CHECK(c.unterminated_quote());
        CHECK_FALSE(c.error_msg().empty());
    }
}

TEST_CASE("converter test invalid split conversions with exceptions") {
    ss::converter<ss::escape<'\\'>, ss::trim<' '>, ss::quote<'"'>,
                  ss::throw_on_error>
        c;

    // mismatched quote
    REQUIRE_EXCEPTION(c.convert<std::string, std::string, double, char>(
        buff(R"(  "just  , some ,   "12.3","a"  )")));
    CHECK_FALSE(c.unterminated_quote());

    // unterminated quote
    REQUIRE_EXCEPTION(c.convert<std::string, std::string, double, std::string>(
        buff(R"(  ju\,st  ,  "so,me"  ,   12.34     ,   "str""ings)")));
    CHECK(c.unterminated_quote());

    // unterminated escape
    REQUIRE_EXCEPTION(c.convert<std::string, std::string, double, std::string>(
        buff(R"(just,some,2,strings\)")));
    CHECK_FALSE(c.unterminated_quote());

    // unterminated escape while quoting
    REQUIRE_EXCEPTION(c.convert<std::string, std::string, double, std::string>(
        buff(R"(just,some,2,"strings\)")));
    CHECK_FALSE(c.unterminated_quote());

    // unterminated escaped quote
    REQUIRE_EXCEPTION(c.convert<std::string, std::string, double, std::string>(
        buff(R"(just,some,2,"strings\")")));
    CHECK(c.unterminated_quote());
}