added field class

This commit is contained in:
Sascha Kuehl 2024-03-10 11:51:07 +01:00
parent 2b552c4383
commit dd94ed1020
4 changed files with 153 additions and 18 deletions

View File

@ -4,40 +4,107 @@
#include "matador/sql/any_type.hpp" #include "matador/sql/any_type.hpp"
#include "matador/sql/any_type_to_visitor.hpp" #include "matador/sql/any_type_to_visitor.hpp"
#include "matador/utils/types.hpp"
#include <optional>
#include <string> #include <string>
namespace matador::sql { namespace matador::sql {
enum class field_type {
Integer,
FloatingPoint,
String,
Boolean,
Blob,
Null
};
template < typename Type, typename Enable = void >
struct field_traits;
template<typename Type>
struct field_traits<Type, typename std::enable_if<std::is_integral<Type>::value && !std::is_same<Type, bool>::value>::type>
{
static field_type type() { return field_type::Integer; }
};
template<typename Type>
struct field_traits<Type, typename std::enable_if<std::is_floating_point<Type>::value>::type>
{
static field_type type() { return field_type::FloatingPoint; }
};
template<typename Type>
struct field_traits<Type, typename std::enable_if<std::is_same<Type, std::string>::value>::type>
{
static field_type type() { return field_type::String; }
};
template<typename Type>
struct field_traits<Type, typename std::enable_if<std::is_same<Type, bool>::value>::type>
{
static field_type type() { return field_type::Boolean; }
};
template<typename Type>
struct field_traits<Type, typename std::enable_if<std::is_same<Type, nullptr_t>::value>::type>
{
static field_type type() { return field_type::Null; }
};
template<typename Type>
struct field_traits<Type, typename std::enable_if<std::is_same<Type, utils::blob>::value>::type>
{
static field_type type() { return field_type::Blob; }
};
class field class field
{ {
public: public:
explicit field(std::string name);
template<typename Type>
field(std::string name, Type value)
: name_(std::move(name))
, value_(value)
, type_(field_traits<Type>::type()) {}
field(const field &x) = default;
field& operator=(const field &x) = default;
field(field &&x) noexcept;
field& operator=(field &&x) noexcept;
template<typename Type>
field& operator=(Type value) {
value_ = std::move(value);
type_ = field_traits<Type>::type();
return *this;
}
[[nodiscard]] const std::string& name() const; [[nodiscard]] const std::string& name() const;
template<class Type> template<class Type>
Type as() const std::optional<Type> as() const
{ {
const Type* ptr= std::get_if<Type>(&value_);
if (ptr) {
return *ptr;
}
any_type_to_visitor<Type> visitor; any_type_to_visitor<Type> visitor;
std::visit(visitor, const_cast<any_type&>(value_)); std::visit(visitor, const_cast<any_type&>(value_));
return visitor.result; return visitor.result;
} }
bool is_integer() const; [[nodiscard]] std::string str() const;
bool is_floating_point() const;
bool is_bool() const; [[nodiscard]] bool is_integer() const;
bool is_string() const; [[nodiscard]] bool is_floating_point() const;
bool is_blob() const; [[nodiscard]] bool is_bool() const;
bool is_null() const; [[nodiscard]] bool is_string() const;
[[nodiscard]] bool is_blob() const;
[[nodiscard]] bool is_null() const;
friend std::ostream& operator<<(std::ostream &out, const field &col); friend std::ostream& operator<<(std::ostream &out, const field &col);
private: private:
std::string name_; std::string name_;
any_type value_; any_type value_;
field_type type_;
}; };
} }

View File

@ -2,6 +2,28 @@
namespace matador::sql { namespace matador::sql {
field::field(std::string name)
: name_(std::move(name)), value_(nullptr), type_(field_traits<nullptr_t>::type())
{}
field::field(field &&x) noexcept
: name_(std::move(x.name_)), value_(std::move(x.value_)), type_(x.type_)
{
x.value_ = nullptr;
x.type_ = field_type::Null;
}
field &field::operator=(field &&x) noexcept
{
name_ = std::move(x.name_);
value_ = std::move(x.value_);
type_ = x.type_;
x.value_ = nullptr;
x.type_ = field_type::Null;
return *this;
}
const std::string &field::name() const const std::string &field::name() const
{ {
return name_; return name_;
@ -9,36 +31,43 @@ const std::string &field::name() const
std::ostream &operator<<(std::ostream &out, const field &col) std::ostream &operator<<(std::ostream &out, const field &col)
{ {
out << col.str();
return out; return out;
} }
std::string field::str() const
{
return as<std::string>().value();
}
bool field::is_integer() const bool field::is_integer() const
{ {
return false; return type_ == field_type::Integer;
} }
bool field::is_floating_point() const bool field::is_floating_point() const
{ {
return false; return type_ == field_type::FloatingPoint;
} }
bool field::is_bool() const bool field::is_bool() const
{ {
return false; return type_ == field_type::Boolean;
} }
bool field::is_string() const bool field::is_string() const
{ {
return false; return type_ == field_type::String;
} }
bool field::is_blob() const bool field::is_blob() const
{ {
return false; return type_ == field_type::Blob;
} }
bool field::is_null() const bool field::is_null() const
{ {
return false; return type_ == field_type::Null;
} }
} }

View File

@ -38,7 +38,8 @@ add_executable(tests
DummyConnection.cpp DummyConnection.cpp
EntityQueryBuilderTest.cpp EntityQueryBuilderTest.cpp
models/author.hpp models/author.hpp
models/book.hpp) models/book.hpp
FieldTest.cpp)
target_link_libraries(tests PRIVATE target_link_libraries(tests PRIVATE
Catch2::Catch2WithMain Catch2::Catch2WithMain

38
test/FieldTest.cpp Normal file
View File

@ -0,0 +1,38 @@
#include <catch2/catch_test_macros.hpp>
#include "matador/sql/field.hpp"
using namespace matador::sql;
TEST_CASE("Field test", "[field]") {
field f("name");
REQUIRE(f.name() == "name");
REQUIRE(f.is_null());
REQUIRE(!f.is_integer());
REQUIRE(!f.is_floating_point());
REQUIRE(!f.is_blob());
REQUIRE(!f.is_bool());
REQUIRE(!f.is_string());
f = 7UL;
REQUIRE(!f.is_null());
REQUIRE(f.is_integer());
REQUIRE(!f.is_floating_point());
REQUIRE(!f.is_blob());
REQUIRE(!f.is_bool());
REQUIRE(!f.is_string());
auto int_val = f.as<int>();
REQUIRE(int_val.has_value());
REQUIRE(int_val.value() == 7);
auto float_val = f.as<float>();
REQUIRE(float_val.has_value());
REQUIRE(float_val.value() == 7.0);
auto str_val = f.as<std::string>();
REQUIRE(str_val.has_value());
REQUIRE(str_val.value() == "7");
auto bool_val = f.as<bool>();
REQUIRE(bool_val.has_value());
REQUIRE(bool_val.value());
}