2020-03-27 17:28:33 +01:00
|
|
|
/*
|
|
|
|
This file is part of TALER
|
|
|
|
Copyright (C) 2014-2020 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 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 General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public
|
|
|
|
License along with TALER; see the file COPYING. If not, see
|
|
|
|
<http://www.gnu.org/licenses/>
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @file testing/testing_api_cmd_deposits_get.c
|
|
|
|
* @brief Implement the testing CMDs for the /deposits/ GET operations.
|
|
|
|
* @author Marcello Stanisci
|
|
|
|
*/
|
|
|
|
#include "platform.h"
|
|
|
|
#include "taler_json_lib.h"
|
|
|
|
#include <gnunet/gnunet_curl_lib.h>
|
|
|
|
#include "taler_testing_lib.h"
|
|
|
|
|
|
|
|
/**
|
|
|
|
* State for a "track transaction" CMD.
|
|
|
|
*/
|
|
|
|
struct TrackTransactionState
|
|
|
|
{
|
|
|
|
|
|
|
|
/**
|
|
|
|
* If non NULL, will provide a WTID to be compared against
|
|
|
|
* the one returned by the "track transaction" operation.
|
|
|
|
*/
|
|
|
|
const char *bank_transfer_reference;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The WTID associated by the transaction being tracked.
|
|
|
|
*/
|
|
|
|
struct TALER_WireTransferIdentifierRawP wtid;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Expected HTTP response code.
|
|
|
|
*/
|
|
|
|
unsigned int expected_response_code;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Reference to any operation that can provide a transaction.
|
|
|
|
* Will be the transaction to track.
|
|
|
|
*/
|
|
|
|
const char *transaction_reference;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Index of the coin involved in the transaction. Recall:
|
|
|
|
* at the exchange, the tracking is done _per coin_.
|
|
|
|
*/
|
|
|
|
unsigned int coin_index;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Handle to the "track transaction" pending operation.
|
|
|
|
*/
|
|
|
|
struct TALER_EXCHANGE_DepositGetHandle *tth;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Interpreter state.
|
|
|
|
*/
|
|
|
|
struct TALER_TESTING_Interpreter *is;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Checks what is returned by the "track transaction" operation.
|
|
|
|
* Checks that the HTTP response code is acceptable, and - if the
|
|
|
|
* right reference is non NULL - that the wire transfer subject
|
|
|
|
* line matches our expectations.
|
|
|
|
*
|
|
|
|
* @param cls closure.
|
2020-04-05 22:05:38 +02:00
|
|
|
* @param hr HTTP response details
|
2020-03-27 17:28:33 +01:00
|
|
|
* @param exchange_pub public key of the exchange
|
|
|
|
* @param wtid wire transfer identifier, NULL if exchange did not
|
|
|
|
* execute the transaction yet.
|
|
|
|
* @param execution_time actual or planned execution time for the
|
|
|
|
* wire transfer.
|
|
|
|
* @param coin_contribution contribution to the total amount of
|
|
|
|
* the deposited coin (can be NULL).
|
|
|
|
*/
|
|
|
|
static void
|
|
|
|
deposit_wtid_cb (void *cls,
|
2020-04-05 22:05:38 +02:00
|
|
|
const struct TALER_EXCHANGE_HttpResponse *hr,
|
2020-03-27 17:28:33 +01:00
|
|
|
const struct TALER_ExchangePublicKeyP *exchange_pub,
|
|
|
|
const struct TALER_WireTransferIdentifierRawP *wtid,
|
|
|
|
struct GNUNET_TIME_Absolute execution_time,
|
|
|
|
const struct TALER_Amount *coin_contribution)
|
|
|
|
{
|
|
|
|
struct TrackTransactionState *tts = cls;
|
|
|
|
struct TALER_TESTING_Interpreter *is = tts->is;
|
|
|
|
struct TALER_TESTING_Command *cmd = &is->commands[is->ip];
|
|
|
|
|
|
|
|
(void) coin_contribution;
|
|
|
|
(void) exchange_pub;
|
|
|
|
(void) execution_time;
|
|
|
|
tts->tth = NULL;
|
2020-04-05 22:05:38 +02:00
|
|
|
if (tts->expected_response_code != hr->http_status)
|
2020-03-27 17:28:33 +01:00
|
|
|
{
|
|
|
|
GNUNET_log (GNUNET_ERROR_TYPE_ERROR,
|
2020-04-05 22:05:38 +02:00
|
|
|
"Unexpected response code %u/%d to command %s in %s:%u\n",
|
|
|
|
hr->http_status,
|
|
|
|
(int) hr->ec,
|
2020-03-27 17:28:33 +01:00
|
|
|
cmd->label,
|
|
|
|
__FILE__,
|
|
|
|
__LINE__);
|
2020-04-05 22:05:38 +02:00
|
|
|
json_dumpf (hr->reply,
|
|
|
|
stderr,
|
|
|
|
0);
|
2020-03-27 17:28:33 +01:00
|
|
|
TALER_TESTING_interpreter_fail (is);
|
|
|
|
return;
|
|
|
|
}
|
2020-04-05 22:05:38 +02:00
|
|
|
switch (hr->http_status)
|
2020-03-27 17:28:33 +01:00
|
|
|
{
|
|
|
|
case MHD_HTTP_OK:
|
|
|
|
tts->wtid = *wtid;
|
|
|
|
if (NULL != tts->bank_transfer_reference)
|
|
|
|
{
|
|
|
|
const struct TALER_TESTING_Command *bank_transfer_cmd;
|
|
|
|
const struct TALER_WireTransferIdentifierRawP *wtid_want;
|
|
|
|
|
|
|
|
/* _this_ wire transfer subject line. */
|
2020-04-05 22:05:38 +02:00
|
|
|
bank_transfer_cmd
|
|
|
|
= TALER_TESTING_interpreter_lookup_command (is,
|
|
|
|
tts->bank_transfer_reference);
|
2020-03-27 17:28:33 +01:00
|
|
|
if (NULL == bank_transfer_cmd)
|
|
|
|
{
|
|
|
|
GNUNET_break (0);
|
|
|
|
TALER_TESTING_interpreter_fail (is);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (GNUNET_OK !=
|
2020-04-05 22:05:38 +02:00
|
|
|
TALER_TESTING_get_trait_wtid (bank_transfer_cmd,
|
|
|
|
0,
|
|
|
|
&wtid_want))
|
2020-03-27 17:28:33 +01:00
|
|
|
{
|
|
|
|
GNUNET_break (0);
|
|
|
|
TALER_TESTING_interpreter_fail (is);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Compare that expected and gotten subjects match. */
|
|
|
|
if (0 != GNUNET_memcmp (wtid,
|
|
|
|
wtid_want))
|
|
|
|
{
|
|
|
|
GNUNET_break (0);
|
|
|
|
TALER_TESTING_interpreter_fail (tts->is);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case MHD_HTTP_ACCEPTED:
|
|
|
|
/* allowed, nothing to check here */
|
|
|
|
break;
|
|
|
|
case MHD_HTTP_NOT_FOUND:
|
|
|
|
/* allowed, nothing to check here */
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
GNUNET_break (0);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
TALER_TESTING_interpreter_next (tts->is);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Run the command.
|
|
|
|
*
|
|
|
|
* @param cls closure.
|
|
|
|
* @param cmd the command to execute.
|
|
|
|
* @param is the interpreter state.
|
|
|
|
*/
|
|
|
|
static void
|
|
|
|
track_transaction_run (void *cls,
|
|
|
|
const struct TALER_TESTING_Command *cmd,
|
|
|
|
struct TALER_TESTING_Interpreter *is)
|
|
|
|
{
|
|
|
|
struct TrackTransactionState *tts = cls;
|
|
|
|
const struct TALER_TESTING_Command *transaction_cmd;
|
|
|
|
const struct TALER_CoinSpendPrivateKeyP *coin_priv;
|
|
|
|
struct TALER_CoinSpendPublicKeyP coin_pub;
|
|
|
|
const json_t *contract_terms;
|
|
|
|
const json_t *wire_details;
|
|
|
|
struct GNUNET_HashCode h_wire_details;
|
|
|
|
struct GNUNET_HashCode h_contract_terms;
|
|
|
|
const struct TALER_MerchantPrivateKeyP *merchant_priv;
|
|
|
|
|
|
|
|
(void) cmd;
|
|
|
|
tts->is = is;
|
|
|
|
transaction_cmd
|
|
|
|
= TALER_TESTING_interpreter_lookup_command (tts->is,
|
|
|
|
tts->transaction_reference);
|
|
|
|
if (NULL == transaction_cmd)
|
|
|
|
{
|
|
|
|
GNUNET_break (0);
|
|
|
|
TALER_TESTING_interpreter_fail (tts->is);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (GNUNET_OK !=
|
|
|
|
TALER_TESTING_get_trait_coin_priv (transaction_cmd,
|
|
|
|
tts->coin_index,
|
|
|
|
&coin_priv))
|
|
|
|
{
|
|
|
|
GNUNET_break (0);
|
|
|
|
TALER_TESTING_interpreter_fail (tts->is);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
GNUNET_CRYPTO_eddsa_key_get_public (&coin_priv->eddsa_priv,
|
|
|
|
&coin_pub.eddsa_pub);
|
|
|
|
|
|
|
|
/* Get the strings.. */
|
|
|
|
if (GNUNET_OK !=
|
|
|
|
TALER_TESTING_get_trait_wire_details (transaction_cmd,
|
|
|
|
0,
|
|
|
|
&wire_details))
|
|
|
|
{
|
|
|
|
GNUNET_break (0);
|
|
|
|
TALER_TESTING_interpreter_fail (tts->is);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (GNUNET_OK !=
|
|
|
|
TALER_TESTING_get_trait_contract_terms (transaction_cmd,
|
|
|
|
0,
|
|
|
|
&contract_terms))
|
|
|
|
{
|
|
|
|
GNUNET_break (0);
|
|
|
|
TALER_TESTING_interpreter_fail (tts->is);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( (NULL == wire_details) ||
|
|
|
|
(NULL == contract_terms) )
|
|
|
|
{
|
|
|
|
GNUNET_break (0);
|
|
|
|
TALER_TESTING_interpreter_fail (tts->is);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Should not fail here, json has been parsed already */
|
|
|
|
GNUNET_assert
|
|
|
|
( (GNUNET_OK ==
|
|
|
|
TALER_JSON_merchant_wire_signature_hash (wire_details,
|
|
|
|
&h_wire_details)) &&
|
|
|
|
(GNUNET_OK ==
|
|
|
|
TALER_JSON_hash (contract_terms,
|
|
|
|
&h_contract_terms)) );
|
|
|
|
|
|
|
|
if (GNUNET_OK !=
|
|
|
|
TALER_TESTING_get_trait_merchant_priv (transaction_cmd,
|
|
|
|
0,
|
|
|
|
&merchant_priv))
|
|
|
|
{
|
|
|
|
GNUNET_break (0);
|
|
|
|
TALER_TESTING_interpreter_fail (tts->is);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
tts->tth = TALER_EXCHANGE_deposits_get (is->exchange,
|
|
|
|
merchant_priv,
|
|
|
|
&h_wire_details,
|
|
|
|
&h_contract_terms,
|
|
|
|
&coin_pub,
|
|
|
|
&deposit_wtid_cb,
|
|
|
|
tts);
|
|
|
|
GNUNET_assert (NULL != tts->tth);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Cleanup the state from a "track transaction" CMD, and possibly
|
|
|
|
* cancel a operation thereof.
|
|
|
|
*
|
|
|
|
* @param cls closure.
|
|
|
|
* @param cmd the command which is being cleaned up.
|
|
|
|
*/
|
|
|
|
static void
|
|
|
|
track_transaction_cleanup (void *cls,
|
|
|
|
const struct TALER_TESTING_Command *cmd)
|
|
|
|
{
|
|
|
|
struct TrackTransactionState *tts = cls;
|
|
|
|
|
|
|
|
if (NULL != tts->tth)
|
|
|
|
{
|
|
|
|
GNUNET_log (GNUNET_ERROR_TYPE_WARNING,
|
|
|
|
"Command %u (%s) did not complete\n",
|
|
|
|
tts->is->ip,
|
|
|
|
cmd->label);
|
|
|
|
TALER_EXCHANGE_deposits_get_cancel (tts->tth);
|
|
|
|
tts->tth = NULL;
|
|
|
|
}
|
|
|
|
GNUNET_free (tts);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Offer internal data from a "track transaction" CMD.
|
|
|
|
*
|
|
|
|
* @param cls closure.
|
|
|
|
* @param[out] ret result (could be anything).
|
|
|
|
* @param trait name of the trait.
|
|
|
|
* @param index index number of the object to offer.
|
|
|
|
* @return #GNUNET_OK on success.
|
|
|
|
*/
|
|
|
|
static int
|
|
|
|
track_transaction_traits (void *cls,
|
|
|
|
const void **ret,
|
|
|
|
const char *trait,
|
|
|
|
unsigned int index)
|
|
|
|
{
|
|
|
|
struct TrackTransactionState *tts = cls;
|
|
|
|
struct TALER_TESTING_Trait traits[] = {
|
|
|
|
TALER_TESTING_make_trait_wtid (0, &tts->wtid),
|
|
|
|
TALER_TESTING_trait_end ()
|
|
|
|
};
|
|
|
|
|
|
|
|
return TALER_TESTING_get_trait (traits,
|
|
|
|
ret,
|
|
|
|
trait,
|
|
|
|
index);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Create a "track transaction" command.
|
|
|
|
*
|
|
|
|
* @param label the command label.
|
|
|
|
* @param transaction_reference reference to a deposit operation,
|
|
|
|
* will be used to get the input data for the track.
|
|
|
|
* @param coin_index index of the coin involved in the transaction.
|
|
|
|
* @param expected_response_code expected HTTP response code.
|
|
|
|
* @param bank_transfer_reference reference to a command that
|
|
|
|
* can offer a WTID so as to check that against what WTID
|
|
|
|
* the tracked operation has. Set as NULL if not needed.
|
|
|
|
* @return the command.
|
|
|
|
*/
|
|
|
|
struct TALER_TESTING_Command
|
|
|
|
TALER_TESTING_cmd_track_transaction (const char *label,
|
|
|
|
const char *transaction_reference,
|
|
|
|
unsigned int coin_index,
|
|
|
|
unsigned int expected_response_code,
|
|
|
|
const char *bank_transfer_reference)
|
|
|
|
{
|
|
|
|
struct TrackTransactionState *tts;
|
|
|
|
|
|
|
|
tts = GNUNET_new (struct TrackTransactionState);
|
|
|
|
tts->transaction_reference = transaction_reference;
|
|
|
|
tts->expected_response_code = expected_response_code;
|
|
|
|
tts->bank_transfer_reference = bank_transfer_reference;
|
|
|
|
tts->coin_index = coin_index;
|
|
|
|
{
|
|
|
|
struct TALER_TESTING_Command cmd = {
|
|
|
|
.cls = tts,
|
|
|
|
.label = label,
|
|
|
|
.run = &track_transaction_run,
|
|
|
|
.cleanup = &track_transaction_cleanup,
|
|
|
|
.traits = &track_transaction_traits
|
|
|
|
};
|
|
|
|
|
|
|
|
return cmd;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* end of testing_api_cmd_deposits_get.c */
|