This repository has been archived on 2025-09-14. You can view files and clone it, but cannot push or open issues or pull requests.
Files
zhangyang-zerotierone/ext/libpqxx-7.7.3/src/transaction_base.cxx
2022-06-24 10:12:36 -07:00

540 lines
13 KiB
C++

/** Common code and definitions for the transaction classes.
*
* pqxx::transaction_base defines the interface for any abstract class that
* represents a database transaction.
*
* Copyright (c) 2000-2022, Jeroen T. Vermeulen.
*
* See COPYING for copyright license. If you did not receive a file called
* COPYING with this source code, please notify the distributor of this
* mistake, or contact the author.
*/
#include "pqxx-source.hxx"
#include <cstring>
#include <stdexcept>
#include "pqxx/internal/header-pre.hxx"
#include "pqxx/connection.hxx"
#include "pqxx/internal/concat.hxx"
#include "pqxx/internal/encodings.hxx"
#include "pqxx/internal/gates/connection-transaction.hxx"
#include "pqxx/internal/gates/transaction-transaction_focus.hxx"
#include "pqxx/result.hxx"
#include "pqxx/transaction_base.hxx"
#include "pqxx/transaction_focus.hxx"
#include "pqxx/internal/header-post.hxx"
using namespace std::literals;
namespace
{
/// Return a query pointer for the command "ROLLBACK".
/** Concentrates constructions so as to minimise the number of allocations.
* This way, the string gets allocated once and then all subsequent invocations
* copy shared_ptr instances to the same string.
*/
std::shared_ptr<std::string> make_rollback_cmd()
{
static auto const cmd{std::make_shared<std::string>("ROLLBACK")};
return cmd;
}
} // namespace
pqxx::transaction_base::transaction_base(connection &c) :
m_conn{c}, m_rollback_cmd{make_rollback_cmd()}
{}
pqxx::transaction_base::transaction_base(
connection &c, std::string_view tname) :
m_conn{c}, m_name{tname}, m_rollback_cmd{make_rollback_cmd()}
{}
pqxx::transaction_base::~transaction_base()
{
try
{
if (not std::empty(m_pending_error))
PQXX_UNLIKELY
process_notice(
internal::concat("UNPROCESSED ERROR: ", m_pending_error, "\n"));
if (m_registered)
{
m_conn.process_notice(
internal::concat(description(), " was never closed properly!\n"));
pqxx::internal::gate::connection_transaction{conn()}
.unregister_transaction(this);
}
}
catch (std::exception const &e)
{
try
{
process_notice(internal::concat(e.what(), "\n"));
}
catch (std::exception const &)
{
process_notice(e.what());
}
}
}
void pqxx::transaction_base::register_transaction()
{
pqxx::internal::gate::connection_transaction{conn()}.register_transaction(
this);
m_registered = true;
}
void pqxx::transaction_base::commit()
{
check_pending_error();
// Check previous status code. Caller should only call this function if
// we're in "implicit" state, but multiple commits are silently accepted.
switch (m_status)
{
case status::active: // Just fine. This is what we expect.
break;
case status::aborted:
throw usage_error{internal::concat(
"Attempt to commit previously aborted ", description())};
case status::committed:
// Transaction has been committed already. This is not exactly proper
// behaviour, but throwing an exception here would only give the impression
// that an abort is needed--which would only confuse things further at this
// stage.
// Therefore, multiple commits are accepted, though under protest.
m_conn.process_notice(
internal::concat(description(), " committed more than once.\n"));
return;
case status::in_doubt:
// Transaction may or may not have been committed. The only thing we can
// really do is keep telling the caller that the transaction is in doubt.
throw in_doubt_error{internal::concat(
description(), " committed again while in an indeterminate state.")};
default: throw internal_error{"pqxx::transaction: invalid status code."};
}
// Tricky one. If stream is nested in transaction but inside the same scope,
// the commit() will come before the stream is closed. Which means the
// commit is premature. Punish this swiftly and without fail to discourage
// the habit from forming.
if (m_focus != nullptr)
throw failure{internal::concat(
"Attempt to commit ", description(), " with ", m_focus->description(),
" still open.")};
// Check that we're still connected (as far as we know--this is not an
// absolute thing!) before trying to commit. If the connection was broken
// already, the commit would fail anyway but this way at least we don't
// remain in-doubt as to whether the backend got the commit order at all.
if (not m_conn.is_open())
throw broken_connection{
"Broken connection to backend; cannot complete transaction."};
try
{
do_commit();
m_status = status::committed;
}
catch (in_doubt_error const &)
{
m_status = status::in_doubt;
throw;
}
catch (std::exception const &)
{
m_status = status::aborted;
throw;
}
close();
}
void pqxx::transaction_base::do_abort()
{
if (m_rollback_cmd)
direct_exec(m_rollback_cmd);
}
void pqxx::transaction_base::abort()
{
// Check previous status code. Quietly accept multiple aborts to
// simplify emergency bailout code.
switch (m_status)
{
case status::active:
try
{
do_abort();
}
catch (std::exception const &e)
{
m_conn.process_notice(internal::concat(e.what(), "\n"));
}
break;
case status::aborted: return;
case status::committed:
throw usage_error{internal::concat(
"Attempt to abort previously committed ", description())};
case status::in_doubt:
// Aborting an in-doubt transaction is probably a reasonably sane response
// to an insane situation. Log it, but do not fail.
m_conn.process_notice(internal::concat(
"Warning: ", description(),
" aborted after going into indeterminate state; "
"it may have been executed anyway.\n"));
return;
default: throw internal_error{"Invalid transaction status."};
}
m_status = status::aborted;
close();
}
std::string PQXX_COLD pqxx::transaction_base::quote_raw(zview bin) const
{
return conn().quote(binary_cast(bin));
}
namespace
{
/// Guard command execution against clashes with pipelines and such.
/** A transaction can have only one focus at a time. Command execution is the
* most basic example of a transaction focus.
*/
class PQXX_PRIVATE command : pqxx::transaction_focus
{
public:
command(pqxx::transaction_base &tx, std::string_view oname) :
transaction_focus{tx, "command"sv, oname}
{
register_me();
}
~command() { unregister_me(); }
};
} // namespace
pqxx::result
pqxx::transaction_base::exec(std::string_view query, std::string_view desc)
{
check_pending_error();
command cmd{*this, desc};
switch (m_status)
{
case status::active: break;
case status::committed:
case status::aborted:
case status::in_doubt: {
std::string const n{
std::empty(desc) ? "" : internal::concat("'", desc, "' ")};
throw usage_error{internal::concat(
"Could not execute command ", n, ": transaction is already closed.")};
}
default: throw internal_error{"pqxx::transaction: invalid status code."};
}
return direct_exec(query, desc);
}
pqxx::result pqxx::transaction_base::exec_n(
result::size_type rows, zview query, std::string_view desc)
{
#include "pqxx/internal/ignore-deprecated-pre.hxx"
result const r{exec(query, desc)};
#include "pqxx/internal/ignore-deprecated-post.hxx"
if (std::size(r) != rows)
{
std::string const N{
std::empty(desc) ? "" : internal::concat("'", desc, "'")};
throw unexpected_rows{internal::concat(
"Expected ", rows, " row(s) of data from query ", N, ", got ",
std::size(r), ".")};
}
return r;
}
void pqxx::transaction_base::check_rowcount_prepared(
zview statement, result::size_type expected_rows,
result::size_type actual_rows)
{
if (actual_rows != expected_rows)
throw unexpected_rows{internal::concat(
"Expected ", expected_rows, " row(s) of data from prepared statement '",
statement, "', got ", actual_rows, ".")};
}
void pqxx::transaction_base::check_rowcount_params(
std::size_t expected_rows, std::size_t actual_rows)
{
if (actual_rows != expected_rows)
throw unexpected_rows{internal::concat(
"Expected ", expected_rows,
" row(s) of data from parameterised query, got ", actual_rows, ".")};
}
pqxx::result pqxx::transaction_base::internal_exec_prepared(
zview statement, internal::c_params const &args)
{
command cmd{*this, statement};
return pqxx::internal::gate::connection_transaction{conn()}.exec_prepared(
statement, args);
}
pqxx::result pqxx::transaction_base::internal_exec_params(
zview query, internal::c_params const &args)
{
command cmd{*this, query};
return pqxx::internal::gate::connection_transaction{conn()}.exec_params(
query, args);
}
void pqxx::transaction_base::set_variable(
std::string_view var, std::string_view value)
{
#include "pqxx/internal/ignore-deprecated-pre.hxx"
conn().set_variable(var, value);
#include "pqxx/internal/ignore-deprecated-post.hxx"
}
std::string pqxx::transaction_base::get_variable(std::string_view var)
{
#include "pqxx/internal/ignore-deprecated-pre.hxx"
return conn().get_variable(var);
#include "pqxx/internal/ignore-deprecated-post.hxx"
}
void pqxx::transaction_base::close() noexcept
{
try
{
try
{
check_pending_error();
}
catch (std::exception const &e)
{
m_conn.process_notice(e.what());
}
if (m_registered)
{
m_registered = false;
pqxx::internal::gate::connection_transaction{conn()}
.unregister_transaction(this);
}
if (m_status != status::active)
return;
if (m_focus != nullptr)
PQXX_UNLIKELY
m_conn.process_notice(internal::concat(
"Closing ", description(), " with ", m_focus->description(),
" still open.\n"));
try
{
abort();
}
catch (std::exception const &e)
{
m_conn.process_notice(e.what());
}
}
catch (std::exception const &e)
{
try
{
m_conn.process_notice(e.what());
}
catch (std::exception const &)
{}
}
}
namespace
{
[[nodiscard]] std::string_view
get_classname(pqxx::transaction_focus const *focus)
{
return (focus == nullptr) ? ""sv : focus->classname();
}
[[nodiscard]] std::string_view
get_obj_name(pqxx::transaction_focus const *focus)
{
return (focus == nullptr) ? ""sv : focus->name();
}
} // namespace
void pqxx::transaction_base::register_focus(transaction_focus *new_focus)
{
internal::check_unique_register(
m_focus, get_classname(m_focus), get_obj_name(m_focus), new_focus,
get_classname(new_focus), get_obj_name(new_focus));
m_focus = new_focus;
}
void pqxx::transaction_base::unregister_focus(
transaction_focus *new_focus) noexcept
{
try
{
pqxx::internal::check_unique_unregister(
m_focus, get_classname(m_focus), get_obj_name(m_focus), new_focus,
get_classname(new_focus), get_obj_name(new_focus));
m_focus = nullptr;
}
catch (std::exception const &e)
{
m_conn.process_notice(internal::concat(e.what(), "\n"));
}
}
pqxx::result pqxx::transaction_base::direct_exec(
std::string_view cmd, std::string_view desc)
{
check_pending_error();
return pqxx::internal::gate::connection_transaction{conn()}.exec(cmd, desc);
}
pqxx::result pqxx::transaction_base::direct_exec(
std::shared_ptr<std::string> cmd, std::string_view desc)
{
check_pending_error();
return pqxx::internal::gate::connection_transaction{conn()}.exec(cmd, desc);
}
void pqxx::transaction_base::register_pending_error(zview err) noexcept
{
if (std::empty(m_pending_error) and not std::empty(err))
{
try
{
m_pending_error = err;
}
catch (std::exception const &e)
{
try
{
PQXX_UNLIKELY
process_notice("UNABLE TO PROCESS ERROR\n");
process_notice(e.what());
process_notice("ERROR WAS:");
process_notice(err);
}
catch (...)
{}
}
}
}
void pqxx::transaction_base::register_pending_error(std::string &&err) noexcept
{
if (std::empty(m_pending_error) and not std::empty(err))
{
try
{
m_pending_error = std::move(err);
}
catch (std::exception const &e)
{
try
{
PQXX_UNLIKELY
process_notice("UNABLE TO PROCESS ERROR\n");
process_notice(e.what());
process_notice("ERROR WAS:");
process_notice(err);
}
catch (...)
{}
}
}
}
void pqxx::transaction_base::check_pending_error()
{
if (not std::empty(m_pending_error))
{
std::string err;
err.swap(m_pending_error);
throw failure{err};
}
}
std::string pqxx::transaction_base::description() const
{
return internal::describe_object("transaction", name());
}
void pqxx::transaction_focus::register_me()
{
pqxx::internal::gate::transaction_transaction_focus{m_trans}.register_focus(
this);
m_registered = true;
}
void pqxx::transaction_focus::unregister_me() noexcept
{
pqxx::internal::gate::transaction_transaction_focus{m_trans}
.unregister_focus(this);
m_registered = false;
}
void pqxx::transaction_focus::reg_pending_error(
std::string const &err) noexcept
{
pqxx::internal::gate::transaction_transaction_focus{m_trans}
.register_pending_error(err);
}