6#if defined(_WIN32) || defined(_WIN64)
11#include "SqlConnection.hpp"
12#include "SqlDataBinder.hpp"
13#include "SqlQuery.hpp"
14#include "SqlServerType.hpp"
21#include <source_location>
31template <
typename QueryObject>
33 { queryObject.ToSql() } -> std::convertible_to<std::string>;
37class SqlVariantRowCursor;
67 [[nodiscard]] LIGHTWEIGHT_API
bool IsAlive() const noexcept;
69 [[nodiscard]] LIGHTWEIGHT_API
bool IsPrepared() const noexcept;
72 [[nodiscard]] LIGHTWEIGHT_API
SqlConnection& Connection() noexcept;
75 [[nodiscard]] LIGHTWEIGHT_API
SqlConnection const& Connection() const noexcept;
78 [[nodiscard]] LIGHTWEIGHT_API
SqlErrorInfo LastError() const;
85 std::string_view
const& tableAlias)
const;
88 [[nodiscard]] LIGHTWEIGHT_API SQLHSTMT NativeHandle() const noexcept;
94 LIGHTWEIGHT_API
void Prepare(std::string_view query) &;
96 LIGHTWEIGHT_API
SqlStatement Prepare(std::string_view query) &&;
106 [[nodiscard]] std::
string const& PreparedQuery() const noexcept;
108 template <SqlInputParameterBinder Arg>
109 void BindInputParameter(SQLSMALLINT columnIndex, Arg const& arg);
111 template <SqlInputParameterBinder Arg, typename ColumnName>
112 void BindInputParameter(SQLSMALLINT columnIndex, Arg const& arg, ColumnName&& columnNameHint);
117 template <SqlOutputColumnBinder... Args>
118 void BindOutputColumns(Args*... args);
126 template <typename... Records>
127 requires(((std::is_class_v<Records> && std::is_aggregate_v<Records>) && ...))
128 void BindOutputColumnsToRecord(Records*... records);
130 template <SqlOutputColumnBinder T>
131 void BindOutputColumn(SQLUSMALLINT columnIndex, T* arg);
134 template <SqlInputParameterBinder... Args>
135 void Execute(Args const&... args);
138 LIGHTWEIGHT_API
void ExecuteWithVariants(std::vector<
SqlVariant> const& args);
148 template <SqlInputParameterBatchBinder FirstColumnBatch, std::ranges::contiguous_range... MoreColumnBatches>
149 void ExecuteBatchNative(FirstColumnBatch const& firstColumnBatch, MoreColumnBatches const&... moreColumnBatches);
159 template <SqlInputParameterBatchBinder FirstColumnBatch, std::ranges::range... MoreColumnBatches>
160 void ExecuteBatchSoft(FirstColumnBatch const& firstColumnBatch, MoreColumnBatches const&... moreColumnBatches);
167 template <SqlInputParameterBatchBinder FirstColumnBatch, std::ranges::range... MoreColumnBatches>
168 void ExecuteBatch(FirstColumnBatch const& firstColumnBatch, MoreColumnBatches const&... moreColumnBatches);
171 LIGHTWEIGHT_API
void ExecuteDirect(std::string_view const& query,
172 std::source_location location = std::source_location::current());
176 std::source_location location = std::source_location::current());
179 template <typename Callable>
181 void MigrateDirect(Callable const& callable, std::source_location location = std::source_location::current());
185 template <typename T>
187 [[nodiscard]] std::optional<T> ExecuteDirectScalar(const std::string_view& query,
188 std::source_location location = std::source_location::current());
190 template <typename T>
192 [[nodiscard]] T ExecuteDirectScalar(const std::string_view& query,
193 std::source_location location = std::source_location::current());
197 template <typename T>
199 [[nodiscard]] std::optional<T> ExecuteDirectScalar(
SqlQueryObject auto const& query,
200 std::source_location location = std::source_location::current());
202 template <typename T>
204 [[nodiscard]] T ExecuteDirectScalar(
SqlQueryObject auto const& query,
205 std::source_location location = std::source_location::current());
208 [[nodiscard]] LIGHTWEIGHT_API
size_t NumRowsAffected() const;
211 [[nodiscard]] LIGHTWEIGHT_API
size_t NumColumnsAffected() const;
214 [[nodiscard]] LIGHTWEIGHT_API
size_t LastInsertId(std::string_view tableName);
222 [[nodiscard]] LIGHTWEIGHT_API
bool FetchRow();
224 [[nodiscard]] LIGHTWEIGHT_API std::expected<
bool,
SqlErrorInfo> TryFetchRow(
225 std::source_location location = std::source_location::current()) noexcept;
230 void CloseCursor() noexcept;
236 SqlVariantRowCursor GetVariantRowCursor() noexcept;
241 template <SqlGetColumnNativeType T>
242 [[nodiscard]]
bool GetColumn(SQLUSMALLINT column, T* result) const;
245 template <SqlGetColumnNativeType T>
246 [[nodiscard]] T GetColumn(SQLUSMALLINT column) const;
251 template <SqlGetColumnNativeType T>
252 [[nodiscard]] std::optional<T> GetNullableColumn(SQLUSMALLINT column) const;
255 LIGHTWEIGHT_API
void RequireSuccess(SQLRETURN error,
256 std::source_location sourceLocation = std::source_location::current()) const;
257 LIGHTWEIGHT_API
void PlanPostExecuteCallback(std::function<
void()>&& cb) override;
258 LIGHTWEIGHT_API
void PlanPostProcessOutputColumn(std::function<
void()>&& cb) override;
259 [[nodiscard]] LIGHTWEIGHT_API SqlServerType ServerType() const noexcept override;
260 LIGHTWEIGHT_API
void ProcessPostExecuteCallbacks();
262 LIGHTWEIGHT_API
void RequireIndicators();
263 LIGHTWEIGHT_API SQLLEN* GetIndicatorForColumn(SQLUSMALLINT column) noexcept;
267 std::unique_ptr<Data,
void (*)(Data*)> m_data;
270 std::string m_preparedQuery;
271 SQLSMALLINT m_expectedParameterCount {};
288 m_stmt { other.m_stmt }
290 other.m_stmt =
nullptr;
297 m_stmt = other.m_stmt;
298 other.m_stmt =
nullptr;
306 SQLCloseCursor(m_stmt->NativeHandle());
312 return m_stmt->NumRowsAffected();
318 return m_stmt->NumColumnsAffected();
324 template <SqlOutputColumnBinder... Args>
327 m_stmt->BindOutputColumns(args...);
330 template <SqlOutputColumnBinder T>
331 LIGHTWEIGHT_FORCE_INLINE
void BindOutputColumn(SQLUSMALLINT columnIndex, T* arg)
333 m_stmt->BindOutputColumn(columnIndex, arg);
337 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE
bool FetchRow()
339 return m_stmt->FetchRow();
345 template <SqlGetColumnNativeType T>
346 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE
bool GetColumn(SQLUSMALLINT column, T* result)
const
348 return m_stmt->GetColumn<T>(column, result);
352 template <SqlGetColumnNativeType T>
353 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE T
GetColumn(SQLUSMALLINT column)
const
355 return m_stmt->GetColumn<T>(column);
361 template <SqlGetColumnNativeType T>
362 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE std::optional<T>
GetNullableColumn(SQLUSMALLINT column)
const
364 return m_stmt->GetNullableColumn<T>(column);
371struct [[nodiscard]] SqlSentinelIterator
375class [[nodiscard]] SqlVariantRowIterator
378 explicit SqlVariantRowIterator(SqlSentinelIterator )
noexcept:
384 _numResultColumns {
static_cast<SQLUSMALLINT
>(cursor.NumColumnsAffected()) },
387 _row.reserve(_numResultColumns);
391 SqlVariantRow& operator*() noexcept
396 SqlVariantRow
const& operator*() const noexcept
401 SqlVariantRowIterator& operator++() noexcept
407 for (
auto const i: std::views::iota(SQLUSMALLINT(1), SQLUSMALLINT(_numResultColumns + 1)))
408 _row.emplace_back(_cursor->GetColumn<
SqlVariant>(i));
413 bool operator!=(SqlSentinelIterator )
const noexcept
418 bool operator!=(SqlVariantRowIterator
const& )
const noexcept
425 SQLUSMALLINT _numResultColumns = 0;
430class [[nodiscard]] SqlVariantRowCursor
434 _resultCursor { std::move(cursor) }
438 SqlVariantRowIterator begin() noexcept
440 return SqlVariantRowIterator { _resultCursor };
443 static SqlSentinelIterator end() noexcept
445 return SqlSentinelIterator {};
476 _connection { &conn }
483 using difference_type = bool;
484 using value_type = T;
486 iterator& operator++()
488 _is_end = !_stmt->FetchRow();
494 _stmt->CloseCursor();
499 LIGHTWEIGHT_FORCE_INLINE value_type operator*()
noexcept
503 Reflection::EnumerateMembers(res, [
this]<
size_t I>(
auto&& value) {
504 auto tmp = _stmt->GetColumn<
typename Reflection::MemberTypeOf<I, value_type>::ValueType>(I + 1);
511 LIGHTWEIGHT_FORCE_INLINE
constexpr bool operator!=(iterator
const& other)
const noexcept
513 return _is_end != other._is_end;
516 constexpr iterator(std::default_sentinel_t )
noexcept:
522 _stmt { std::make_unique<SqlStatement>(conn) }
526 LIGHTWEIGHT_FORCE_INLINE
SqlStatement& Statement()
noexcept
532 bool _is_end =
false;
533 std::unique_ptr<SqlStatement> _stmt;
538 auto it = iterator { *_connection };
539 auto& stmt = it.Statement();
540 stmt.
Prepare(it.Statement().Query(RecordTableName<T>).Select().template Fields<T>().All());
546 iterator end()
noexcept
548 return iterator { std::default_sentinel };
556inline LIGHTWEIGHT_FORCE_INLINE
bool SqlStatement::IsAlive() const noexcept
558 return m_connection && m_connection->
IsAlive() && m_hStmt !=
nullptr;
561inline LIGHTWEIGHT_FORCE_INLINE
bool SqlStatement::IsPrepared() const noexcept
563 return !m_preparedQuery.empty();
568 return *m_connection;
573 return *m_connection;
588 Prepare(queryObject.ToSql());
593 return Prepare(queryObject.ToSql());
596inline LIGHTWEIGHT_FORCE_INLINE std::string
const& SqlStatement::PreparedQuery() const noexcept
598 return m_preparedQuery;
601template <SqlOutputColumnBinder... Args>
607 ((++i, RequireSuccess(SqlDataBinder<Args>::OutputColumn(m_hStmt, i, args, GetIndicatorForColumn(i), *
this))), ...);
610template <
typename... Records>
611 requires(((std::is_class_v<Records> && std::is_aggregate_v<Records>) && ...))
617 ((Reflection::EnumerateMembers(*records,
618 [
this, &i]<
size_t I,
typename FieldType>(FieldType& value) {
620 RequireSuccess(SqlDataBinder<FieldType>::OutputColumn(
621 m_hStmt, i, &value, GetIndicatorForColumn(i), *
this));
626template <SqlOutputColumnBinder T>
627inline LIGHTWEIGHT_FORCE_INLINE
void SqlStatement::BindOutputColumn(SQLUSMALLINT columnIndex, T* arg)
632 SqlDataBinder<T>::OutputColumn(m_hStmt, columnIndex, arg, GetIndicatorForColumn(columnIndex), *
this));
635template <SqlInputParameterBinder Arg>
636inline LIGHTWEIGHT_FORCE_INLINE
void SqlStatement::BindInputParameter(SQLSMALLINT columnIndex, Arg
const& arg)
639 m_expectedParameterCount = (std::numeric_limits<
decltype(m_expectedParameterCount)>::max)();
640 RequireSuccess(SqlDataBinder<Arg>::InputParameter(m_hStmt, columnIndex, arg, *
this));
643template <SqlInputParameterBinder Arg,
typename ColumnName>
644inline LIGHTWEIGHT_FORCE_INLINE
void SqlStatement::BindInputParameter(SQLSMALLINT columnIndex,
646 ColumnName&& columnNameHint)
649 BindInputParameter(columnIndex, arg);
652template <SqlInputParameterBinder... Args>
661 if (!(m_expectedParameterCount == (std::numeric_limits<
decltype(m_expectedParameterCount)>::max)()
662 &&
sizeof...(args) == 0)
663 && !(m_expectedParameterCount ==
sizeof...(args)))
664 throw std::invalid_argument {
"Invalid argument count" };
669 RequireSuccess(SqlDataBinder<Args>::InputParameter(m_hStmt, i, args, *
this))),
672 auto const result = SQLExecute(m_hStmt);
674 if (result != SQL_NO_DATA && result != SQL_SUCCESS && result != SQL_SUCCESS_WITH_INFO)
677 ProcessPostExecuteCallbacks();
682concept SqlNativeContiguousValueConcept =
683 std::same_as<T, bool>
684 || std::same_as<T, char>
685 || std::same_as<T, unsigned char>
686 || std::same_as<T, wchar_t>
687 || std::same_as<T, std::int16_t>
688 || std::same_as<T, std::uint16_t>
689 || std::same_as<T, std::int32_t>
690 || std::same_as<T, std::uint32_t>
691 || std::same_as<T, std::int64_t>
692 || std::same_as<T, std::uint64_t>
693 || std::same_as<T, float>
694 || std::same_as<T, double>
695 || std::same_as<T, SqlDate>
696 || std::same_as<T, SqlTime>
697 || std::same_as<T, SqlDateTime>
698 || std::same_as<T, SqlFixedString<T::Capacity, typename T::value_type, T::PostRetrieveOperation>>;
700template <
typename FirstColumnBatch,
typename... MoreColumnBatches>
701concept SqlNativeBatchable =
702 std::ranges::contiguous_range<FirstColumnBatch>
703 && (std::ranges::contiguous_range<MoreColumnBatches> && ...)
704 && SqlNativeContiguousValueConcept<std::ranges::range_value_t<FirstColumnBatch>>
705 && (SqlNativeContiguousValueConcept<std::ranges::range_value_t<MoreColumnBatches>> && ...);
709template <SqlInputParameterBatchBinder FirstColumnBatch, std::ranges::contiguous_range... MoreColumnBatches>
711 MoreColumnBatches
const&... moreColumnBatches)
713 static_assert(SqlNativeBatchable<FirstColumnBatch, MoreColumnBatches...>,
714 "Must be a supported native contiguous element type.");
716 if (m_expectedParameterCount != 1 +
sizeof...(moreColumnBatches))
717 throw std::invalid_argument {
"Invalid number of columns" };
719 const auto rowCount = std::ranges::size(firstColumnBatch);
720 if (!((std::size(moreColumnBatches) == rowCount) && ...))
721 throw std::invalid_argument {
"Uneven number of rows" };
727 RequireSuccess(SQLSetStmtAttr(m_hStmt, SQL_ATTR_PARAMSET_SIZE, (SQLPOINTER) rowCount, 0));
728 RequireSuccess(SQLSetStmtAttr(m_hStmt, SQL_ATTR_PARAM_BIND_OFFSET_PTR, &rowStart, 0));
729 RequireSuccess(SQLSetStmtAttr(m_hStmt, SQL_ATTR_PARAM_BIND_TYPE, SQL_PARAM_BIND_BY_COLUMN, 0));
730 RequireSuccess(SQLSetStmtAttr(m_hStmt, SQL_ATTR_PARAM_OPERATION_PTR, SQL_PARAM_PROCEED, 0));
731 RequireSuccess(SqlDataBinder<std::remove_cvref_t<
decltype(*std::ranges::data(firstColumnBatch))>>::
732 InputParameter(m_hStmt, 1, *std::ranges::data(firstColumnBatch), *
this));
733 SQLUSMALLINT column = 1;
734 (RequireSuccess(SqlDataBinder<std::remove_cvref_t<
decltype(*std::ranges::data(moreColumnBatches))>>::
735 InputParameter(m_hStmt, ++column, *std::ranges::data(moreColumnBatches), *
this)), ...);
736 RequireSuccess(SQLExecute(m_hStmt));
737 ProcessPostExecuteCallbacks();
741template <SqlInputParameterBatchBinder FirstColumnBatch, std::ranges::range... MoreColumnBatches>
743 MoreColumnBatches
const&... moreColumnBatches)
747 if constexpr (SqlNativeBatchable<FirstColumnBatch, MoreColumnBatches...>)
753template <SqlInputParameterBatchBinder FirstColumnBatch, std::ranges::range... MoreColumnBatches>
755 MoreColumnBatches
const&... moreColumnBatches)
757 if (m_expectedParameterCount != 1 +
sizeof...(moreColumnBatches))
758 throw std::invalid_argument {
"Invalid number of columns" };
760 const auto rowCount = std::ranges::size(firstColumnBatch);
761 if (!((std::size(moreColumnBatches) == rowCount) && ...))
762 throw std::invalid_argument {
"Uneven number of rows" };
764 for (
auto const rowIndex: std::views::iota(
size_t { 0 }, rowCount))
767 [&]<SqlInputParameterBinder... ColumnValues>(ColumnValues
const&... columnsInRow) {
768 SQLUSMALLINT column = 0;
769 ((++column, SqlDataBinder<ColumnValues>::InputParameter(m_hStmt, column, columnsInRow, *this)), ...);
770 RequireSuccess(SQLExecute(m_hStmt));
771 ProcessPostExecuteCallbacks();
773 std::make_tuple(std::ref(*std::ranges::next(std::ranges::begin(firstColumnBatch), rowIndex)),
774 std::ref(*std::ranges::next(std::ranges::begin(moreColumnBatches), rowIndex))...));
778template <SqlGetColumnNativeType T>
782 RequireSuccess(SqlDataBinder<T>::GetColumn(m_hStmt, column, result, &indicator, *
this));
783 return indicator != SQL_NULL_DATA;
790concept SqlNullableType = (std::same_as<T, SqlVariant> || IsSpecializationOf<std::optional, T>);
794template <SqlGetColumnNativeType T>
799 RequireSuccess(SqlDataBinder<T>::GetColumn(m_hStmt, column, &result, &indicator, *
this));
800 if constexpr (!detail::SqlNullableType<T>)
801 if (indicator == SQL_NULL_DATA)
802 throw std::runtime_error {
"Column value is NULL" };
806template <SqlGetColumnNativeType T>
811 RequireSuccess(SqlDataBinder<T>::GetColumn(m_hStmt, column, &result, &indicator, *
this));
812 if (indicator == SQL_NULL_DATA)
814 return { std::move(result) };
818 std::source_location location)
823template <
typename Callable>
824 requires std::invocable<Callable, SqlMigrationQueryBuilder&>
829 auto const queries = migration.GetPlan().ToSql();
830 for (
auto const& query: queries)
835 requires(!std::same_as<T, SqlVariant>)
838 auto const _ = detail::Finally([
this] { CloseCursor(); });
839 ExecuteDirect(query, location);
840 RequireSuccess(FetchRow());
841 return GetNullableColumn<T>(1);
845 requires(std::same_as<T, SqlVariant>)
848 auto const _ = detail::Finally([
this] { CloseCursor(); });
849 ExecuteDirect(query, location);
850 RequireSuccess(FetchRow());
851 if (
auto result = GetNullableColumn<T>(1); result.has_value())
857 requires(!std::same_as<T, SqlVariant>)
859 std::source_location location)
861 return ExecuteDirectScalar<T>(query.ToSql(), location);
865 requires(std::same_as<T, SqlVariant>)
868 return ExecuteDirectScalar<T>(query.ToSql(), location);
874 SQLFreeStmt(m_hStmt, SQL_CLOSE);
Represents a connection to a SQL database.
bool IsAlive() const noexcept
Tests if the connection is still active.
SqlQueryFormatter const & QueryFormatter() const noexcept
Retrieves a query formatter suitable for the SQL server being connected.
virtual void OnExecute(std::string_view const &query)=0
Invoked when a prepared query is executed.
void OnBindInputParameter(std::string_view const &name, T &&value)
Invoked when an input parameter is bound.
virtual void OnFetchEnd()=0
Invoked when fetching is done.
static SqlLogger & GetLogger()
Retrieves the currently configured logger.
Query builder for building SQL migration queries.
API Entry point for building SQL queries.
API for reading an SQL query result set.
LIGHTWEIGHT_FORCE_INLINE T GetColumn(SQLUSMALLINT column) const
Retrieves the value of the column at the given index for the currently selected row.
LIGHTWEIGHT_FORCE_INLINE void BindOutputColumns(Args *... args)
LIGHTWEIGHT_FORCE_INLINE std::optional< T > GetNullableColumn(SQLUSMALLINT column) const
LIGHTWEIGHT_FORCE_INLINE bool FetchRow()
Fetches the next row of the result set.
LIGHTWEIGHT_FORCE_INLINE size_t NumColumnsAffected() const
Retrieves the number of columns affected by the last query.
LIGHTWEIGHT_FORCE_INLINE bool GetColumn(SQLUSMALLINT column, T *result) const
LIGHTWEIGHT_FORCE_INLINE size_t NumRowsAffected() const
Retrieves the number of rows affected by the last query.
SQL query result row iterator.
High level API for (prepared) raw SQL statements.
void ExecuteBatch(FirstColumnBatch const &firstColumnBatch, MoreColumnBatches const &... moreColumnBatches)
LIGHTWEIGHT_API SqlQueryBuilder QueryAs(std::string_view const &table, std::string_view const &tableAlias) const
Creates a new query builder for the given table with an alias, compatible with the SQL server being c...
void ExecuteBatchNative(FirstColumnBatch const &firstColumnBatch, MoreColumnBatches const &... moreColumnBatches)
LIGHTWEIGHT_API SqlConnection & Connection() noexcept
Retrieves the connection associated with this statement.
void BindOutputColumnsToRecord(Records *... records)
LIGHTWEIGHT_API SQLHSTMT NativeHandle() const noexcept
Retrieves the native handle of the statement.
SqlVariantRowCursor GetVariantRowCursor() noexcept
Retrieves the variant row cursor for reading an SQL query result of unknown column types and column c...
SqlResultCursor GetResultCursor() noexcept
Retrieves the result cursor for reading an SQL query result.
void Execute(Args const &... args)
Binds the given arguments to the prepared statement and executes it.
void CloseCursor() noexcept
LIGHTWEIGHT_API SqlErrorInfo LastError() const
Retrieves the last error information with respect to this SQL statement handle.
void ExecuteBatchSoft(FirstColumnBatch const &firstColumnBatch, MoreColumnBatches const &... moreColumnBatches)
LIGHTWEIGHT_API void ExecuteDirect(std::string_view const &query, std::source_location location=std::source_location::current())
Executes the given query directly.
void MigrateDirect(Callable const &callable, std::source_location location=std::source_location::current())
Executes an SQL migration query, as created b the callback.
std::optional< T > GetNullableColumn(SQLUSMALLINT column) const
LIGHTWEIGHT_API bool FetchRow()
std::optional< T > ExecuteDirectScalar(const std::string_view &query, std::source_location location=std::source_location::current())
void BindOutputColumns(Args *... args)
LIGHTWEIGHT_API void Prepare(std::string_view query) &
bool GetColumn(SQLUSMALLINT column, T *result) const
LIGHTWEIGHT_API SqlStatement()
Construct a new SqlStatement object, using a new connection, and connect to the default database.
Represents an SQL query object, that provides a ToSql() method.
constexpr auto SqlNullValue
Represents an ODBC SQL error.
static SqlErrorInfo fromStatementHandle(SQLHSTMT hStmt)
Constructs an ODBC error info object from the given ODBC statement handle.
Represents a value that can be any of the supported SQL data types.