From 685837ad280c83abae5f8a14f91c5968b720df8c Mon Sep 17 00:00:00 2001 From: Christian Grothoff Date: Sat, 19 Mar 2022 15:25:43 +0100 Subject: [PATCH] simplify auditor_api_handle.c, do not modify global context with respect to 'Expect' header --- src/lib/auditor_api_deposit_confirmation.c | 4 + src/lib/auditor_api_exchanges.c | 17 -- src/lib/auditor_api_handle.c | 203 ++++++++------------- 3 files changed, 77 insertions(+), 147 deletions(-) diff --git a/src/lib/auditor_api_deposit_confirmation.c b/src/lib/auditor_api_deposit_confirmation.c index 18155fe2c..2c5bd2b41 100644 --- a/src/lib/auditor_api_deposit_confirmation.c +++ b/src/lib/auditor_api_deposit_confirmation.c @@ -359,6 +359,10 @@ TALER_AUDITOR_deposit_confirmation ( dh->ctx.headers, &handle_deposit_confirmation_finished, dh); + /* Disable 100 continue processing */ + GNUNET_CURL_extend_headers (dh->job, + curl_slist_append (NULL, + "Expect:")); return dh; } diff --git a/src/lib/auditor_api_exchanges.c b/src/lib/auditor_api_exchanges.c index 136212258..7327f11b2 100644 --- a/src/lib/auditor_api_exchanges.c +++ b/src/lib/auditor_api_exchanges.c @@ -193,17 +193,6 @@ handle_exchanges_finished (void *cls, } -/** - * Submit an /exchanges request to the auditor and get the - * auditor's response. If the auditor's reply is not - * well-formed, we return an HTTP status code of zero to @a cb. - * - * @param auditor the auditor handle; the auditor must be ready to operate - * @param cb the callback to call when a reply for this request is available - * @param cb_cls closure for the above callback - * @return a handle for this request; NULL if the inputs are invalid (i.e. - * signatures fail to verify). In this case, the callback is not called. - */ struct TALER_AUDITOR_ListExchangesHandle * TALER_AUDITOR_list_exchanges (struct TALER_AUDITOR_Handle *auditor, TALER_AUDITOR_ListExchangesResultCallback cb, @@ -247,12 +236,6 @@ TALER_AUDITOR_list_exchanges (struct TALER_AUDITOR_Handle *auditor, } -/** - * Cancel a list exchanges request. This function cannot be used - * on a request handle if a response is already served for it. - * - * @param leh the list exchanges request handle - */ void TALER_AUDITOR_list_exchanges_cancel ( struct TALER_AUDITOR_ListExchangesHandle *leh) diff --git a/src/lib/auditor_api_handle.c b/src/lib/auditor_api_handle.c index a5478ef9b..9edb1115e 100644 --- a/src/lib/auditor_api_handle.c +++ b/src/lib/auditor_api_handle.c @@ -1,6 +1,6 @@ /* This file is part of TALER - Copyright (C) 2014-2020 Taler Systems SA + Copyright (C) 2014-2022 Taler Systems SA 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 @@ -75,12 +75,6 @@ enum AuditorHandleState }; -/** - * Data for the request to get the /version of a auditor. - */ -struct VersionRequest; - - /** * Handle to the auditor */ @@ -111,7 +105,12 @@ struct TALER_AUDITOR_Handle * Data for the request to get the /version of a auditor, * NULL once we are past stage #MHS_INIT. */ - struct VersionRequest *vr; + struct GNUNET_CURL_Job *vr; + + /** + * The url for the @e vr job. + */ + char *vr_url; /** * Task for retrying /version request. @@ -144,50 +143,12 @@ struct TALER_AUDITOR_Handle /* ***************** Internal /version fetching ************* */ -/** - * Data for the request to get the /version of a auditor. - */ -struct VersionRequest -{ - /** - * The connection to auditor this request handle will use - */ - struct TALER_AUDITOR_Handle *auditor; - - /** - * The url for this handle - */ - char *url; - - /** - * Entry for this request with the `struct GNUNET_CURL_Context`. - */ - struct GNUNET_CURL_Job *job; - -}; - - -/** - * Release memory occupied by a version request. - * Note that this does not cancel the request - * itself. - * - * @param vr request to free - */ -static void -free_version_request (struct VersionRequest *vr) -{ - GNUNET_free (vr->url); - GNUNET_free (vr); -} - - /** * Decode the JSON in @a resp_obj from the /version response and store the data * in the @a key_data. * * @param[in] resp_obj JSON object to parse - * @param[out] auditor where to store the results we decoded + * @param[in,out] auditor where to store the results we decoded * @param[out] vc where to store version compatibility data * @return #TALER_EC_NONE on success */ @@ -200,6 +161,7 @@ decode_version_json (const json_t *resp_obj, unsigned int age; unsigned int revision; unsigned int current; + char dummy; const char *ver; struct GNUNET_JSON_Specification spec[] = { GNUNET_JSON_spec_string ("version", @@ -224,14 +186,16 @@ decode_version_json (const json_t *resp_obj, return TALER_EC_GENERIC_JSON_INVALID; } if (3 != sscanf (ver, - "%u:%u:%u", + "%u:%u:%u%c", ¤t, &revision, - &age)) + &age, + &dummy)) { GNUNET_break_op (0); return TALER_EC_GENERIC_VERSION_MALFORMED; } + GNUNET_free (auditor->version); auditor->version = GNUNET_strdup (ver); vi->version = auditor->version; *vc = TALER_AUDITOR_VC_MATCH; @@ -264,7 +228,7 @@ request_version (void *cls); * Callback used when downloading the reply to a /version request * is complete. * - * @param cls the `struct VersionRequest` + * @param cls the `struct TALER_AUDITOR_Handle` * @param response_code HTTP response code, 0 on error * @param gresp_obj parsed JSON result, NULL on error, must be a `const json_t *` */ @@ -273,21 +237,19 @@ version_completed_cb (void *cls, long response_code, const void *gresp_obj) { + struct TALER_AUDITOR_Handle *auditor = cls; const json_t *resp_obj = gresp_obj; - struct VersionRequest *vr = cls; - struct TALER_AUDITOR_Handle *auditor = vr->auditor; enum TALER_AUDITOR_VersionCompatibility vc; struct TALER_AUDITOR_HttpResponse hr = { .reply = resp_obj, .http_status = (unsigned int) response_code }; + auditor->vr = NULL; GNUNET_log (GNUNET_ERROR_TYPE_INFO, "Received version from URL `%s' with status %ld.\n", - vr->url, + auditor->url, response_code); - free_version_request (vr); - auditor->vr = NULL; vc = TALER_AUDITOR_VC_PROTOCOL_ERROR; switch (response_code) { @@ -357,6 +319,43 @@ version_completed_cb (void *cls, } +/** + * Initiate download of /version from the auditor. + * + * @param cls auditor where to download /version from + */ +static void +request_version (void *cls) +{ + struct TALER_AUDITOR_Handle *auditor = cls; + CURL *eh; + + auditor->retry_task = NULL; + GNUNET_assert (NULL == auditor->vr); + GNUNET_log (GNUNET_ERROR_TYPE_INFO, + "Requesting auditor version with URL `%s'.\n", + auditor->vr_url); + eh = TALER_AUDITOR_curl_easy_get_ (auditor->vr_url); + if (NULL == eh) + { + GNUNET_break (0); + auditor->retry_delay = EXCHANGE_LIB_BACKOFF (auditor->retry_delay); + auditor->retry_task = GNUNET_SCHEDULER_add_delayed (auditor->retry_delay, + &request_version, + auditor); + return; + } + GNUNET_break (CURLE_OK == + curl_easy_setopt (eh, + CURLOPT_TIMEOUT, + (long) 300)); + auditor->vr = GNUNET_CURL_job_add (auditor->ctx, + eh, + &version_completed_cb, + auditor); +} + + /* ********************* library internal API ********* */ @@ -371,8 +370,7 @@ enum GNUNET_GenericReturnValue TALER_AUDITOR_handle_is_ready_ (struct TALER_AUDITOR_Handle *h) { GNUNET_log (GNUNET_ERROR_TYPE_INFO, - "Checking if auditor %p (%s) is now ready: %s\n", - h, + "Checking if auditor at `%s` is now ready: %s\n", h->url, (MHD_VERSION == h->state) ? "yes" : "no"); return (MHS_VERSION == h->state) ? GNUNET_YES : GNUNET_NO; @@ -401,103 +399,48 @@ TALER_AUDITOR_connect (struct GNUNET_CURL_Context *ctx, { struct TALER_AUDITOR_Handle *auditor; - /* Disable 100 continue processing */ - GNUNET_break (GNUNET_OK == - GNUNET_CURL_append_header (ctx, - "Expect:")); auditor = GNUNET_new (struct TALER_AUDITOR_Handle); + auditor->version_cb = version_cb; + auditor->version_cb_cls = version_cb_cls; auditor->retry_delay = GNUNET_TIME_UNIT_SECONDS; /* start slowly */ auditor->ctx = ctx; auditor->url = GNUNET_strdup (url); - auditor->version_cb = version_cb; - auditor->version_cb_cls = version_cb_cls; + auditor->vr_url = TALER_AUDITOR_path_to_url_ (auditor, + "/version"); + if (NULL == auditor->vr_url) + { + GNUNET_break (0); + GNUNET_free (auditor->url); + GNUNET_free (auditor); + return NULL; + } auditor->retry_task = GNUNET_SCHEDULER_add_now (&request_version, auditor); GNUNET_log (GNUNET_ERROR_TYPE_INFO, - "Connecting to auditor at URL `%s' (%p).\n", - url, - auditor); + "Connecting to auditor at URL `%s'.\n", + url); return auditor; } -/** - * Initiate download of /version from the auditor. - * - * @param cls auditor where to download /version from - */ -static void -request_version (void *cls) -{ - struct TALER_AUDITOR_Handle *auditor = cls; - struct VersionRequest *vr; - CURL *eh; - - auditor->retry_task = NULL; - GNUNET_assert (NULL == auditor->vr); - vr = GNUNET_new (struct VersionRequest); - vr->auditor = auditor; - vr->url = TALER_AUDITOR_path_to_url_ (auditor, - "/version"); - if (NULL == vr->url) - { - struct TALER_AUDITOR_HttpResponse hr = { - .ec = TALER_EC_GENERIC_CONFIGURATION_INVALID - }; - - auditor->version_cb (auditor->version_cb_cls, - &hr, - NULL, - TALER_AUDITOR_VC_PROTOCOL_ERROR); - GNUNET_free (vr); - return; - } - GNUNET_log (GNUNET_ERROR_TYPE_INFO, - "Requesting auditor version with URL `%s'.\n", - vr->url); - eh = TALER_AUDITOR_curl_easy_get_ (vr->url); - if (NULL == eh) - { - GNUNET_break (0); - auditor->retry_delay = EXCHANGE_LIB_BACKOFF (auditor->retry_delay); - auditor->retry_task = GNUNET_SCHEDULER_add_delayed (auditor->retry_delay, - &request_version, - auditor); - GNUNET_free (vr->url); - GNUNET_free (vr); - return; - } - GNUNET_break (CURLE_OK == - curl_easy_setopt (eh, - CURLOPT_TIMEOUT, - (long) 300)); - vr->job = GNUNET_CURL_job_add (auditor->ctx, - eh, - &version_completed_cb, - vr); - auditor->vr = vr; -} - - void TALER_AUDITOR_disconnect (struct TALER_AUDITOR_Handle *auditor) { GNUNET_log (GNUNET_ERROR_TYPE_INFO, - "Disconnecting from auditor at URL `%s' (%p).\n", - auditor->url, - auditor); + "Disconnecting from auditor at URL `%s'.\n", + auditor->url); if (NULL != auditor->vr) { - GNUNET_CURL_job_cancel (auditor->vr->job); - free_version_request (auditor->vr); + GNUNET_CURL_job_cancel (auditor->vr); auditor->vr = NULL; } - GNUNET_free (auditor->version); if (NULL != auditor->retry_task) { GNUNET_SCHEDULER_cancel (auditor->retry_task); auditor->retry_task = NULL; } + GNUNET_free (auditor->version); + GNUNET_free (auditor->vr_url); GNUNET_free (auditor->url); GNUNET_free (auditor); }