added column names and values generator and tests

This commit is contained in:
Sascha Kuehl 2023-11-12 22:27:19 +01:00
parent 30f2126225
commit 4ed01a617d
14 changed files with 374 additions and 23 deletions

View File

@ -28,5 +28,6 @@ add_subdirectory(src)
add_subdirectory(test) add_subdirectory(test)
add_subdirectory(backends) add_subdirectory(backends)
add_executable(query main.cpp) add_executable(query main.cpp
include/matador/sql/column_name_generator.hpp)
target_link_libraries(query matador) target_link_libraries(query matador)

View File

@ -0,0 +1,18 @@
#ifndef QUERY_ANY_TYPE_HPP
#define QUERY_ANY_TYPE_HPP
#include <string>
#include <variant>
namespace matador::sql {
using any_type = std::variant<
char, short, int, long, long long,
unsigned char, unsigned short, unsigned int, unsigned long, unsigned long long,
bool,
float, double,
const char*,
std::string>;
}
#endif //QUERY_ANY_TYPE_HPP

View File

@ -6,16 +6,56 @@
#include "matador/utils/access.hpp" #include "matador/utils/access.hpp"
#include "matador/utils/field_attributes.hpp" #include "matador/utils/field_attributes.hpp"
#include "matador/utils/identifiable.hpp"
#include "matador/utils/identifier.hpp"
#include <vector> #include <vector>
namespace matador::utils { namespace matador::utils {
class identifiable;
enum class cascade_type; enum class cascade_type;
} }
namespace matador::sql { namespace matador::sql {
namespace detail {
class null_identifiable : public utils::identifiable
{
public:
void reset(const utils::identifier &) override {};
[[nodiscard]] bool has_primary_key() const override { return false; }
[[nodiscard]] const utils::identifier& primary_key() const override { return id_; }
utils::identifier& primary_key() override { return id_; }
[[nodiscard]] utils::identifier create_identifier() const override { return utils::identifier(id_); }
private:
utils::identifier id_;
};
static null_identifiable null_id;
}
class fk_column_generator : public utils::identifier_serializer
{
public:
column generate(const char *id, utils::identifiable &x);
void serialize(short &i, const utils::field_attributes &attributes) override;
void serialize(int &i, const utils::field_attributes &attributes) override;
void serialize(long &i, const utils::field_attributes &attributes) override;
void serialize(long long int &i, const utils::field_attributes &attributes) override;
void serialize(unsigned short &i, const utils::field_attributes &attributes) override;
void serialize(unsigned int &i, const utils::field_attributes &attributes) override;
void serialize(unsigned long &i, const utils::field_attributes &attributes) override;
void serialize(unsigned long long int &i, const utils::field_attributes &attributes) override;
void serialize(std::string &string, const utils::field_attributes &attributes) override;
void serialize(utils::null_type_t &type, const utils::field_attributes &attributes) override;
private:
utils::identifiable &identifiable_{detail::null_id};
const char *id_{};
column column_;
};
class column_generator class column_generator
{ {
private: private:
@ -37,21 +77,23 @@ public:
template < class V > template < class V >
void on_primary_key(const char *, V &x, typename std::enable_if<std::is_integral<V>::value && !std::is_same<bool, V>::value>::type* = 0); void on_primary_key(const char *, V &x, typename std::enable_if<std::is_integral<V>::value && !std::is_same<bool, V>::value>::type* = 0);
void on_primary_key(const char *id, std::string &pk, size_t size); void on_primary_key(const char *id, std::string &pk, size_t size);
void on_revision(const char *id, unsigned long long &/*rev*/); void on_revision(const char *id, unsigned long long &rev);
template<typename Type> template<typename Type>
void on_attribute(const char *id, Type &x, const utils::field_attributes &attr = utils::null_attributes); void on_attribute(const char *id, Type &x, const utils::field_attributes &attr = utils::null_attributes);
void on_belongs_to(const char *id, utils::identifiable &x, utils::cascade_type); void on_belongs_to(const char *id, utils::identifiable &x, utils::cascade_type);
void on_has_one(const char *id, utils::identifiable &x, utils::cascade_type); void on_has_one(const char *id, utils::identifiable &x, utils::cascade_type);
// void on_has_many(const char *, abstract_container &, const char *, const char *, cascade_type) {} template<class ContainerType>
// void on_has_many(const char *, abstract_container &, cascade_type) {} void on_has_many(const char *, ContainerType &, const char *, const char *, utils::cascade_type) {}
template<class ContainerType>
void on_has_many(const char *, ContainerType &, utils::cascade_type) {}
private: private:
size_t index_ = 0; size_t index_ = 0;
std::vector<column> &columns_; std::vector<column> &columns_;
// typed_column_identifier_serializer column_identifier_serializer_; fk_column_generator fk_column_generator_;
}; };
template<typename V> template<typename V>

View File

@ -0,0 +1,59 @@
#ifndef QUERY_COLUMN_NAME_GENERATOR_HPP
#define QUERY_COLUMN_NAME_GENERATOR_HPP
#include "matador/utils/access.hpp"
#include "matador/utils/field_attributes.hpp"
#include <string>
#include <vector>
namespace matador::utils {
class identifiable;
}
namespace matador::sql {
class column_name_generator
{
private:
explicit column_name_generator(std::vector<std::string> &column_names);
public:
~column_name_generator() = default;
template < class Type >
static std::vector<std::string> generate()
{
std::vector<std::string> columns;
column_name_generator gen(columns);
Type obj;
matador::utils::access::process(gen, obj);
return std::move(columns);
}
template < class V >
void on_primary_key(const char *id, V &, typename std::enable_if<std::is_integral<V>::value && !std::is_same<bool, V>::value>::type* = 0)
{
column_names_.emplace_back(id);
}
void on_primary_key(const char *id, std::string &, size_t);
void on_revision(const char *id, unsigned long long &/*rev*/);
template<typename Type>
void on_attribute(const char *id, Type &, const utils::field_attributes &/*attr*/ = utils::null_attributes)
{
column_names_.emplace_back(id);
}
void on_belongs_to(const char *id, utils::identifiable &, utils::cascade_type);
void on_has_one(const char *id, utils::identifiable &, utils::cascade_type);
// void on_has_many(const char *, abstract_container &, const char *, const char *, cascade_type) {}
// void on_has_many(const char *, abstract_container &, cascade_type) {}
private:
std::vector<std::string> &column_names_;
};
}
#endif //QUERY_COLUMN_NAME_GENERATOR_HPP

View File

@ -1,20 +1,10 @@
#ifndef QUERY_KEY_VALUE_PAIR_HPP #ifndef QUERY_KEY_VALUE_PAIR_HPP
#define QUERY_KEY_VALUE_PAIR_HPP #define QUERY_KEY_VALUE_PAIR_HPP
#include <any> #include "matador/sql/any_type.hpp"
#include <string>
#include <variant>
namespace matador::sql { namespace matador::sql {
using any_type = std::variant<
char, short, int, long, long long,
unsigned char, unsigned short, unsigned int, unsigned long, unsigned long long,
bool,
float, double,
const char*,
std::string>;
class key_value_pair class key_value_pair
{ {
public: public:

View File

@ -3,6 +3,7 @@
#include "matador/sql/column.hpp" #include "matador/sql/column.hpp"
#include "matador/sql/column_generator.hpp" #include "matador/sql/column_generator.hpp"
#include "matador/sql/column_name_generator.hpp"
#include "matador/sql/key_value_pair.hpp" #include "matador/sql/key_value_pair.hpp"
#include "matador/sql/query_result.hpp" #include "matador/sql/query_result.hpp"
#include "matador/sql/record.hpp" #include "matador/sql/record.hpp"
@ -158,6 +159,11 @@ public:
using query_intermediate::query_intermediate; using query_intermediate::query_intermediate;
query_into_intermediate into(const std::string &table, std::initializer_list<std::string> column_names); query_into_intermediate into(const std::string &table, std::initializer_list<std::string> column_names);
template<class Type>
query_into_intermediate into(const std::string &table)
{
return into(table, column_name_generator::generate<Type>());
}
}; };
class query_execute_where_intermediate : public query_execute_finish class query_execute_where_intermediate : public query_execute_finish

View File

@ -0,0 +1,70 @@
#ifndef QUERY_VALUE_GENERATOR_HPP
#define QUERY_VALUE_GENERATOR_HPP
#include "matador/sql/any_type.hpp"
#include "matador/utils/access.hpp"
#include "matador/utils/field_attributes.hpp"
#include "matador/utils/identifiable.hpp"
#include <vector>
namespace matador::utils {
enum class cascade_type;
}
namespace matador::sql {
class value_generator
{
private:
explicit value_generator(std::vector<any_type> &values);
public:
~value_generator() = default;
template < class Type >
static std::vector<any_type> generate()
{
std::vector<any_type> values;
value_generator gen(values);
Type obj;
matador::utils::access::process(gen, obj);
return std::move(values);
}
template<typename ValueType>
void on_primary_key(const char *, ValueType &x, typename std::enable_if<std::is_integral<ValueType>::value && !std::is_same<bool, ValueType>::value>::type* = 0)
{
append(x);
}
void on_primary_key(const char *id, std::string &pk, size_t size);
void on_revision(const char *id, unsigned long long &rev);
template < class Type >
void on_attribute(const char *, Type &x, const utils::field_attributes &/*attr*/ = utils::null_attributes)
{
append(x);
}
void on_attribute(const char *id, char *x, const utils::field_attributes &/*attr*/ = utils::null_attributes);
void on_attribute(const char *id, std::string &x, const utils::field_attributes &/*attr*/ = utils::null_attributes);
void on_belongs_to(const char *id, utils::identifiable &x, utils::cascade_type);
void on_has_one(const char *id, utils::identifiable &x, utils::cascade_type);
template<class ContainerType>
void on_has_many(const char *, ContainerType &, const char *, const char *, utils::cascade_type) {}
template<class ContainerType>
void on_has_many(const char *, ContainerType &, utils::cascade_type) {}
private:
template<typename Type>
void append(Type &value)
{
values_.emplace_back(value);
}
private:
std::vector<any_type> &values_;
};
}
#endif //QUERY_VALUE_GENERATOR_HPP

View File

@ -12,7 +12,8 @@ set(SQL_SOURCES
sql/session.cpp sql/session.cpp
sql/backend_provider.cpp sql/backend_provider.cpp
sql/query_result_impl.cpp sql/query_result_impl.cpp
sql/column_generator.cpp) sql/column_generator.cpp
sql/column_name_generator.cpp)
set(SQL_HEADER set(SQL_HEADER
../include/matador/sql/dialect.hpp ../include/matador/sql/dialect.hpp
@ -33,7 +34,10 @@ set(SQL_HEADER
../include/matador/sql/session.hpp ../include/matador/sql/session.hpp
../include/matador/sql/backend_provider.hpp ../include/matador/sql/backend_provider.hpp
../include/matador/sql/query_result_impl.hpp ../include/matador/sql/query_result_impl.hpp
../include/matador/sql/column_generator.hpp) ../include/matador/sql/column_generator.hpp
../include/matador/sql/column_name_generator.hpp
../include/matador/sql/value_generator.hpp
../include/matador/sql/any_type.hpp)
set(UTILS_HEADER set(UTILS_HEADER
../include/matador/utils/field_attributes.hpp ../include/matador/utils/field_attributes.hpp
@ -52,7 +56,8 @@ set(UTILS_SOURCES
sql/condition.cpp sql/condition.cpp
utils/library.cpp utils/library.cpp
utils/os.cpp utils/os.cpp
utils/identifier.cpp) utils/identifier.cpp
sql/value_generator.cpp)
add_library(matador STATIC ${SQL_SOURCES} ${SQL_HEADER} ${UTILS_SOURCES} ${UTILS_HEADER}) add_library(matador STATIC ${SQL_SOURCES} ${SQL_HEADER} ${UTILS_SOURCES} ${UTILS_HEADER})
target_include_directories(matador PUBLIC ${PROJECT_SOURCE_DIR}/include) target_include_directories(matador PUBLIC ${PROJECT_SOURCE_DIR}/include)

View File

@ -1,5 +1,7 @@
#include "matador/sql/column_generator.hpp" #include "matador/sql/column_generator.hpp"
#include "matador/utils/identifiable.hpp"
namespace matador::sql { namespace matador::sql {
column_generator::column_generator(std::vector<column> &columns) column_generator::column_generator(std::vector<column> &columns)
@ -17,7 +19,8 @@ void column_generator::on_revision(const char *id, unsigned long long int &x)
void column_generator::on_belongs_to(const char *id, utils::identifiable &x, utils::cascade_type) void column_generator::on_belongs_to(const char *id, utils::identifiable &x, utils::cascade_type)
{ {
columns_.push_back(fk_column_generator_.generate(id, x));
// x.primary_key().serialize(fk_column_generator)
} }
void column_generator::on_has_one(const char *id, utils::identifiable &x, utils::cascade_type) void column_generator::on_has_one(const char *id, utils::identifiable &x, utils::cascade_type)
@ -25,4 +28,62 @@ void column_generator::on_has_one(const char *id, utils::identifiable &x, utils:
} }
column fk_column_generator::generate(const char *id, utils::identifiable &x)
{
identifiable_ = x;
id_ = id;
x.primary_key().serialize(*this);
return column_;
}
void fk_column_generator::serialize(short &i, const utils::field_attributes &attributes)
{
column_.
}
void fk_column_generator::serialize(int &i, const utils::field_attributes &attributes)
{
}
void fk_column_generator::serialize(long &i, const utils::field_attributes &attributes)
{
}
void fk_column_generator::serialize(long long int &i, const utils::field_attributes &attributes)
{
}
void fk_column_generator::serialize(unsigned short &i, const utils::field_attributes &attributes)
{
}
void fk_column_generator::serialize(unsigned int &i, const utils::field_attributes &attributes)
{
}
void fk_column_generator::serialize(unsigned long &i, const utils::field_attributes &attributes)
{
}
void fk_column_generator::serialize(unsigned long long int &i, const utils::field_attributes &attributes)
{
}
void fk_column_generator::serialize(std::string &string, const utils::field_attributes &attributes)
{
}
void fk_column_generator::serialize(utils::null_type_t &type, const utils::field_attributes &attributes)
{
}
} }

View File

@ -0,0 +1,29 @@
#include "matador/sql/column_name_generator.hpp"
namespace matador::sql {
column_name_generator::column_name_generator(std::vector<std::string> &column_names)
: column_names_(column_names)
{}
void column_name_generator::on_primary_key(const char *id, std::string &, size_t)
{
column_names_.emplace_back(id);
}
void column_name_generator::on_revision(const char *id, unsigned long long int &)
{
column_names_.emplace_back(id);
}
void column_name_generator::on_belongs_to(const char *id, utils::identifiable &, utils::cascade_type)
{
column_names_.emplace_back(id);
}
void column_name_generator::on_has_one(const char *id, utils::identifiable &, utils::cascade_type)
{
column_names_.emplace_back(id);
}
}

View File

@ -0,0 +1,36 @@
#include "matador/sql/value_generator.hpp"
namespace matador::sql {
value_generator::value_generator(std::vector<any_type> &values)
: values_(values)
{}
void value_generator::on_primary_key(const char *, std::string &pk, size_t)
{
append(pk);
}
void value_generator::on_revision(const char *, unsigned long long int &rev)
{
append(rev);
}
void value_generator::on_attribute(const char *, char *x, const utils::field_attributes &)
{
append(x);
}
void value_generator::on_attribute(const char *, std::string &x, const utils::field_attributes &)
{
append(x);
}
void value_generator::on_belongs_to(const char *, utils::identifiable &x, utils::cascade_type) {
}
void value_generator::on_has_one(const char *, utils::identifiable &x, utils::cascade_type) {
}
}

View File

@ -16,7 +16,8 @@ add_executable(tests builder.cpp
backend_provider.cpp backend_provider.cpp
connection.cpp connection.cpp
product.hpp product.hpp
column_generator.cpp) column_generator.cpp
column_name_generator.cpp)
target_link_libraries(tests PRIVATE target_link_libraries(tests PRIVATE
Catch2::Catch2WithMain Catch2::Catch2WithMain
matador matador

View File

@ -0,0 +1,30 @@
#include <catch2/catch_test_macros.hpp>
#include "matador/sql/column_name_generator.hpp"
#include "product.hpp"
using namespace matador::sql;
TEST_CASE("Generate column names from object", "[column name generator]") {
auto columns = column_name_generator::generate<matador::test::product>();
const std::vector<std::string> expected_columns = {
"product_name",
"supplier_id",
"category_id",
"quantity_per_unit",
"unit_price",
"units_in_stock",
"units_in_order",
"reorder_level",
"discontinued"
};
REQUIRE(!columns.empty());
REQUIRE(columns.size() == expected_columns.size());
for (size_t i = 0; i != expected_columns.size(); ++i) {
REQUIRE(expected_columns[i] == columns[i]);
}
}

View File

@ -7,6 +7,7 @@
#include "product.hpp" #include "product.hpp"
using namespace matador::sql; using namespace matador::sql;
using namespace matador::test;
TEST_CASE("Execute create table statement", "[connection]") { TEST_CASE("Execute create table statement", "[connection]") {
connection_pool<connection> pool("sqlite://sqlite.db", 4); connection_pool<connection> pool("sqlite://sqlite.db", 4);
@ -21,7 +22,9 @@ TEST_CASE("Execute create table statement", "[connection]") {
REQUIRE(res.second == R"(CREATE TABLE "person" ("id" BIGINT NOT NULL PRIMARY KEY, "name" VARCHAR(255), "age" INTEGER))"); REQUIRE(res.second == R"(CREATE TABLE "person" ("id" BIGINT NOT NULL PRIMARY KEY, "name" VARCHAR(255), "age" INTEGER))");
res = s.create().table<matador::test::product>("product").execute(); res = s.create().table<product>("product").execute();
REQUIRE(res.second == R"(CREATE TABLE "person" ("id" BIGINT NOT NULL PRIMARY KEY, "name" VARCHAR(255), "age" INTEGER))");
} }
TEST_CASE("Execute drop table statement", "[connection]") { TEST_CASE("Execute drop table statement", "[connection]") {