/* This file is part of TALER Copyright (C) 2023 Taler Systems SA TALER is free software; you can redistribute it and/or modify it under the terms of the GNU Affero General Public License as published by the Free Software Foundation; either version 3, or (at your option) any later version. TALER is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Affero General Public License for more details. You should have received a copy of the GNU Affero General Public License along with TALER; see the file COPYING. If not, see */ /** * @file taler-merchant-wirewatch.c * @brief Process that imports information about incoming bank transfers into the merchant backend * @author Christian Grothoff */ #include "platform.h" #include #include #include #include #include "taler_merchant_bank_lib.h" #include "taler_merchantdb_lib.h" #include "taler_merchantdb_plugin.h" /** * Timeout for the bank interaction. Rather long as we should do long-polling * and do not want to wake up too often. */ #define BANK_TIMEOUT GNUNET_TIME_relative_multiply (GNUNET_TIME_UNIT_MINUTES, \ 5) /** * Information about a watch job. */ struct Watch { /** * Kept in a DLL. */ struct Watch *next; /** * Kept in a DLL. */ struct Watch *prev; /** * Next task to run, if any. */ struct GNUNET_SCHEDULER_Task *task; /** * Dynamically adjusted long polling time-out. */ struct GNUNET_TIME_Relative bank_timeout; /** * For which instance are we importing bank transfers? */ char *instance_id; /** * For which account are we importing bank transfers? */ struct TALER_FullPayto payto_uri; /** * Bank history request. */ struct TALER_MERCHANT_BANK_CreditHistoryHandle *hh; /** * Start row for the bank interaction. Exclusive. */ uint64_t start_row; /** * Artificial delay to use between API calls. Used to * throttle on failures. */ struct GNUNET_TIME_Relative delay; /** * When did we start our last HTTP request? */ struct GNUNET_TIME_Absolute start_time; /** * How long should long-polling take at least? */ struct GNUNET_TIME_Absolute long_poll_timeout; /** * Login data for the bank. */ struct TALER_MERCHANT_BANK_AuthenticationData ad; /** * Set to true if we found a transaction in the last iteration. */ bool found; }; /** * Head of active watches. */ static struct Watch *w_head; /** * Tail of active watches. */ static struct Watch *w_tail; /** * The merchant's configuration. */ static const struct GNUNET_CONFIGURATION_Handle *cfg; /** * Our database plugin. */ static struct TALER_MERCHANTDB_Plugin *db_plugin; /** * Handle to the context for interacting with the bank. */ static struct GNUNET_CURL_Context *ctx; /** * Scheduler context for running the @e ctx. */ static struct GNUNET_CURL_RescheduleContext *rc; /** * Event handler to learn that the configuration changed * and we should shutdown (to be restarted). */ static struct GNUNET_DB_EventHandler *eh; /** * Value to return from main(). 0 on success, non-zero on errors. */ static int global_ret; /** * How many transactions should we fetch at most per batch? */ static unsigned int batch_size = 32; /** * #GNUNET_YES if we are in test mode and should exit when idle. */ static int test_mode; /** * #GNUNET_YES if we are in persistent mode and do * not exit on #config_changed. */ static int persist_mode; /** * Set to true if we are shutting down due to a * configuration change. */ static bool config_changed_flag; /** * Save progress in DB. */ static void save (struct Watch *w) { enum GNUNET_DB_QueryStatus qs; qs = db_plugin->update_wirewatch_progress (db_plugin->cls, w->instance_id, w->payto_uri, w->start_row); if (qs < 0) { GNUNET_log (GNUNET_ERROR_TYPE_WARNING, "Failed to persist wirewatch progress for %s/%s (%d)\n", w->instance_id, w->payto_uri.full_payto, qs); GNUNET_SCHEDULER_shutdown (); global_ret = EXIT_FAILURE; } } /** * Free resources of @a w. * * @param w watch job to terminate */ static void end_watch (struct Watch *w) { if (NULL != w->task) { GNUNET_SCHEDULER_cancel (w->task); w->task = NULL; } if (NULL != w->hh) { TALER_MERCHANT_BANK_credit_history_cancel (w->hh); w->hh = NULL; } GNUNET_free (w->instance_id); GNUNET_free (w->payto_uri.full_payto); TALER_MERCHANT_BANK_auth_free (&w->ad); GNUNET_CONTAINER_DLL_remove (w_head, w_tail, w); GNUNET_free (w); } /** * We're being aborted with CTRL-C (or SIGTERM). Shut down. * * @param cls closure */ static void shutdown_task (void *cls) { (void) cls; GNUNET_log (GNUNET_ERROR_TYPE_INFO, "Running shutdown\n"); while (NULL != w_head) { struct Watch *w = w_head; save (w); end_watch (w); } if (NULL != eh) { db_plugin->event_listen_cancel (eh); eh = NULL; } TALER_MERCHANTDB_plugin_unload (db_plugin); db_plugin = NULL; cfg = NULL; if (NULL != ctx) { GNUNET_CURL_fini (ctx); ctx = NULL; } if (NULL != rc) { GNUNET_CURL_gnunet_rc_destroy (rc); rc = NULL; } } /** * Parse @a subject from wire transfer into @a wtid and @a exchange_url. * * @param subject wire transfer subject to parse; * format is "$WTID $URL" * @param[out] wtid wire transfer ID to extract * @param[out] exchange_url set to exchange URL * @return #GNUNET_OK on success */ static enum GNUNET_GenericReturnValue parse_subject (const char *subject, struct TALER_WireTransferIdentifierRawP *wtid, char **exchange_url) { const char *space; space = strchr (subject, ' '); if (NULL == space) return GNUNET_NO; if (GNUNET_OK != GNUNET_STRINGS_string_to_data (subject, space - subject, wtid, sizeof (*wtid))) return GNUNET_NO; space++; if (! TALER_url_valid_charset (space)) return GNUNET_NO; if ( (0 != strncasecmp ("http://", space, strlen ("http://"))) && (0 != strncasecmp ("https://", space, strlen ("https://"))) ) return GNUNET_NO; *exchange_url = GNUNET_strdup (space); return GNUNET_OK; } /** * Run next iteration. * * @param cls a `struct Watch *` */ static void do_work (void *cls); /** * Callbacks of this type are used to serve the result of asking * the bank for the credit transaction history. * * @param cls a `struct Watch *` * @param http_status HTTP response code, #MHD_HTTP_OK (200) for successful status request * 0 if the bank's reply is bogus (fails to follow the protocol), * #MHD_HTTP_NO_CONTENT if there are no more results; on success the * last callback is always of this status (even if `abs(num_results)` were * already returned). * @param ec detailed error code * @param serial_id monotonically increasing counter corresponding to the transaction * @param details details about the wire transfer * @return #GNUNET_OK to continue, #GNUNET_SYSERR to abort iteration */ static enum GNUNET_GenericReturnValue credit_cb ( void *cls, unsigned int http_status, enum TALER_ErrorCode ec, uint64_t serial_id, const struct TALER_MERCHANT_BANK_CreditDetails *details) { struct Watch *w = cls; switch (http_status) { case 0: GNUNET_log (GNUNET_ERROR_TYPE_WARNING, "Invalid HTTP response (HTTP status: 0, %d) from bank\n", ec); w->delay = GNUNET_TIME_STD_BACKOFF (w->delay); break; case MHD_HTTP_OK: { enum GNUNET_DB_QueryStatus qs; char *exchange_url; struct TALER_WireTransferIdentifierRawP wtid; GNUNET_log (GNUNET_ERROR_TYPE_INFO, "Received wire transfer `%s' over %s\n", details->wire_subject, TALER_amount2s (&details->amount)); w->found = true; if (GNUNET_OK != parse_subject (details->wire_subject, &wtid, &exchange_url)) { GNUNET_log (GNUNET_ERROR_TYPE_INFO, "Skipping transfer %llu (%s): not from exchange\n", (unsigned long long) serial_id, details->wire_subject); w->start_row = serial_id; return GNUNET_OK; } /* FIXME-Performance-Optimization: consider grouping multiple inserts into one bigger transaction with just one notify. */ qs = db_plugin->insert_transfer (db_plugin->cls, w->instance_id, exchange_url, &wtid, &details->amount, details->credit_account_uri, true /* confirmed */); if (GNUNET_DB_STATUS_SUCCESS_NO_RESULTS == qs) { struct TALER_Amount total; struct TALER_Amount wfee; struct TALER_Amount eamount; struct GNUNET_TIME_Timestamp timestamp; bool have_esig; bool verified; qs = db_plugin->lookup_transfer (db_plugin->cls, w->instance_id, exchange_url, &wtid, &total, &wfee, &eamount, ×tamp, &have_esig, &verified); if (GNUNET_DB_STATUS_SUCCESS_NO_RESULTS == qs) { GNUNET_log (GNUNET_ERROR_TYPE_WARNING, "Inserting transfer for %s into database failed. Is the credit account %s configured correctly?\n", w->instance_id, details->credit_account_uri.full_payto); } if (GNUNET_DB_STATUS_SUCCESS_ONE_RESULT == qs) { if (0 != TALER_amount_cmp (&total, &details->amount)) { GNUNET_log (GNUNET_ERROR_TYPE_WARNING, "Inserting transfer for %s into database failed. An entry exists for a different transfer amount (%s)!\n", w->instance_id, TALER_amount2s (&total)); } else { GNUNET_log (GNUNET_ERROR_TYPE_INFO, "Inserting transfer for %s into database failed. An equivalent entry already exists.\n", w->instance_id); } } } GNUNET_free (exchange_url); if (qs < 0) { GNUNET_break (0); GNUNET_SCHEDULER_shutdown (); w->hh = NULL; return GNUNET_SYSERR; } /* Success => reset back-off timer! */ w->delay = GNUNET_TIME_UNIT_ZERO; { struct GNUNET_DB_EventHeaderP es = { .size = htons (sizeof (es)), .type = htons (TALER_DBEVENT_MERCHANT_WIRE_TRANSFER_CONFIRMED) }; db_plugin->event_notify (db_plugin->cls, &es, NULL, 0); } } w->start_row = serial_id; return GNUNET_OK; case MHD_HTTP_NO_CONTENT: save (w); /* Delay artificially if server returned before long-poll timeout */ if (! w->found) w->delay = GNUNET_TIME_absolute_get_remaining (w->long_poll_timeout); break; case MHD_HTTP_NOT_FOUND: /* configuration likely wrong, wait at least 1 minute, backoff up to 15 minutes! */ w->delay = GNUNET_TIME_relative_max (GNUNET_TIME_UNIT_MINUTES, GNUNET_TIME_STD_BACKOFF (w->delay)); GNUNET_log (GNUNET_ERROR_TYPE_WARNING, "Bank claims account is unknown, waiting for %s before trying again\n", GNUNET_TIME_relative2s (w->delay, true)); break; case MHD_HTTP_GATEWAY_TIMEOUT: GNUNET_log (GNUNET_ERROR_TYPE_INFO, "Gateway timeout, adjusting long polling threshold\n"); /* Limit new timeout at request delay */ w->bank_timeout = GNUNET_TIME_relative_min (GNUNET_TIME_absolute_get_duration ( w->start_time), w->bank_timeout); /* set the timeout a bit earlier */ w->bank_timeout = GNUNET_TIME_relative_subtract (w->bank_timeout, GNUNET_TIME_UNIT_SECONDS); /* do not allow it to go to zero */ w->bank_timeout = GNUNET_TIME_relative_max (w->bank_timeout, GNUNET_TIME_UNIT_SECONDS); w->delay = GNUNET_TIME_STD_BACKOFF (w->delay); break; default: /* Something went wrong, try again, but with back-off */ w->delay = GNUNET_TIME_STD_BACKOFF (w->delay); GNUNET_log (GNUNET_ERROR_TYPE_WARNING, "Unexpected HTTP status code %u(%d) from bank\n", http_status, ec); break; } w->hh = NULL; if (test_mode && (! w->found)) { GNUNET_log (GNUNET_ERROR_TYPE_INFO, "No transactions found and in test mode. Ending watch!\n"); end_watch (w); if (NULL == w_head) GNUNET_SCHEDULER_shutdown (); return GNUNET_OK; } w->task = GNUNET_SCHEDULER_add_delayed (w->delay, &do_work, w); return GNUNET_OK; } static void do_work (void *cls) { struct Watch *w = cls; w->task = NULL; w->found = false; w->long_poll_timeout = GNUNET_TIME_relative_to_absolute (w->bank_timeout); w->start_time = GNUNET_TIME_absolute_get (); w->hh = TALER_MERCHANT_BANK_credit_history (ctx, &w->ad, w->start_row, batch_size, test_mode ? GNUNET_TIME_UNIT_ZERO : w->bank_timeout, &credit_cb, w); if (NULL == w->hh) { GNUNET_break (0); GNUNET_SCHEDULER_shutdown (); return; } } /** * Function called with information about a accounts * the wirewatcher should monitor. * * @param cls closure (NULL) * @param instance instance that owns the account * @param payto_uri account URI * @param credit_facade_url URL for the credit facade * @param credit_facade_credentials account access credentials * @param last_serial last transaction serial (inclusive) we have seen from this account */ static void start_watch ( void *cls, const char *instance, struct TALER_FullPayto payto_uri, const char *credit_facade_url, const json_t *credit_facade_credentials, uint64_t last_serial) { struct Watch *w = GNUNET_new (struct Watch); (void) cls; w->bank_timeout = BANK_TIMEOUT; if (GNUNET_OK != TALER_MERCHANT_BANK_auth_parse_json (credit_facade_credentials, credit_facade_url, &w->ad)) { GNUNET_log (GNUNET_ERROR_TYPE_ERROR, "Failed to parse authentication data of `%s/%s'\n", instance, payto_uri.full_payto); GNUNET_free (w); GNUNET_SCHEDULER_shutdown (); global_ret = EXIT_NOTCONFIGURED; return; } GNUNET_CONTAINER_DLL_insert (w_head, w_tail, w); w->instance_id = GNUNET_strdup (instance); w->payto_uri.full_payto = GNUNET_strdup (payto_uri.full_payto); w->start_row = last_serial; w->task = GNUNET_SCHEDULER_add_now (&do_work, w); } /** * Function called on configuration change events received from Postgres. We * shutdown (and systemd should restart us). * * @param cls closure (NULL) * @param extra additional event data provided * @param extra_size number of bytes in @a extra */ static void config_changed (void *cls, const void *extra, size_t extra_size) { (void) cls; (void) extra; (void) extra_size; GNUNET_log (GNUNET_ERROR_TYPE_INFO, "Configuration changed, %s\n", 0 == persist_mode ? "restarting" : "reinitializing"); config_changed_flag = true; GNUNET_SCHEDULER_shutdown (); } /** * First task. * * @param cls closure, NULL * @param args remaining command-line arguments * @param cfgfile name of the configuration file used (for saving, can be NULL!) * @param c configuration */ static void run (void *cls, char *const *args, const char *cfgfile, const struct GNUNET_CONFIGURATION_Handle *c) { (void) args; (void) cfgfile; cfg = c; GNUNET_SCHEDULER_add_shutdown (&shutdown_task, NULL); ctx = GNUNET_CURL_init (&GNUNET_CURL_gnunet_scheduler_reschedule, &rc); rc = GNUNET_CURL_gnunet_rc_create (ctx); if (NULL == ctx) { GNUNET_break (0); GNUNET_SCHEDULER_shutdown (); global_ret = EXIT_FAILURE; return; } if (NULL == (db_plugin = TALER_MERCHANTDB_plugin_load (cfg))) { GNUNET_log (GNUNET_ERROR_TYPE_ERROR, "Failed to initialize DB subsystem\n"); GNUNET_SCHEDULER_shutdown (); global_ret = EXIT_NOTCONFIGURED; return; } if (GNUNET_OK != db_plugin->connect (db_plugin->cls)) { GNUNET_log (GNUNET_ERROR_TYPE_ERROR, "Failed to connect to database. Consider running taler-merchant-dbinit!\n"); GNUNET_SCHEDULER_shutdown (); global_ret = EXIT_FAILURE; return; } { struct GNUNET_DB_EventHeaderP es = { .size = htons (sizeof (es)), .type = htons (TALER_DBEVENT_MERCHANT_ACCOUNTS_CHANGED) }; eh = db_plugin->event_listen (db_plugin->cls, &es, GNUNET_TIME_UNIT_FOREVER_REL, &config_changed, NULL); } { enum GNUNET_DB_QueryStatus qs; qs = db_plugin->select_wirewatch_accounts (db_plugin->cls, &start_watch, NULL); if (qs < 0) { GNUNET_log (GNUNET_ERROR_TYPE_ERROR, "Failed to obtain wirewatch accounts from database\n"); GNUNET_SCHEDULER_shutdown (); global_ret = EXIT_NO_RESTART; return; } if ( (NULL == w_head) && (GNUNET_YES == test_mode) ) { GNUNET_log (GNUNET_ERROR_TYPE_INFO, "No active wirewatch accounts in database and in test mode. Exiting.\n"); GNUNET_SCHEDULER_shutdown (); global_ret = EXIT_SUCCESS; return; } } } /** * The main function of taler-merchant-wirewatch * * @param argc number of arguments from the command line * @param argv command line arguments * @return 0 ok, 1 on error */ int main (int argc, char *const *argv) { struct GNUNET_GETOPT_CommandLineOption options[] = { GNUNET_GETOPT_option_flag ('p', "persist", "run in persist mode and do not exit on configuration changes", &persist_mode), GNUNET_GETOPT_option_timetravel ('T', "timetravel"), GNUNET_GETOPT_option_flag ('t', "test", "run in test mode and exit when idle", &test_mode), GNUNET_GETOPT_option_version (VERSION "-" VCS_VERSION), GNUNET_GETOPT_OPTION_END }; enum GNUNET_GenericReturnValue ret; TALER_OS_init (); do { config_changed_flag = false; ret = GNUNET_PROGRAM_run ( argc, argv, "taler-merchant-wirewatch", gettext_noop ( "background process that watches for incoming wire transfers to the merchant bank account"), options, &run, NULL); } while ( (1 == persist_mode) && config_changed_flag); if (GNUNET_SYSERR == ret) return EXIT_INVALIDARGUMENT; if (GNUNET_NO == ret) return EXIT_SUCCESS; return global_ret; } /* end of taler-exchange-wirewatch.c */