#include "test_parser1.hpp" TEST_CASE("test file not found") { unique_file_name f{"test_parser"}; { ss::parser p{f.name, ","}; CHECK_FALSE(p.valid()); } { ss::parser p{f.name, ","}; CHECK_FALSE(p.valid()); } try { ss::parser p{f.name, ","}; FAIL("Expected exception..."); } catch (const std::exception& e) { CHECK_FALSE(std::string{e.what()}.empty()); } } template void test_various_cases() { unique_file_name f{"test_parser"}; std::vector data = {{1, 2, "x"}, {3, 4, "y"}, {5, 6, "z"}, {7, 8, "u"}, {9, 10, "v"}, {11, 12, "w"}}; make_and_write(f.name, data); auto csv_data_buffer = make_buffer(f.name); { auto [p, _] = make_parser(f.name, ","); ss::parser p0{std::move(p)}; p = std::move(p0); std::vector i; auto [p2, __] = make_parser(f.name, ","); std::vector i2; auto move_rotate = [&p = p, &p0 = p0] { auto p1 = std::move(p); p0 = std::move(p1); p = std::move(p0); }; while (!p.eof()) { move_rotate(); auto a = p.template get_next(); i.emplace_back(ss::to_object(a)); } for (const auto& a : p2.template iterate()) { i2.emplace_back(ss::to_object(a)); } CHECK_EQ(i, data); CHECK_EQ(i2, data); } { auto [p, _] = make_parser(f.name, ","); std::vector i; auto [p2, __] = make_parser(f.name, ","); std::vector i2; auto [p3, ___] = make_parser(f.name, ","); std::vector i3; std::vector expected = {std::begin(data) + 1, std::end(data)}; using tup = std::tuple; p.ignore_next(); while (!p.eof()) { auto a = p.template get_next(); i.emplace_back(ss::to_object(a)); } p2.ignore_next(); for (const auto& a : p2.template iterate()) { i2.emplace_back(ss::to_object(a)); } p3.ignore_next(); for (auto it = p3.template iterate().begin(); it != p3.template iterate().end(); ++it) { i3.emplace_back(ss::to_object(*it)); } CHECK_EQ(i, expected); CHECK_EQ(i2, expected); CHECK_EQ(i3, expected); } { auto [p, _] = make_parser(f.name, ","); std::vector i; auto [p2, __] = make_parser(f.name, ","); std::vector i2; while (!p.eof()) { i.push_back(p.template get_object()); } for (auto&& a : p2.template iterate_object()) { i2.push_back(std::move(a)); } CHECK_EQ(i, data); CHECK_EQ(i2, data); } { auto [p, _] = make_parser(f.name, ","); std::vector i; for (auto&& a : p.template iterate_object()) { i.push_back(std::move(a)); } CHECK_EQ(i, data); } { auto [p, _] = make_parser(f.name, ","); std::vector i; auto [p2, __] = make_parser(f.name, ","); std::vector i2; using tup = std::tuple; while (!p.eof()) { i.push_back(p.template get_object()); } for (auto it = p2.template iterate_object().begin(); it != p2.template iterate_object().end(); it++) { i2.push_back({it->i, it->d, it->s}); } CHECK_EQ(i, data); CHECK_EQ(i2, data); } { auto [p, _] = make_parser(f.name, ","); std::vector i; using tup = std::tuple; for (auto&& a : p.template iterate_object()) { i.push_back(std::move(a)); } CHECK_EQ(i, data); } { auto [p, _] = make_parser(f.name, ","); std::vector i; while (!p.eof()) { i.push_back(p.template get_next()); } CHECK_EQ(i, data); } { auto [p, _] = make_parser(f.name, ","); std::vector i; for (auto&& a : p.template iterate()) { i.push_back(std::move(a)); } CHECK_EQ(i, data); } { constexpr int excluded = 3; auto [p, _] = make_parser(f.name, ","); std::vector i; auto [p2, __] = make_parser(f.name, ","); std::vector i2; while (!p.eof()) { try { auto a = p.template get_object, double, std::string>(); if (p.valid()) { i.push_back(a); } } catch (...) { // ignore }; } if (!ss::setup::throw_on_error) { for (auto&& a : p2.template iterate_object, double, std::string>()) { if (p2.valid()) { i2.push_back(std::move(a)); } } } std::vector expected; for (auto& x : data) { if (x.i != excluded) { expected.push_back(x); } } std::copy_if(data.begin(), data.end(), expected.begin(), [&](const X& x) { return x.i != excluded; }); CHECK_EQ(i, expected); if (!ss::setup::throw_on_error) { CHECK_EQ(i2, expected); } } { auto [p, _] = make_parser(f.name, ","); std::vector i; auto [p2, __] = make_parser(f.name, ","); std::vector i2; while (!p.eof()) { try { auto a = p.template get_object, double, std::string>(); if (p.valid()) { i.push_back(a); } } catch (...) { // ignore } } if (!ss::setup::throw_on_error) { for (auto&& a : p2.template iterate_object, double, std::string>()) { if (p2.valid()) { i2.push_back(std::move(a)); } } } std::vector expected = {{3, 4, "y"}}; CHECK_EQ(i, expected); if (!ss::setup::throw_on_error) { CHECK_EQ(i2, expected); } } { unique_file_name empty_f{"test_parser"}; std::vector empty_data = {}; make_and_write(empty_f.name, empty_data); auto [p, _] = make_parser(empty_f.name, ","); std::vector i; auto [p2, __] = make_parser(empty_f.name, ","); std::vector i2; while (!p.eof()) { i.push_back(p.template get_next()); } for (auto&& a : p2.template iterate()) { i2.push_back(std::move(a)); } CHECK(i.empty()); CHECK(i2.empty()); } } TEST_CASE("parser test various cases") { test_various_cases(); test_various_cases(); test_various_cases(); test_various_cases(); test_various_cases(); test_various_cases(); } using test_tuple = std::tuple; struct test_struct { int i; double d; char c; auto tied() { return std::tie(i, d, c); } }; static inline void expect_test_struct(const test_struct&) { } template void test_composite_conversion() { unique_file_name f{"test_parser"}; { std::ofstream out{f.name}; for (auto& i : {"10,a,11.1", "10,20,11.1", "junk", "10,11.1", "1,11.1,a", "junk", "10,junk", "11,junk", "10,11.1,c", "10,20", "10,22.2,f"}) { out << i << std::endl; } } auto [p, _] = make_parser(f.name, ","); auto fail = [] { FAIL(""); }; auto expect_error = [](auto error) { CHECK(!error.empty()); }; auto ignore_error = [] {}; REQUIRE(p.valid()); REQUIRE_FALSE(p.eof()); { constexpr static auto expectedData = std::tuple{10, 'a', 11.1}; auto [d1, d2, d3, d4] = p.template try_next(fail) .template or_else(fail) .template or_else( [&](auto&& data) { CHECK_EQ(data, expectedData); }) .on_error(fail) .template or_else(fail) .values(); REQUIRE(p.valid()); REQUIRE_FALSE(d1); REQUIRE_FALSE(d2); REQUIRE(d3); REQUIRE_FALSE(d4); CHECK_EQ(*d3, expectedData); } { REQUIRE(!p.eof()); constexpr static auto expectedData = std::tuple{10, 20, 11.1}; auto [d1, d2, d3, d4] = p.template try_next( [&](auto& i1, auto i2, double d) { CHECK_EQ(std::tie(i1, i2, d), expectedData); }) .on_error(fail) .template or_object(fail) .on_error(fail) .template or_else(fail) .on_error(fail) .template or_else(fail) .values(); REQUIRE(p.valid()); REQUIRE(d1); REQUIRE_FALSE(d2); REQUIRE_FALSE(d3); REQUIRE_FALSE(d4); CHECK_EQ(*d1, expectedData); } { REQUIRE(!p.eof()); auto [d1, d2, d3, d4, d5] = p.template try_object(fail) .on_error(expect_error) .template or_else(fail) .template or_else(fail) .template or_else(fail) .template or_else(fail) .values(); REQUIRE_FALSE(p.valid()); REQUIRE_FALSE(d1); REQUIRE_FALSE(d2); REQUIRE_FALSE(d3); REQUIRE_FALSE(d4); REQUIRE_FALSE(d5); } { REQUIRE(!p.eof()); auto [d1, d2] = p.template try_next([](auto& i, auto& d) { REQUIRE_EQ(std::tie(i, d), std::tuple{10, 11.1}); }) .template or_else([](auto&, auto&) { FAIL(""); }) .values(); REQUIRE(p.valid()); REQUIRE(d1); REQUIRE_FALSE(d2); } { REQUIRE(!p.eof()); auto [d1, d2] = p.template try_next([](auto&, auto&) { FAIL(""); }) .template or_else(expect_test_struct) .values(); REQUIRE(p.valid()); REQUIRE_FALSE(d1); REQUIRE(d2); CHECK_EQ(d2->tied(), std::tuple{1, 11.1, 'a'}); } { REQUIRE(!p.eof()); auto [d1, d2, d3, d4, d5] = p.template try_next(fail) .template or_object() .template or_else(expect_test_struct) .template or_else(fail) .template or_else>(fail) .on_error(ignore_error) .on_error(expect_error) .values(); REQUIRE_FALSE(p.valid()); REQUIRE_FALSE(d1); REQUIRE_FALSE(d2); REQUIRE_FALSE(d3); REQUIRE_FALSE(d4); REQUIRE_FALSE(d5); } { REQUIRE(!p.eof()); auto [d1, d2] = p.template try_next>() .on_error(ignore_error) .on_error(fail) .template or_else>(fail) .on_error(ignore_error) .on_error(fail) .on_error(ignore_error) .values(); REQUIRE(p.valid()); REQUIRE(d1); REQUIRE_FALSE(d2); CHECK_EQ(*d1, std::tuple{10, std::nullopt}); } { REQUIRE_FALSE(p.eof()); auto [d1, d2] = p.template try_next>() .on_error(fail) .template or_else>(fail) .on_error(fail) .values(); REQUIRE(p.valid()); REQUIRE(d1); REQUIRE_FALSE(d2); CHECK_EQ(*d1, std::tuple{11, std::variant{"junk"}}); } { REQUIRE(!p.eof()); auto [d1, d2] = p.template try_object() .template or_else(fail) .values(); REQUIRE(p.valid()); REQUIRE(d1); REQUIRE_FALSE(d2); CHECK_EQ(d1->tied(), std::tuple{10, 11.1, 'c'}); } { REQUIRE_FALSE(p.eof()); auto [d1, d2, d3, d4] = p.template try_next([] { return false; }) .template or_else([](auto&) { return false; }) .template or_else() .template or_else(fail) .values(); REQUIRE(p.valid()); REQUIRE_FALSE(d1); REQUIRE_FALSE(d2); REQUIRE(d3); REQUIRE_FALSE(d4); CHECK_EQ(d3.value(), std::tuple{10, 20}); } { REQUIRE(!p.eof()); auto [d1, d2, d3, d4] = p.template try_object( [] { return false; }) .template or_else([](auto&) { return false; }) .template or_object() .template or_else(fail) .values(); REQUIRE(p.valid()); REQUIRE_FALSE(d1); REQUIRE_FALSE(d2); REQUIRE(d3); REQUIRE_FALSE(d4); CHECK_EQ(d3->tied(), std::tuple{10, 22.2, 'f'}); } CHECK(p.eof()); } // various scenarios TEST_CASE("parser test composite conversion") { test_composite_conversion(); test_composite_conversion(); }