LCOV - code coverage report
Current view: top level - backend - taler-merchant-httpd_private-post-transfers.c (source / functions) Coverage Total Hit
Test: coverage.info Lines: 58.8 % 51 30
Test Date: 2025-11-06 19:31:41 Functions: 100.0 % 1 1

            Line data    Source code
       1              : /*
       2              :   This file is part of TALER
       3              :   (C) 2014-2023, 2025 Taler Systems SA
       4              : 
       5              :   TALER is free software; you can redistribute it and/or modify it under the
       6              :   terms of the GNU Affero General Public License as published by the Free Software
       7              :   Foundation; either version 3, or (at your option) any later version.
       8              : 
       9              :   TALER is distributed in the hope that it will be useful, but WITHOUT ANY
      10              :   WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR
      11              :   A PARTICULAR PURPOSE.  See the GNU General Public License for more details.
      12              : 
      13              :   You should have received a copy of the GNU General Public License along with
      14              :   TALER; see the file COPYING.  If not, see <http://www.gnu.org/licenses/>
      15              : */
      16              : /**
      17              :  * @file taler-merchant-httpd_private-post-transfers.c
      18              :  * @brief implement API for registering wire transfers
      19              :  * @author Marcello Stanisci
      20              :  * @author Christian Grothoff
      21              :  */
      22              : #include "platform.h"
      23              : #include <jansson.h>
      24              : #include <taler/taler_signatures.h>
      25              : #include <taler/taler_json_lib.h>
      26              : #include <taler/taler_dbevents.h>
      27              : #include "taler-merchant-httpd_exchanges.h"
      28              : #include "taler-merchant-httpd_helper.h"
      29              : #include "taler-merchant-httpd_private-post-transfers.h"
      30              : 
      31              : 
      32              : /**
      33              :  * How often do we retry the simple INSERT database transaction?
      34              :  */
      35              : #define MAX_RETRIES 5
      36              : 
      37              : 
      38              : MHD_RESULT
      39           10 : TMH_private_post_transfers (const struct TMH_RequestHandler *rh,
      40              :                             struct MHD_Connection *connection,
      41              :                             struct TMH_HandlerContext *hc)
      42              : {
      43              :   struct TALER_FullPayto payto_uri;
      44              :   const char *exchange_url;
      45              :   struct TALER_WireTransferIdentifierRawP wtid;
      46              :   struct TALER_Amount amount;
      47              :   struct GNUNET_JSON_Specification spec[] = {
      48           10 :     TALER_JSON_spec_amount_any ("credit_amount",
      49              :                                 &amount),
      50           10 :     GNUNET_JSON_spec_fixed_auto ("wtid",
      51              :                                  &wtid),
      52           10 :     TALER_JSON_spec_full_payto_uri ("payto_uri",
      53              :                                     &payto_uri),
      54           10 :     TALER_JSON_spec_web_url ("exchange_url",
      55              :                              &exchange_url),
      56           10 :     GNUNET_JSON_spec_end ()
      57              :   };
      58              :   enum GNUNET_GenericReturnValue res;
      59              :   enum GNUNET_DB_QueryStatus qs;
      60              : 
      61           10 :   res = TALER_MHD_parse_json_data (connection,
      62           10 :                                    hc->request_body,
      63              :                                    spec);
      64           10 :   if (GNUNET_OK != res)
      65              :     return (GNUNET_NO == res)
      66              :       ? MHD_YES
      67            0 :       : MHD_NO;
      68           10 :   GNUNET_log (GNUNET_ERROR_TYPE_DEBUG,
      69              :               "New inbound wire transfer over %s to %s from %s\n",
      70              :               TALER_amount2s (&amount),
      71              :               payto_uri.full_payto,
      72              :               exchange_url);
      73              : 
      74              :   /* Check if transfer data is in database, if not, add it. */
      75           10 :   for (unsigned int retry = 0; retry<MAX_RETRIES; retry++)
      76              :   {
      77           10 :     TMH_db->preflight (TMH_db->cls);
      78           10 :     if (GNUNET_OK !=
      79           10 :         TMH_db->start (TMH_db->cls,
      80              :                        "post-transfers"))
      81              :     {
      82            0 :       GNUNET_break (0);
      83            0 :       return TALER_MHD_reply_with_error (connection,
      84              :                                          MHD_HTTP_INTERNAL_SERVER_ERROR,
      85              :                                          TALER_EC_GENERIC_DB_START_FAILED,
      86              :                                          "transfer");
      87              :     }
      88           10 :     qs = TMH_db->insert_transfer (TMH_db->cls,
      89           10 :                                   hc->instance->settings.id,
      90              :                                   exchange_url,
      91              :                                   &wtid,
      92              :                                   &amount,
      93              :                                   payto_uri,
      94              :                                   0 /* no bank serial known! */);
      95           10 :     switch (qs)
      96              :     {
      97            0 :     case GNUNET_DB_STATUS_HARD_ERROR:
      98            0 :       GNUNET_break (0);
      99            0 :       TMH_db->rollback (TMH_db->cls);
     100            0 :       return TALER_MHD_reply_with_error (connection,
     101              :                                          MHD_HTTP_INTERNAL_SERVER_ERROR,
     102              :                                          TALER_EC_GENERIC_DB_STORE_FAILED,
     103              :                                          "insert_transfer");
     104            0 :     case GNUNET_DB_STATUS_SOFT_ERROR:
     105            0 :       TMH_db->rollback (TMH_db->cls);
     106            0 :       continue;
     107            0 :     case GNUNET_DB_STATUS_SUCCESS_NO_RESULTS:
     108              :       /* Must mean the bank account is unknown! */
     109            0 :       TMH_db->rollback (TMH_db->cls);
     110            0 :       return TALER_MHD_reply_with_error (
     111              :         connection,
     112              :         MHD_HTTP_CONFLICT,
     113              :         TALER_EC_MERCHANT_PRIVATE_POST_TRANSFERS_CONFLICTING_SUBMISSION,
     114            0 :         payto_uri.full_payto);
     115           10 :     case GNUNET_DB_STATUS_SUCCESS_ONE_RESULT:
     116           10 :       break;
     117              :     }
     118              :     {
     119           10 :       struct GNUNET_DB_EventHeaderP es = {
     120           10 :         .size = htons (sizeof (es)),
     121           10 :         .type = htons (TALER_DBEVENT_MERCHANT_WIRE_TRANSFER_CONFIRMED)
     122              :       };
     123              : 
     124           10 :       TMH_db->event_notify (TMH_db->cls,
     125              :                             &es,
     126              :                             NULL,
     127              :                             0);
     128              :     }
     129           10 :     qs = TMH_db->commit (TMH_db->cls);
     130           10 :     switch (qs)
     131              :     {
     132            0 :     case GNUNET_DB_STATUS_HARD_ERROR:
     133            0 :       GNUNET_break (0);
     134            0 :       TMH_db->rollback (TMH_db->cls);
     135            0 :       return TALER_MHD_reply_with_error (connection,
     136              :                                          MHD_HTTP_INTERNAL_SERVER_ERROR,
     137              :                                          TALER_EC_GENERIC_DB_COMMIT_FAILED,
     138              :                                          NULL);
     139            0 :     case GNUNET_DB_STATUS_SOFT_ERROR:
     140            0 :       TMH_db->rollback (TMH_db->cls);
     141            0 :       continue;
     142           10 :     case GNUNET_DB_STATUS_SUCCESS_NO_RESULTS:
     143              :     case GNUNET_DB_STATUS_SUCCESS_ONE_RESULT:
     144           10 :       GNUNET_log (GNUNET_ERROR_TYPE_INFO,
     145              :                   "post-transfer committed successfully\n");
     146           10 :       break;
     147              :     }
     148           10 :     break;
     149              :   }
     150           10 :   return TALER_MHD_reply_static (connection,
     151              :                                  MHD_HTTP_NO_CONTENT,
     152              :                                  NULL,
     153              :                                  NULL,
     154              :                                  0);
     155              : }
     156              : 
     157              : 
     158              : /* end of taler-merchant-httpd_private-post-transfers.c */
        

Generated by: LCOV version 2.0-1