added sequence-specific code (create, drop and nextval)

This commit is contained in:
sascha 2026-02-17 13:31:06 +01:00
parent 80426c5293
commit 5668f1060b
17 changed files with 174 additions and 8 deletions

View File

@ -2,8 +2,8 @@ CPMAddPackage("gh:catchorg/Catch2@3.7.1")
list(APPEND CMAKE_MODULE_PATH ${catch2_SOURCE_DIR}/extras) list(APPEND CMAKE_MODULE_PATH ${catch2_SOURCE_DIR}/extras)
#set(POSTGRES_CONNECTION_STRING "postgres://test:test123!@127.0.0.1:15442/matador") set(POSTGRES_CONNECTION_STRING "postgres://test:test123!@127.0.0.1:15442/matador")
set(POSTGRES_CONNECTION_STRING "postgres://test:test123!@127.0.0.1:5432/matador") #set(POSTGRES_CONNECTION_STRING "postgres://test:test123!@127.0.0.1:5432/matador")
configure_file(Connection.hpp.in ${PROJECT_BINARY_DIR}/backends/postgres/test/connection.hpp @ONLY IMMEDIATE) configure_file(Connection.hpp.in ${PROJECT_BINARY_DIR}/backends/postgres/test/connection.hpp @ONLY IMMEDIATE)

View File

@ -48,6 +48,11 @@ public:
[[nodiscard]] const utils::identifier &primary_key() const { return proxy_->primary_key(); } [[nodiscard]] const utils::identifier &primary_key() const { return proxy_->primary_key(); }
void primary_key(const utils::identifier &pk) { proxy_->primary_key(pk); } void primary_key(const utils::identifier &pk) { proxy_->primary_key(pk); }
[[nodiscard]] bool is_persistent() const { return proxy_->is_persistent(); }
[[nodiscard]] bool is_transient() const { return !proxy_->is_transient(); }
[[nodiscard]] bool is_detached() const { return !proxy_->is_detached(); }
[[nodiscard]] bool is_removed() const { return !proxy_->is_removed(); }
private: private:
std::shared_ptr<object_proxy<Type> > proxy_{}; std::shared_ptr<object_proxy<Type> > proxy_{};
}; };

View File

@ -11,7 +11,8 @@ public:
query_drop_intermediate(); query_drop_intermediate();
executable_query table(const table &tab); executable_query table(const table &tab);
// executable_query schema(const std::string &schema_name); executable_query sequence(const std::string &sequence_name);
executable_query schema(const std::string &schema_name);
}; };
} }

View File

@ -17,6 +17,8 @@ class query_select_intermediate : public query_intermediate
public: public:
explicit query_select_intermediate(const std::vector<table_column>& columns); explicit query_select_intermediate(const std::vector<table_column>& columns);
fetchable_query nextval(const std::string& sequence_name);
template<typename... Tables> template<typename... Tables>
query_from_intermediate from(const Tables&... tables) { query_from_intermediate from(const Tables&... tables) {
std::vector<table> v { tables... }; std::vector<table> v { tables... };

View File

@ -126,6 +126,20 @@ private:
std::vector<table_column> columns_; std::vector<table_column> columns_;
}; };
class query_select_nextval_part final : public query_part {
public:
explicit query_select_nextval_part(std::string sequence_name);
[[nodiscard]] const std::string& sequence_name() const;
private:
void accept(query_part_visitor &visitor) override;
private:
std::string sequence_name_;
};
/** /**
* Represents the SQL FROM part * Represents the SQL FROM part
*/ */
@ -443,6 +457,19 @@ private:
std::list<table_constraint> constraints_; std::list<table_constraint> constraints_;
}; };
class query_create_sequence_part final : public query_part {
public:
explicit query_create_sequence_part(std::string sequence_name);
[[nodiscard]] const std::string& sequence_name() const;
private:
void accept(query_part_visitor &visitor) override;
private:
std::string sequence_name_;
};
class query_create_schema_part final : public query_part { class query_create_schema_part final : public query_part {
public: public:
explicit query_create_schema_part(std::string schema); explicit query_create_schema_part(std::string schema);
@ -479,6 +506,19 @@ private:
class table table_; class table table_;
}; };
class query_drop_sequence_part final : public query_part {
public:
explicit query_drop_sequence_part(std::string sequence_name);
[[nodiscard]] const std::string& sequence_name() const;
private:
void accept(query_part_visitor &visitor) override;
private:
std::string sequence_name_;
};
class query_drop_schema_part final : public query_part { class query_drop_schema_part final : public query_part {
public: public:
explicit query_drop_schema_part(std::string schema_); explicit query_drop_schema_part(std::string schema_);

View File

@ -45,6 +45,7 @@ protected:
void visit(internal::query_drop_key_constraint_part_by_constraint &part) override; void visit(internal::query_drop_key_constraint_part_by_constraint &part) override;
void visit(internal::query_select_part &part) override; void visit(internal::query_select_part &part) override;
void visit(internal::query_select_nextval_part &part) override;
void visit(internal::query_from_part &part) override; void visit(internal::query_from_part &part) override;
void visit(internal::query_join_table_part &part) override; void visit(internal::query_join_table_part &part) override;
void visit(internal::query_join_query_part &part) override; void visit(internal::query_join_query_part &part) override;
@ -72,10 +73,12 @@ protected:
void visit(internal::query_create_table_part &part) override; void visit(internal::query_create_table_part &part) override;
void visit(internal::query_create_table_columns_part& part) override; void visit(internal::query_create_table_columns_part& part) override;
void visit(internal::query_create_table_constraints_part& part) override; void visit(internal::query_create_table_constraints_part& part) override;
void visit(internal::query_create_sequence_part &part) override;
void visit(internal::query_create_schema_part& part) override; void visit(internal::query_create_schema_part& part) override;
void visit(internal::query_drop_part &part) override; void visit(internal::query_drop_part &part) override;
void visit(internal::query_drop_table_part &part) override; void visit(internal::query_drop_table_part &part) override;
void visit(internal::query_drop_sequence_part &part) override;
void visit(internal::query_drop_schema_part& part) override; void visit(internal::query_drop_schema_part& part) override;
static std::string build_table_name(sql::dialect_token token, const sql::dialect &d, const table& t); static std::string build_table_name(sql::dialect_token token, const sql::dialect &d, const table& t);

View File

@ -17,6 +17,7 @@ class query_add_foreign_key_reference_part;
class query_add_primary_key_constraint_part; class query_add_primary_key_constraint_part;
// Select // Select
class query_select_part; class query_select_part;
class query_select_nextval_part;
class query_from_part; class query_from_part;
class query_join_table_part; class query_join_table_part;
class query_join_query_part; class query_join_query_part;
@ -45,12 +46,16 @@ class query_create_part;
class query_create_table_part; class query_create_table_part;
class query_create_table_columns_part; class query_create_table_columns_part;
class query_create_table_constraints_part; class query_create_table_constraints_part;
// Create Sequence
class query_create_sequence_part;
// Create schema // Create schema
class query_create_schema_part; class query_create_schema_part;
// Drop common // Drop common
class query_drop_part; class query_drop_part;
// Drop table // Drop table
class query_drop_table_part; class query_drop_table_part;
// Drop sequence
class query_drop_sequence_part;
// Drop schema // Drop schema
class query_drop_schema_part; class query_drop_schema_part;
} }
@ -70,6 +75,7 @@ public:
virtual void visit(internal::query_drop_key_constraint_part_by_constraint &part) = 0; virtual void visit(internal::query_drop_key_constraint_part_by_constraint &part) = 0;
virtual void visit(internal::query_select_part &part) = 0; virtual void visit(internal::query_select_part &part) = 0;
virtual void visit(internal::query_select_nextval_part &part) = 0;
virtual void visit(internal::query_from_part &part) = 0; virtual void visit(internal::query_from_part &part) = 0;
virtual void visit(internal::query_join_table_part &part) = 0; virtual void visit(internal::query_join_table_part &part) = 0;
virtual void visit(internal::query_join_query_part &part) = 0; virtual void visit(internal::query_join_query_part &part) = 0;
@ -97,10 +103,12 @@ public:
virtual void visit(internal::query_create_table_part &part) = 0; virtual void visit(internal::query_create_table_part &part) = 0;
virtual void visit(internal::query_create_table_columns_part &part) = 0; virtual void visit(internal::query_create_table_columns_part &part) = 0;
virtual void visit(internal::query_create_table_constraints_part &part) = 0; virtual void visit(internal::query_create_table_constraints_part &part) = 0;
virtual void visit(internal::query_create_sequence_part &part) = 0;
virtual void visit(internal::query_create_schema_part &part) = 0; virtual void visit(internal::query_create_schema_part &part) = 0;
virtual void visit(internal::query_drop_part &part) = 0; virtual void visit(internal::query_drop_part &part) = 0;
virtual void visit(internal::query_drop_table_part &part) = 0; virtual void visit(internal::query_drop_table_part &part) = 0;
virtual void visit(internal::query_drop_sequence_part &part) = 0;
virtual void visit(internal::query_drop_schema_part &part) = 0; virtual void visit(internal::query_drop_schema_part &part) = 0;
}; };

View File

@ -145,6 +145,7 @@ public:
[[nodiscard]] const std::string& end_binary_data() const; [[nodiscard]] const std::string& end_binary_data() const;
[[nodiscard]] const std::string& end_quote() const; [[nodiscard]] const std::string& end_quote() const;
[[nodiscard]] const std::string& end_string_data() const; [[nodiscard]] const std::string& end_string_data() const;
[[nodiscard]] const std::string& exists() const;
[[nodiscard]] const std::string& foreign_key() const; [[nodiscard]] const std::string& foreign_key() const;
[[nodiscard]] const std::string& from() const; [[nodiscard]] const std::string& from() const;
[[nodiscard]] const std::string& group_by() const; [[nodiscard]] const std::string& group_by() const;
@ -157,6 +158,7 @@ public:
[[nodiscard]] const std::string& join() const; [[nodiscard]] const std::string& join() const;
[[nodiscard]] const std::string& like() const; [[nodiscard]] const std::string& like() const;
[[nodiscard]] const std::string& limit() const; [[nodiscard]] const std::string& limit() const;
[[nodiscard]] const std::string& nextval() const;
[[nodiscard]] const std::string& not_() const; [[nodiscard]] const std::string& not_() const;
[[nodiscard]] const std::string& not_null() const; [[nodiscard]] const std::string& not_null() const;
[[nodiscard]] const std::string& offset() const; [[nodiscard]] const std::string& offset() const;
@ -171,6 +173,7 @@ public:
[[nodiscard]] const std::string& schema() const; [[nodiscard]] const std::string& schema() const;
[[nodiscard]] const std::string& select() const; [[nodiscard]] const std::string& select() const;
[[nodiscard]] const std::string& set() const; [[nodiscard]] const std::string& set() const;
[[nodiscard]] const std::string& sequence() const;
[[nodiscard]] const std::string& start_quote() const; [[nodiscard]] const std::string& start_quote() const;
[[nodiscard]] const std::string& string_quote() const; [[nodiscard]] const std::string& string_quote() const;
[[nodiscard]] const std::string& table() const; [[nodiscard]] const std::string& table() const;
@ -192,6 +195,7 @@ private:
std::string default_schema_name_; std::string default_schema_name_;
token_to_string_map tokens_ { token_to_string_map tokens_ {
{dialect_token::AddConstraint, "ADD CONSTRAINT"}, {dialect_token::AddConstraint, "ADD CONSTRAINT"},
{dialect_token::Alter, "ALTER"}, {dialect_token::Alter, "ALTER"},
@ -215,6 +219,7 @@ private:
{dialect_token::EndBinaryData, "'"}, {dialect_token::EndBinaryData, "'"},
{dialect_token::EndQuote, "\""}, {dialect_token::EndQuote, "\""},
{dialect_token::EndStringData, "'"}, {dialect_token::EndStringData, "'"},
{dialect_token::Exists, "EXISTS"},
{dialect_token::ForeignKey, "FOREIGN KEY"}, {dialect_token::ForeignKey, "FOREIGN KEY"},
{dialect_token::From, "FROM"}, {dialect_token::From, "FROM"},
{dialect_token::GroupBy, "GROUP BY"}, {dialect_token::GroupBy, "GROUP BY"},
@ -227,6 +232,7 @@ private:
{dialect_token::Join, "LEFT JOIN"}, {dialect_token::Join, "LEFT JOIN"},
{dialect_token::Like, "LIKE"}, {dialect_token::Like, "LIKE"},
{dialect_token::Limit, "LIMIT"}, {dialect_token::Limit, "LIMIT"},
{dialect_token::NextVal, "NEXTVAL"},
{dialect_token::None, ""}, {dialect_token::None, ""},
{dialect_token::Not, "NOT"}, {dialect_token::Not, "NOT"},
{dialect_token::NotNull, "NOT NULL"}, {dialect_token::NotNull, "NOT NULL"},
@ -242,6 +248,7 @@ private:
{dialect_token::Schema, "SCHEMA"}, {dialect_token::Schema, "SCHEMA"},
{dialect_token::Select, "SELECT"}, {dialect_token::Select, "SELECT"},
{dialect_token::Set, "SET"}, {dialect_token::Set, "SET"},
{dialect_token::Sequence, "SEQUENCE"},
{dialect_token::StartQuote, "\""}, {dialect_token::StartQuote, "\""},
{dialect_token::StringQuote, "'"}, {dialect_token::StringQuote, "'"},
{dialect_token::Table, "TABLE"}, {dialect_token::Table, "TABLE"},
@ -250,7 +257,6 @@ private:
{dialect_token::Values, "VALUES"}, {dialect_token::Values, "VALUES"},
{dialect_token::Where, "WHERE"} {dialect_token::Where, "WHERE"}
}; };
data_type_to_string_map data_types_ { data_type_to_string_map data_types_ {
{utils::basic_type::Int8, "TINYINT"}, {utils::basic_type::Int8, "TINYINT"},
{utils::basic_type::Int16, "SMALLINT"}, {utils::basic_type::Int16, "SMALLINT"},

View File

@ -29,6 +29,7 @@ enum class dialect_token : uint8_t {
EndBinaryData, EndBinaryData,
EndQuote, EndQuote,
EndStringData, EndStringData,
Exists,
ForeignKey, ForeignKey,
From, From,
GroupBy, GroupBy,
@ -41,6 +42,7 @@ enum class dialect_token : uint8_t {
Join, Join,
Like, Like,
Limit, Limit,
NextVal,
None, None,
Not, Not,
NotNull, NotNull,
@ -56,6 +58,7 @@ enum class dialect_token : uint8_t {
Schema, Schema,
Select, Select,
Set, Set,
Sequence,
StartQuote, StartQuote,
StringQuote, StringQuote,
Table, Table,

View File

@ -13,6 +13,7 @@ enum class sql_command {
Create, Create,
CreateTable, CreateTable,
CreateSchema, CreateSchema,
CreateSequence,
CreateDatabase, CreateDatabase,
Update, Update,
Insert, Insert,
@ -21,6 +22,7 @@ enum class sql_command {
Drop, Drop,
DropTable, DropTable,
DropSchema, DropSchema,
DropSequence,
DropDatabase, DropDatabase,
Alter, Alter,
AlterTable, AlterTable,

View File

@ -18,7 +18,9 @@ query_create_table_intermediate query_create_intermediate::table(const class tab
context_->parts.push_back(std::make_unique<internal::query_create_table_part>(tab)); context_->parts.push_back(std::make_unique<internal::query_create_table_part>(tab));
return {context_}; return {context_};
} }
query_create_sequence_intermediate query_create_intermediate::sequence(const std::string& sequence_name) { query_create_sequence_intermediate query_create_intermediate::sequence(const std::string& sequence_name) {
context_->parts.push_back(std::make_unique<internal::query_create_sequence_part>(sequence_name));
return {context_}; return {context_};
} }

View File

@ -5,7 +5,6 @@
#include "matador/query/query_data.hpp" #include "matador/query/query_data.hpp"
namespace matador::query { namespace matador::query {
query_drop_intermediate::query_drop_intermediate() { query_drop_intermediate::query_drop_intermediate() {
context_->parts.push_back(std::make_unique<internal::query_drop_part>()); context_->parts.push_back(std::make_unique<internal::query_drop_part>());
} }
@ -15,4 +14,13 @@ executable_query query_drop_intermediate::table(const class table &tab) {
return {context_}; return {context_};
} }
executable_query query_drop_intermediate::sequence(const std::string& sequence_name) {
context_->parts.push_back(std::make_unique<internal::query_drop_sequence_part>(sequence_name));
return {context_};
}
executable_query query_drop_intermediate::schema(const std::string& schema_name) {
context_->parts.push_back(std::make_unique<internal::query_drop_schema_part>(schema_name));
return {context_};
}
} }

View File

@ -11,9 +11,11 @@ query_select_intermediate::query_select_intermediate(const std::vector<table_col
context_->parts.push_back(std::make_unique<internal::query_select_part>(columns)); context_->parts.push_back(std::make_unique<internal::query_select_part>(columns));
} }
// query_from_intermediate query_select_intermediate::from(const table& tab) { fetchable_query query_select_intermediate::nextval(const std::string& sequence_name) {
// return from(std::vector{tab}); context_->parts.push_back(std::make_unique<internal::query_select_nextval_part>(sequence_name));
// } return {context_};
}
query_from_intermediate query_select_intermediate::from(const std::vector<table>& tables) { query_from_intermediate query_select_intermediate::from(const std::vector<table>& tables) {
context_->parts.push_back(std::make_unique<internal::query_from_part>(tables)); context_->parts.push_back(std::make_unique<internal::query_from_part>(tables));
for (const auto& tab : tables) { for (const auto& tab : tables) {

View File

@ -134,6 +134,19 @@ const std::vector<table_column> &query_select_part::columns() const {
return columns_; return columns_;
} }
query_select_nextval_part::query_select_nextval_part(std::string sequence_name)
: query_part(sql::dialect_token::NextVal)
, sequence_name_(std::move(sequence_name)){
}
const std::string& query_select_nextval_part::sequence_name() const {
return sequence_name_;
}
void query_select_nextval_part::accept(query_part_visitor& visitor) {
visitor.visit(*this);
}
query_from_part::query_from_part(std::vector<table> tables) query_from_part::query_from_part(std::vector<table> tables)
: query_part(sql::dialect_token::From) : query_part(sql::dialect_token::From)
, tables_(std::move(tables)) { , tables_(std::move(tables)) {
@ -419,6 +432,18 @@ void query_create_table_constraints_part::accept(query_part_visitor &visitor) {
visitor.visit(*this); visitor.visit(*this);
} }
query_create_sequence_part::query_create_sequence_part(std::string sequence_name)
: query_part(sql::dialect_token::Sequence)
, sequence_name_(std::move(sequence_name)) {}
const std::string& query_create_sequence_part::sequence_name() const {
return sequence_name_;
}
void query_create_sequence_part::accept(query_part_visitor& visitor) {
visitor.visit(*this);
}
query_create_schema_part::query_create_schema_part(std::string schema) query_create_schema_part::query_create_schema_part(std::string schema)
: query_part(sql::dialect_token::Schema) : query_part(sql::dialect_token::Schema)
, schema_(std::move(schema)) { , schema_(std::move(schema)) {
@ -453,6 +478,19 @@ void query_drop_table_part::accept(query_part_visitor &visitor) {
visitor.visit(*this); visitor.visit(*this);
} }
query_drop_sequence_part::query_drop_sequence_part(std::string sequence_name)
: query_part(sql::dialect_token::Table)
, sequence_name_(std::move(sequence_name)) {
}
const std::string& query_drop_sequence_part::sequence_name() const {
return sequence_name_;
}
void query_drop_sequence_part::accept(query_part_visitor& visitor) {
visitor.visit(*this);
}
query_drop_schema_part::query_drop_schema_part(std::string schema_) query_drop_schema_part::query_drop_schema_part(std::string schema_)
: query_part(sql::dialect_token::Schema) : query_part(sql::dialect_token::Schema)
, schema_(std::move(schema_)) { , schema_(std::move(schema_)) {

View File

@ -94,6 +94,11 @@ void query_builder::visit(internal::query_select_part &part) {
build_fetchable_columns(query_, part.columns(), *dialect_); build_fetchable_columns(query_, part.columns(), *dialect_);
} }
void query_builder::visit(internal::query_select_nextval_part& part) {
query_.sql += dialect_->nextval() + "('" + part.sequence_name() + "')";
prepare_prototype(query_.prototype, part.sequence_name());
}
void query_builder::visit(internal::query_from_part &part) { void query_builder::visit(internal::query_from_part &part) {
query_.table_name = part.tables().front().name(); query_.table_name = part.tables().front().name();
query_.sql += " " + dialect_->from() + " "; query_.sql += " " + dialect_->from() + " ";
@ -301,6 +306,11 @@ void query_builder::visit(internal::query_create_table_constraints_part& part) {
query_.sql += result; query_.sql += result;
} }
void query_builder::visit(internal::query_create_sequence_part& part) {
query_.command = sql::sql_command::CreateSequence;
query_.sql += " " + dialect_->sequence() + " " + dialect_->prepare_identifier_string(part.sequence_name());
}
void query_builder::visit( internal::query_create_schema_part& part ) { void query_builder::visit( internal::query_create_schema_part& part ) {
query_.command = sql::sql_command::CreateSchema; query_.command = sql::sql_command::CreateSchema;
query_.sql += " " + dialect_->schema() + " " + dialect_->prepare_identifier_string(part.schema()); query_.sql += " " + dialect_->schema() + " " + dialect_->prepare_identifier_string(part.schema());
@ -341,6 +351,11 @@ void query_builder::visit(internal::query_set_part &part) {
query_.sql += result; query_.sql += result;
} }
void query_builder::visit(internal::query_drop_sequence_part& part) {
query_.command = sql::sql_command::DropSequence;
query_.sql += " " + dialect_->sequence() + " " + dialect_->prepare_identifier_string(part.sequence_name());
}
void query_builder::visit(internal::query_drop_table_part &part) { void query_builder::visit(internal::query_drop_table_part &part) {
query_.table_name = part.table().name(); query_.table_name = part.table().name();
query_.sql += " " + build_table_name(part.token(), *dialect_, query_.table_name); query_.sql += " " + build_table_name(part.token(), *dialect_, query_.table_name);

View File

@ -171,6 +171,10 @@ const std::string &dialect::end_string_data() const {
return token_at(dialect_token::EndStringData); return token_at(dialect_token::EndStringData);
} }
const std::string& dialect::exists() const {
return token_at(dialect_token::Exists);
}
const std::string &dialect::from() const { const std::string &dialect::from() const {
return token_at(dialect_token::From); return token_at(dialect_token::From);
} }
@ -235,6 +239,10 @@ const std::string &dialect::limit() const {
return token_at(dialect_token::Limit); return token_at(dialect_token::Limit);
} }
const std::string& dialect::nextval() const {
return token_at(dialect_token::NextVal);
}
const std::string &dialect::not_() const { const std::string &dialect::not_() const {
return token_at(dialect_token::Not); return token_at(dialect_token::Not);
} }
@ -291,6 +299,10 @@ const std::string &dialect::set() const {
return token_at(dialect_token::Set); return token_at(dialect_token::Set);
} }
const std::string& dialect::sequence() const {
return token_at(dialect_token::Sequence);
}
const std::string &dialect::start_quote() const { const std::string &dialect::start_quote() const {
return token_at(dialect_token::StartQuote); return token_at(dialect_token::StartQuote);
} }

View File

@ -111,6 +111,25 @@ TEST_CASE_METHOD(QueryFixture, "Execute insert with returning", "[query][insert]
REQUIRE(rec->at(0).as<uint32_t>() == 7); REQUIRE(rec->at(0).as<uint32_t>() == 7);
} }
TEST_CASE_METHOD(QueryFixture, "Test sequence", "[query][sequence]") {
auto result = query::create()
.sequence("person_seq")
.execute(db);
REQUIRE(result.is_ok());
auto next_id = query::select()
.nextval("person_seq")
.fetch_value<uint32_t>(db);
REQUIRE(next_id.is_ok());
REQUIRE(*next_id == 1);
result = query::drop()
.sequence("person_seq")
.execute(db);
REQUIRE(result.is_ok());
}
TEST_CASE_METHOD(QueryFixture, "Execute insert statement", "[query][insert]") { TEST_CASE_METHOD(QueryFixture, "Execute insert statement", "[query][insert]") {
auto res = query::create() auto res = query::create()
.table("person") .table("person")