| 1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798 |
- #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
|