261 lines
8.7 KiB
C++
261 lines
8.7 KiB
C++
#include <catch2/catch_test_macros.hpp>
|
|
|
|
#include "QueryFixture.hpp"
|
|
|
|
#include <matador/query/criteria.hpp>
|
|
#include <matador/query/query.hpp>
|
|
|
|
#include "matador/object/attribute_definition_generator.hpp"
|
|
|
|
#include "matador/sql/connection.hpp"
|
|
#include "matador/sql/table.hpp"
|
|
|
|
#include "matador/utils/placeholder.hpp"
|
|
|
|
using namespace matador::test;
|
|
using namespace matador::object;
|
|
using namespace matador::sql;
|
|
using namespace matador::query;
|
|
using namespace matador::utils;
|
|
|
|
TEST_CASE_METHOD(QueryFixture, "Test alter table sql statement", "[query][alter][table]") {
|
|
auto result = query::alter()
|
|
.table("employees")
|
|
.add_constraint("FK_employees_dep_id")
|
|
.foreign_key("dep_id"_col)
|
|
.references("departments", {"id"_col})
|
|
.str(*db);
|
|
|
|
REQUIRE(result == R"(ALTER TABLE "employees" ADD CONSTRAINT FK_employees_dep_id FOREIGN KEY ("dep_id") REFERENCES departments ("id"))");
|
|
result = query::alter()
|
|
.table("employees")
|
|
.drop_constraint("FK_employees_dep_id")
|
|
.str(*db);
|
|
|
|
REQUIRE(result == R"(ALTER TABLE "employees" DROP CONSTRAINT FK_employees_dep_id)");
|
|
}
|
|
|
|
TEST_CASE_METHOD(QueryFixture, "Test create table sql statement string", "[query]") {
|
|
auto result = query::create()
|
|
.table({"person"}, {
|
|
make_pk_column<uint32_t>("id"),
|
|
make_column<std::string>("name", 255),
|
|
make_column<unsigned short>("age")
|
|
}).str(*db);
|
|
|
|
REQUIRE(result == R"##(CREATE TABLE "person" ("id" BIGINT NOT NULL, "name" VARCHAR(255) NOT NULL, "age" INTEGER NOT NULL, CONSTRAINT PK_person PRIMARY KEY (id)))##");
|
|
|
|
auto ctx = query::create()
|
|
.table("person", {
|
|
make_pk_column<uint32_t>("id"),
|
|
make_column<std::string>("name", {255, constraints::UNIQUE}, null_option::NOT_NULL),
|
|
make_column<unsigned short>("age"),
|
|
make_fk_column<uint32_t>("address", "address", "id")
|
|
}).compile(*db);
|
|
|
|
REQUIRE(ctx.sql == R"##(CREATE TABLE "person" ("id" BIGINT NOT NULL, "name" VARCHAR(255) NOT NULL UNIQUE, "age" INTEGER NOT NULL, "address" BIGINT NOT NULL, CONSTRAINT PK_person PRIMARY KEY (id)))##");
|
|
REQUIRE(ctx.additional_commands.size() == 1);
|
|
REQUIRE(ctx.additional_commands[0].sql == R"##(ALTER TABLE "person" ADD CONSTRAINT FK_person_address FOREIGN KEY ("address") REFERENCES address(id))##");
|
|
}
|
|
|
|
TEST_CASE_METHOD(QueryFixture, "Test drop table sql statement string", "[query]") {
|
|
const auto result = query::drop()
|
|
.table("person")
|
|
.str(*db);
|
|
|
|
REQUIRE(result == R"(DROP TABLE "person")");
|
|
}
|
|
|
|
TEST_CASE_METHOD(QueryFixture, "Test select all columns with asterisk", "[query][select][asterisk]") {
|
|
const auto result = query::select()
|
|
.from("person")
|
|
.str(*db);
|
|
|
|
REQUIRE(result == R"(SELECT * FROM "person")");
|
|
}
|
|
|
|
TEST_CASE_METHOD(QueryFixture, "Test select sql statement string", "[query]") {
|
|
const auto result = query::select({"id", "name", "age"})
|
|
.from("person")
|
|
.str(*db);
|
|
|
|
REQUIRE(result == R"(SELECT "id", "name", "age" FROM "person")");
|
|
}
|
|
|
|
TEST_CASE_METHOD(QueryFixture, "Test insert sql statement string", "[query]") {
|
|
const auto result = query::insert()
|
|
.into("person", {
|
|
"id", "name", "age"
|
|
})
|
|
.values({7U, "george", 65U})
|
|
.str(*db);
|
|
|
|
REQUIRE(result == R"(INSERT INTO "person" ("id", "name", "age") VALUES (7, 'george', 65))");
|
|
}
|
|
|
|
TEST_CASE_METHOD(QueryFixture, "Test update sql statement string", "[query]") {
|
|
auto result = query::update("person")
|
|
.set({
|
|
{"id", 7U},
|
|
{"name", "george"},
|
|
{"age", 65U}
|
|
})
|
|
.str(*db);
|
|
|
|
REQUIRE(result == R"(UPDATE "person" SET "id"=7, "name"='george', "age"=65)");
|
|
|
|
result = query::update("person")
|
|
.set({
|
|
{"id", 7U},
|
|
{"name", "george"},
|
|
{"age", 65U}
|
|
})
|
|
.where("id"_col > 9)
|
|
.order_by("id").asc()
|
|
.limit(3)
|
|
.offset(2)
|
|
.str(*db);
|
|
|
|
REQUIRE(result == R"(UPDATE "person" SET "id"=7, "name"='george', "age"=65 WHERE "id" > 9 ORDER BY "id" ASC LIMIT 3 OFFSET 2)");
|
|
}
|
|
|
|
TEST_CASE_METHOD(QueryFixture, "Test update limit sql statement", "[query][update][limit]") {
|
|
const auto result = query::update("person")
|
|
.set({{"id", 7U}, {"name", "george"}, {"age", 65U}})
|
|
.where("name"_col == "george")
|
|
.order_by("id"_col).asc()
|
|
.limit(2)
|
|
.str(*db);
|
|
|
|
REQUIRE(result == R"(UPDATE "person" SET "id"=7, "name"='george', "age"=65 WHERE "name" = 'george' ORDER BY "id" ASC LIMIT 2)");
|
|
}
|
|
|
|
TEST_CASE_METHOD(QueryFixture, "Test delete sql statement string", "[query]") {
|
|
const auto result = query::remove().from("person").str(*db);
|
|
|
|
REQUIRE(result == R"(DELETE FROM "person")");
|
|
}
|
|
|
|
TEST_CASE_METHOD(QueryFixture, "Test delete limit sql statement", "[query][delete][limit]") {
|
|
const auto result = query::remove()
|
|
.from("person")
|
|
.where("name"_col == "george")
|
|
.order_by("id"_col).asc()
|
|
.limit(2)
|
|
.str(*db);
|
|
|
|
REQUIRE(result == R"(DELETE FROM "person" WHERE "name" = 'george' ORDER BY "id" ASC LIMIT 2)");
|
|
}
|
|
|
|
TEST_CASE_METHOD(QueryFixture, "Test select sql statement string with where clause", "[query]") {
|
|
auto result = query::select({"id", "name", "age"})
|
|
.from("person")
|
|
.where("id"_col == 8 && "age"_col > 50)
|
|
.str(*db);
|
|
|
|
REQUIRE(result == R"(SELECT "id", "name", "age" FROM "person" WHERE ("id" = 8 AND "age" > 50))");
|
|
|
|
result = query::select({"id", "name", "age"})
|
|
.from("person")
|
|
.where("id"_col == _ && "age"_col > 50)
|
|
.str(*db);
|
|
|
|
REQUIRE(result == R"(SELECT "id", "name", "age" FROM "person" WHERE ("id" = ? AND "age" > 50))");
|
|
}
|
|
|
|
TEST_CASE_METHOD(QueryFixture, "Test insert sql statement with placeholder", "[query]") {
|
|
auto result = query::insert()
|
|
.into("person", {"id", "name", "age"})
|
|
.values({_, _, _})
|
|
.str(*db);
|
|
|
|
REQUIRE(result == R"(INSERT INTO "person" ("id", "name", "age") VALUES (?, ?, ?))");
|
|
|
|
result = query::insert()
|
|
.into("person", {"id", "name", "age"})
|
|
.values({9, "george", _})
|
|
.str(*db);
|
|
|
|
REQUIRE(result == R"(INSERT INTO "person" ("id", "name", "age") VALUES (9, 'george', ?))");
|
|
}
|
|
|
|
TEST_CASE_METHOD(QueryFixture, "Test select sql statement string with order by", "[query]") {
|
|
const auto result = query::select({"id", "name", "age"})
|
|
.from("person")
|
|
.order_by("name"_col).asc()
|
|
.str(*db);
|
|
|
|
REQUIRE(result == R"(SELECT "id", "name", "age" FROM "person" ORDER BY "name" ASC)");
|
|
}
|
|
|
|
TEST_CASE_METHOD(QueryFixture, "Test select sql statement string with group by", "[query]") {
|
|
const auto result = query::select({"id", "name", "age"})
|
|
.from("person")
|
|
.group_by("age"_col)
|
|
.str(*db);
|
|
|
|
REQUIRE(result == R"(SELECT "id", "name", "age" FROM "person" GROUP BY "age")");
|
|
}
|
|
|
|
TEST_CASE_METHOD(QueryFixture, "Test select sql statement string with offset and limit", "[query]") {
|
|
const auto result = query::select({"id", "name", "age"})
|
|
.from("person")
|
|
.order_by("id"_col).asc()
|
|
.limit(20)
|
|
.offset(10)
|
|
.str(*db);
|
|
|
|
REQUIRE(result == R"(SELECT "id", "name", "age" FROM "person" ORDER BY "id" ASC LIMIT 20 OFFSET 10)");
|
|
}
|
|
|
|
TEST_CASE_METHOD(QueryFixture, "Test create, insert and select a blob column", "[query][blob]") {
|
|
auto result = query::create()
|
|
.table("person", {
|
|
make_pk_column<uint32_t>("id"),
|
|
make_column<std::string>("name", 255),
|
|
make_column<blob>("data")
|
|
})
|
|
.str(*db);
|
|
|
|
REQUIRE(result == R"##(CREATE TABLE "person" ("id" BIGINT NOT NULL, "name" VARCHAR(255) NOT NULL, "data" BLOB NOT NULL, CONSTRAINT PK_person PRIMARY KEY (id)))##");
|
|
|
|
result = query::insert()
|
|
.into("person", {"id", "name", "data"})
|
|
.values({7U, "george", blob{1, 'A', 3, 4}})
|
|
.str(*db);
|
|
|
|
REQUIRE(result == R"(INSERT INTO "person" ("id", "name", "data") VALUES (7, 'george', X'01410304'))");
|
|
|
|
result = query::select({"id", "name", "data"})
|
|
.from("person")
|
|
.str(*db);
|
|
|
|
REQUIRE(result == R"(SELECT "id", "name", "data" FROM "person")");
|
|
}
|
|
|
|
TEST_CASE_METHOD(QueryFixture, "Test select statement with join_left", "[query][statement][join_left]") {
|
|
const auto result = query::select({"f.id", "ap.brand", "f.pilot_name"})
|
|
.from({"flight", "f"})
|
|
.join_left({"airplane", "ap"})
|
|
.on("f.airplane_id"_col == "ap.id"_col)
|
|
.str(*db);
|
|
|
|
REQUIRE(result == R"(SELECT "f"."id", "ap"."brand", "f"."pilot_name" FROM "flight" "f" LEFT JOIN "airplane" "ap" ON "f"."airplane_id" = "ap"."id")");
|
|
}
|
|
|
|
// TEST_CASE_METHOD(QueryFixture, "Select statement with aliased columns", "[query][select][alias]") {
|
|
// using namespace matador::test;
|
|
// connection noop("noop://noop.db");
|
|
// schema scm("noop");
|
|
// scm.attach<author>("authors");
|
|
// scm.attach<book>("books");
|
|
//
|
|
//
|
|
// const auto result = query::select<author>(scm)
|
|
// .from("authors"_tab.as("T01"))
|
|
// .str(*db);
|
|
//
|
|
// const auto expected_sql = R"(SELECT "T01"."id", "T01"."first_name", "T01"."last_name", "T01"."date_of_birth", "T01"."year_of_birth", "T01"."distinguished" FROM "authors" "T01")";
|
|
//
|
|
// REQUIRE(result == expected_sql);
|
|
// }
|