#include "catch2/catch_test_macros.hpp" #include "SessionFixture.hpp" #include "connection.hpp" #include "matador/query/session.hpp" #include "models/author.hpp" #include "models/book.hpp" using namespace matador; using namespace matador::query; using namespace matador::object; using namespace matador::test; namespace matador::test { template struct colorlist_pk_generator { unsigned int id{}; std::string name; collection colors; colorlist_pk_generator() = default; colorlist_pk_generator(std::string name, const std::vector& color_names) : name(std::move(name)) , colors(color_names) {} template void process(Operator &op) { namespace field = matador::access; field::primary_key(op, "id", id, PkAttribute); field::attribute(op, "name", name, VarChar511); field::has_many(op, "colors", colors, "colorlist_id", utils::CascadeAllFetchLazy); } }; using colorlist_identity = colorlist_pk_generator; using colorlist_sequence = colorlist_pk_generator; using colorlist_table = colorlist_pk_generator; } template object_ptr create_author() { auto s_king = make_object("Steven King"); s_king->books.push_back(make_object("Carrie", nullobj, 1974)); s_king->books.push_back(make_object("The Shining", nullobj, 1977)); s_king->books.push_back(make_object("It", nullobj, 1986)); s_king->books.push_back(make_object("Misery", nullobj, 1987)); s_king->books.push_back(make_object("The Dark Tower: The Gunslinger", nullobj, 1982)); return s_king; } template void validate_author_state(const object_ptr& ptr, object_state expected_state) { REQUIRE(ptr.is_state(expected_state)); for (auto &b: ptr->books) { REQUIRE(b.is_state(expected_state)); } } namespace matador::utils { template < typename ValueType > std::ostream& operator<<(std::ostream& os, const result& value) { if (value) { return os; } return os << "Error: " << value.err(); } } TEST_CASE_METHOD(SessionFixture, "Test insert object with has many relation", "[session][insert][has_many]") { const auto result = schema.attach("books") .and_then( [this] { return schema.attach("authors"); } ) .and_then([this] { return schema.create(db); } ); REQUIRE(result.is_ok()); session ses({bus, connection::dns, 4}, schema); auto s_king = make_object(1, "Steven", "King", "21.9.1947", 1956, false); s_king->books.push_back(make_object(2, "Carrie", nullobj, 1974)); s_king->books.push_back(make_object(3, "The Shining", nullobj, 1977)); s_king->books.push_back(make_object(4, "It", nullobj, 1986)); s_king->books.push_back(make_object(5, "Misery", nullobj, 1987)); s_king->books.push_back(make_object(6, "The Dark Tower: The Gunslinger", nullobj, 1982)); validate_author_state(s_king, object_state::Transient); auto res = ses.insert(s_king); REQUIRE(res); validate_author_state(s_king, object_state::Persistent); } TEST_CASE_METHOD(SessionFixture, "Test insert object with has many relation with identity", "[session][insert][has_many][identity]") { const auto result = schema.attach("books") .and_then( [this] { return schema.attach("authors"); } ) .and_then([this] { return schema.create(db); } ); REQUIRE(result.is_ok()); session ses({bus, connection::dns, 4}, schema); auto s_king = create_author(); validate_author_state(s_king, object_state::Transient); auto res = ses.insert(s_king); REQUIRE(res.is_ok()); validate_author_state(s_king, object_state::Persistent); } TEST_CASE_METHOD(SessionFixture, "Test insert object with has many relation with sequence", "[session][insert][has_many][sequence]") { const auto result = schema.attach("books") .and_then( [this] { return schema.attach("authors"); } ) .and_then([this] { return schema.create(db); } ); REQUIRE(result.is_ok()); session ses({bus, connection::dns, 4}, schema); auto s_king = create_author(); validate_author_state(s_king, object_state::Transient); auto res = ses.insert(s_king); REQUIRE(res.is_ok()); validate_author_state(s_king, object_state::Persistent); } TEST_CASE_METHOD(SessionFixture, "Test insert object with has many relation with table sequence", "[session][insert][has_many][table_sequence]") { const auto result = schema.attach("books") .and_then( [this] { return schema.attach("authors"); } ) .and_then([this] { return schema.create(db); } ); REQUIRE(result.is_ok()); session ses({bus, connection::dns, 4}, schema); auto s_king = create_author(); validate_author_state(s_king, object_state::Transient); auto res = ses.insert(s_king); REQUIRE(res.is_ok()); validate_author_state(s_king, object_state::Persistent); auto author_result = ses.find(s_king->id); REQUIRE(author_result); auto author = *author_result; REQUIRE(author.is_persistent()); REQUIRE(author->books.size() == 5); } TEST_CASE_METHOD(SessionFixture, "Test insert object with has many primitive relation with identity", "[session][insert][has_many][primitive][identity]") { const auto result = schema.attach("colorlist") .and_then([this] { return schema.create(db); } ); REQUIRE(result.is_ok()); session ses({bus, connection::dns, 4}, schema); const auto c = make_object("rgb", std::vector{"red", "green", "blue"}); auto res = ses.insert(c); REQUIRE(res.is_ok()); auto colorlist_result = ses.find(c->id); REQUIRE(colorlist_result); auto clist = *colorlist_result; REQUIRE(clist.is_persistent()); REQUIRE(clist->colors.size() == 3); std::vector expected_colors{"red", "green", "blue"}; for (const auto &cstr: clist->colors) { REQUIRE(std::find(expected_colors.begin(), expected_colors.end(), cstr) != expected_colors.end()); } } TEST_CASE_METHOD(SessionFixture, "Test insert object with has many primitive relation with table sequence", "[session][insert][has_many][primitive][table_sequence]") { const auto result = schema.attach("colorlist") .and_then([this] { return schema.create(db); } ); REQUIRE(result.is_ok()); session ses({bus, connection::dns, 4}, schema); const auto c = make_object("rgb", std::vector{"red", "green", "blue"}); auto res = ses.insert(c); REQUIRE(res.is_ok()); auto colorlist_result = ses.find(c->id); REQUIRE(colorlist_result); auto clist = *colorlist_result; REQUIRE(clist.is_persistent()); REQUIRE(clist->colors.size() == 3); std::vector expected_colors{"red", "green", "blue"}; for (const auto &cstr: clist->colors) { REQUIRE(std::find(expected_colors.begin(), expected_colors.end(), cstr) != expected_colors.end()); } } TEST_CASE_METHOD(SessionFixture, "Test insert object with has many primitive relation with sequence", "[session][insert][has_many][primitive][sequence]") { const auto result = schema.attach("colorlist") .and_then([this] { return schema.create(db); } ); REQUIRE(result.is_ok()); session ses({bus, connection::dns, 4}, schema); const auto c = make_object("rgb", std::vector{"red", "green", "blue"}); auto res = ses.insert(c); REQUIRE(res.is_ok()); auto colorlist_result = ses.find(c->id); REQUIRE(colorlist_result); auto clist = *colorlist_result; REQUIRE(clist.is_persistent()); REQUIRE(clist->colors.size() == 3); std::vector expected_colors{"red", "green", "blue"}; for (const auto &cstr: clist->colors) { REQUIRE(std::find(expected_colors.begin(), expected_colors.end(), cstr) != expected_colors.end()); } }