mirror of
https://github.com/VCMP-SqMod/SqMod.git
synced 2024-11-08 00:37:15 +01:00
1926 lines
71 KiB
C++
1926 lines
71 KiB
C++
#pragma once
|
|
|
|
// ------------------------------------------------------------------------------------------------
|
|
#include "Core/Buffer.hpp"
|
|
#include "Core/Utility.hpp"
|
|
#include "Library/Utils/Vector.hpp"
|
|
|
|
// ------------------------------------------------------------------------------------------------
|
|
#include <Poco/Data/Session.h>
|
|
#include <Poco/Data/Statement.h>
|
|
#include <Poco/Data/RecordSet.h>
|
|
#include <Poco/Data/SessionPool.h>
|
|
|
|
// ------------------------------------------------------------------------------------------------
|
|
namespace std { // NOLINT(cert-dcl58-cpp)
|
|
|
|
/* ------------------------------------------------------------------------------------------------
|
|
* Allows the StackStrF to write its data into a std::ostream type.
|
|
*/
|
|
inline std::ostream & operator << (std::ostream & out, const ::Sqrat::StackStrF & str)
|
|
{
|
|
if (str.mLen)
|
|
{
|
|
out.write(str.mPtr, str.mLen);
|
|
}
|
|
return out;
|
|
}
|
|
|
|
/* ------------------------------------------------------------------------------------------------
|
|
* Allows the StackStrF to write its data into a std::ostringstream type.
|
|
*/
|
|
inline std::ostringstream & operator << (std::ostringstream & out, const ::Sqrat::StackStrF & str)
|
|
{
|
|
if (str.mLen)
|
|
{
|
|
out.write(str.mPtr, str.mLen);
|
|
}
|
|
return out;
|
|
}
|
|
|
|
} // Namespace:: std
|
|
|
|
// ------------------------------------------------------------------------------------------------
|
|
namespace Poco { // NOLINT(modernize-concat-nested-namespaces)
|
|
namespace Data {
|
|
|
|
/* ------------------------------------------------------------------------------------------------
|
|
* Implementation of AbstractBinding for shared ownership binding of values.
|
|
* Because we cannot take references to script variables, we use this as a proxy.
|
|
*/
|
|
template <class T> struct ReferenceBinding : public AbstractBinding
|
|
{
|
|
using ValType = T;
|
|
using ValPtr = SharedPtr<ValType>;
|
|
using Type = ReferenceBinding<ValType>;
|
|
using Ptr = SharedPtr<Type>;
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Base constructor.
|
|
*/
|
|
explicit ReferenceBinding(const ValPtr& val, const std::string& name = "", Direction direction = PD_IN)
|
|
: AbstractBinding(name, direction), m_Value(val), m_Bound(false)
|
|
{
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Destructor.
|
|
*/
|
|
~ReferenceBinding() override = default;
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Retrieve columns occupied.
|
|
*/
|
|
SQMOD_NODISCARD std::size_t numOfColumnsHandled() const override
|
|
{
|
|
return TypeHandler<T>::size();
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Retrieve rows occupied.
|
|
*/
|
|
SQMOD_NODISCARD std::size_t numOfRowsHandled() const override
|
|
{
|
|
return 1;
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Check if binding is available.
|
|
*/
|
|
SQMOD_NODISCARD bool canBind() const override
|
|
{
|
|
return !m_Bound;
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Bind the value.
|
|
*/
|
|
void bind(std::size_t pos) override
|
|
{
|
|
poco_assert_dbg(!getBinder().isNull());
|
|
TypeHandler<T>::bind(pos, *m_Value, getBinder(), getDirection());
|
|
m_Bound = true;
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Reset the binding.
|
|
*/
|
|
void reset() override
|
|
{
|
|
m_Bound = false;
|
|
AbstractBinder::Ptr pBinder = getBinder();
|
|
poco_assert_dbg (!pBinder.isNull());
|
|
pBinder->reset();
|
|
}
|
|
|
|
private:
|
|
|
|
ValPtr m_Value;
|
|
bool m_Bound;
|
|
};
|
|
|
|
/* ------------------------------------------------------------------------------------------------
|
|
* Implementation of AbstractBinding for shared ownership binding of values.
|
|
* Because we cannot take references to script variables, we use this as a proxy.
|
|
*/
|
|
template <class T> struct ReferenceBinding<std::vector<T>> : public AbstractBinding
|
|
{
|
|
using ValType = std::vector<T>;
|
|
using ValPtr = SharedPtr<ValType>;
|
|
using Ptr = SharedPtr<Binding<ValType>>;
|
|
using Iterator = typename ValType::const_iterator;
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Base constructor.
|
|
*/
|
|
explicit ReferenceBinding(const ValPtr& val, const std::string& name = "", Direction direction = PD_IN)
|
|
: AbstractBinding(name, direction), m_Value(val), m_Begin(), m_End()
|
|
{
|
|
if (PD_IN == direction && m_Value->size() == 0)
|
|
{
|
|
throw BindingException("It is illegal to bind to an empty data collection");
|
|
}
|
|
m_Begin = m_Value->begin();
|
|
m_End = m_Value->end();
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Destructor.
|
|
*/
|
|
~ReferenceBinding() override = default;
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Retrieve columns occupied.
|
|
*/
|
|
SQMOD_NODISCARD std::size_t numOfColumnsHandled() const override
|
|
{
|
|
return TypeHandler<T>::size();
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Retrieve rows occupied.
|
|
*/
|
|
SQMOD_NODISCARD std::size_t numOfRowsHandled() const override
|
|
{
|
|
return m_Value->size();
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Check if binding is available.
|
|
*/
|
|
SQMOD_NODISCARD bool canBind() const override
|
|
{
|
|
return (m_Begin != m_End);
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Bind the value.
|
|
*/
|
|
void bind(std::size_t pos) override
|
|
{
|
|
poco_assert_dbg(!getBinder().isNull());
|
|
poco_assert_dbg(canBind());
|
|
TypeHandler<T>::bind(pos, *m_Begin, getBinder(), getDirection());
|
|
++m_Begin;
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Reset the binding.
|
|
*/
|
|
void reset() override
|
|
{
|
|
m_Begin = m_Value->begin();
|
|
m_End = m_Value->end();
|
|
}
|
|
|
|
private:
|
|
|
|
ValPtr m_Value;
|
|
Iterator m_Begin;
|
|
Iterator m_End;
|
|
};
|
|
|
|
/* ------------------------------------------------------------------------------------------------
|
|
* Implementation of AbstractExtraction for shared ownership binding of values.
|
|
* Because we cannot take references to script variables, we use this as a proxy.
|
|
*/
|
|
template <class T>
|
|
class ReferenceExtraction: public AbstractExtraction
|
|
{
|
|
public:
|
|
using ValType = T;
|
|
using Result = SharedPtr<T>;
|
|
using ValPtr = SharedPtr<ValType>;
|
|
using Type = Extraction<ValType>;
|
|
using Ptr = SharedPtr<Type>;
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Creates an Extraction object at specified position. Uses an empty object T as default value.
|
|
*/
|
|
explicit ReferenceExtraction(const Result& result, const Position& pos = Position(0))
|
|
: AbstractExtraction(Limit::LIMIT_UNLIMITED, pos.value())
|
|
, m_Result(result),m_Default(),m_Extracted(false),m_Null(false)
|
|
{
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Creates an Extraction object at specified position. Uses the provided def object as default value.
|
|
*/
|
|
ReferenceExtraction(const Result& result, const T& def, const Position& pos = Position(0))
|
|
: AbstractExtraction(Limit::LIMIT_UNLIMITED, pos.value())
|
|
, m_Result(result), m_Default(def), m_Extracted(false), m_Null(false)
|
|
{
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Creates an Extraction object at specified position. Uses the provided def object as default value.
|
|
*/
|
|
ReferenceExtraction(const Result& result, T&& def, const Position& pos = Position(0))
|
|
: AbstractExtraction(Limit::LIMIT_UNLIMITED, pos.value())
|
|
, m_Result(result), m_Default(std::move(def)), m_Extracted(false), m_Null(false)
|
|
{
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Destroys the Extraction object.
|
|
*/
|
|
~ReferenceExtraction() override = default;
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Retrieve the number of columns handled.
|
|
*/
|
|
SQMOD_NODISCARD std::size_t numOfColumnsHandled() const override
|
|
{
|
|
return TypeHandler<T>::size();
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Retrieve the number of rows handled.
|
|
*/
|
|
SQMOD_NODISCARD std::size_t numOfRowsHandled() const override
|
|
{
|
|
return m_Extracted ? 1u : 0;
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Retrieve the number of rows allowed.
|
|
*/
|
|
SQMOD_NODISCARD std::size_t numOfRowsAllowed() const override
|
|
{
|
|
return 1u;
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Check if null.
|
|
*/
|
|
SQMOD_NODISCARD bool isNull(std::size_t /*row*/) const override
|
|
{
|
|
return m_Null;
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Extract the value.
|
|
*/
|
|
std::size_t extract(std::size_t pos) override
|
|
{
|
|
if (m_Extracted)
|
|
{
|
|
throw ExtractException("value already extracted");
|
|
}
|
|
|
|
m_Extracted = true;
|
|
AbstractExtractor::Ptr pExt = getExtractor();
|
|
TypeHandler<T>::extract(pos, *m_Result, m_Default, pExt);
|
|
m_Null = isValueNull<T>(*m_Result, pExt->isNull(pos));
|
|
|
|
return 1u;
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Reset state.
|
|
*/
|
|
void reset() override
|
|
{
|
|
m_Extracted = false;
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* See if a value was extracted.
|
|
*/
|
|
SQMOD_NODISCARD bool canExtract() const override
|
|
{
|
|
return !m_Extracted;
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Create a preparation instance for this type.
|
|
*/
|
|
AbstractPreparation::Ptr createPreparation(AbstractPreparator::Ptr& pPrep, std::size_t pos) override
|
|
{
|
|
return new Preparation<T>(pPrep, pos, *m_Result);
|
|
}
|
|
|
|
private:
|
|
|
|
Result m_Result;
|
|
T m_Default;
|
|
bool m_Extracted;
|
|
bool m_Null;
|
|
};
|
|
|
|
/* ------------------------------------------------------------------------------------------------
|
|
* Implementation of AbstractExtraction for shared ownership binding of values.
|
|
* Because we cannot take references to script variables, we use this as a proxy.
|
|
*/
|
|
template <class T>
|
|
class ReferenceExtraction<std::vector<T>>: public AbstractExtraction
|
|
{
|
|
public:
|
|
using ValType = std::vector<T>;
|
|
using Result = SharedPtr<std::vector<T>>;
|
|
using ValPtr = SharedPtr<ValType>;
|
|
using Type = Extraction<ValType>;
|
|
using Ptr = SharedPtr<Type>;
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Creates an Extraction object at specified position. Uses an empty object T as default value.
|
|
*/
|
|
explicit ReferenceExtraction(const Result& result, const Position& pos = Position(0))
|
|
: AbstractExtraction(Limit::LIMIT_UNLIMITED, pos.value())
|
|
, m_Result(result), m_Default(), m_Nulls()
|
|
{
|
|
m_Result->clear();
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Creates an Extraction object at specified position. Uses the provided def object as default value.
|
|
*/
|
|
ReferenceExtraction(const Result& result, const T& def, const Position& pos = Position(0))
|
|
: AbstractExtraction(Limit::LIMIT_UNLIMITED, pos.value())
|
|
, m_Result(result), m_Default(def), m_Nulls()
|
|
{
|
|
m_Result->clear();
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Destroys the Extraction object.
|
|
*/
|
|
~ReferenceExtraction() override = default;
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Retrieve the number of columns handled.
|
|
*/
|
|
SQMOD_NODISCARD std::size_t numOfColumnsHandled() const override
|
|
{
|
|
return TypeHandler<T>::size();
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Retrieve the number of rows handled.
|
|
*/
|
|
SQMOD_NODISCARD std::size_t numOfRowsHandled() const override
|
|
{
|
|
return static_cast<std::size_t>(m_Result->size());
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Retrieve the number of rows allowed.
|
|
*/
|
|
SQMOD_NODISCARD std::size_t numOfRowsAllowed() const override
|
|
{
|
|
return getLimit();
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Check if null.
|
|
*/
|
|
SQMOD_NODISCARD bool isNull(std::size_t row) const override
|
|
{
|
|
try
|
|
{
|
|
return m_Nulls.at(row);
|
|
}
|
|
catch (std::out_of_range& ex)
|
|
{
|
|
throw RangeException(ex.what());
|
|
}
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Extract the value.
|
|
*/
|
|
std::size_t extract(std::size_t pos) override
|
|
{
|
|
AbstractExtractor::Ptr ext = getExtractor();
|
|
m_Result->push_back(m_Default);
|
|
TypeHandler<T>::extract(pos, m_Result->back(), m_Default, ext);
|
|
m_Nulls.push_back(isValueNull(m_Result->back(), ext->isNull(pos)));
|
|
return 1u;
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Create a preparation instance for this type.
|
|
*/
|
|
AbstractPreparation::Ptr createPreparation(AbstractPreparator::Ptr& pPrep, std::size_t pos) override
|
|
{
|
|
return new Preparation<T>(pPrep, pos, m_Default);
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Reset state.
|
|
*/
|
|
void reset() override
|
|
{
|
|
m_Nulls.clear();
|
|
}
|
|
|
|
protected:
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Retrieve the container with the extracted values.
|
|
*/
|
|
SQMOD_NODISCARD const std::vector<T>& result() const
|
|
{
|
|
return *m_Result;
|
|
}
|
|
|
|
private:
|
|
Result m_Result;
|
|
T m_Default;
|
|
std::deque< bool > m_Nulls;
|
|
};
|
|
|
|
// ------------------------------------------------------------------------------------------------
|
|
template <> std::size_t ReferenceExtraction< std::vector<bool> >::extract(std::size_t pos)
|
|
{
|
|
AbstractExtractor::Ptr ext = getExtractor();
|
|
bool tmp = m_Default;
|
|
TypeHandler<bool>::extract(pos, tmp, m_Default, ext);
|
|
m_Result->push_back(tmp);
|
|
m_Nulls.push_back(ext->isNull(pos));
|
|
return 1u;
|
|
}
|
|
|
|
} // Namespace:: Data
|
|
} // Namespace:: Poco
|
|
|
|
// ------------------------------------------------------------------------------------------------
|
|
namespace SqMod {
|
|
|
|
// ------------------------------------------------------------------------------------------------
|
|
using namespace Poco::Data::Keywords;
|
|
// ------------------------------------------------------------------------------------------------
|
|
using Poco::Data::Session;
|
|
using Poco::Data::Statement;
|
|
using Poco::Data::RecordSet;
|
|
using Poco::Data::SessionPool;
|
|
using Poco::Data::ReferenceBinding;
|
|
|
|
// ------------------------------------------------------------------------------------------------
|
|
struct SqDataSession;
|
|
struct SqDataStatement;
|
|
struct SqDataRecordSet;
|
|
|
|
/* ------------------------------------------------------------------------------------------------
|
|
* Utility used to transform optimal argument type to stored type.
|
|
*/
|
|
template < class T > struct SqDataBindingOpt
|
|
{
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Optimal argument type. Unchanged when not specialized.
|
|
*/
|
|
using Type = T;
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Convert the optimal type to the stored type. Does nothing special in this case.
|
|
*/
|
|
inline static Type & Get(Type & v) { return v; }
|
|
inline static const Type & Get(const Type & v) { return v; }
|
|
// --------------------------------------------------------------------------------------------
|
|
inline static void Put(T & o, Type & v) { o = v; }
|
|
inline static void Put(T & o, const Type & v) { o = v; }
|
|
};
|
|
|
|
/* ------------------------------------------------------------------------------------------------
|
|
* Specialization of SqDataBindingOpt for std::string type.
|
|
*/
|
|
template < > struct SqDataBindingOpt< String >
|
|
{
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Optimal argument type.
|
|
*/
|
|
using Type = StackStrF;
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Convert the optimal type to the stored type.
|
|
*/
|
|
inline static String Get(Type & v) { return v.ToStr(); }
|
|
inline static String Get(const Type & v) { return v.ToStr(); }
|
|
// --------------------------------------------------------------------------------------------
|
|
inline static void Put(String & o, Type & v) { o.assign(v.mPtr, v.GetSize()); }
|
|
inline static void Put(String & o, const Type & v) { o.assign(v.mPtr, v.GetSize()); }
|
|
};
|
|
|
|
/* ------------------------------------------------------------------------------------------------
|
|
* Wrapper around values that must be passed as reference. Manual lifetime management!
|
|
*/
|
|
template < class T > struct SqDataBinding
|
|
{
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Reference binding type.
|
|
*/
|
|
using Binding = ReferenceBinding< T >;
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Optimal type to receive a value of this type as function argument. Mainly for strings.
|
|
*/
|
|
using OptimalType = typename SqDataBindingOpt< T >::Type;
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Same as OptimalType but preferably with a reference qualifier to avoid copies.
|
|
*/
|
|
using OptimalArg = typename std::conditional< std::is_same< T, OptimalType >::value, T, OptimalType & >::type;
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Reference value.
|
|
*/
|
|
typename Binding::ValPtr mV{};
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Default constructor. Initializes to a default value.
|
|
*/
|
|
SqDataBinding()
|
|
: mV(new T())
|
|
{
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Explicit constructor. Initializes to a specific value.
|
|
*/
|
|
explicit SqDataBinding(OptimalArg v)
|
|
: mV(new T())
|
|
{
|
|
SqDataBindingOpt< T >::Put(*mV, v);
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Copy constructor.
|
|
*/
|
|
SqDataBinding(const SqDataBinding &) = default;
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Move constructor.
|
|
*/
|
|
SqDataBinding(SqDataBinding &&) noexcept = default;
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Destroys the Statement.
|
|
*/
|
|
~SqDataBinding() = default;
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Assignment operator.
|
|
*/
|
|
SqDataBinding & operator = (const SqDataBinding &) = default;
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Move assignment.
|
|
*/
|
|
SqDataBinding & operator = (SqDataBinding &&) noexcept = default;
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Retrieve a value from the instance.
|
|
*/
|
|
SQMOD_NODISCARD const T & Get() const { return *mV; }
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Modify a value from the instance.
|
|
*/
|
|
void Set(OptimalArg v) { SqDataBindingOpt< T >::Put(*mV, v); }
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Modify a value from the instance.
|
|
*/
|
|
SqDataBinding & SetEx(OptimalArg v) { SqDataBindingOpt< T >::Put(*mV, v); return *this; }
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Use the value to a statement.
|
|
*/
|
|
SqDataBinding & Use(SqDataStatement & stmt);
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Use the value to a statement with a specific name.
|
|
*/
|
|
SqDataBinding & UseAs(SqDataStatement & stmt, StackStrF & name);
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Bind the value to a statement.
|
|
*/
|
|
SqDataBinding & Bind(SqDataStatement & stmt);
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Bind the value to a statement with a specific name.
|
|
*/
|
|
SqDataBinding & BindAs(SqDataStatement & stmt, StackStrF & name);
|
|
};
|
|
|
|
/* ------------------------------------------------------------------------------------------------
|
|
* A Session holds a connection to a Database and creates Statement objects.
|
|
*/
|
|
struct SqDataSession : public Session
|
|
{
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Creates a new session, using the given connection (must be in "connection:///info" format).
|
|
*/
|
|
SqDataSession(StackStrF & conn, SQInteger timeout)
|
|
: Session(conn.ToStr(), ConvTo< size_t >::From(timeout))
|
|
{
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Creates a new session, using the given connector, and connection information.
|
|
*/
|
|
SqDataSession(StackStrF & conn, StackStrF & info, SQInteger timeout)
|
|
: Session(conn.ToStr(), info.ToStr(), ConvTo< size_t >::From(timeout))
|
|
{
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Creates a session by copying another one.
|
|
*/
|
|
SqDataSession(const SqDataSession &) = default;
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Creates a session by moving another one.
|
|
*/
|
|
SqDataSession(SqDataSession &&) noexcept = default;
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Creates a session by copying another one.
|
|
*/
|
|
SqDataSession(const Session & s)
|
|
: Session(s)
|
|
{
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Creates a session by moving another one.
|
|
*/
|
|
SqDataSession(Session && s) noexcept
|
|
: Session(std::move(s))
|
|
{
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Destroys the Session.
|
|
*/
|
|
~SqDataSession() = default;
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Assignment operator.
|
|
*/
|
|
SqDataSession & operator = (const SqDataSession &) = default;
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Move assignment.
|
|
*/
|
|
SqDataSession & operator = (SqDataSession &&) noexcept = default;
|
|
#ifdef SQMOD_POCO_HAS_MYSQL
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Implements string escape in MySQL.
|
|
*/
|
|
LightObj MySQLEscapeString(StackStrF & str);
|
|
#endif
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Opens the session using the supplied string.
|
|
* Can also be used with default empty string to reconnect a disconnected session.
|
|
*/
|
|
void Open(StackStrF & connect) { open(connect.ToStr()); }
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Closes the session.
|
|
*/
|
|
void Close() { close(); }
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Returns true if session is connected, false otherwise.
|
|
*/
|
|
bool IsConnected() { return isConnected(); }
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Closes the session and opens it.
|
|
*/
|
|
void Reconnect() { return reconnect(); }
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Returns true if the session is good and can be used, false otherwise.
|
|
*/
|
|
SQMOD_NODISCARD bool IsGood() { return isGood(); }
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Sets the session login timeout value.
|
|
*/
|
|
void SetLoginTimeout(SQInteger timeout) { setLoginTimeout(ConvTo< size_t >::From(timeout)); }
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Returns the session login timeout value.
|
|
*/
|
|
SQMOD_NODISCARD SQInteger GetLoginTimeout() const { return static_cast< SQInteger >(getLoginTimeout()); }
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Sets the session connection timeout value.
|
|
*/
|
|
void SetConnectionTimeout(SQInteger timeout) { setConnectionTimeout(ConvTo< size_t >::From(timeout)); }
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Returns the session connection timeout value.
|
|
*/
|
|
SQInteger GetConnectionTimeout() { return static_cast< SQInteger >(getConnectionTimeout()); }
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Starts a transaction.
|
|
*/
|
|
void Begin() { begin(); }
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Commits and ends a transaction.
|
|
*/
|
|
void Commit() { commit(); }
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Rolls back and ends a transaction.
|
|
*/
|
|
void Rollback() { rollback(); }
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Returns true if session has transaction capabilities.
|
|
*/
|
|
SQMOD_NODISCARD bool CanTransact() { return canTransact(); }
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Returns true if a transaction is in progress, false otherwise.
|
|
*/
|
|
SQMOD_NODISCARD bool IsTransaction() { return isTransaction(); }
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Sets the transaction isolation level.
|
|
*/
|
|
void SetTransactionIsolation(SQInteger ti) { setTransactionIsolation(static_cast< uint32_t >(ti)); }
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Returns the transaction isolation level.
|
|
*/
|
|
SQMOD_NODISCARD SQInteger GetTransactionIsolation() { return static_cast< SQInteger >(getTransactionIsolation()); }
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Returns true if the transaction isolation level corresponding to the supplied bitmask is supported.
|
|
*/
|
|
bool HasTransactionIsolation(SQInteger ti) { return hasTransactionIsolation(static_cast< uint32_t >(ti)); }
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Returns true if the transaction isolation level corresponds to the supplied bitmask.
|
|
*/
|
|
bool IsTransactionIsolation(SQInteger ti) { return isTransactionIsolation(static_cast< uint32_t >(ti)); }
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Returns the connector name for this session.
|
|
*/
|
|
SQMOD_NODISCARD std::string GetConnector() const { return connector(); }
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Returns the URI for this session.
|
|
*/
|
|
SQMOD_NODISCARD std::string GetURI() const { return uri(); }
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Utility function that teturns the URI formatted from supplied arguments as "connector:///info".
|
|
*/
|
|
SQMOD_NODISCARD static std::string BuildURI(StackStrF & connector, StackStrF & info)
|
|
{
|
|
return uri(connector.ToStr(), info.ToStr());
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Set the state of a feature.
|
|
*/
|
|
void SetFeature(bool state, StackStrF & name) { setFeature(name.ToStr(), state); }
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Look up the state of a feature.
|
|
*/
|
|
SQMOD_NODISCARD bool GetFeature(StackStrF & name) const { return getFeature(name.ToStr()); }
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Set the value of a property.
|
|
*/
|
|
void SetProperty(const LightObj & value, StackStrF & name);
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Look up the value of a property.
|
|
*/
|
|
SQMOD_NODISCARD LightObj GetProperty(StackStrF & name) const;
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Retrieve a statement from this session.
|
|
*/
|
|
SQMOD_NODISCARD SqDataStatement GetStatement(StackStrF & data);
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Retrieve a record-set from this session.
|
|
*/
|
|
SQMOD_NODISCARD SqDataRecordSet GetRecordSet(StackStrF & data);
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Create a statement and execute the given query immediately.
|
|
*/
|
|
SqDataSession & Execute(StackStrF & query);
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Create a statement and execute the given query whenever possible.
|
|
*/
|
|
SqDataSession & ExecuteAsync(StackStrF & query);
|
|
};
|
|
|
|
/* ------------------------------------------------------------------------------------------------
|
|
* Statement is used to execute SQL statements.
|
|
*/
|
|
struct SqDataStatementResult : public SqChainedInstances< SqDataStatementResult >
|
|
{
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Watched result.
|
|
*/
|
|
Statement::Result mRes;
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Script callback.
|
|
*/
|
|
Function mFunc;
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Reference to the statement.
|
|
*/
|
|
LightObj mStmt;
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Reference to self.
|
|
*/
|
|
LightObj mSelf;
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Base constructor.
|
|
*/
|
|
explicit SqDataStatementResult(const Statement::Result & r, LightObj && stmt)
|
|
: mRes(r), mFunc(), mStmt(std::move(stmt)), mSelf()
|
|
{
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Copy constructor.
|
|
*/
|
|
SqDataStatementResult(const SqDataStatementResult &) = delete;
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Move constructor.
|
|
*/
|
|
SqDataStatementResult(SqDataStatementResult &&) = delete;
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Destructor.
|
|
*/
|
|
~SqDataStatementResult()
|
|
{
|
|
// Forget about this instance
|
|
UnchainInstance();
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Assignment operator.
|
|
*/
|
|
SqDataStatementResult & operator = (const SqDataStatementResult &) = delete;
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Move assignment.
|
|
*/
|
|
SqDataStatementResult & operator = (SqDataStatementResult &&) = delete;
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Bind a callback and wait for completion.
|
|
*/
|
|
LightObj & Bind(Function & fn)
|
|
{
|
|
mFunc = std::move(fn);
|
|
// Reference self to prevent destruction
|
|
mSelf = LightObj(this);
|
|
// Remember this instance
|
|
ChainInstance();
|
|
// Return the statement
|
|
return mStmt;
|
|
}
|
|
};
|
|
|
|
/* ------------------------------------------------------------------------------------------------
|
|
* Statement is used to execute SQL statements.
|
|
*/
|
|
struct SqDataStatement : public Statement
|
|
{
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Creates the Statement for the given Session.
|
|
*/
|
|
explicit SqDataStatement(SqDataSession & session)
|
|
: Statement(session)
|
|
{
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Creates the Statement for the given Session and adds initial SQL code.
|
|
*/
|
|
explicit SqDataStatement(SqDataSession & session, StackStrF & data)
|
|
: Statement(session)
|
|
{
|
|
Add(data);
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Copy constructor. If the statement has been executed asynchronously and has not been
|
|
* synchronized prior to copy operation (i.e. is copied while executing), this constructor shall synchronize it.
|
|
*/
|
|
SqDataStatement(const SqDataStatement &) = default;
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Move constructor.
|
|
*/
|
|
SqDataStatement(SqDataStatement &&) noexcept = default;
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Destroys the Statement.
|
|
*/
|
|
~SqDataStatement() = default;
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Assignment operator.
|
|
*/
|
|
SqDataStatement & operator = (const SqDataStatement &) = default;
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Move assignment.
|
|
*/
|
|
SqDataStatement & operator = (SqDataStatement &&) noexcept = default;
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Concatenates data with the SQL statement string.
|
|
*/
|
|
SqDataStatement & Add(StackStrF & data)
|
|
{
|
|
Statement::operator<<(data);
|
|
// Allow chaining
|
|
return *this;
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Returns true if statement was marked for asynchronous execution.
|
|
*/
|
|
bool GetAsync()
|
|
{
|
|
return isAsync();
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Sets the asynchronous flag. This setting does not affect the statement's capability
|
|
* to be executed synchronously by directly calling Execute().
|
|
*/
|
|
SqDataStatement & SetAsync(bool async)
|
|
{
|
|
setAsync(async);
|
|
return *this;
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Returns true if the statement was initialized (i.e. not executed yet).
|
|
*/
|
|
bool Initialized()
|
|
{
|
|
return initialized();
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Returns true if the statement was paused (a range limit stopped it and there is more work to do).
|
|
*/
|
|
bool Paused()
|
|
{
|
|
return paused();
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Returns true if the statement was completely executed or false if a range limit stopped it
|
|
* and there is more work to do. When no limit is set, it will always return true after calling execute().
|
|
*/
|
|
bool Done()
|
|
{
|
|
return done();
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Returns true if statement is in a state that allows the internal storage to be modified.
|
|
*/
|
|
bool CanModifyStorage()
|
|
{
|
|
return canModifyStorage();
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Returns the internal storage type for the statement.
|
|
*/
|
|
SQInteger Storage() const
|
|
{
|
|
return static_cast< SQInteger >(storage());
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Returns the internal storage type for the statement.
|
|
*/
|
|
const std::string & GetStorage() const
|
|
{
|
|
return getStorage();
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Sets the internal storage type for the statement.
|
|
*/
|
|
void SetStorage(StackStrF & storage)
|
|
{
|
|
setStorage(storage.ToStr());
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Returns the number of columns returned for current data set.
|
|
* Default value indicates current data set (if any).
|
|
*/
|
|
SQInteger GetColumnsExtracted(int data_set) const
|
|
{
|
|
return static_cast< SQInteger >(columnsExtracted(data_set));
|
|
}
|
|
SQInteger ColumnsExtracted() const
|
|
{
|
|
return static_cast< SQInteger >(columnsExtracted());
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Returns the number of rows returned for current data set during last statement execution.
|
|
* Default value indicates current data set (if any).
|
|
*/
|
|
SQInteger GetRowsExtracted(int data_set) const
|
|
{
|
|
return static_cast< SQInteger >(rowsExtracted(data_set));
|
|
}
|
|
SQInteger RowsExtracted() const
|
|
{
|
|
return static_cast< SQInteger >(rowsExtracted());
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Returns the number of rows extracted so far for the data set.
|
|
* Default value indicates current data set (if any).
|
|
*/
|
|
SQInteger GetSubTotalRowCount(int data_set) const
|
|
{
|
|
return static_cast< SQInteger >(subTotalRowCount(data_set));
|
|
}
|
|
SQInteger SubTotalRowCount() const
|
|
{
|
|
return static_cast< SQInteger >(subTotalRowCount());
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Returns the number of extraction storage buffers associated with the current data set.
|
|
*/
|
|
SQInteger ExtractionCount() const
|
|
{
|
|
return static_cast< SQInteger >(extractionCount());
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Returns the number of data sets associated with the statement.
|
|
*/
|
|
SQInteger DataSetCount() const
|
|
{
|
|
return static_cast< SQInteger >(dataSetCount());
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Returns the index of the next data set.
|
|
*/
|
|
SQInteger NextDataSet()
|
|
{
|
|
return static_cast< SQInteger >(nextDataSet());
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Returns the index of the previous data set.
|
|
*/
|
|
SQInteger PreviousDataSet()
|
|
{
|
|
return static_cast< SQInteger >(previousDataSet());
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Returns false if the current data set index points to the last data set. Otherwise, it returns true.
|
|
*/
|
|
bool HasMoreDataSets() const
|
|
{
|
|
return hasMoreDataSets();
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Executes the statement synchronously or asynchronously.
|
|
*/
|
|
SQInteger Execute()
|
|
{
|
|
return static_cast< SQInteger >(execute(true));
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Executes the statement synchronously or asynchronously.
|
|
*/
|
|
SQInteger Execute_(bool reset)
|
|
{
|
|
return static_cast< SQInteger >(execute(reset));
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Executes the statement synchronously or asynchronously. Returns itself instead of rows.
|
|
*/
|
|
SqDataStatement & ExecuteChained()
|
|
{
|
|
execute(true);
|
|
// Allow chaining
|
|
return *this;
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Executes the statement synchronously or asynchronously. Returns itself instead of rows.
|
|
*/
|
|
SqDataStatement & ExecuteChained_(bool reset)
|
|
{
|
|
execute(reset);
|
|
// Allow chaining
|
|
return *this;
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Executes the statement asynchronously.
|
|
*/
|
|
LightObj ExecuteAsync()
|
|
{
|
|
return LightObj(SqTypeIdentity< SqDataStatementResult >{}, SqVM(), executeAsync(true), LightObj(this));
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Executes the statement asynchronously.
|
|
*/
|
|
LightObj ExecuteAsync_(bool reset)
|
|
{
|
|
return LightObj(SqTypeIdentity< SqDataStatementResult >{}, SqVM(), executeAsync(true), LightObj(this));
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Executes the statement asynchronously.
|
|
*/
|
|
SqDataStatement & ExecuteAsyncChained()
|
|
{
|
|
executeAsync(true);
|
|
return *this;
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Executes the statement asynchronously.
|
|
*/
|
|
SqDataStatement & ExecuteAsyncChained_(bool reset)
|
|
{
|
|
executeAsync(reset);
|
|
return *this;
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Waits for the execution completion for asynchronous statements or returns immediately for synchronous ones.
|
|
*/
|
|
SQInteger Wait(SQInteger milliseconds)
|
|
{
|
|
return static_cast< SQInteger >(wait(static_cast< long >(milliseconds)));
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Resets the Statement so that it can be filled with a new SQL command.
|
|
*/
|
|
SqDataStatement & Reset(SqDataSession & session)
|
|
{
|
|
reset(session);
|
|
return *this;
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Swaps the statement with another one.
|
|
*/
|
|
void Swap(SqDataStatement & stmt)
|
|
{
|
|
swap(stmt);
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Bind a reference to the statement.
|
|
*/
|
|
SqDataStatement & Use(LightObj & obj)
|
|
{
|
|
UseEx(obj, String(), Poco::Data::AbstractBinding::PD_IN);
|
|
return *this;
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Bind a named reference to the statement.
|
|
*/
|
|
SqDataStatement & UseAs(LightObj & obj, StackStrF & name)
|
|
{
|
|
UseEx(obj, String(name.mPtr, name.GetSize()), Poco::Data::AbstractBinding::PD_IN);
|
|
return *this;
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Internal function used internally to bind a reference to the statement.
|
|
*/
|
|
void UseEx(LightObj & obj, const std::string & name, Poco::Data::AbstractBinding::Direction dir);
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Internal function used internally to bind a instance reference to the statement.
|
|
*/
|
|
void UseInst_(LightObj & obj, const std::string & name, Poco::Data::AbstractBinding::Direction dir);
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Bind a reference to the statement and mark it as input (i.e alias of Use).
|
|
*/
|
|
SqDataStatement & In(LightObj & obj)
|
|
{
|
|
return Use(obj);
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Bind a named reference to the statement mark it as input (i.e alias of UseAs).
|
|
*/
|
|
SqDataStatement & InAs(LightObj & obj, StackStrF & name)
|
|
{
|
|
return UseAs(obj, name);
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Bind a reference to the statement and mark it as output.
|
|
*/
|
|
SqDataStatement & Out(LightObj & obj)
|
|
{
|
|
UseEx(obj, String(), Poco::Data::AbstractBinding::PD_OUT);
|
|
return *this;
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Bind a named reference to the statement and mark it as output.
|
|
*/
|
|
SqDataStatement & OutAs(LightObj & obj, StackStrF & name)
|
|
{
|
|
UseEx(obj, String(name.mPtr, name.GetSize()), Poco::Data::AbstractBinding::PD_OUT);
|
|
return *this;
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Bind a value to the statement.
|
|
*/
|
|
SqDataStatement & Bind(LightObj & obj)
|
|
{
|
|
BindEx(obj, String(), Poco::Data::AbstractBinding::PD_IN);
|
|
return *this;
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Bind a named value to the statement.
|
|
*/
|
|
SqDataStatement & BindAs(LightObj & obj, StackStrF & name)
|
|
{
|
|
BindEx(obj, String(name.mPtr, name.GetSize()), Poco::Data::AbstractBinding::PD_IN);
|
|
return *this;
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Internal function used internally to bind a value to the statement.
|
|
*/
|
|
void BindEx(LightObj & obj, const std::string & name, Poco::Data::AbstractBinding::Direction dir);
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Internal function used internally to bind a instance value to the statement.
|
|
*/
|
|
void BindInst_(LightObj & obj, const std::string & name, Poco::Data::AbstractBinding::Direction dir);
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Bind a value/reference to the statement and mark it as input/output.
|
|
*/
|
|
SqDataStatement & Io(LightObj & obj)
|
|
{
|
|
if (obj.GetType() == OT_INSTANCE)
|
|
{
|
|
UseInst_(obj, String(), Poco::Data::AbstractBinding::PD_IN_OUT);
|
|
}
|
|
else
|
|
{
|
|
Bind(obj);
|
|
}
|
|
return *this;
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Register a single extraction with the statement.
|
|
*/
|
|
SqDataStatement & Into(LightObj & obj);
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Register a single extraction with the statement with a default value.
|
|
*/
|
|
SqDataStatement & Into_(LightObj & obj, LightObj & def);
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Sets a limit on the maximum number of rows a select is allowed to return.
|
|
*/
|
|
SqDataStatement & Limit1(SQInteger limit) { return Limit3(limit, false, false); }
|
|
SqDataStatement & Limit2(SQInteger limit, bool hard) { return Limit3(limit, hard, false); }
|
|
SqDataStatement & Limit3(SQInteger limit, bool hard, bool lower)
|
|
{
|
|
(*this), Poco::Data::Limit(static_cast< Poco::Data::Limit::SizeT >(limit), hard, lower);
|
|
return *this;
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Sets a an extraction range for the maximum number of rows a select is allowed to return.
|
|
*/
|
|
SqDataStatement & Range(SQInteger lower, SQInteger upper) { return RangeEx(lower, upper, false); }
|
|
SqDataStatement & RangeEx(SQInteger lower, SQInteger upper, bool hard)
|
|
{
|
|
(*this), Poco::Data::Range(static_cast< Poco::Data::Limit::SizeT >(lower),
|
|
static_cast< Poco::Data::Limit::SizeT >(upper), hard);
|
|
return *this;
|
|
}
|
|
};
|
|
|
|
// ------------------------------------------------------------------------------------------------
|
|
template < class T > inline SqDataBinding< T > & SqDataBinding< T >::Use(SqDataStatement & stmt)
|
|
{
|
|
stmt.addBind(new Poco::Data::ReferenceBinding< T >(mV, String(), Poco::Data::AbstractBinding::PD_IN));
|
|
return *this;
|
|
}
|
|
|
|
// ------------------------------------------------------------------------------------------------
|
|
template < class T > inline SqDataBinding< T > & SqDataBinding< T >::UseAs(SqDataStatement & stmt, StackStrF & name)
|
|
{
|
|
stmt.addBind(new Poco::Data::ReferenceBinding< T >(mV, name.ToStr(), Poco::Data::AbstractBinding::PD_IN));
|
|
return *this;
|
|
}
|
|
|
|
// ------------------------------------------------------------------------------------------------
|
|
template < class T > inline SqDataBinding< T > & SqDataBinding< T >::Bind(SqDataStatement & stmt)
|
|
{
|
|
stmt.addBind(new Poco::Data::CopyBinding< T >(*mV, String(), Poco::Data::AbstractBinding::PD_IN));
|
|
return *this;
|
|
}
|
|
|
|
// ------------------------------------------------------------------------------------------------
|
|
template < class T > inline SqDataBinding< T > & SqDataBinding< T >::BindAs(SqDataStatement & stmt, StackStrF & name)
|
|
{
|
|
stmt.addBind(new Poco::Data::CopyBinding< T >(*mV, name.ToStr(), Poco::Data::AbstractBinding::PD_IN));
|
|
return *this;
|
|
}
|
|
|
|
/* ------------------------------------------------------------------------------------------------
|
|
* RecordSet provides access to data returned from a query.
|
|
*/
|
|
struct SqDataRecordSet : public RecordSet
|
|
{
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Creates the RecordSet for the given statement.
|
|
*/
|
|
SqDataRecordSet(SqDataStatement & stmt)
|
|
: RecordSet(stmt)
|
|
{
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Creates the RecordSet for the given query.
|
|
*/
|
|
SqDataRecordSet(SqDataSession & session, StackStrF & query)
|
|
: RecordSet(session, query.ToStr())
|
|
{
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Copy constructor. If the statement has been executed asynchronously and has not been
|
|
* synchronized prior to copy operation (i.e. is copied while executing), this constructor shall synchronize it.
|
|
*/
|
|
SqDataRecordSet(const SqDataRecordSet &) = default;
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Move constructor.
|
|
*/
|
|
SqDataRecordSet(SqDataRecordSet &&) noexcept = default;
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Destroys the RecordSet.
|
|
*/
|
|
~SqDataRecordSet() = default;
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Assignment operator.
|
|
*/
|
|
SqDataRecordSet & operator = (const SqDataRecordSet &) = default;
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Move assignment.
|
|
*/
|
|
SqDataRecordSet & operator = (SqDataRecordSet &&) noexcept = default;
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Returns the number of rows in the RecordSet. The number of rows reported is dependent on filtering.
|
|
*/
|
|
SQInteger RowCount() const
|
|
{
|
|
return static_cast< SQInteger >(rowCount());
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Returns the number of rows extracted during the last statement execution.
|
|
* The number of rows reported is independent of filtering.
|
|
*/
|
|
SQInteger ExtractedRowCount() const
|
|
{
|
|
return static_cast< SQInteger >(extractedRowCount());
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Returns the total number of rows in the RecordSet.
|
|
* The number of rows reported is independent of filtering.
|
|
*/
|
|
SQInteger GetTotalRowCount() const
|
|
{
|
|
return static_cast< SQInteger >(getTotalRowCount());
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Explicitly sets the total row count.
|
|
*/
|
|
void SetTotalRowCount(SQInteger totalRowCount)
|
|
{
|
|
setTotalRowCount(ClampL< SQInteger, size_t >(totalRowCount));
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Implicitly sets the total row count. The supplied sql must return exactly one column and one row.
|
|
* The returned value must be an unsigned integer. The value is set as the total number of rows.
|
|
*/
|
|
void SetTotalRowCountQ(StackStrF & sql)
|
|
{
|
|
setTotalRowCount(sql.ToStr());
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Returns the number of columns in the recordset.
|
|
*/
|
|
SQInteger ColumnCount() const
|
|
{
|
|
return static_cast< SQInteger >(columnCount());
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Moves the row cursor to the first row.
|
|
* Returns true if there is at least one row in the RecordSet, false otherwise.
|
|
*/
|
|
bool MoveFirst()
|
|
{
|
|
return moveFirst();
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Moves the row cursor to the next row.
|
|
* Returns true if the row is available, or false if the end of the record set has been reached and no more rows are available.
|
|
*/
|
|
bool MoveNext()
|
|
{
|
|
return moveNext();
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Moves the row cursor to the previous row.
|
|
* Returns true if the row is available, or false if there are no more rows available.
|
|
*/
|
|
bool MovePrevious()
|
|
{
|
|
return movePrevious();
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Moves the row cursor to the last row.
|
|
* Returns true if there is at least one row in the RecordSet, false otherwise.
|
|
*/
|
|
bool MoveLast()
|
|
{
|
|
return moveLast();
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Resets the RecordSet and assigns a new statement.
|
|
* Should be called after the given statement has been reset, assigned a new SQL statement, and executed.
|
|
* Does not remove the associated RowFilter or RowFormatter.
|
|
*/
|
|
void Reset(const SqDataStatement & stmt)
|
|
{
|
|
reset(stmt);
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Retrieve the value in the given column (index) of the current row if the value is not NULL.
|
|
*/
|
|
LightObj GetValueAt(SQInteger idx)
|
|
{
|
|
size_t i = ClampL< SQInteger, size_t >(idx);
|
|
return GetValueImpl(value(i), columnType(i), LightObj{});
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Retrieve the value in the given column (index) of the current row if the value is not NULL, or `fallback` otherwise.
|
|
*/
|
|
LightObj GetValueAtOr(SQInteger idx, const LightObj & fallback)
|
|
{
|
|
size_t i = ClampL< SQInteger, size_t >(idx);
|
|
return GetValueImpl(value(i), columnType(i), fallback);
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Retrieve the value in the given column (name) of the current row if the value is not NULL.
|
|
*/
|
|
LightObj GetValue(StackStrF & name)
|
|
{
|
|
String s(name.ToStr());
|
|
return GetValueImpl(value(s), columnType(s), LightObj{});
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Retrieve the value in the given column (name) of the current row if the value is not NULL, or `fallback` otherwise.
|
|
*/
|
|
LightObj GetValueOr(StackStrF & name, const LightObj & fallback)
|
|
{
|
|
String s(name.ToStr());
|
|
return GetValueImpl(value(s), columnType(s), fallback);
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Retrieve the type for the column at specified position.
|
|
*/
|
|
SQInteger ColumnTypeAt(SQInteger pos) const
|
|
{
|
|
return static_cast< SQInteger >(columnType(ClampL< SQInteger, size_t >(pos)));
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Retrieve the type for the column with specified name.
|
|
*/
|
|
SQInteger ColumnType(StackStrF & name) const
|
|
{
|
|
return static_cast< SQInteger >(columnType(name.ToStr()));
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Retrieve column name for the column at specified position.
|
|
*/
|
|
const std::string & ColumnName(SQInteger pos) const
|
|
{
|
|
return columnName(ClampL< SQInteger, size_t >(pos));
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Retrieve column maximum length for the column at specified position.
|
|
*/
|
|
SQInteger ColumnLengthAt(SQInteger pos) const
|
|
{
|
|
return static_cast< SQInteger >(columnLength(ClampL< SQInteger, size_t >(pos)));
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Retrieve column maximum length for the column with specified name.
|
|
*/
|
|
SQInteger ColumnLength(StackStrF & name) const
|
|
{
|
|
return static_cast< SQInteger >(columnLength(name.ToStr()));
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Retrieve column precision for the column at specified position.
|
|
* Valid for floating point fields only (zero for other data types).
|
|
*/
|
|
SQInteger ColumnPrecisionAt(SQInteger pos) const
|
|
{
|
|
return static_cast< SQInteger >(columnPrecision(ClampL< SQInteger, size_t >(pos)));
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Retrieve column precision for the column with specified name.
|
|
* Valid for floating point fields only (zero for other data types).
|
|
*/
|
|
SQInteger ColumnPrecision(StackStrF & name) const
|
|
{
|
|
return static_cast< SQInteger >(columnPrecision(name.ToStr()));
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Returns true if column value of the current row is null.
|
|
*/
|
|
bool IsNull(StackStrF & name) const
|
|
{
|
|
return isNull(name.ToStr());
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Returns true if recordset is filtered.
|
|
*/
|
|
bool IsFiltered() const
|
|
{
|
|
return isFiltered();
|
|
}
|
|
|
|
protected:
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Retrieve the value in the given column (index) of the current row if the value is not NULL, or `fallback` otherwise.
|
|
*/
|
|
static LightObj GetValueImpl(const Poco::Dynamic::Var & v, Poco::Data::MetaColumn::ColumnDataType t, const LightObj & fb)
|
|
{
|
|
// Is null?
|
|
if (v.isEmpty())
|
|
{
|
|
return fb; // Use fallback
|
|
}
|
|
// Identify type
|
|
switch (t)
|
|
{
|
|
case Poco::Data::MetaColumn::FDT_BOOL:
|
|
return LightObj(SqInPlace{}, SqVM(), v.convert< bool >());
|
|
case Poco::Data::MetaColumn::FDT_INT8:
|
|
case Poco::Data::MetaColumn::FDT_INT16:
|
|
case Poco::Data::MetaColumn::FDT_INT32:
|
|
#ifndef _SQ64
|
|
return LightObj(SqInPlace{}, SqVM(), v.convert< SQInteger >());
|
|
#endif
|
|
case Poco::Data::MetaColumn::FDT_INT64:
|
|
return LightObj(SqInPlace{}, SqVM(), v.convert< int64_t >());
|
|
case Poco::Data::MetaColumn::FDT_UINT8:
|
|
case Poco::Data::MetaColumn::FDT_UINT16:
|
|
case Poco::Data::MetaColumn::FDT_UINT32:
|
|
#ifndef _SQ64
|
|
return LightObj(SqInPlace{}, SqVM(), v.convert< SQUnsignedInteger >());
|
|
#endif
|
|
case Poco::Data::MetaColumn::FDT_UINT64:
|
|
return LightObj(SqInPlace{}, SqVM(), v.convert< uint64_t >());
|
|
case Poco::Data::MetaColumn::FDT_FLOAT:
|
|
return LightObj(SqInPlace{}, SqVM(), v.convert< float >());
|
|
case Poco::Data::MetaColumn::FDT_DOUBLE:
|
|
return LightObj(SqInPlace{}, SqVM(), v.convert< double >());
|
|
case Poco::Data::MetaColumn::FDT_STRING:
|
|
case Poco::Data::MetaColumn::FDT_WSTRING:
|
|
return LightObj(SqInPlace{}, SqVM(), v.convert< std::string >());
|
|
case Poco::Data::MetaColumn::FDT_BLOB:
|
|
case Poco::Data::MetaColumn::FDT_CLOB:
|
|
case Poco::Data::MetaColumn::FDT_DATE:
|
|
case Poco::Data::MetaColumn::FDT_TIME:
|
|
case Poco::Data::MetaColumn::FDT_TIMESTAMP:
|
|
STHROWF("This type of data is currently not implemented.");
|
|
default:
|
|
STHROWF("Unknown or unsupported type");
|
|
}
|
|
SQ_UNREACHABLE
|
|
// Unreachable
|
|
return LightObj();
|
|
}
|
|
};
|
|
|
|
/* ------------------------------------------------------------------------------------------------
|
|
* SessionPool implements session pooling for POCO Data.
|
|
*/
|
|
struct SqDataSessionPool : public SessionPool
|
|
{
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Creates the SessionPool for sessions with the given connector and connection string.
|
|
*/
|
|
SqDataSessionPool(StackStrF & connector, StackStrF & connection)
|
|
: SessionPool(connector.ToStr(), connection.ToStr())
|
|
{
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Creates the SessionPool for sessions with the given connector and connection string.
|
|
*/
|
|
SqDataSessionPool(StackStrF & connector, int min_ses, int max_ses, int idle_time, StackStrF & connection)
|
|
: SessionPool(connector.ToStr(), connection.ToStr(), min_ses, max_ses, idle_time)
|
|
{
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Copy constructor (disabled).
|
|
*/
|
|
SqDataSessionPool(const SqDataSessionPool &) = delete;
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Move constructor (disabled).
|
|
*/
|
|
SqDataSessionPool(SqDataSessionPool &&) noexcept = delete;
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Destroys the SessionPool.
|
|
*/
|
|
~SqDataSessionPool() = default;
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Assignment operator (disabled).
|
|
*/
|
|
SqDataSessionPool & operator = (const SqDataSessionPool &) = delete;
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Move assignment (disabled).
|
|
*/
|
|
SqDataSessionPool & operator = (SqDataSessionPool &&) noexcept = delete;
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Retrieve a Session.
|
|
*/
|
|
LightObj Get()
|
|
{
|
|
return LightObj(SqTypeIdentity< SqDataSession >{}, SqVM(), get());
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Retrieve a Session with requested property set.
|
|
*/
|
|
LightObj GetWithProperty(const LightObj & value, StackStrF & name)
|
|
{
|
|
switch (value.GetType())
|
|
{
|
|
case OT_NULL: {
|
|
return LightObj(SqTypeIdentity< SqDataSession >{}, SqVM(), get(name.ToStr(), nullptr));
|
|
} break;
|
|
case OT_INTEGER: {
|
|
return LightObj(SqTypeIdentity< SqDataSession >{}, SqVM(), get(name.ToStr(), value.Cast< SQInteger >()));
|
|
} break;
|
|
case OT_FLOAT: {
|
|
return LightObj(SqTypeIdentity< SqDataSession >{}, SqVM(), get(name.ToStr(), value.Cast< SQFloat >()));
|
|
} break;
|
|
case OT_BOOL: {
|
|
return LightObj(SqTypeIdentity< SqDataSession >{}, SqVM(), get(name.ToStr(), value.Cast< bool >()));
|
|
} break;
|
|
case OT_STRING: {
|
|
return LightObj(SqTypeIdentity< SqDataSession >{}, SqVM(), get(name.ToStr(), value.Cast< std::string >()));
|
|
} break;
|
|
default: STHROWF("Unsupported property value type");
|
|
}
|
|
// Should never get here
|
|
SQ_UNREACHABLE
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Retrieve a Session with requested feature set.
|
|
*/
|
|
LightObj GetWithFeature(bool value, StackStrF & name)
|
|
{
|
|
return LightObj(SqTypeIdentity< SqDataSession >{}, SqVM(), get(name.ToStr(), value));
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Returns the maximum number of sessions the SessionPool will manage.
|
|
*/
|
|
SQInteger GetCapacity() const
|
|
{
|
|
return capacity();
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Returns the number of sessions currently in use.
|
|
*/
|
|
SQInteger GetUsed() const
|
|
{
|
|
return used();
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Returns the number of idle sessions.
|
|
*/
|
|
SQInteger GetIdle() const
|
|
{
|
|
return idle();
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Returns the number of not connected active sessions.
|
|
*/
|
|
SQInteger GetDead()
|
|
{
|
|
return dead();
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Returns the number of allocated sessions.
|
|
*/
|
|
SQInteger GetAllocated() const
|
|
{
|
|
return allocated();
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Returns the number of available (idle + remaining capacity) sessions.
|
|
*/
|
|
SQInteger GetAvailable() const
|
|
{
|
|
return available();
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Returns the name for this pool.
|
|
*/
|
|
String GetName() const
|
|
{
|
|
return name();
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Returns the name formatted from supplied arguments as "connector:///connection".
|
|
*/
|
|
static String GetName_(StackStrF & connector, StackStrF & connection)
|
|
{
|
|
return name(connector.ToStr(), connection.ToStr());
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Sets feature for all the sessions.
|
|
*/
|
|
SqDataSessionPool & SetFeature(StackStrF & name, bool state)
|
|
{
|
|
setFeature(name.ToStr(), state);
|
|
return *this;
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Returns the requested feature.
|
|
*/
|
|
bool GetFeature(StackStrF & name)
|
|
{
|
|
return getFeature(name.ToStr());
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Sets property for all sessions.
|
|
*/
|
|
SqDataSessionPool & SetProperty(const LightObj & value, StackStrF & name)
|
|
{
|
|
switch (value.GetType())
|
|
{
|
|
case OT_NULL: {
|
|
setProperty(name.ToStr(), Poco::Any(nullptr));
|
|
} break;
|
|
case OT_INTEGER: {
|
|
setProperty(name.ToStr(), Poco::Any(value.Cast< SQInteger >()));
|
|
} break;
|
|
case OT_FLOAT: {
|
|
setProperty(name.ToStr(), Poco::Any(value.Cast< SQFloat >()));
|
|
} break;
|
|
case OT_BOOL: {
|
|
setProperty(name.ToStr(), Poco::Any(value.Cast<bool>()));
|
|
} break;
|
|
case OT_STRING: {
|
|
setProperty(name.ToStr(), Poco::Any(value.Cast<std::string>()));
|
|
} break;
|
|
default: STHROWF("Unsupported property value type");
|
|
}
|
|
return *this;
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Returns the requested property.
|
|
*/
|
|
LightObj GetProperty(StackStrF & name);
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Shuts down the session pool.
|
|
*/
|
|
void Shutdown()
|
|
{
|
|
shutdown();
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Returns true if session pool is active (not shut down).
|
|
*/
|
|
bool IsActive() const
|
|
{
|
|
return isActive();
|
|
}
|
|
};
|
|
|
|
} // Namespace:: SqMod
|