Lightweight 0.1.0
Loading...
Searching...
No Matches
SqlBinary.hpp
1#pragma once
2
3#include "../SqlColumnTypeDefinitions.hpp"
4#include "BasicStringBinder.hpp"
5#include "Core.hpp"
6
7#include <cassert>
8#include <cstdint>
9#include <vector>
10
11/// @brief Represents a binary data type.
12///
13/// This class is a thin wrapper around std::vector<uint8_t> to represent binary data types efficiently.
14///
15/// @ingroup DataTypes
16class SqlBinary final: public std::vector<uint8_t>
17{
18 public:
19 using std::vector<uint8_t>::vector;
20
21 constexpr auto operator<=>(SqlBinary const&) const noexcept = default;
22
23 private:
24 friend struct SqlDataBinder<SqlBinary>;
25
26 mutable SQLLEN _indicator = 0;
27};
28
29template <>
30struct SqlDataBinder<SqlBinary>
31{
32 static constexpr auto ColumnType = SqlColumnTypeDefinitions::Binary { 255 };
33
34 static LIGHTWEIGHT_FORCE_INLINE SQLRETURN InputParameter(SQLHSTMT stmt,
35 SQLUSMALLINT column,
36 SqlBinary const& value,
37 SqlDataBinderCallback& /*cb*/) noexcept
38 {
39 value._indicator = static_cast<SQLLEN>(value.size());
40 return SQLBindParameter(stmt,
41 column,
42 SQL_PARAM_INPUT,
43 SQL_C_BINARY,
44 SQL_LONGVARBINARY,
45 value.size(),
46 0,
47 (SQLPOINTER) value.data(),
48 0,
49 &value._indicator);
50 }
51
52 static LIGHTWEIGHT_FORCE_INLINE SQLRETURN OutputColumn(
53 SQLHSTMT stmt, SQLUSMALLINT column, SqlBinary* result, SQLLEN* indicator, SqlDataBinderCallback& cb) noexcept
54 {
55 if (result->empty())
56 result->resize(255);
57
58 cb.PlanPostProcessOutputColumn([stmt, column, result, indicator]() {
59 if (*indicator == SQL_NULL_DATA)
60 // The data is NULL.
61 result->clear();
62 else if (*indicator == SQL_NO_TOTAL)
63 // We have a truncation and the server does not know how much data is left.
64 result->resize(result->size() - 1);
65 else if (*indicator <= static_cast<SQLLEN>(result->size()))
66 result->resize(*indicator);
67 else
68 {
69 // We have a truncation and the server knows how much data is left.
70 // Extend the buffer and fetch the rest via SQLGetData.
71
72 auto const totalCharsRequired = *indicator;
73 result->resize(totalCharsRequired + 1);
74 auto const sqlResult = SQLGetData(
75 stmt, column, SQL_C_BINARY, (SQLPOINTER) result->data(), totalCharsRequired + 1, indicator);
76 (void) sqlResult;
77 assert(SQL_SUCCEEDED(sqlResult));
78 assert(*indicator == totalCharsRequired);
79 result->resize(totalCharsRequired);
80 }
81 });
82
83 return SQLBindCol(stmt, column, SQL_C_BINARY, (SQLPOINTER) result->data(), 255, indicator);
84 }
85
86 static LIGHTWEIGHT_FORCE_INLINE SQLRETURN GetColumn(SQLHSTMT stmt,
87 SQLUSMALLINT column,
88 SqlBinary* result,
89 SQLLEN* indicator,
90 SqlDataBinderCallback const& /*cb*/) noexcept
91 {
92 if (result->empty())
93 result->resize(255);
94
95 return detail::GetArrayData<SQL_C_BINARY>(stmt, column, result, indicator);
96 }
97
98 static LIGHTWEIGHT_FORCE_INLINE std::string Inspect(SqlBinary const& value)
99 {
100 return std::format("SqlBinary(size={})", value.size());
101 }
102};
Represents a binary data type.
Definition SqlBinary.hpp:17