fix #5281 for exchange: do preflight check that an old transaction is no longer running by accident

This commit is contained in:
Christian Grothoff 2018-03-12 11:33:10 +01:00
parent 1ae2ba3d0a
commit a166ca7fec
No known key found for this signature in database
GPG Key ID: 939E6BE1E29FC3CC
22 changed files with 157 additions and 40 deletions

View File

@ -3906,8 +3906,11 @@ transact (Analysis analysis,
GNUNET_break (0); GNUNET_break (0);
return GNUNET_SYSERR; return GNUNET_SYSERR;
} }
ret = edb->start (edb->cls, edb->preflight (edb->cls,
esession); esession);
ret = edb->start (edb->cls,
esession,
"auditor");
if (GNUNET_OK != ret) if (GNUNET_OK != ret)
{ {
GNUNET_break (0); GNUNET_break (0);

View File

@ -1412,8 +1412,11 @@ run (void *cls,
GNUNET_SCHEDULER_shutdown (); GNUNET_SCHEDULER_shutdown ();
return; return;
} }
ret = edb->start (edb->cls, edb->preflight (edb->cls,
esession); esession);
ret = edb->start (edb->cls,
esession,
"wire auditor");
if (GNUNET_OK != ret) if (GNUNET_OK != ret)
{ {
GNUNET_break (0); GNUNET_break (0);

View File

@ -1203,9 +1203,12 @@ run_reserve_closures (void *cls)
GNUNET_SCHEDULER_shutdown (); GNUNET_SCHEDULER_shutdown ();
return; return;
} }
db_plugin->preflight (db_plugin->cls,
session);
if (GNUNET_OK != if (GNUNET_OK !=
db_plugin->start (db_plugin->cls, db_plugin->start (db_plugin->cls,
session)) session,
"aggregator reserve closures"))
{ {
GNUNET_log (GNUNET_ERROR_TYPE_ERROR, GNUNET_log (GNUNET_ERROR_TYPE_ERROR,
"Failed to start database transaction!\n"); "Failed to start database transaction!\n");
@ -1413,7 +1416,8 @@ run_aggregation (void *cls)
transaction to mark all* of the selected deposits as minor! */ transaction to mark all* of the selected deposits as minor! */
if (GNUNET_OK != if (GNUNET_OK !=
db_plugin->start (db_plugin->cls, db_plugin->start (db_plugin->cls,
session)) session,
"aggregator mark tiny transactions"))
{ {
GNUNET_log (GNUNET_ERROR_TYPE_ERROR, GNUNET_log (GNUNET_ERROR_TYPE_ERROR,
"Failed to start database transaction!\n"); "Failed to start database transaction!\n");
@ -1767,9 +1771,12 @@ run_transfers (void *cls)
GNUNET_SCHEDULER_shutdown (); GNUNET_SCHEDULER_shutdown ();
return; return;
} }
db_plugin->preflight (db_plugin->cls,
session);
if (GNUNET_OK != if (GNUNET_OK !=
db_plugin->start (db_plugin->cls, db_plugin->start (db_plugin->cls,
session)) session,
"aggregator run transfer"))
{ {
GNUNET_log (GNUNET_ERROR_TYPE_ERROR, GNUNET_log (GNUNET_ERROR_TYPE_ERROR,
"Failed to start database transaction!\n"); "Failed to start database transaction!\n");

View File

@ -41,6 +41,7 @@
* errors, generates an error message for @a connection. * errors, generates an error message for @a connection.
* *
* @param connection MHD connection to run @a cb for * @param connection MHD connection to run @a cb for
* @param name name of the transaction (for debugging)
* @param[out] set to MHD response code, if transaction failed * @param[out] set to MHD response code, if transaction failed
* @param cb callback implementing transaction logic * @param cb callback implementing transaction logic
* @param cb_cls closure for @a cb, must be read-only! * @param cb_cls closure for @a cb, must be read-only!
@ -48,6 +49,7 @@
*/ */
int int
TEH_DB_run_transaction (struct MHD_Connection *connection, TEH_DB_run_transaction (struct MHD_Connection *connection,
const char *name,
int *mhd_ret, int *mhd_ret,
TEH_DB_TransactionCallback cb, TEH_DB_TransactionCallback cb,
void *cb_cls) void *cb_cls)
@ -64,13 +66,16 @@ TEH_DB_run_transaction (struct MHD_Connection *connection,
TALER_EC_DB_SETUP_FAILED); TALER_EC_DB_SETUP_FAILED);
return GNUNET_SYSERR; return GNUNET_SYSERR;
} }
TEH_plugin->preflight (TEH_plugin->cls,
session);
for (unsigned int retries = 0;retries < MAX_TRANSACTION_COMMIT_RETRIES; retries++) for (unsigned int retries = 0;retries < MAX_TRANSACTION_COMMIT_RETRIES; retries++)
{ {
enum GNUNET_DB_QueryStatus qs; enum GNUNET_DB_QueryStatus qs;
if (GNUNET_OK != if (GNUNET_OK !=
TEH_plugin->start (TEH_plugin->cls, TEH_plugin->start (TEH_plugin->cls,
session)) session,
name))
{ {
GNUNET_break (0); GNUNET_break (0);
if (NULL != mhd_ret) if (NULL != mhd_ret)

View File

@ -54,6 +54,7 @@ typedef enum GNUNET_DB_QueryStatus
* errors, generates an error message for @a connection. * errors, generates an error message for @a connection.
* *
* @param connection MHD connection to run @a cb for * @param connection MHD connection to run @a cb for
* @param name name of the transaction (for debugging)
* @param[out] set to MHD response code, if transaction failed * @param[out] set to MHD response code, if transaction failed
* @param cb callback implementing transaction logic * @param cb callback implementing transaction logic
* @param cb_cls closure for @a cb, must be read-only! * @param cb_cls closure for @a cb, must be read-only!
@ -61,6 +62,7 @@ typedef enum GNUNET_DB_QueryStatus
*/ */
int int
TEH_DB_run_transaction (struct MHD_Connection *connection, TEH_DB_run_transaction (struct MHD_Connection *connection,
const char *name,
int *mhd_ret, int *mhd_ret,
TEH_DB_TransactionCallback cb, TEH_DB_TransactionCallback cb,
void *cb_cls); void *cb_cls);

View File

@ -289,6 +289,7 @@ verify_and_execute_deposit (struct MHD_Connection *connection,
dc.deposit = deposit; dc.deposit = deposit;
if (GNUNET_OK != if (GNUNET_OK !=
TEH_DB_run_transaction (connection, TEH_DB_run_transaction (connection,
"execute deposit",
&mhd_ret, &mhd_ret,
&deposit_transaction, &deposit_transaction,
&dc)) &dc))

View File

@ -711,6 +711,7 @@ reload_keys_denom_iter (void *cls,
arc.revocation_master_sig = revocation_master_sig; arc.revocation_master_sig = revocation_master_sig;
if (GNUNET_OK != if (GNUNET_OK !=
TEH_DB_run_transaction (NULL, TEH_DB_run_transaction (NULL,
"add denomination key revocations",
NULL, NULL,
&add_revocations_transaction, &add_revocations_transaction,
&arc)) &arc))
@ -739,6 +740,7 @@ reload_keys_denom_iter (void *cls,
if (GNUNET_OK != if (GNUNET_OK !=
TEH_DB_run_transaction (NULL, TEH_DB_run_transaction (NULL,
"add denomination key",
NULL, NULL,
&add_denomination_transaction, &add_denomination_transaction,
(void *) dki)) (void *) dki))

View File

@ -396,6 +396,7 @@ verify_and_execute_payback (struct MHD_Connection *connection,
pc.coin = coin; pc.coin = coin;
if (GNUNET_OK != if (GNUNET_OK !=
TEH_DB_run_transaction (connection, TEH_DB_run_transaction (connection,
"run payback",
&mhd_ret, &mhd_ret,
&payback_transaction, &payback_transaction,
&pc)) &pc))

View File

@ -203,6 +203,7 @@ TEH_REFRESH_handler_refresh_link (struct TEH_RequestHandler *rh,
ctx.mlist = json_array (); ctx.mlist = json_array ();
if (GNUNET_OK != if (GNUNET_OK !=
TEH_DB_run_transaction (connection, TEH_DB_run_transaction (connection,
"run link",
&mhd_ret, &mhd_ret,
&refresh_link_transaction, &refresh_link_transaction,
&ctx)) &ctx))

View File

@ -364,6 +364,7 @@ handle_refresh_melt (struct MHD_Connection *connection,
if (GNUNET_OK != if (GNUNET_OK !=
TEH_DB_run_transaction (connection, TEH_DB_run_transaction (connection,
"run melt",
&mhd_ret, &mhd_ret,
&refresh_melt_transaction, &refresh_melt_transaction,
rmc)) rmc))

View File

@ -587,6 +587,7 @@ handle_refresh_reveal_json (struct MHD_Connection *connection,
/* do transactional work */ /* do transactional work */
if (GNUNET_OK == if (GNUNET_OK ==
TEH_DB_run_transaction (connection, TEH_DB_run_transaction (connection,
"run reveal",
&res, &res,
&refresh_reveal_transaction, &refresh_reveal_transaction,
rctx)) rctx))

View File

@ -452,6 +452,7 @@ verify_and_execute_refund (struct MHD_Connection *connection,
} }
if (GNUNET_OK != if (GNUNET_OK !=
TEH_DB_run_transaction (connection, TEH_DB_run_transaction (connection,
"run refund",
&mhd_ret, &mhd_ret,
&refund_transaction, &refund_transaction,
(void *) refund)) (void *) refund))

View File

@ -144,6 +144,7 @@ TEH_RESERVE_handler_reserve_status (struct TEH_RequestHandler *rh,
rsc.rh = NULL; rsc.rh = NULL;
if (GNUNET_OK != if (GNUNET_OK !=
TEH_DB_run_transaction (connection, TEH_DB_run_transaction (connection,
"get reserve status",
&mhd_ret, &mhd_ret,
&reserve_status_transaction, &reserve_status_transaction,
&rsc)) &rsc))

View File

@ -492,6 +492,7 @@ TEH_RESERVE_handler_reserve_withdraw (struct TEH_RequestHandler *rh,
if (GNUNET_OK != if (GNUNET_OK !=
TEH_DB_run_transaction (connection, TEH_DB_run_transaction (connection,
"run reserve withdraw",
&mhd_ret, &mhd_ret,
&withdraw_transaction, &withdraw_transaction,
&wc)) &wc))

View File

@ -292,6 +292,7 @@ check_and_handle_track_transaction_request (struct MHD_Connection *connection,
if (GNUNET_OK != if (GNUNET_OK !=
TEH_DB_run_transaction (connection, TEH_DB_run_transaction (connection,
"handle track transaction",
&mhd_ret, &mhd_ret,
&track_transaction_transaction, &track_transaction_transaction,
&ctx)) &ctx))

View File

@ -474,6 +474,7 @@ TEH_TRACKING_handler_track_transfer (struct TEH_RequestHandler *rh,
return MHD_YES; /* parse error */ return MHD_YES; /* parse error */
if (GNUNET_OK != if (GNUNET_OK !=
TEH_DB_run_transaction (connection, TEH_DB_run_transaction (connection,
"run track transfer",
&mhd_ret, &mhd_ret,
&track_transfer_transaction, &track_transfer_transaction,
&ctx)) &ctx))

View File

@ -422,9 +422,12 @@ find_transfers (void *cls)
GNUNET_SCHEDULER_shutdown (); GNUNET_SCHEDULER_shutdown ();
return; return;
} }
db_plugin->preflight (db_plugin->cls,
session);
if (GNUNET_OK != if (GNUNET_OK !=
db_plugin->start (db_plugin->cls, db_plugin->start (db_plugin->cls,
session)) session,
"wirewatch check for incoming wire transfers"))
{ {
GNUNET_log (GNUNET_ERROR_TYPE_ERROR, GNUNET_log (GNUNET_ERROR_TYPE_ERROR,
"Failed to start database transaction!\n"); "Failed to start database transaction!\n");

View File

@ -447,7 +447,8 @@ do_deposit (struct Command *cmd)
/* finally, actually perform the DB operation */ /* finally, actually perform the DB operation */
if ( (GNUNET_OK != if ( (GNUNET_OK !=
plugin->start (plugin->cls, plugin->start (plugin->cls,
session)) || session,
"aggregator-test-1")) ||
(GNUNET_OK != (GNUNET_OK !=
plugin->insert_deposit (plugin->cls, plugin->insert_deposit (plugin->cls,
session, session,
@ -1150,7 +1151,8 @@ run (void *cls)
&issue.properties.denom_hash); &issue.properties.denom_hash);
if ( (GNUNET_OK != if ( (GNUNET_OK !=
plugin->start (plugin->cls, plugin->start (plugin->cls,
session)) || session,
"aggregator-test-2")) ||
(GNUNET_DB_STATUS_SUCCESS_ONE_RESULT != (GNUNET_DB_STATUS_SUCCESS_ONE_RESULT !=
plugin->insert_denomination_info (plugin->cls, plugin->insert_denomination_info (plugin->cls,
session, session,

View File

@ -1159,7 +1159,8 @@ interpret (struct PERF_TALER_EXCHANGEDB_interpreter_state *state)
case PERF_TALER_EXCHANGEDB_CMD_START_TRANSACTION: case PERF_TALER_EXCHANGEDB_CMD_START_TRANSACTION:
GNUNET_break (GNUNET_OK == GNUNET_break (GNUNET_OK ==
state->plugin->start (state->plugin->cls, state->plugin->start (state->plugin->cls,
state->session)); state->session,
"perf-interpreter"));
break; break;
case PERF_TALER_EXCHANGEDB_CMD_COMMIT_TRANSACTION: case PERF_TALER_EXCHANGEDB_CMD_COMMIT_TRANSACTION:

View File

@ -1,6 +1,6 @@
/* /*
This file is part of TALER This file is part of TALER
Copyright (C) 2014, 2015, 2016, 2017 GNUnet e.V. Copyright (C) 2014, 2015, 2016, 2017, 2018 GNUnet e.V.
TALER is free software; you can redistribute it and/or modify it under the TALER is free software; you can redistribute it and/or modify it under the
terms of the GNU General Public License as published by the Free Software terms of the GNU General Public License as published by the Free Software
@ -60,6 +60,11 @@ struct TALER_EXCHANGEDB_Session
*/ */
PGconn *conn; PGconn *conn;
/**
* Name of the current transaction, for debugging.
*/
const char *transaction_name;
}; };
@ -1533,11 +1538,14 @@ postgres_get_session (void *cls)
* *
* @param cls the `struct PostgresClosure` with the plugin-specific state * @param cls the `struct PostgresClosure` with the plugin-specific state
* @param session the database connection * @param session the database connection
* @param name unique name identifying the transaction (for debugging)
* must point to a constant
* @return #GNUNET_OK on success * @return #GNUNET_OK on success
*/ */
static int static int
postgres_start (void *cls, postgres_start (void *cls,
struct TALER_EXCHANGEDB_Session *session) struct TALER_EXCHANGEDB_Session *session,
const char *name)
{ {
PGresult *result; PGresult *result;
ExecStatusType ex; ExecStatusType ex;
@ -1552,9 +1560,11 @@ postgres_start (void *cls,
PQerrorMessage (session->conn)); PQerrorMessage (session->conn));
GNUNET_break (0); GNUNET_break (0);
PQclear (result); PQclear (result);
session->transaction_name = NULL;
return GNUNET_SYSERR; return GNUNET_SYSERR;
} }
PQclear (result); PQclear (result);
session->transaction_name = name;
return GNUNET_OK; return GNUNET_OK;
} }
@ -1577,6 +1587,7 @@ postgres_rollback (void *cls,
GNUNET_break (PGRES_COMMAND_OK == GNUNET_break (PGRES_COMMAND_OK ==
PQresultStatus (result)); PQresultStatus (result));
PQclear (result); PQclear (result);
session->transaction_name = NULL;
} }
@ -1594,10 +1605,50 @@ postgres_commit (void *cls,
struct GNUNET_PQ_QueryParam params[] = { struct GNUNET_PQ_QueryParam params[] = {
GNUNET_PQ_query_param_end GNUNET_PQ_query_param_end
}; };
enum GNUNET_DB_QueryStatus qs;
return GNUNET_PQ_eval_prepared_non_select (session->conn, qs = GNUNET_PQ_eval_prepared_non_select (session->conn,
"do_commit", "do_commit",
params); params);
session->transaction_name = NULL;
return qs;
}
/**
* Do a pre-flight check that we are not in an uncommitted transaction.
* If we are, try to commit the previous transaction and output a warning.
* Does not return anything, as we will continue regardless of the outcome.
*
* @param cls the `struct PostgresClosure` with the plugin-specific state
* @param session the database connection
*/
static void
postgres_preflight (void *cls,
struct TALER_EXCHANGEDB_Session *session)
{
PGresult *result;
ExecStatusType status;
if (NULL == session->transaction_name)
return; /* all good */
result = PQexec (session->conn,
"COMMIT");
status = PQresultStatus (result);
if (PGRES_COMMAND_OK == status)
{
GNUNET_log (GNUNET_ERROR_TYPE_ERROR,
"BUG: Preflight check committed transaction `%s'!\n",
session->transaction_name);
}
else
{
GNUNET_log (GNUNET_ERROR_TYPE_ERROR,
"BUG: Preflight check failed to commit transaction `%s'!\n",
session->transaction_name);
}
session->transaction_name = NULL;
PQclear (result);
} }
@ -6363,6 +6414,7 @@ libtaler_plugin_exchangedb_postgres_init (void *cls)
plugin->create_tables = &postgres_create_tables; plugin->create_tables = &postgres_create_tables;
plugin->start = &postgres_start; plugin->start = &postgres_start;
plugin->commit = &postgres_commit; plugin->commit = &postgres_commit;
plugin->preflight = &postgres_preflight;
plugin->rollback = &postgres_rollback; plugin->rollback = &postgres_rollback;
plugin->insert_denomination_info = &postgres_insert_denomination_info; plugin->insert_denomination_info = &postgres_insert_denomination_info;
plugin->get_denomination_info = &postgres_get_denomination_info; plugin->get_denomination_info = &postgres_get_denomination_info;

View File

@ -1545,7 +1545,8 @@ run (void *cls)
FAILIF (GNUNET_OK != FAILIF (GNUNET_OK !=
plugin->start (plugin->cls, plugin->start (plugin->cls,
session)); session,
"test-1"));
/* test DB is empty */ /* test DB is empty */
FAILIF (GNUNET_DB_STATUS_SUCCESS_NO_RESULTS != FAILIF (GNUNET_DB_STATUS_SUCCESS_NO_RESULTS !=
@ -1909,7 +1910,8 @@ run (void *cls)
session)); session));
FAILIF (GNUNET_OK != FAILIF (GNUNET_OK !=
plugin->start (plugin->cls, plugin->start (plugin->cls,
session)); session,
"test-2"));
FAILIF (GNUNET_DB_STATUS_SUCCESS_ONE_RESULT != FAILIF (GNUNET_DB_STATUS_SUCCESS_ONE_RESULT !=
plugin->mark_deposit_tiny (plugin->cls, plugin->mark_deposit_tiny (plugin->cls,
session, session,
@ -1928,7 +1930,8 @@ run (void *cls)
&deposit)); &deposit));
FAILIF (GNUNET_OK != FAILIF (GNUNET_OK !=
plugin->start (plugin->cls, plugin->start (plugin->cls,
session)); session,
"test-3"));
FAILIF (GNUNET_DB_STATUS_SUCCESS_NO_RESULTS != FAILIF (GNUNET_DB_STATUS_SUCCESS_NO_RESULTS !=
plugin->test_deposit_done (plugin->cls, plugin->test_deposit_done (plugin->cls,
session, session,
@ -1992,9 +1995,12 @@ run (void *cls)
FAILIF (GNUNET_DB_STATUS_SUCCESS_NO_RESULTS != FAILIF (GNUNET_DB_STATUS_SUCCESS_NO_RESULTS !=
plugin->commit (plugin->cls, plugin->commit (plugin->cls,
session)); session));
plugin->preflight (plugin->cls,
session);
FAILIF (GNUNET_OK != FAILIF (GNUNET_OK !=
plugin->start (plugin->cls, plugin->start (plugin->cls,
session)); session,
"test-4"));
FAILIF (GNUNET_DB_STATUS_SUCCESS_NO_RESULTS != FAILIF (GNUNET_DB_STATUS_SUCCESS_NO_RESULTS !=
plugin->insert_denomination_revocation (plugin->cls, plugin->insert_denomination_revocation (plugin->cls,
session, session,
@ -2002,9 +2008,12 @@ run (void *cls)
&master_sig)); &master_sig));
plugin->rollback (plugin->cls, plugin->rollback (plugin->cls,
session); session);
plugin->preflight (plugin->cls,
session);
FAILIF (GNUNET_OK != FAILIF (GNUNET_OK !=
plugin->start (plugin->cls, plugin->start (plugin->cls,
session)); session,
"test-5"));
{ {
struct TALER_MasterSignatureP msig; struct TALER_MasterSignatureP msig;
uint64_t rev_rowid; uint64_t rev_rowid;
@ -2164,6 +2173,8 @@ run (void *cls)
FAILIF (GNUNET_OK != FAILIF (GNUNET_OK !=
test_wire_fees (session)); test_wire_fees (session));
plugin->preflight (plugin->cls,
session);
result = 0; result = 0;

View File

@ -1152,11 +1152,14 @@ struct TALER_EXCHANGEDB_Plugin
* *
* @param cls the @e cls of this struct with the plugin-specific state * @param cls the @e cls of this struct with the plugin-specific state
* @param session connection to use * @param session connection to use
* @param name unique name identifying the transaction (for debugging),
* must point to a constant
* @return #GNUNET_OK on success * @return #GNUNET_OK on success
*/ */
int int
(*start) (void *cls, (*start) (void *cls,
struct TALER_EXCHANGEDB_Session *session); struct TALER_EXCHANGEDB_Session *session,
const char *name);
/** /**
@ -1171,6 +1174,19 @@ struct TALER_EXCHANGEDB_Plugin
struct TALER_EXCHANGEDB_Session *session); struct TALER_EXCHANGEDB_Session *session);
/**
* Do a pre-flight check that we are not in an uncommitted transaction.
* If we are, try to commit the previous transaction and output a warning.
* Does not return anything, as we will continue regardless of the outcome.
*
* @param cls the `struct PostgresClosure` with the plugin-specific state
* @param session the database connection
*/
void
(*preflight) (void *cls,
struct TALER_EXCHANGEDB_Session *session);
/** /**
* Abort/rollback a transaction. * Abort/rollback a transaction.
* *