Lightweight 0.1.0
Loading...
Searching...
No Matches
SqlDateTime.hpp
1// SPDX-License-Identifier: Apache-2.0
2
3#pragma once
4
5#include "../SqlColumnTypeDefinitions.hpp"
6#include "Core.hpp"
7
8#include <chrono>
9#include <format>
10
11/// Represents a date and time to efficiently write to or read from a database.
12///
13/// @see SqlDate, SqlTime
14/// @ingroup DataTypes
16{
17 using native_type = std::chrono::system_clock::time_point;
18 using duration_type = std::chrono::system_clock::duration;
19
20 /// Returns the current date and time.
21 [[nodiscard]] static LIGHTWEIGHT_FORCE_INLINE SqlDateTime Now() noexcept
22 {
23 return SqlDateTime { std::chrono::system_clock::now() };
24 }
25
26 constexpr SqlDateTime() noexcept = default;
27 constexpr SqlDateTime(SqlDateTime&&) noexcept = default;
28 constexpr SqlDateTime& operator=(SqlDateTime&&) noexcept = default;
29 constexpr SqlDateTime(SqlDateTime const&) noexcept = default;
30 constexpr SqlDateTime& operator=(SqlDateTime const& other) noexcept = default;
31 constexpr ~SqlDateTime() noexcept = default;
32
33 constexpr bool operator==(SqlDateTime const& other) const noexcept
34 {
35 return value() == other.value();
36 }
37
38 constexpr bool operator!=(SqlDateTime const& other) const noexcept
39 {
40 return !(*this == other);
41 }
42
43 /// Constructs a date and time from individual components.
44 LIGHTWEIGHT_FORCE_INLINE constexpr SqlDateTime(std::chrono::year_month_day ymd,
45 std::chrono::hh_mm_ss<duration_type> time) noexcept:
46 sqlValue {
47 .year = (SQLSMALLINT) (int) ymd.year(),
48 .month = (SQLUSMALLINT) (unsigned) ymd.month(),
49 .day = (SQLUSMALLINT) (unsigned) ymd.day(),
50 .hour = (SQLUSMALLINT) time.hours().count(),
51 .minute = (SQLUSMALLINT) time.minutes().count(),
52 .second = (SQLUSMALLINT) time.seconds().count(),
53 .fraction =
54 (SQLUINTEGER) (std::chrono::duration_cast<std::chrono::nanoseconds>(time.to_duration()).count() / 100)
55 * 100,
56 }
57 {
58 }
59
60 /// Constructs a date and time from individual components.
61 LIGHTWEIGHT_FORCE_INLINE constexpr SqlDateTime(
62 std::chrono::year year,
63 std::chrono::month month,
64 std::chrono::day day,
65 std::chrono::hours hour,
66 std::chrono::minutes minute,
67 std::chrono::seconds second,
68 std::chrono::nanoseconds nanosecond = std::chrono::nanoseconds(0)) noexcept:
69 sqlValue {
70 .year = (SQLSMALLINT) (int) year,
71 .month = (SQLUSMALLINT) (unsigned) month,
72 .day = (SQLUSMALLINT) (unsigned) day,
73 .hour = (SQLUSMALLINT) hour.count(),
74 .minute = (SQLUSMALLINT) minute.count(),
75 .second = (SQLUSMALLINT) second.count(),
76 .fraction = (SQLUINTEGER) (nanosecond.count() / 100) * 100,
77 }
78 {
79 }
80
81 /// Constructs a date and time from a time point.
82 LIGHTWEIGHT_FORCE_INLINE constexpr SqlDateTime(std::chrono::system_clock::time_point value) noexcept:
83 sqlValue { ConvertToSqlValue(value) }
84 {
85 }
86
87 LIGHTWEIGHT_FORCE_INLINE constexpr operator native_type() const noexcept
88 {
89 return value();
90 }
91
92 static LIGHTWEIGHT_FORCE_INLINE SQL_TIMESTAMP_STRUCT constexpr ConvertToSqlValue(native_type value) noexcept
93 {
94 using namespace std::chrono;
95 auto const totalDays = floor<days>(value);
96 auto const ymd = year_month_day { totalDays };
97 auto const hms = hh_mm_ss<duration_type> { std::chrono::duration_cast<duration_type>(
98 floor<nanoseconds>(value - totalDays)) };
99 return ConvertToSqlValue(ymd, hms);
100 }
101
102 static LIGHTWEIGHT_FORCE_INLINE SQL_TIMESTAMP_STRUCT constexpr ConvertToSqlValue(
103 std::chrono::year_month_day ymd, std::chrono::hh_mm_ss<duration_type> hms) noexcept
104 {
105 // clang-format off
106 // NB: The fraction field is in 100ns units.
107 return SQL_TIMESTAMP_STRUCT {
108 .year = (SQLSMALLINT) (int) ymd.year(),
109 .month = (SQLUSMALLINT) (unsigned) ymd.month(),
110 .day = (SQLUSMALLINT) (unsigned) ymd.day(),
111 .hour = (SQLUSMALLINT) hms.hours().count(),
112 .minute = (SQLUSMALLINT) hms.minutes().count(),
113 .second = (SQLUSMALLINT) hms.seconds().count(),
114 .fraction = (SQLUINTEGER) (((std::chrono::duration_cast<std::chrono::nanoseconds>(hms.to_duration()).count() % 1'000'000'000llu) / 100) * 100)
115 };
116 // clang-format on
117 }
118
119 static LIGHTWEIGHT_FORCE_INLINE native_type constexpr ConvertToNative(SQL_TIMESTAMP_STRUCT const& time) noexcept
120 {
121 // clang-format off
122 using namespace std::chrono;
123 auto const ymd = year_month_day { year { time.year } / month { time.month } / day { time.day } };
124 auto const hms = hh_mm_ss<duration_type> {
125 duration_cast<duration_type>(
126 hours { time.hour }
127 + minutes { time.minute }
128 + seconds { time.second }
129 + nanoseconds { time.fraction }
130 )
131 };
132 return sys_days { ymd } + hms.to_duration();
133 // clang-format on
134 }
135
136 /// Returns the current date and time.
137 [[nodiscard]] constexpr LIGHTWEIGHT_FORCE_INLINE native_type value() const noexcept
138 {
139 return ConvertToNative(sqlValue);
140 }
141
142 LIGHTWEIGHT_FORCE_INLINE SqlDateTime& operator+=(duration_type duration) noexcept
143 {
144 *this = SqlDateTime { value() + duration };
145 return *this;
146 }
147
148 LIGHTWEIGHT_FORCE_INLINE SqlDateTime& operator-=(duration_type duration) noexcept
149 {
150 *this = SqlDateTime { value() - duration };
151 return *this;
152 }
153
154 friend LIGHTWEIGHT_FORCE_INLINE SqlDateTime operator+(SqlDateTime dateTime, duration_type duration) noexcept
155 {
156 auto tmp = dateTime.value() + duration;
157 return SqlDateTime(tmp);
158 // return SqlDateTime { dateTime.value() + duration };
159 }
160
161 friend LIGHTWEIGHT_FORCE_INLINE SqlDateTime operator-(SqlDateTime dateTime, duration_type duration) noexcept
162 {
163 return SqlDateTime { dateTime.value() - duration };
164 }
165
166 SQL_TIMESTAMP_STRUCT sqlValue {};
167};
168
169template <>
170struct std::formatter<SqlDateTime>: std::formatter<std::string>
171{
172 LIGHTWEIGHT_FORCE_INLINE auto format(SqlDateTime const& value, std::format_context& ctx) const
173 -> std::format_context::iterator
174 {
175 return std::formatter<std::string>::format(std::format("{:04}-{:02}-{:02} {:02}:{:02}:{:02}.{:09}",
176 value.sqlValue.year,
177 value.sqlValue.month,
178 value.sqlValue.day,
179 value.sqlValue.hour,
180 value.sqlValue.minute,
181 value.sqlValue.second,
182 value.sqlValue.fraction),
183 ctx);
184 }
185};
186
187template <>
188struct SqlDataBinder<SqlDateTime::native_type>
189{
190 static LIGHTWEIGHT_FORCE_INLINE SQLRETURN GetColumn(SQLHSTMT stmt,
191 SQLUSMALLINT column,
192 SqlDateTime::native_type* result,
193 SQLLEN* indicator,
194 SqlDataBinderCallback const& /*cb*/) noexcept
195 {
196 SQL_TIMESTAMP_STRUCT sqlValue {};
197 auto const rc = SQLGetData(stmt, column, SQL_C_TYPE_TIMESTAMP, &sqlValue, sizeof(sqlValue), indicator);
198 if (SQL_SUCCEEDED(rc))
199 *result = SqlDateTime::ConvertToNative(sqlValue);
200 return rc;
201 }
202};
203
204template <>
205struct LIGHTWEIGHT_API SqlDataBinder<SqlDateTime>
206{
207 static constexpr auto ColumnType = SqlColumnTypeDefinitions::DateTime {};
208
209 static LIGHTWEIGHT_FORCE_INLINE SQLRETURN InputParameter(SQLHSTMT stmt,
210 SQLUSMALLINT column,
211 SqlDateTime const& value,
212 SqlDataBinderCallback& /*cb*/) noexcept
213 {
214 return SQLBindParameter(stmt,
215 column,
216 SQL_PARAM_INPUT,
217 SQL_C_TIMESTAMP,
218 SQL_TYPE_TIMESTAMP,
219 27,
220 7,
221 (SQLPOINTER) &value.sqlValue,
222 sizeof(value),
223 nullptr);
224 }
225
226 static LIGHTWEIGHT_FORCE_INLINE SQLRETURN OutputColumn(SQLHSTMT stmt,
227 SQLUSMALLINT column,
228 SqlDateTime* result,
229 SQLLEN* indicator,
230 SqlDataBinderCallback& /*cb*/) noexcept
231 {
232 // TODO: handle indicator to check for NULL values
233 *indicator = sizeof(result->sqlValue);
234 return SQLBindCol(stmt, column, SQL_C_TYPE_TIMESTAMP, &result->sqlValue, 0, indicator);
235 }
236
237 static LIGHTWEIGHT_FORCE_INLINE SQLRETURN GetColumn(SQLHSTMT stmt,
238 SQLUSMALLINT column,
239 SqlDateTime* result,
240 SQLLEN* indicator,
241 SqlDataBinderCallback const& /*cb*/) noexcept
242 {
243 return SQLGetData(stmt, column, SQL_C_TYPE_TIMESTAMP, &result->sqlValue, sizeof(result->sqlValue), indicator);
244 }
245
246 static LIGHTWEIGHT_FORCE_INLINE std::string Inspect(SqlDateTime const& value) noexcept
247 {
248 return std::format("{}", value);
249 }
250};
LIGHTWEIGHT_FORCE_INLINE constexpr SqlDateTime(std::chrono::system_clock::time_point value) noexcept
Constructs a date and time from a time point.
LIGHTWEIGHT_FORCE_INLINE constexpr SqlDateTime(std::chrono::year year, std::chrono::month month, std::chrono::day day, std::chrono::hours hour, std::chrono::minutes minute, std::chrono::seconds second, std::chrono::nanoseconds nanosecond=std::chrono::nanoseconds(0)) noexcept
Constructs a date and time from individual components.
constexpr LIGHTWEIGHT_FORCE_INLINE native_type value() const noexcept
Returns the current date and time.
LIGHTWEIGHT_FORCE_INLINE constexpr SqlDateTime(std::chrono::year_month_day ymd, std::chrono::hh_mm_ss< duration_type > time) noexcept
Constructs a date and time from individual components.
static LIGHTWEIGHT_FORCE_INLINE SqlDateTime Now() noexcept
Returns the current date and time.