mirror of
https://github.com/zerotier/ZeroTierOne.git
synced 2025-01-06 21:18:42 +00:00
99 lines
2.8 KiB
C++
99 lines
2.8 KiB
C++
|
#include "../test_helpers.hxx"
|
||
|
|
||
|
#include "pqxx/composite"
|
||
|
#include "pqxx/transaction"
|
||
|
|
||
|
namespace
|
||
|
{
|
||
|
void test_composite()
|
||
|
{
|
||
|
pqxx::connection conn;
|
||
|
pqxx::work tx{conn};
|
||
|
tx.exec0("CREATE TYPE pqxxfoo AS (a integer, b text)");
|
||
|
auto const r{tx.exec1("SELECT '(5,hello)'::pqxxfoo")};
|
||
|
|
||
|
int a;
|
||
|
std::string b;
|
||
|
pqxx::parse_composite(r[0].view(), a, b);
|
||
|
|
||
|
PQXX_CHECK_EQUAL(a, 5, "Integer composite field came back wrong.");
|
||
|
PQXX_CHECK_EQUAL(b, "hello", "String composite field came back wrong.");
|
||
|
}
|
||
|
|
||
|
|
||
|
void test_composite_escapes()
|
||
|
{
|
||
|
pqxx::connection conn;
|
||
|
pqxx::work tx{conn};
|
||
|
pqxx::row r;
|
||
|
tx.exec0("CREATE TYPE pqxxsingle AS (x text)");
|
||
|
std::string s;
|
||
|
|
||
|
r = tx.exec1(R"--(SELECT '("a""b")'::pqxxsingle)--");
|
||
|
pqxx::parse_composite(r[0].view(), s);
|
||
|
PQXX_CHECK_EQUAL(
|
||
|
s, "a\"b", "Double-double-quotes escaping did not parse correctly.");
|
||
|
|
||
|
r = tx.exec1(R"--(SELECT '("a\"b")'::pqxxsingle)--");
|
||
|
pqxx::parse_composite(r[0].view(), s);
|
||
|
PQXX_CHECK_EQUAL(s, "a\"b", "Backslash escaping did not parse correctly.");
|
||
|
}
|
||
|
|
||
|
|
||
|
void test_composite_handles_nulls()
|
||
|
{
|
||
|
pqxx::connection conn;
|
||
|
pqxx::work tx{conn};
|
||
|
pqxx::row r;
|
||
|
|
||
|
tx.exec0("CREATE TYPE pqxxnull AS (a integer)");
|
||
|
int nonnull;
|
||
|
r = tx.exec1("SELECT '()'::pqxxnull");
|
||
|
PQXX_CHECK_THROWS(
|
||
|
pqxx::parse_composite(r[0].view(), nonnull), pqxx::conversion_error,
|
||
|
"No conversion error when reading a null into a nulless variable.");
|
||
|
std::optional<int> nullable{5};
|
||
|
pqxx::parse_composite(r[0].view(), nullable);
|
||
|
PQXX_CHECK(
|
||
|
not nullable.has_value(), "Null integer came out as having a value.");
|
||
|
|
||
|
tx.exec0("CREATE TYPE pqxxnulls AS (a integer, b integer)");
|
||
|
std::optional<int> a{2}, b{4};
|
||
|
r = tx.exec1("SELECT '(,)'::pqxxnulls");
|
||
|
pqxx::parse_composite(r[0].view(), a, b);
|
||
|
PQXX_CHECK(not a.has_value(), "Null first integer stored as value.");
|
||
|
PQXX_CHECK(not b.has_value(), "Null second integer stored as value.");
|
||
|
}
|
||
|
|
||
|
|
||
|
void test_composite_renders_to_string()
|
||
|
{
|
||
|
pqxx::connection conn;
|
||
|
pqxx::work tx{conn};
|
||
|
char buf[1000];
|
||
|
|
||
|
pqxx::composite_into_buf(
|
||
|
std::begin(buf), std::end(buf), 355, "foo", "b\na\\r");
|
||
|
PQXX_CHECK_EQUAL(
|
||
|
std::string{buf}, "(355,\"foo\",\"b\na\\\\r\")",
|
||
|
"Composite was not rendered as expected.");
|
||
|
|
||
|
tx.exec0("CREATE TYPE pqxxcomp AS (a integer, b text, c text)");
|
||
|
auto const r{tx.exec1("SELECT '" + std::string{buf} + "'::pqxxcomp")};
|
||
|
|
||
|
int a;
|
||
|
std::string b, c;
|
||
|
bool const nonnull{r[0].composite_to(a, b, c)};
|
||
|
PQXX_CHECK(nonnull, "Mistaken nullness.");
|
||
|
PQXX_CHECK_EQUAL(a, 355, "Int came back wrong.");
|
||
|
PQXX_CHECK_EQUAL(b, "foo", "Simple string came back wrong.");
|
||
|
PQXX_CHECK_EQUAL(c, "b\na\\r", "Escaping went wrong.");
|
||
|
}
|
||
|
|
||
|
|
||
|
PQXX_REGISTER_TEST(test_composite);
|
||
|
PQXX_REGISTER_TEST(test_composite_escapes);
|
||
|
PQXX_REGISTER_TEST(test_composite_handles_nulls);
|
||
|
PQXX_REGISTER_TEST(test_composite_renders_to_string);
|
||
|
} // namespace
|