Make the API-agnostic iterator work.

This allows to build local histories to be matched
against the ones returned by the bank; in particular,
this version builds those stories regardless of /history
or /history-range being run.
This commit is contained in:
Marcello Stanisci 2019-04-09 17:14:13 +02:00
parent 074b011d01
commit f3d80b58f6
No known key found for this signature in database
GPG Key ID: 8D526861953F4C0F
5 changed files with 219 additions and 30 deletions

View File

@ -76,6 +76,20 @@ run (void *cls,
NULL, /* start */ NULL, /* start */
5), 5),
#if 0
/**
* Just a dummy call to check if the logic doesn't crash.
*/
TALER_TESTING_cmd_bank_history_range_with_dates
("history-0-range",
bank_url,
EXCHANGE_ACCOUNT_NUMBER,
TALER_BANK_DIRECTION_BOTH,
GNUNET_NO,
GNUNET_TIME_UNIT_ZERO_ABS,
GNUNET_TIME_UNIT_FOREVER_ABS),
#endif
TALER_TESTING_cmd_fakebank_transfer_with_subject TALER_TESTING_cmd_fakebank_transfer_with_subject
("deposit-1", ("deposit-1",
"KUDOS:5.01", "KUDOS:5.01",

View File

@ -1,4 +1,4 @@
/* /*
This file is part of TALER This file is part of TALER
Copyright (C) 2018 Taler Systems SA Copyright (C) 2018 Taler Systems SA
@ -40,19 +40,16 @@
*/ */
struct HistoryState struct HistoryState
{ {
/** /**
* Base URL of the bank offering the "history" operation. * Base URL of the bank offering the "history" operation.
*/ */
const char *bank_url; const char *bank_url;
/** /**
* Account number to ask the history for. * Account number to ask the history for.
*/ */
uint64_t account_no; uint64_t account_no;
/** /**
* Which type of records we are interested: in-transfers * Which type of records we are interested: in-transfers
* / out-transfers / rejected transfers. * / out-transfers / rejected transfers.
@ -65,7 +62,9 @@ struct HistoryState
const char *start_row_reference; const char *start_row_reference;
/** /**
* How many rows we want in the result. * How many rows we want in the result, _at most_. In
* the case of /history-range, we fake this value with
* INT64_MAX.
*/ */
long long num_results; long long num_results;
@ -263,6 +262,45 @@ print_expected (struct History *h,
} }
} }
/**
* Tell if the current item is beyond the allowed limit.
*
* @param total current number of items in the built history list.
* Note, this is the list we build locally and compare with
* what the server returned.
* @param hs the history CMD state.
* @param pos current item to be evaluated or not (if the list
* has already enough elements).
* @return GNUNET_OK / GNUNET_NO.
*/
static int
build_history_hit_limit (uint64_t total,
const struct HistoryState *hs,
const struct TALER_TESTING_Command *pos)
{
/* "/history-range" case. */
if (GNUNET_TIME_UNIT_FOREVER_ABS.abs_value_us !=
hs->start_date.abs_value_us)
{
const struct GNUNET_TIME_Absolute *timestamp;
GNUNET_assert (GNUNET_OK ==
TALER_TESTING_get_trait_absolute_time (pos,
0,
&timestamp));
GNUNET_assert (GNUNET_TIME_UNIT_FOREVER_ABS.abs_value_us !=
hs->end_date.abs_value_us);
return timestamp->abs_value_us >= hs->end_date.abs_value_us;
}
return total >= hs->num_results;
}
/** /**
* This function constructs the list of history elements that * This function constructs the list of history elements that
* interest the account number of the caller. It has two main * interest the account number of the caller. It has two main
@ -305,9 +343,9 @@ build_history (struct TALER_TESTING_Interpreter *is,
if (NULL != hs->start_row_reference) if (NULL != hs->start_row_reference)
{ {
TALER_LOG_INFO ("`%s': start row given via reference `%s'\n", TALER_LOG_INFO
TALER_TESTING_interpreter_get_current_label ("`%s': start row given via reference `%s'\n",
(is), TALER_TESTING_interpreter_get_current_label (is),
hs->start_row_reference); hs->start_row_reference);
add_incoming_cmd = TALER_TESTING_interpreter_lookup_command add_incoming_cmd = TALER_TESTING_interpreter_lookup_command
(is, hs->start_row_reference); (is, hs->start_row_reference);
@ -317,7 +355,6 @@ build_history (struct TALER_TESTING_Interpreter *is,
} }
GNUNET_assert (0 != hs->num_results); GNUNET_assert (0 != hs->num_results);
if (0 == is->ip) if (0 == is->ip)
{ {
*rh = NULL; *rh = NULL;
@ -344,9 +381,9 @@ build_history (struct TALER_TESTING_Interpreter *is,
if (NULL == row_id_start) if (NULL == row_id_start)
ok = GNUNET_YES; ok = GNUNET_YES;
/* This loop counts how many commands in the list off _all_ /* This loop counts how many commands _later than "start"_ belong
* the commands belong to the history of the caller. This * to the history of the caller. This is stored in the @var total
* is stored in the @var total variable. */ * variable. */
for (unsigned int off = start;off != end + inc; off += inc) for (unsigned int off = start;off != end + inc; off += inc)
{ {
const struct TALER_TESTING_Command *pos = &is->commands[off]; const struct TALER_TESTING_Command *pos = &is->commands[off];
@ -358,11 +395,12 @@ build_history (struct TALER_TESTING_Interpreter *is,
* that do not offer a "row_id" trait. Such skipped CMDs are * that do not offer a "row_id" trait. Such skipped CMDs are
* not interesting for building a history. * not interesting for building a history.
*/ */
if (GNUNET_OK != TALER_TESTING_get_trait_uint64 (pos,
if (GNUNET_OK != TALER_TESTING_get_trait_uint64 0,
(pos, 0, &row_id)) &row_id))
continue; continue;
/* Seek "/history" starting row. */
if (NULL != row_id_start) if (NULL != row_id_start)
{ {
if (*row_id_start == *row_id) if (*row_id_start == *row_id)
@ -373,12 +411,32 @@ build_history (struct TALER_TESTING_Interpreter *is,
continue; continue;
} }
} }
/* Seek "/history-range" starting row, _if_ that's the case */
if (GNUNET_TIME_UNIT_FOREVER_ABS.abs_value_us !=
hs->start_date.abs_value_us)
{
const struct GNUNET_TIME_Absolute *timestamp;
TALER_TESTING_get_trait_absolute_time (pos,
0,
&timestamp);
if (hs->start_date.abs_value_us <= timestamp->abs_value_us)
{
total = 0;
ok = GNUNET_YES;
continue;
}
}
/* when 'start' was _not_ given, then ok == GNUNET_YES */ /* when 'start' was _not_ given, then ok == GNUNET_YES */
if (GNUNET_NO == ok) if (GNUNET_NO == ok)
continue; /* skip until we find the marker */ continue; /* skip until we find the marker */
if (total >= hs->num_results * inc) if (build_history_hit_limit (total,
break; /* hit limit specified by command */ hs,
pos))
break;
cancelled = test_cancelled (is, off); cancelled = test_cancelled (is, off);
@ -435,7 +493,6 @@ build_history (struct TALER_TESTING_Interpreter *is,
if (NULL == row_id_start) if (NULL == row_id_start)
ok = GNUNET_YES; ok = GNUNET_YES;
/** /**
* This loop _only_ populates the array of history elements. * This loop _only_ populates the array of history elements.
*/ */
@ -457,12 +514,32 @@ build_history (struct TALER_TESTING_Interpreter *is,
if (*row_id_start == *row_id) if (*row_id_start == *row_id)
{ {
/* Doesn't count, start is excluded from output. */ /* Warning: this zeroing is superfluous, as total doesn't
* get incremented if 'start' was given and couldn't be found.
*/
total = 0; total = 0;
ok = GNUNET_YES; ok = GNUNET_YES;
continue; continue;
} }
} }
/* Seek "/history-range" starting row, _if_ that's the case */
if (GNUNET_TIME_UNIT_FOREVER_ABS.abs_value_us !=
hs->start_date.abs_value_us)
{
const struct GNUNET_TIME_Absolute *timestamp;
TALER_TESTING_get_trait_absolute_time (pos,
0,
&timestamp);
if (hs->start_date.abs_value_us <= timestamp->abs_value_us)
{
total = 0;
ok = GNUNET_YES;
continue;
}
}
if (GNUNET_NO == ok) if (GNUNET_NO == ok)
{ {
TALER_LOG_INFO ("Skip on `%s'\n", TALER_LOG_INFO ("Skip on `%s'\n",
@ -470,7 +547,9 @@ build_history (struct TALER_TESTING_Interpreter *is,
continue; /* skip until we find the marker */ continue; /* skip until we find the marker */
} }
if (total >= hs->num_results * inc) if (build_history_hit_limit (total,
hs,
pos))
{ {
TALER_LOG_INFO ("hit limit specified by command\n"); TALER_LOG_INFO ("hit limit specified by command\n");
break; break;
@ -490,6 +569,12 @@ build_history (struct TALER_TESTING_Interpreter *is,
(unsigned long long) *credit_account_no, (unsigned long long) *credit_account_no,
(unsigned long long) hs->account_no); (unsigned long long) hs->account_no);
/**
* Discard transactions where the audited account played
* _both_ the credit and the debit roles, but _only if_
* the audit goes on both directions.. This needs more
* explaination!
*/
if ( ( (0 != (hs->direction & TALER_BANK_DIRECTION_CREDIT)) && if ( ( (0 != (hs->direction & TALER_BANK_DIRECTION_CREDIT)) &&
(hs->account_no == *credit_account_no)) && (hs->account_no == *credit_account_no)) &&
( (0 != (hs->direction & TALER_BANK_DIRECTION_DEBIT)) && ( (0 != (hs->direction & TALER_BANK_DIRECTION_DEBIT)) &&
@ -513,6 +598,10 @@ build_history (struct TALER_TESTING_Interpreter *is,
GNUNET_assert (NULL != bank_hostname); GNUNET_assert (NULL != bank_hostname);
bank_hostname += 3; bank_hostname += 3;
/* Next two blocks only put the 'direction' and 'banking'
* information. */
/* Asked for credit, and account got the credit. */
if ( (0 != (hs->direction & TALER_BANK_DIRECTION_CREDIT)) && if ( (0 != (hs->direction & TALER_BANK_DIRECTION_CREDIT)) &&
(hs->account_no == *credit_account_no)) (hs->account_no == *credit_account_no))
{ {
@ -528,6 +617,8 @@ build_history (struct TALER_TESTING_Interpreter *is,
bank_hostname, bank_hostname,
(unsigned long long) *debit_account_no); (unsigned long long) *debit_account_no);
} }
/* Asked for debit, and account got the debit. */
if ( (0 != (hs->direction & TALER_BANK_DIRECTION_DEBIT)) && if ( (0 != (hs->direction & TALER_BANK_DIRECTION_DEBIT)) &&
(hs->account_no == *debit_account_no)) (hs->account_no == *debit_account_no))
{ {
@ -543,6 +634,9 @@ build_history (struct TALER_TESTING_Interpreter *is,
bank_hostname, bank_hostname,
(unsigned long long) *credit_account_no); (unsigned long long) *credit_account_no);
} }
/* This block _completes_ the information of the current item,
* with amount / subject / exchange URL. */
if ( ( (0 != (hs->direction & TALER_BANK_DIRECTION_CREDIT)) && if ( ( (0 != (hs->direction & TALER_BANK_DIRECTION_CREDIT)) &&
(hs->account_no == *credit_account_no)) || (hs->account_no == *credit_account_no)) ||
( (0 != (hs->direction & TALER_BANK_DIRECTION_DEBIT)) && ( (0 != (hs->direction & TALER_BANK_DIRECTION_DEBIT)) &&
@ -774,8 +868,9 @@ history_run (void *cls,
if (NULL == history_cmd) if (NULL == history_cmd)
TALER_TESTING_FAIL (is); TALER_TESTING_FAIL (is);
if (GNUNET_OK != TALER_TESTING_get_trait_uint64 if (GNUNET_OK != TALER_TESTING_get_trait_uint64 (history_cmd,
(history_cmd, 0, &row_id_ptr)) 0,
&row_id_ptr))
TALER_TESTING_FAIL (is); TALER_TESTING_FAIL (is);
row_id = *row_id_ptr; row_id = *row_id_ptr;
@ -830,10 +925,13 @@ history_range_run (void *cls,
if (GNUNET_OK != TALER_TESTING_get_trait_absolute_time if (GNUNET_OK != TALER_TESTING_get_trait_absolute_time
(history_cmd, 0, &start_date)) (history_cmd, 0, &start_date))
TALER_TESTING_FAIL (is); TALER_TESTING_FAIL (is);
hs->start_date = *start_date;
} }
else else
{ {
GNUNET_assert (UINT64_MAX != hs->start_date.abs_value_us); /* no trait wanted. */
GNUNET_assert (GNUNET_TIME_UNIT_FOREVER_ABS.abs_value_us !=
hs->start_date.abs_value_us);
start_date = &hs->start_date; start_date = &hs->start_date;
} }
@ -851,10 +949,13 @@ history_range_run (void *cls,
if (GNUNET_OK != TALER_TESTING_get_trait_absolute_time if (GNUNET_OK != TALER_TESTING_get_trait_absolute_time
(history_cmd, 0, &end_date)) (history_cmd, 0, &end_date))
TALER_TESTING_FAIL (is); TALER_TESTING_FAIL (is);
hs->end_date = *end_date;
} }
else else
{ {
GNUNET_assert (UINT64_MAX != hs->end_date.abs_value_us); /* no trait wanted. */
GNUNET_assert (GNUNET_TIME_UNIT_FOREVER_ABS.abs_value_us !=
hs->end_date.abs_value_us);
end_date = &hs->end_date; end_date = &hs->end_date;
} }
@ -933,6 +1034,8 @@ TALER_TESTING_cmd_bank_history
hs->start_row_reference = start_row_reference; hs->start_row_reference = start_row_reference;
hs->num_results = num_results; hs->num_results = num_results;
hs->ascending = ascending; hs->ascending = ascending;
hs->start_date = GNUNET_TIME_UNIT_FOREVER_ABS;
hs->end_date = GNUNET_TIME_UNIT_FOREVER_ABS;
struct TALER_TESTING_Command cmd = { struct TALER_TESTING_Command cmd = {
.label = label, .label = label,
@ -962,7 +1065,10 @@ TALER_TESTING_cmd_bank_history
* @param end_row_reference reference to a command that can * @param end_row_reference reference to a command that can
* offer a absolute time to use as the 'end' argument * offer a absolute time to use as the 'end' argument
* for "/history-range". * for "/history-range".
* @param num_result how many rows we want in the result. * @param num_results how many results we want from the bank; NOTE,
* this value is NOT used to issue any "delta" parameter in
* the HTTP request. Rather, it is only checked against
* the results returned by the bank.
* *
* @return the command. * @return the command.
*/ */
@ -1016,7 +1122,10 @@ TALER_TESTING_cmd_bank_history_range
* of "/history-range". * of "/history-range".
* @param end_date value for the 'end' argument * @param end_date value for the 'end' argument
* of "/history-range". * of "/history-range".
* @param num_result how many rows we want in the result. * @param num_results how many results we want from the bank; NOTE,
* this value is NOT used to issue any "delta" parameter in
* the HTTP request. Rather, it is only checked against
* the results returned by the bank.
* *
* @return the command. * @return the command.
*/ */

View File

@ -1195,7 +1195,6 @@ run (void *cls,
const char *cfgfile, const char *cfgfile,
const struct GNUNET_CONFIGURATION_Handle *cfg) const struct GNUNET_CONFIGURATION_Handle *cfg)
{ {
static struct GNUNET_HashCode zero;
struct GNUNET_TIME_Relative lookahead_sign; struct GNUNET_TIME_Relative lookahead_sign;
struct GNUNET_CRYPTO_EddsaPrivateKey *eddsa_priv; struct GNUNET_CRYPTO_EddsaPrivateKey *eddsa_priv;

View File

@ -109,6 +109,74 @@ TALER_TESTING_cmd_bank_history
const char *start_row_reference, const char *start_row_reference,
long long num_results); long long num_results);
/**
* Make a "history-range" CMD, picking dates from the arguments.
*
* @param label command label.
* @param bank_url base URL of the bank offering the "history"
* operation.
* @param account_no bank account number to ask the history for.
* @param direction which direction this operation is interested.
* @param ascending if GNUNET_YES, the bank will return the rows
* in ascending (= chronological) order.
* @param start_date value for the 'start' argument
* of "/history-range".
* @param end_date value for the 'end' argument
* of "/history-range".
* @param num_results how many results we want from the bank; NOTE,
* this value is NOT used to issue any "delta" parameter in
* the HTTP request. Rather, it is only checked against
* the results returned by the bank.
* @return the command.
*/
struct TALER_TESTING_Command
TALER_TESTING_cmd_bank_history_range_with_dates
(const char *label,
const char *bank_url,
uint64_t account_no,
enum TALER_BANK_Direction direction,
unsigned int ascending,
struct GNUNET_TIME_Absolute start_date,
struct GNUNET_TIME_Absolute end_date,
long long num_results);
/**
* Make a "history-range" CMD, picking dates from traits.
*
* @param label command label.
* @param bank_url base URL of the bank offering the "history"
* operation.
* @param account_no bank account number to ask the history for.
* @param direction which direction this operation is interested.
* @param ascending if GNUNET_YES, the bank will return the rows
* in ascending (= chronological) order.
* @param start_row_reference reference to a command that can
* offer a absolute time to use as the 'start' argument
* for "/history-range".
* @param end_row_reference reference to a command that can
* offer a absolute time to use as the 'end' argument
* for "/history-range".
* @param num_results how many results we want from the bank; NOTE,
* this value is NOT used to issue any "delta" parameter in
* the HTTP request. Rather, it is only checked against
* the results returned by the bank.
*
* @return the command.
*/
struct TALER_TESTING_Command
TALER_TESTING_cmd_bank_history_range
(const char *label,
const char *bank_url,
uint64_t account_no,
enum TALER_BANK_Direction direction,
unsigned int ascending,
const char *start_row_reference,
const char *end_row_reference,
long long num_results);
/** /**
* Create a "reject" CMD. * Create a "reject" CMD.
* *

View File

@ -52,7 +52,6 @@ TALER_TESTING_get_trait_absolute_time
index); index);
} }
/** /**
* Offer a absolute time. * Offer a absolute time.
* *