2015-08-08 20:21:13 +02:00
|
|
|
/*
|
|
|
|
This file is part of TALER
|
2016-01-19 14:39:00 +01:00
|
|
|
Copyright (C) 2015 GNUnet e.V.
|
2015-08-08 20:21:13 +02:00
|
|
|
|
|
|
|
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, If not, see
|
|
|
|
<http://www.gnu.org/licenses/>
|
|
|
|
*/
|
|
|
|
/**
|
2016-03-01 15:35:04 +01:00
|
|
|
* @file exchange-lib/exchange_api_common.c
|
|
|
|
* @brief common functions for the exchange API
|
2015-08-08 20:21:13 +02:00
|
|
|
* @author Christian Grothoff
|
|
|
|
*/
|
|
|
|
#include "platform.h"
|
2016-03-01 15:35:04 +01:00
|
|
|
#include "exchange_api_common.h"
|
2016-03-19 15:54:21 +01:00
|
|
|
#include "taler_json_lib.h"
|
2016-04-17 17:45:15 +02:00
|
|
|
#include <gnunet/gnunet_curl_lib.h>
|
2016-03-01 15:35:04 +01:00
|
|
|
#include "exchange_api_handle.h"
|
2015-08-08 20:21:13 +02:00
|
|
|
#include "taler_signatures.h"
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
2016-03-01 15:35:04 +01:00
|
|
|
* Verify a coins transaction history as returned by the exchange.
|
2015-08-08 20:21:13 +02:00
|
|
|
*
|
|
|
|
* @param currency expected currency for the coin
|
|
|
|
* @param coin_pub public key of the coin
|
|
|
|
* @param history history of the coin in json encoding
|
|
|
|
* @param[out] total how much of the coin has been spent according to @a history
|
|
|
|
* @return #GNUNET_OK if @a history is valid, #GNUNET_SYSERR if not
|
|
|
|
*/
|
|
|
|
int
|
2016-03-01 15:35:04 +01:00
|
|
|
TALER_EXCHANGE_verify_coin_history_ (const char *currency,
|
2016-04-11 20:16:58 +02:00
|
|
|
const struct TALER_CoinSpendPublicKeyP *coin_pub,
|
|
|
|
json_t *history,
|
|
|
|
struct TALER_Amount *total)
|
2015-08-08 20:21:13 +02:00
|
|
|
{
|
|
|
|
size_t len;
|
|
|
|
size_t off;
|
|
|
|
|
|
|
|
if (NULL == history)
|
|
|
|
{
|
|
|
|
GNUNET_break_op (0);
|
|
|
|
return GNUNET_SYSERR;
|
|
|
|
}
|
|
|
|
len = json_array_size (history);
|
|
|
|
if (0 == len)
|
|
|
|
{
|
|
|
|
GNUNET_break_op (0);
|
|
|
|
return GNUNET_SYSERR;
|
|
|
|
}
|
|
|
|
TALER_amount_get_zero (currency,
|
|
|
|
total);
|
|
|
|
for (off=0;off<len;off++)
|
|
|
|
{
|
|
|
|
json_t *transaction;
|
|
|
|
struct TALER_Amount amount;
|
2015-11-11 14:44:08 +01:00
|
|
|
struct TALER_CoinSpendSignatureP sig;
|
|
|
|
void *details;
|
|
|
|
size_t details_size;
|
|
|
|
const char *type;
|
2016-03-19 15:54:21 +01:00
|
|
|
struct GNUNET_JSON_Specification spec[] = {
|
|
|
|
TALER_JSON_spec_amount ("amount",
|
2015-08-08 20:21:13 +02:00
|
|
|
&amount),
|
2016-03-19 15:54:21 +01:00
|
|
|
GNUNET_JSON_spec_string ("type",
|
2015-11-11 14:44:08 +01:00
|
|
|
&type),
|
2016-03-19 15:54:21 +01:00
|
|
|
GNUNET_JSON_spec_fixed_auto ("signature",
|
2015-11-11 14:44:08 +01:00
|
|
|
&sig),
|
2016-03-19 15:54:21 +01:00
|
|
|
GNUNET_JSON_spec_varsize ("details",
|
2015-11-11 14:44:08 +01:00
|
|
|
&details,
|
|
|
|
&details_size),
|
2016-03-19 15:54:21 +01:00
|
|
|
GNUNET_JSON_spec_end()
|
2015-08-08 20:21:13 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
transaction = json_array_get (history,
|
|
|
|
off);
|
|
|
|
if (GNUNET_OK !=
|
2016-03-19 15:54:21 +01:00
|
|
|
GNUNET_JSON_parse (transaction,
|
|
|
|
spec,
|
|
|
|
NULL, NULL))
|
2015-08-08 20:21:13 +02:00
|
|
|
{
|
|
|
|
GNUNET_break_op (0);
|
|
|
|
return GNUNET_SYSERR;
|
|
|
|
}
|
2015-11-11 14:44:08 +01:00
|
|
|
if (0 == strcasecmp (type,
|
|
|
|
"DEPOSIT"))
|
2015-08-08 20:21:13 +02:00
|
|
|
{
|
2015-11-11 14:44:08 +01:00
|
|
|
const struct TALER_DepositRequestPS *dr;
|
|
|
|
struct TALER_Amount dr_amount;
|
2015-08-08 20:21:13 +02:00
|
|
|
|
2015-11-11 14:44:08 +01:00
|
|
|
if (details_size != sizeof (struct TALER_DepositRequestPS))
|
|
|
|
{
|
|
|
|
GNUNET_break_op (0);
|
2016-03-19 15:54:21 +01:00
|
|
|
GNUNET_JSON_parse_free (spec);
|
2015-11-11 14:44:08 +01:00
|
|
|
return GNUNET_SYSERR;
|
2015-08-08 20:21:13 +02:00
|
|
|
}
|
2015-11-11 14:44:08 +01:00
|
|
|
dr = (const struct TALER_DepositRequestPS *) details;
|
|
|
|
if (details_size != ntohl (dr->purpose.size))
|
2015-08-08 20:21:13 +02:00
|
|
|
{
|
2015-11-11 14:44:08 +01:00
|
|
|
GNUNET_break_op (0);
|
2016-03-19 15:54:21 +01:00
|
|
|
GNUNET_JSON_parse_free (spec);
|
2015-11-11 14:44:08 +01:00
|
|
|
return GNUNET_SYSERR;
|
|
|
|
}
|
|
|
|
if (GNUNET_OK !=
|
|
|
|
GNUNET_CRYPTO_eddsa_verify (TALER_SIGNATURE_WALLET_COIN_DEPOSIT,
|
|
|
|
&dr->purpose,
|
|
|
|
&sig.eddsa_signature,
|
|
|
|
&coin_pub->eddsa_pub))
|
2015-08-08 20:21:13 +02:00
|
|
|
{
|
2015-11-11 14:44:08 +01:00
|
|
|
GNUNET_break_op (0);
|
2016-03-19 15:54:21 +01:00
|
|
|
GNUNET_JSON_parse_free (spec);
|
2015-11-11 14:44:08 +01:00
|
|
|
return GNUNET_SYSERR;
|
|
|
|
}
|
|
|
|
|
|
|
|
TALER_amount_ntoh (&dr_amount,
|
|
|
|
&dr->amount_with_fee);
|
|
|
|
if (0 != TALER_amount_cmp (&dr_amount,
|
|
|
|
&amount))
|
2015-08-08 20:21:13 +02:00
|
|
|
{
|
|
|
|
GNUNET_break (0);
|
2016-03-19 15:54:21 +01:00
|
|
|
GNUNET_JSON_parse_free (spec);
|
2015-08-08 20:21:13 +02:00
|
|
|
return GNUNET_SYSERR;
|
|
|
|
}
|
2015-11-11 14:44:08 +01:00
|
|
|
}
|
|
|
|
else if (0 == strcasecmp (type,
|
|
|
|
"MELT"))
|
|
|
|
{
|
|
|
|
const struct TALER_RefreshMeltCoinAffirmationPS *rm;
|
|
|
|
struct TALER_Amount rm_amount;
|
|
|
|
|
|
|
|
if (details_size != sizeof (struct TALER_RefreshMeltCoinAffirmationPS))
|
|
|
|
{
|
|
|
|
GNUNET_break_op (0);
|
2016-03-19 15:54:21 +01:00
|
|
|
GNUNET_JSON_parse_free (spec);
|
2015-11-11 14:44:08 +01:00
|
|
|
return GNUNET_SYSERR;
|
|
|
|
}
|
|
|
|
rm = (const struct TALER_RefreshMeltCoinAffirmationPS *) details;
|
|
|
|
if (details_size != ntohl (rm->purpose.size))
|
|
|
|
{
|
|
|
|
GNUNET_break_op (0);
|
2016-03-19 15:54:21 +01:00
|
|
|
GNUNET_JSON_parse_free (spec);
|
2015-11-11 14:44:08 +01:00
|
|
|
return GNUNET_SYSERR;
|
|
|
|
}
|
|
|
|
if (GNUNET_OK !=
|
|
|
|
GNUNET_CRYPTO_eddsa_verify (TALER_SIGNATURE_WALLET_COIN_MELT,
|
|
|
|
&rm->purpose,
|
|
|
|
&sig.eddsa_signature,
|
|
|
|
&coin_pub->eddsa_pub))
|
|
|
|
{
|
|
|
|
GNUNET_break_op (0);
|
2016-03-19 15:54:21 +01:00
|
|
|
GNUNET_JSON_parse_free (spec);
|
2015-11-11 14:44:08 +01:00
|
|
|
return GNUNET_SYSERR;
|
|
|
|
}
|
|
|
|
TALER_amount_ntoh (&rm_amount,
|
|
|
|
&rm->amount_with_fee);
|
|
|
|
if (0 != TALER_amount_cmp (&rm_amount,
|
|
|
|
&amount))
|
|
|
|
{
|
|
|
|
GNUNET_break_op (0);
|
2016-03-19 15:54:21 +01:00
|
|
|
GNUNET_JSON_parse_free (spec);
|
2015-11-11 14:44:08 +01:00
|
|
|
return GNUNET_SYSERR;
|
2015-08-08 20:21:13 +02:00
|
|
|
}
|
2015-11-11 14:44:08 +01:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2015-08-08 20:21:13 +02:00
|
|
|
/* signature not supported, new version on server? */
|
2015-11-11 14:44:08 +01:00
|
|
|
GNUNET_break_op (0);
|
2016-03-19 15:54:21 +01:00
|
|
|
GNUNET_JSON_parse_free (spec);
|
2015-08-08 20:21:13 +02:00
|
|
|
return GNUNET_SYSERR;
|
|
|
|
}
|
|
|
|
if (GNUNET_OK !=
|
|
|
|
TALER_amount_add (total,
|
|
|
|
total,
|
|
|
|
&amount))
|
|
|
|
{
|
2016-03-01 15:35:04 +01:00
|
|
|
/* overflow in history already!? inconceivable! Bad exchange! */
|
2015-08-08 20:21:13 +02:00
|
|
|
GNUNET_break_op (0);
|
2016-03-19 15:54:21 +01:00
|
|
|
GNUNET_JSON_parse_free (spec);
|
2015-08-08 20:21:13 +02:00
|
|
|
return GNUNET_SYSERR;
|
|
|
|
}
|
2016-03-19 15:54:21 +01:00
|
|
|
GNUNET_JSON_parse_free (spec);
|
2015-08-08 20:21:13 +02:00
|
|
|
}
|
|
|
|
return GNUNET_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-03-01 15:35:04 +01:00
|
|
|
/* end of exchange_api_common.c */
|