diff options
| m--------- | contrib/gana | 0 | ||||
| -rw-r--r-- | src/exchange/taler-exchange-httpd_reserves_get.c | 311 | ||||
| -rw-r--r-- | src/exchangedb/pg_reserves_in_insert.c | 6 | 
3 files changed, 143 insertions, 174 deletions
| diff --git a/contrib/gana b/contrib/gana -Subproject 1ec4596bf4925ee24fc06d3e74d2a553b823987 +Subproject bf43b20a0362ac19bcf1bab9c33215e55d8d9f3 diff --git a/src/exchange/taler-exchange-httpd_reserves_get.c b/src/exchange/taler-exchange-httpd_reserves_get.c index 88f7aca9..003db926 100644 --- a/src/exchange/taler-exchange-httpd_reserves_get.c +++ b/src/exchange/taler-exchange-httpd_reserves_get.c @@ -1,6 +1,6 @@  /*    This file is part of TALER -  Copyright (C) 2014-2022 Taler Systems SA +  Copyright (C) 2014-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 @@ -63,6 +63,16 @@ struct ReservePoller    struct GNUNET_TIME_Absolute timeout;    /** +   * Public key of the reserve the inquiry is about. +   */ +  struct TALER_ReservePublicKeyP reserve_pub; + +  /** +   * Balance of the reserve, set in the callback. +   */ +  struct TALER_Amount balance; + +  /**     * True if we are still suspended.     */    bool suspended; @@ -84,13 +94,10 @@ static struct ReservePoller *rp_tail;  void  TEH_reserves_get_cleanup ()  { -  struct ReservePoller *rp; - -  while (NULL != (rp = rp_head)) +  for (struct ReservePoller *rp = rp_head; +       NULL != rp; +       rp = rp->next)    { -    GNUNET_CONTAINER_DLL_remove (rp_head, -                                 rp_tail, -                                 rp);      if (rp->suspended)      {        rp->suspended = false; @@ -115,11 +122,14 @@ rp_cleanup (struct TEH_RequestContext *rc)    if (NULL != rp->eh)    {      GNUNET_log (GNUNET_ERROR_TYPE_INFO, -                "Cancelling DB event listening\n"); +                "Cancelling DB event listening on cleanup (odd unless during shutdown)\n");      TEH_plugin->event_listen_cancel (TEH_plugin->cls,                                       rp->eh);      rp->eh = NULL;    } +  GNUNET_CONTAINER_DLL_remove (rp_head, +                               rp_tail, +                               rp);    GNUNET_free (rp);  } @@ -143,20 +153,14 @@ db_event_cb (void *cls,    (void) extra;    (void) extra_size; -  if (NULL == rp) -    return; /* event triggered while main transaction -               was still running */    if (! rp->suspended)      return; /* might get multiple wake-up events */ -  rp->suspended = false;    GNUNET_async_scope_enter (&rc->async_scope_id,                              &old_scope);    GNUNET_log (GNUNET_ERROR_TYPE_INFO,                "Resuming from long-polling on reserve\n");    TEH_check_invariants (); -  GNUNET_CONTAINER_DLL_remove (rp_head, -                               rp_tail, -                               rp); +  rp->suspended = false;    MHD_resume_connection (rp->connection);    TALER_MHD_daemon_trigger ();    TEH_check_invariants (); @@ -164,191 +168,154 @@ db_event_cb (void *cls,  } -/** - * Closure for #reserve_history_transaction. - */ -struct ReserveHistoryContext -{ -  /** -   * Public key of the reserve the inquiry is about. -   */ -  struct TALER_ReservePublicKeyP reserve_pub; - -  /** -   * Balance of the reserve, set in the callback. -   */ -  struct TALER_Amount balance; - -  /** -   * Set to true if we did not find the reserve. -   */ -  bool not_found; -}; - - -/** - * Function implementing /reserves/ GET transaction. - * Execute a /reserves/ GET.  Given the public key of a reserve, - * return the associated transaction history.  Runs the - * transaction logic; IF it returns a non-error code, the transaction - * logic MUST NOT queue a MHD response.  IF it returns an hard error, - * the transaction logic MUST queue a MHD response and set @a mhd_ret. - * IF it returns the soft error code, the function MAY be called again - * to retry and MUST not queue a MHD response. - * - * @param cls a `struct ReserveHistoryContext *` - * @param connection MHD request which triggered the transaction - * @param[out] mhd_ret set to MHD response status for @a connection, - *             if transaction failed (!) - * @return transaction status - */ -static enum GNUNET_DB_QueryStatus -reserve_balance_transaction (void *cls, -                             struct MHD_Connection *connection, -                             MHD_RESULT *mhd_ret) -{ -  struct ReserveHistoryContext *rsc = cls; -  enum GNUNET_DB_QueryStatus qs; - -  qs = TEH_plugin->get_reserve_balance (TEH_plugin->cls, -                                        &rsc->reserve_pub, -                                        &rsc->balance); -  if (GNUNET_DB_STATUS_HARD_ERROR == qs) -  { -    GNUNET_break (0); -    *mhd_ret -      = TALER_MHD_reply_with_error (connection, -                                    MHD_HTTP_INTERNAL_SERVER_ERROR, -                                    TALER_EC_GENERIC_DB_FETCH_FAILED, -                                    "get_reserve_balance"); -  } -  if (GNUNET_DB_STATUS_SUCCESS_NO_RESULTS == qs) -    rsc->not_found = true; -  if (GNUNET_DB_STATUS_SUCCESS_ONE_RESULT == qs) -    rsc->not_found = false; -  return qs; -} - -  MHD_RESULT  TEH_handler_reserves_get (struct TEH_RequestContext *rc,                            const char *const args[1])  { -  struct ReserveHistoryContext rsc; -  struct GNUNET_TIME_Relative timeout = GNUNET_TIME_UNIT_ZERO; -  struct GNUNET_DB_EventHandler *eh = NULL; - -  if (GNUNET_OK != -      GNUNET_STRINGS_string_to_data (args[0], -                                     strlen (args[0]), -                                     &rsc.reserve_pub, -                                     sizeof (rsc.reserve_pub))) -  { -    GNUNET_break_op (0); -    return TALER_MHD_reply_with_error (rc->connection, -                                       MHD_HTTP_BAD_REQUEST, -                                       TALER_EC_GENERIC_RESERVE_PUB_MALFORMED, -                                       args[0]); -  } +  struct ReservePoller *rp = rc->rh_ctx; + +  if (NULL == rp)    { -    const char *long_poll_timeout_ms; +    struct GNUNET_TIME_Relative timeout +      = GNUNET_TIME_UNIT_ZERO; -    long_poll_timeout_ms -      = MHD_lookup_connection_value (rc->connection, -                                     MHD_GET_ARGUMENT_KIND, -                                     "timeout_ms"); -    if (NULL != long_poll_timeout_ms) +    rp = GNUNET_new (struct ReservePoller); +    rp->connection = rc->connection; +    rc->rh_ctx = rp; +    rc->rh_cleaner = &rp_cleanup; +    GNUNET_CONTAINER_DLL_insert (rp_head, +                                 rp_tail, +                                 rp); +    if (GNUNET_OK != +        GNUNET_STRINGS_string_to_data (args[0], +                                       strlen (args[0]), +                                       &rp->reserve_pub, +                                       sizeof (rp->reserve_pub)))      { -      unsigned int timeout_ms; -      char dummy; +      GNUNET_break_op (0); +      return TALER_MHD_reply_with_error (rc->connection, +                                         MHD_HTTP_BAD_REQUEST, +                                         TALER_EC_GENERIC_RESERVE_PUB_MALFORMED, +                                         args[0]); +    } +    { +      const char *long_poll_timeout_ms; -      if (1 != sscanf (long_poll_timeout_ms, -                       "%u%c", -                       &timeout_ms, -                       &dummy)) +      long_poll_timeout_ms +        = MHD_lookup_connection_value (rc->connection, +                                       MHD_GET_ARGUMENT_KIND, +                                       "timeout_ms"); +      if (NULL != long_poll_timeout_ms)        { -        GNUNET_break_op (0); -        return TALER_MHD_reply_with_error (rc->connection, -                                           MHD_HTTP_BAD_REQUEST, -                                           TALER_EC_GENERIC_PARAMETER_MALFORMED, -                                           "timeout_ms (must be non-negative number)"); +        unsigned int timeout_ms; +        char dummy; + +        if (1 != sscanf (long_poll_timeout_ms, +                         "%u%c", +                         &timeout_ms, +                         &dummy)) +        { +          GNUNET_break_op (0); +          return TALER_MHD_reply_with_error (rc->connection, +                                             MHD_HTTP_BAD_REQUEST, +                                             TALER_EC_GENERIC_PARAMETER_MALFORMED, +                                             "timeout_ms (must be non-negative number)"); +        } +        timeout = GNUNET_TIME_relative_multiply (GNUNET_TIME_UNIT_MILLISECONDS, +                                                 timeout_ms);        } -      timeout = GNUNET_TIME_relative_multiply (GNUNET_TIME_UNIT_MILLISECONDS, -                                               timeout_ms);      } +    rp->timeout = GNUNET_TIME_relative_to_absolute (timeout);    } -  if ( (! GNUNET_TIME_relative_is_zero (timeout)) && -       (NULL == rc->rh_ctx) ) + +  if ( (GNUNET_TIME_absolute_is_future (rp->timeout)) && +       (NULL == rp->eh) )    {      struct TALER_ReserveEventP rep = {        .header.size = htons (sizeof (rep)),        .header.type = htons (TALER_DBEVENT_EXCHANGE_RESERVE_INCOMING), -      .reserve_pub = rsc.reserve_pub +      .reserve_pub = rp->reserve_pub      };      GNUNET_log (GNUNET_ERROR_TYPE_INFO,                  "Starting DB event listening\n"); -    eh = TEH_plugin->event_listen (TEH_plugin->cls, -                                   timeout, -                                   &rep.header, -                                   &db_event_cb, -                                   rc); +    rp->eh = TEH_plugin->event_listen ( +      TEH_plugin->cls, +      GNUNET_TIME_absolute_get_remaining (rp->timeout), +      &rep.header, +      &db_event_cb, +      rp);    }    { -    MHD_RESULT mhd_ret; +    enum GNUNET_DB_QueryStatus qs; -    if (GNUNET_OK != -        TEH_DB_run_transaction (rc->connection, -                                "get reserve balance", -                                TEH_MT_REQUEST_OTHER, -                                &mhd_ret, -                                &reserve_balance_transaction, -                                &rsc)) +    qs = TEH_plugin->get_reserve_balance (TEH_plugin->cls, +                                          &rp->reserve_pub, +                                          &rp->balance); +    switch (qs)      { -      if (NULL != eh) +    case GNUNET_DB_STATUS_SOFT_ERROR: +      GNUNET_break (0); /* single-shot query should never have soft-errors */ +      if (NULL != rp->eh) +      {          TEH_plugin->event_listen_cancel (TEH_plugin->cls, -                                         eh); -      return mhd_ret; -    } -  } -  /* generate proper response */ -  if (rsc.not_found) -  { -    struct ReservePoller *rp = rc->rh_ctx; - -    if ( (NULL != rp) || -         (GNUNET_TIME_relative_is_zero (timeout)) ) -    { +                                         rp->eh); +        rp->eh = NULL; +      }        return TALER_MHD_reply_with_error (rc->connection, -                                         MHD_HTTP_NOT_FOUND, -                                         TALER_EC_EXCHANGE_RESERVES_STATUS_UNKNOWN, -                                         args[0]); +                                         MHD_HTTP_INTERNAL_SERVER_ERROR, +                                         TALER_EC_GENERIC_DB_SOFT_FAILURE, +                                         "get_reserve_balance"); +    case GNUNET_DB_STATUS_HARD_ERROR: +      GNUNET_break (0); +      if (NULL != rp->eh) +      { +        TEH_plugin->event_listen_cancel (TEH_plugin->cls, +                                         rp->eh); +        rp->eh = NULL; +      } +      return TALER_MHD_reply_with_error (rc->connection, +                                         MHD_HTTP_INTERNAL_SERVER_ERROR, +                                         TALER_EC_GENERIC_DB_FETCH_FAILED, +                                         "get_reserve_balance"); +    case GNUNET_DB_STATUS_SUCCESS_ONE_RESULT: +      if (NULL != rp->eh) +      { +        TEH_plugin->event_listen_cancel (TEH_plugin->cls, +                                         rp->eh); +        rp->eh = NULL; +      } +      return TALER_MHD_REPLY_JSON_PACK (rc->connection, +                                        MHD_HTTP_OK, +                                        TALER_JSON_pack_amount ("balance", +                                                                &rp->balance)); +    case GNUNET_DB_STATUS_SUCCESS_NO_RESULTS: +      if ( (NULL != rp) || +           (GNUNET_TIME_absolute_is_future (rp->timeout)) ) +      { +        if (NULL != rp->eh) +        { +          TEH_plugin->event_listen_cancel (TEH_plugin->cls, +                                           rp->eh); +          rp->eh = NULL; +        } +        return TALER_MHD_reply_with_error (rc->connection, +                                           MHD_HTTP_NOT_FOUND, +                                           TALER_EC_EXCHANGE_RESERVES_STATUS_UNKNOWN, +                                           args[0]); +      } +      GNUNET_log (GNUNET_ERROR_TYPE_INFO, +                  "Long-polling on reserve for %s\n", +                  GNUNET_STRINGS_relative_time_to_string ( +                    GNUNET_TIME_absolute_get_remaining (rp->timeout), +                    true)); +      rp->suspended = true; +      MHD_suspend_connection (rc->connection); +      return MHD_YES;      } -    GNUNET_log (GNUNET_ERROR_TYPE_INFO, -                "Long-polling on reserve for %s\n", -                GNUNET_STRINGS_relative_time_to_string (timeout, -                                                        GNUNET_YES)); -    rp = GNUNET_new (struct ReservePoller); -    rp->connection = rc->connection; -    rp->timeout = GNUNET_TIME_relative_to_absolute (timeout); -    rp->eh = eh; -    rc->rh_ctx = rp; -    rc->rh_cleaner = &rp_cleanup; -    rp->suspended = true; -    GNUNET_CONTAINER_DLL_insert (rp_head, -                                 rp_tail, -                                 rp); -    MHD_suspend_connection (rc->connection); -    return MHD_YES;    } -  if (NULL != eh) -    TEH_plugin->event_listen_cancel (TEH_plugin->cls, -                                     eh); -  return TALER_MHD_REPLY_JSON_PACK ( -    rc->connection, -    MHD_HTTP_OK, -    TALER_JSON_pack_amount ("balance", -                            &rsc.balance)); +  GNUNET_break (0); +  return MHD_NO;  } diff --git a/src/exchangedb/pg_reserves_in_insert.c b/src/exchangedb/pg_reserves_in_insert.c index 71c57f84..876774de 100644 --- a/src/exchangedb/pg_reserves_in_insert.c +++ b/src/exchangedb/pg_reserves_in_insert.c @@ -722,8 +722,6 @@ TEH_PG_reserves_in_insert (                       &transaction_duplicate[i],                       &conflicts[i],                       &reserve_uuid[i]); -      fprintf (stdout, "reserve uuid : %ld c :%d t:%d\n", reserve_uuid[i], -               conflicts[i], transaction_duplicate[i]);        if (qs2<0)        {          GNUNET_log (GNUNET_ERROR_TYPE_WARNING, @@ -825,7 +823,11 @@ TEH_PG_reserves_in_insert (      cs = TEH_PG_commit (pg);      if (cs < 0) +    { +      for (unsigned int i = 0; i<reserves_length; i++) +        GNUNET_free (notify_s[i]);        return cs; +    }    }  exit: | 
