0
0
mirror of https://github.com/rbock/sqlpp11.git synced 2025-01-14 09:47:58 +08:00
sqlpp11/tests/postgresql/usage/TimeZone.cpp
MeanSquaredError 4295c29e9a
Fix serialization of PostgreSQL time data types (#504)
* Use a non-UTC timezone when testing the PostgreSQL timezone support.
* Fix serialization of time of day values for PostgreSQL
* Fix serialization of PostgreSQL values for "timestamp with time zone" and "time with time zone" fields.
2023-07-24 06:40:18 +02:00

166 lines
6.4 KiB
C++

/**
* Copyright © 2023 Vesselin Atanasov
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are met:
*
* Redistributions of source code must retain the above copyright notice, this
* list of conditions and the following disclaimer.
*
* Redistributions in binary form must reproduce the above copyright notice,
* this list of conditions and the following disclaimer in the documentation
* and/or other materials provided with the distribution.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
* ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
* LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
* POSSIBILITY OF SUCH DAMAGE.
*/
#include <vector>
#include <sqlpp11/postgresql/postgresql.h>
#include <sqlpp11/sqlpp11.h>
#include "make_test_connection.h"
#include "TabDateTime.h"
namespace
{
void save_regular (sqlpp::postgresql::connection& dbc, sqlpp::chrono::microsecond_point tp, std::chrono::microseconds tod, sqlpp::chrono::day_point dp)
{
model::TabDateTime tab {};
dbc(
update(tab)
.set(
tab.c_timepoint = tp,
tab.c_time = tod,
tab.c_day = dp
)
.unconditionally()
);
}
void save_prepared (sqlpp::postgresql::connection& dbc, sqlpp::chrono::microsecond_point tp, std::chrono::microseconds tod, sqlpp::chrono::day_point dp)
{
model::TabDateTime tab {};
auto prepared_update = dbc.prepare(
update(tab)
.set(
tab.c_timepoint = parameter(tab.c_timepoint),
tab.c_time = parameter(tab.c_time),
tab.c_day = parameter(tab.c_day)
)
.unconditionally()
);
prepared_update.params.c_timepoint = tp;
prepared_update.params.c_time = tod;
prepared_update.params.c_day = dp;
dbc(prepared_update);
}
template <typename L, typename R>
void require_equal(int line, const L& l, const R& r)
{
if (l != r)
{
std::cerr << line << ": ";
serialize(::sqlpp::wrap_operand_t<L>{l}, std::cerr);
std::cerr << " != ";
serialize(::sqlpp::wrap_operand_t<R>{r}, std::cerr);
throw std::runtime_error("Unexpected result");
}
}
void check_saved_values(sqlpp::postgresql::connection& dbc, sqlpp::chrono::microsecond_point tp, std::chrono::microseconds tod, sqlpp::chrono::day_point dp)
{
model::TabDateTime tab {};
const auto &rows_1 = dbc(
select(
// c_timepoint as microseconds from the start of the UNIX epoch (1970-01-01 00:00:00 UTC)
sqlpp::verbatim<sqlpp::integer>("floor(extract(epoch from c_timepoint)*1000000)::int8").as(sqlpp::alias::a),
// c_time as microseconds from the start of the day (00:00:00 UTC)
sqlpp::verbatim<sqlpp::integer>("floor(extract(epoch from c_time)*1000000)::int8").as(sqlpp::alias::b),
// c_day as days from 1970-01-01 (timezone is not applicable to date fields)
sqlpp::verbatim<sqlpp::integer>("floor(extract(epoch from c_day)/86400)::int8").as(sqlpp::alias::c)
)
.from(tab)
.unconditionally()
);
// Check if the internal values of our C++ time variables match the internal values of the PostgreSQL date/time fields.
// This tests the conversion of date/time types from C++ to PostgreSQL while skipping the conversion from C++ to PostgreSQL.
const auto &row_1 = rows_1.front();
require_equal(__LINE__, row_1.a.value(), tp.time_since_epoch().count());
require_equal(__LINE__, row_1.b.value(), tod.count());
require_equal(__LINE__, row_1.c.value(), dp.time_since_epoch().count());
// Check if saving date/time variables from C++ to PostgreSQL and then reading them back yields the same values.
// This tests the conversion of date/time types from C++ to PostgreSQL and then back from PostgreSQL to C++.
const auto rows_2 = dbc(select(all_of(tab)).from(tab).unconditionally());
const auto &row_2 = rows_2.front();
require_equal(__LINE__, row_2.c_timepoint.value(), tp);
require_equal(__LINE__, row_2.c_time.value(), tod);
require_equal(__LINE__, row_2.c_day.value(), dp);
}
void test_time_point(sqlpp::postgresql::connection& dbc, sqlpp::chrono::microsecond_point tp)
{
auto dp = date::floor<sqlpp::chrono::days> (tp);
auto tod = tp - dp; // Time of day
// Test time values passed in a regular (non-prepared) statement
save_regular(dbc, tp, tod, dp);
check_saved_values(dbc, tp, tod, dp);
// Test time values passed in a prepared statement
save_prepared(dbc, tp, tod, dp);
check_saved_values(dbc, tp, tod, dp);
}
};
int TimeZone(int, char*[])
{
namespace sql = sqlpp::postgresql;
// We use a time zone with non-zero offset from UTC in order to catch serialization/parsing bugs
auto dbc = sql::make_test_connection("+1");
dbc.execute("DROP TABLE IF EXISTS tabdatetime;");
dbc.execute(
"CREATE TABLE tabdatetime "
"("
"c_timepoint timestamp with time zone,"
"c_time time with time zone,"
"c_day date"
")"
);
model::TabDateTime tab {};
try {
dbc(insert_into(tab).default_values());
std::vector<sqlpp::chrono::microsecond_point> tps {
static_cast<date::sys_days>(date::January/1/1970) + std::chrono::hours{1} + std::chrono::minutes{20} + std::chrono::seconds{14} + std::chrono::microseconds{1},
static_cast<date::sys_days>(date::June/13/1986) + std::chrono::hours{12} + std::chrono::minutes{0} + std::chrono::seconds{1} + std::chrono::microseconds{123},
static_cast<date::sys_days>(date::December/31/2022) + std::chrono::hours{0} + std::chrono::minutes{59} + std::chrono::seconds{59} + std::chrono::microseconds{987654}
};
for (const auto &tp : tps) {
test_time_point(dbc, tp);
}
} catch (const sql::failure& e) {
std::cerr << "Exception: " << e.what() << std::endl;
return 1;
}
return 0;
}