| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445 |
- #include <iostream>
- #include <optional>
- #include <pqxx/stream_to>
- #include <pqxx/transaction>
- #include "../test_helpers.hxx"
- #include "../test_types.hxx"
- namespace
- {
- std::string truncate_sql_error(std::string const &what)
- {
- auto trunc{what.substr(0, what.find('\n'))};
- if (std::size(trunc) > 64)
- trunc = trunc.substr(0, 61) + "...";
- return trunc;
- }
- void test_nonoptionals(pqxx::connection &connection)
- {
- pqxx::work tx{connection};
- auto inserter{pqxx::stream_to::table(tx, {"stream_to_test"})};
- PQXX_CHECK(inserter, "stream_to failed to initialize");
- auto const nonascii{"\u3053\u3093\u306b\u3061\u308f"};
- bytea const binary{'\x00', '\x01', '\x02'},
- text{'f', 'o', 'o', ' ', 'b', 'a', 'r', '\0'};
- inserter << std::make_tuple(
- 1234, "now", 4321, ipv4{8, 8, 4, 4}, "hello nonoptional world", binary);
- inserter << std::make_tuple(
- 5678, "2018-11-17 21:23:00", nullptr, nullptr, nonascii, text);
- inserter << std::make_tuple(910, nullptr, nullptr, nullptr, "\\N", bytea{});
- inserter.complete();
- auto r1{tx.exec1("SELECT * FROM stream_to_test WHERE number0 = 1234")};
- PQXX_CHECK_EQUAL(r1[0].as<int>(), 1234, "Read back wrong first int.");
- PQXX_CHECK_EQUAL(
- r1[4].as<std::string>(), "hello nonoptional world",
- "Read back wrong string.");
- PQXX_CHECK_EQUAL(r1[3].as<ipv4>(), ipv4(8, 8, 4, 4), "Read back wrong ip.");
- PQXX_CHECK_EQUAL(r1[5].as<bytea>(), binary, "Read back wrong bytea.");
- auto r2{tx.exec1("SELECT * FROM stream_to_test WHERE number0 = 5678")};
- PQXX_CHECK_EQUAL(r2[0].as<int>(), 5678, "Wrong int on second row.");
- PQXX_CHECK(r2[2].is_null(), "Field 2 was meant to be null.");
- PQXX_CHECK(r2[3].is_null(), "Field 3 was meant to be null.");
- PQXX_CHECK_EQUAL(r2[4].as<std::string>(), nonascii, "Wrong non-ascii text.");
- tx.commit();
- }
- void test_nonoptionals_fold(pqxx::connection &connection)
- {
- pqxx::work tx{connection};
- auto inserter{pqxx::stream_to::table(tx, {"stream_to_test"})};
- PQXX_CHECK(inserter, "stream_to failed to initialize");
- auto const nonascii{"\u3053\u3093\u306b\u3061\u308f"};
- bytea const binary{'\x00', '\x01', '\x02'},
- text{'f', 'o', 'o', ' ', 'b', 'a', 'r', '\0'};
- inserter.write_values(
- 1234, "now", 4321, ipv4{8, 8, 4, 4}, "hello nonoptional world", binary);
- inserter.write_values(
- 5678, "2018-11-17 21:23:00", nullptr, nullptr, nonascii, text);
- inserter.write_values(910, nullptr, nullptr, nullptr, "\\N", bytea{});
- inserter.complete();
- auto r1{tx.exec1("SELECT * FROM stream_to_test WHERE number0 = 1234")};
- PQXX_CHECK_EQUAL(r1[0].as<int>(), 1234, "Read back wrong first int.");
- PQXX_CHECK_EQUAL(
- r1[4].as<std::string>(), "hello nonoptional world",
- "Read back wrong string.");
- PQXX_CHECK_EQUAL(r1[3].as<ipv4>(), ipv4(8, 8, 4, 4), "Read back wrong ip.");
- PQXX_CHECK_EQUAL(r1[5].as<bytea>(), binary, "Read back wrong bytera.");
- auto r2{tx.exec1("SELECT * FROM stream_to_test WHERE number0 = 5678")};
- PQXX_CHECK_EQUAL(r2[0].as<int>(), 5678, "Wrong int on second row.");
- PQXX_CHECK(r2[2].is_null(), "Field 2 was meant to be null.");
- PQXX_CHECK(r2[3].is_null(), "Field 3 was meant to be null.");
- PQXX_CHECK_EQUAL(r2[4].as<std::string>(), nonascii, "Wrong non-ascii text.");
- tx.commit();
- }
- /// Try to violate stream_to_test's not-null constraint using a stream_to.
- void insert_bad_null_tuple(pqxx::stream_to &inserter)
- {
- inserter << std::make_tuple(
- nullptr, "now", 4321, ipv4{8, 8, 8, 8}, "hello world",
- bytea{'\x00', '\x01', '\x02'});
- inserter.complete();
- }
- void test_bad_null(pqxx::connection &connection)
- {
- pqxx::work tx{connection};
- auto inserter{pqxx::stream_to::table(tx, {"stream_to_test"})};
- PQXX_CHECK(inserter, "stream_to failed to initialize");
- PQXX_CHECK_THROWS(
- insert_bad_null_tuple(inserter), pqxx::not_null_violation,
- "Did not expected not_null_violation when stream_to inserts a bad null.");
- }
- /// Try to violate stream_to_test's not-null construct using a stream_to.
- void insert_bad_null_write(pqxx::stream_to &inserter)
- {
- inserter.write_values(
- nullptr, "now", 4321, ipv4{8, 8, 8, 8}, "hello world",
- bytea{'\x00', '\x01', '\x02'});
- inserter.complete();
- }
- void test_bad_null_fold(pqxx::connection &connection)
- {
- pqxx::work tx{connection};
- auto inserter{pqxx::stream_to::table(tx, {"stream_to_test"})};
- PQXX_CHECK(inserter, "stream_to failed to initialize");
- PQXX_CHECK_THROWS(
- insert_bad_null_write(inserter), pqxx::not_null_violation,
- "Did not expected not_null_violation when stream_to inserts a bad null.");
- }
- void test_too_few_fields(pqxx::connection &connection)
- {
- pqxx::work tx{connection};
- auto inserter{pqxx::stream_to::table(tx, {"stream_to_test"})};
- PQXX_CHECK(inserter, "stream_to failed to initialize");
- try
- {
- inserter << std::make_tuple(1234, "now", 4321, ipv4{8, 8, 8, 8});
- inserter.complete();
- tx.commit();
- PQXX_CHECK_NOTREACHED("stream_from improperly inserted row");
- }
- catch (pqxx::sql_error const &e)
- {
- std::string what{e.what()};
- if (what.find("missing data for column") == std::string::npos)
- throw;
- pqxx::test::expected_exception(
- "Could not insert row: " + truncate_sql_error(what));
- }
- }
- void test_too_few_fields_fold(pqxx::connection &connection)
- {
- pqxx::work tx{connection};
- auto inserter{pqxx::stream_to::table(tx, {"stream_to_test"})};
- PQXX_CHECK(inserter, "stream_to failed to initialize");
- try
- {
- inserter.write_values(1234, "now", 4321, ipv4{8, 8, 8, 8});
- inserter.complete();
- tx.commit();
- PQXX_CHECK_NOTREACHED("stream_from_fold improperly inserted row");
- }
- catch (pqxx::sql_error const &e)
- {
- std::string what{e.what()};
- if (what.find("missing data for column") == std::string::npos)
- throw;
- pqxx::test::expected_exception(
- "Fold - Could not insert row: " + truncate_sql_error(what));
- }
- }
- void test_too_many_fields(pqxx::connection &connection)
- {
- pqxx::work tx{connection};
- auto inserter{pqxx::stream_to::table(tx, {"stream_to_test"})};
- PQXX_CHECK(inserter, "stream_to failed to initialize");
- try
- {
- inserter << std::make_tuple(
- 1234, "now", 4321, ipv4{8, 8, 8, 8}, "hello world",
- bytea{'\x00', '\x01', '\x02'}, 5678);
- inserter.complete();
- tx.commit();
- PQXX_CHECK_NOTREACHED("stream_from improperly inserted row");
- }
- catch (pqxx::sql_error const &e)
- {
- std::string what{e.what()};
- if (what.find("extra data") == std::string::npos)
- throw;
- pqxx::test::expected_exception(
- "Could not insert row: " + truncate_sql_error(what));
- }
- }
- void test_too_many_fields_fold(pqxx::connection &connection)
- {
- pqxx::work tx{connection};
- auto inserter{pqxx::stream_to::table(tx, {"stream_to_test"})};
- PQXX_CHECK(inserter, "stream_to failed to initialize");
- try
- {
- inserter.write_values(
- 1234, "now", 4321, ipv4{8, 8, 8, 8}, "hello world",
- bytea{'\x00', '\x01', '\x02'}, 5678);
- inserter.complete();
- tx.commit();
- PQXX_CHECK_NOTREACHED("stream_from_fold improperly inserted row");
- }
- catch (pqxx::sql_error const &e)
- {
- std::string what{e.what()};
- if (what.find("extra data") == std::string::npos)
- throw;
- pqxx::test::expected_exception(
- "Fold - Could not insert row: " + truncate_sql_error(what));
- }
- }
- void test_stream_to_does_nonnull_optional()
- {
- pqxx::connection conn;
- pqxx::work tx{conn};
- tx.exec0("CREATE TEMP TABLE foo(x integer, y text)");
- auto inserter{pqxx::stream_to::table(tx, {"foo"})};
- inserter.write_values(
- std::optional<int>{368}, std::optional<std::string>{"Text"});
- inserter.complete();
- auto const row{tx.exec1("SELECT x, y FROM foo")};
- PQXX_CHECK_EQUAL(
- row[0].as<std::string>(), "368", "Non-null int optional came out wrong.");
- PQXX_CHECK_EQUAL(
- row[1].as<std::string>(), "Text",
- "Non-null string optional came out wrong.");
- }
- template<template<typename...> class O>
- void test_optional(pqxx::connection &connection)
- {
- pqxx::work tx{connection};
- auto inserter{pqxx::stream_to::table(tx, {"stream_to_test"})};
- PQXX_CHECK(inserter, "stream_to failed to initialize");
- inserter << std::make_tuple(
- 910, O<std::string>{pqxx::nullness<O<std::string>>::null()},
- O<int>{pqxx::nullness<O<int>>::null()},
- O<ipv4>{pqxx::nullness<O<ipv4>>::null()}, "\\N", bytea{});
- inserter.complete();
- tx.commit();
- }
- template<template<typename...> class O>
- void test_optional_fold(pqxx::connection &connection)
- {
- pqxx::work tx{connection};
- auto inserter{pqxx::stream_to::table(tx, {"stream_to_test"})};
- PQXX_CHECK(inserter, "stream_to failed to initialize");
- inserter.write_values(
- 910, O<std::string>{pqxx::nullness<O<std::string>>::null()},
- O<int>{pqxx::nullness<O<int>>::null()},
- O<ipv4>{pqxx::nullness<O<ipv4>>::null()}, "\\N", bytea{});
- inserter.complete();
- tx.commit();
- }
- // As an alternative to a tuple, you can also insert a container.
- void test_container_stream_to()
- {
- pqxx::connection conn;
- pqxx::work tx{conn};
- tx.exec0("CREATE TEMP TABLE test_container(a integer, b integer)");
- auto inserter{pqxx::stream_to::table(tx, {"test_container"})};
- inserter << std::vector{112, 244};
- inserter.complete();
- auto read{tx.exec1("SELECT * FROM test_container")};
- PQXX_CHECK_EQUAL(
- read[0].as<int>(), 112, "stream_to on container went wrong.");
- PQXX_CHECK_EQUAL(
- read[1].as<int>(), 244, "Second container field went wrong.");
- tx.commit();
- }
- void test_variant_fold(pqxx::connection_base &connection)
- {
- pqxx::work tx{connection};
- auto inserter{pqxx::stream_to::table(tx, {"stream_to_test"})};
- PQXX_CHECK(inserter, "stream_to failed to initialize");
- inserter.write_values(
- std::variant<std::string, int>{1234},
- std::variant<float, std::string>{"now"}, 4321, ipv4{8, 8, 8, 8},
- "hello world", bytea{'\x00', '\x01', '\x02'});
- inserter.write_values(
- 5678, "2018-11-17 21:23:00", nullptr, nullptr,
- "\u3053\u3093\u306b\u3061\u308f",
- bytea{'f', 'o', 'o', ' ', 'b', 'a', 'r', '\0'});
- inserter.write_values(910, nullptr, nullptr, nullptr, "\\N", bytea{});
- inserter.complete();
- tx.commit();
- }
- void clear_table(pqxx::connection &conn)
- {
- pqxx::work tx{conn};
- tx.exec0("DELETE FROM stream_to_test");
- tx.commit();
- }
- void test_stream_to()
- {
- pqxx::connection conn;
- pqxx::work tx{conn};
- tx.exec0(
- "CREATE TEMP TABLE stream_to_test ("
- "number0 INT NOT NULL,"
- "ts1 TIMESTAMP NULL,"
- "number2 INT NULL,"
- "addr3 INET NULL,"
- "txt4 TEXT NULL,"
- "bin5 BYTEA NOT NULL"
- ")");
- tx.commit();
- test_nonoptionals(conn);
- clear_table(conn);
- test_nonoptionals_fold(conn);
- clear_table(conn);
- test_bad_null(conn);
- clear_table(conn);
- test_bad_null_fold(conn);
- clear_table(conn);
- test_too_few_fields(conn);
- clear_table(conn);
- test_too_few_fields_fold(conn);
- clear_table(conn);
- test_too_many_fields(conn);
- clear_table(conn);
- test_too_many_fields_fold(conn);
- clear_table(conn);
- test_optional<std::unique_ptr>(conn);
- clear_table(conn);
- test_optional_fold<std::unique_ptr>(conn);
- clear_table(conn);
- test_optional<std::optional>(conn);
- clear_table(conn);
- test_optional_fold<std::optional>(conn);
- clear_table(conn);
- test_variant_fold(conn);
- }
- void test_stream_to_factory_with_static_columns()
- {
- pqxx::connection conn;
- pqxx::work tx{conn};
- tx.exec0("CREATE TEMP TABLE pqxx_stream_to(a integer, b varchar)");
- auto stream{pqxx::stream_to::table(tx, {"pqxx_stream_to"}, {"a", "b"})};
- stream.write_values(3, "three");
- stream.complete();
- auto r{tx.exec1("SELECT a, b FROM pqxx_stream_to")};
- PQXX_CHECK_EQUAL(r[0].as<int>(), 3, "Failed to stream_to a table.");
- PQXX_CHECK_EQUAL(
- r[1].as<std::string>(), "three",
- "Failed to stream_to a string to a table.");
- }
- void test_stream_to_factory_with_dynamic_columns()
- {
- pqxx::connection conn;
- pqxx::work tx{conn};
- tx.exec0("CREATE TEMP TABLE pqxx_stream_to(a integer, b varchar)");
- std::vector<std::string_view> columns{"a", "b"};
- #if defined(PQXX_HAVE_CONCEPTS)
- auto stream{pqxx::stream_to::table(tx, {"pqxx_stream_to"}, columns)};
- #else
- auto stream{pqxx::stream_to::raw_table(
- tx, conn.quote_table({"pqxx_stream_to"}), conn.quote_columns(columns))};
- #endif
- stream.write_values(4, "four");
- stream.complete();
- auto r{tx.exec1("SELECT a, b FROM pqxx_stream_to")};
- PQXX_CHECK_EQUAL(
- r[0].as<int>(), 4, "Failed to stream_to a table with dynamic columns.");
- PQXX_CHECK_EQUAL(
- r[1].as<std::string>(), "four",
- "Failed to stream_to a string to a table with dynamic columns.");
- }
- void test_stream_to_quotes_arguments()
- {
- pqxx::connection conn;
- pqxx::work tx{conn};
- std::string const table{R"--(pqxx_Stream"'x)--"}, column{R"--(a'"b)--"};
- tx.exec0(
- "CREATE TEMP TABLE " + tx.quote_name(table) + "(" + tx.quote_name(column) +
- " integer)");
- auto write{pqxx::stream_to::table(tx, {table}, {column})};
- write.write_values<int>(12);
- write.complete();
- PQXX_CHECK_EQUAL(
- tx.query_value<int>(
- "SELECT " + tx.quote_name(column) + " FROM " + tx.quote_name(table)),
- 12, "Stream wrote wrong value.");
- }
- PQXX_REGISTER_TEST(test_stream_to);
- PQXX_REGISTER_TEST(test_container_stream_to);
- PQXX_REGISTER_TEST(test_stream_to_does_nonnull_optional);
- PQXX_REGISTER_TEST(test_stream_to_factory_with_static_columns);
- PQXX_REGISTER_TEST(test_stream_to_factory_with_dynamic_columns);
- PQXX_REGISTER_TEST(test_stream_to_quotes_arguments);
- } // namespace
|