2015-01-08 18:37:20 +01:00
|
|
|
/*
|
|
|
|
This file is part of TALER
|
2015-04-12 22:34:56 +02:00
|
|
|
Copyright (C) 2014, 2015 Christian Grothoff (and other contributing authors)
|
2015-01-08 18:37:20 +01: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/>
|
|
|
|
*/
|
|
|
|
/**
|
|
|
|
* @file util/json.c
|
|
|
|
* @brief helper functions for JSON processing using libjansson
|
|
|
|
* @author Sree Harsha Totakura <sreeharsha@totakura.in>
|
|
|
|
*/
|
|
|
|
#include "platform.h"
|
2015-06-02 12:52:05 +02:00
|
|
|
#if HAVE_GNUNET_GNUNET_UTIL_TALER_WALLET_LIB_H
|
|
|
|
#include <gnunet/gnunet_util_taler_wallet_lib.h>
|
|
|
|
#endif
|
|
|
|
#if HAVE_GNUNET_GNUNET_UTIL_LIB_H
|
2015-01-08 18:37:20 +01:00
|
|
|
#include <gnunet/gnunet_util_lib.h>
|
2015-06-02 12:52:05 +02:00
|
|
|
#endif
|
2015-01-08 18:37:20 +01:00
|
|
|
#include "taler_util.h"
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Shorthand for exit jumps.
|
|
|
|
*/
|
|
|
|
#define EXITIF(cond) \
|
|
|
|
do { \
|
|
|
|
if (cond) { GNUNET_break (0); goto EXITIF_exit; } \
|
|
|
|
} while (0)
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Shorthand for JSON parsing related exit jumps.
|
|
|
|
*/
|
2015-02-17 17:23:13 +01:00
|
|
|
#define UNPACK_EXITIF(cond) \
|
2015-01-08 18:37:20 +01:00
|
|
|
do { \
|
2015-03-27 19:58:40 +01:00
|
|
|
if (cond) { TALER_json_warn (error); goto EXITIF_exit; } \
|
2015-01-08 18:37:20 +01:00
|
|
|
} while (0)
|
|
|
|
|
2015-01-28 20:53:21 +01:00
|
|
|
|
2015-01-08 18:37:20 +01:00
|
|
|
/**
|
|
|
|
* Convert a TALER amount to a JSON
|
|
|
|
* object.
|
|
|
|
*
|
|
|
|
* @param amount the amount
|
|
|
|
* @return a json object describing the amount
|
|
|
|
*/
|
|
|
|
json_t *
|
2015-03-27 19:58:40 +01:00
|
|
|
TALER_json_from_amount (const struct TALER_Amount *amount)
|
2015-01-08 18:37:20 +01:00
|
|
|
{
|
|
|
|
json_t *j;
|
2015-01-28 20:53:21 +01:00
|
|
|
|
2015-03-18 18:55:41 +01:00
|
|
|
if ( (amount->value != (uint64_t) ((json_int_t) amount->value)) ||
|
|
|
|
(0 > ((json_int_t) amount->value)) )
|
|
|
|
{
|
|
|
|
/* Theoretically, json_int_t can be a 32-bit "long", or we might
|
|
|
|
have a 64-bit value which converted to a 63-bit signed long
|
|
|
|
long causes problems here. So we check. Note that depending
|
|
|
|
on the platform, the compiler may be able to statically tell
|
|
|
|
that at least the first check is always false. */
|
|
|
|
GNUNET_break (0);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
j = json_pack ("{s:s, s:I, s:I}",
|
|
|
|
"currency", amount->currency,
|
|
|
|
"value", (json_int_t) amount->value,
|
|
|
|
"fraction", (json_int_t) amount->fraction);
|
2015-01-08 18:37:20 +01:00
|
|
|
GNUNET_assert (NULL != j);
|
|
|
|
return j;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Convert absolute timestamp to a json string.
|
|
|
|
*
|
2015-03-28 15:42:07 +01:00
|
|
|
* @param stamp the time stamp
|
2015-01-08 18:37:20 +01:00
|
|
|
* @return a json string with the timestamp in @a stamp
|
|
|
|
*/
|
|
|
|
json_t *
|
2015-03-27 19:58:40 +01:00
|
|
|
TALER_json_from_abs (struct GNUNET_TIME_Absolute stamp)
|
2015-01-08 18:37:20 +01:00
|
|
|
{
|
|
|
|
json_t *j;
|
|
|
|
char *mystr;
|
|
|
|
int ret;
|
2015-05-17 17:01:25 +02:00
|
|
|
|
2015-07-05 16:55:01 +02:00
|
|
|
GNUNET_assert (GNUNET_OK ==
|
|
|
|
TALER_round_abs_time (&stamp));
|
2015-05-17 17:01:25 +02:00
|
|
|
if (stamp.abs_value_us == GNUNET_TIME_UNIT_FOREVER_ABS.abs_value_us)
|
2015-06-20 22:53:01 +02:00
|
|
|
return json_string ("/never/");
|
2015-04-08 22:28:52 +02:00
|
|
|
ret = GNUNET_asprintf (&mystr,
|
2015-07-05 16:55:01 +02:00
|
|
|
"/Date(%llu)/",
|
|
|
|
(unsigned long long) (stamp.abs_value_us / (1000LL * 1000LL)));
|
2015-01-08 18:37:20 +01:00
|
|
|
GNUNET_assert (ret > 0);
|
|
|
|
j = json_string (mystr);
|
|
|
|
GNUNET_free (mystr);
|
|
|
|
return j;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2015-01-19 21:53:23 +01:00
|
|
|
/**
|
|
|
|
* Convert a signature (with purpose) to a JSON object representation.
|
|
|
|
*
|
|
|
|
* @param purpose purpose of the signature
|
|
|
|
* @param signature the signature
|
|
|
|
* @return the JSON reporesentation of the signature with purpose
|
|
|
|
*/
|
|
|
|
json_t *
|
2015-03-27 19:58:40 +01:00
|
|
|
TALER_json_from_eddsa_sig (const struct GNUNET_CRYPTO_EccSignaturePurpose *purpose,
|
2015-03-09 12:29:41 +01:00
|
|
|
const struct GNUNET_CRYPTO_EddsaSignature *signature)
|
2015-01-19 21:53:23 +01:00
|
|
|
{
|
|
|
|
json_t *root;
|
|
|
|
json_t *el;
|
|
|
|
|
|
|
|
root = json_object ();
|
|
|
|
|
|
|
|
el = json_integer ((json_int_t) ntohl (purpose->size));
|
|
|
|
json_object_set_new (root, "size", el);
|
|
|
|
|
|
|
|
el = json_integer ((json_int_t) ntohl (purpose->purpose));
|
|
|
|
json_object_set_new (root, "purpose", el);
|
|
|
|
|
2015-03-27 19:58:40 +01:00
|
|
|
el = TALER_json_from_data (purpose,
|
2015-03-09 13:07:27 +01:00
|
|
|
ntohl (purpose->size));
|
2015-04-08 22:28:52 +02:00
|
|
|
json_object_set_new (root, "eddsa_val", el);
|
2015-03-09 13:07:27 +01:00
|
|
|
|
2015-03-27 19:58:40 +01:00
|
|
|
el = TALER_json_from_data (signature,
|
2015-01-28 19:48:41 +01:00
|
|
|
sizeof (struct GNUNET_CRYPTO_EddsaSignature));
|
2015-04-08 22:28:52 +02:00
|
|
|
json_object_set_new (root, "eddsa_sig", el);
|
2015-03-09 12:29:41 +01:00
|
|
|
|
|
|
|
return root;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2015-03-15 16:39:06 +01:00
|
|
|
/**
|
|
|
|
* Convert RSA public key to JSON.
|
|
|
|
*
|
|
|
|
* @param pk public key to convert
|
|
|
|
* @return corresponding JSON encoding
|
|
|
|
*/
|
|
|
|
json_t *
|
2015-03-27 19:58:40 +01:00
|
|
|
TALER_json_from_rsa_public_key (struct GNUNET_CRYPTO_rsa_PublicKey *pk)
|
2015-03-15 16:39:06 +01:00
|
|
|
{
|
|
|
|
char *buf;
|
|
|
|
size_t buf_len;
|
|
|
|
json_t *ret;
|
|
|
|
|
|
|
|
buf_len = GNUNET_CRYPTO_rsa_public_key_encode (pk,
|
|
|
|
&buf);
|
2015-03-27 19:58:40 +01:00
|
|
|
ret = TALER_json_from_data (buf,
|
2015-03-15 16:39:06 +01:00
|
|
|
buf_len);
|
|
|
|
GNUNET_free (buf);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2015-05-17 17:28:18 +02:00
|
|
|
/**
|
|
|
|
* Convert JSON to RSA public key.
|
|
|
|
*
|
2015-06-12 11:00:04 +02:00
|
|
|
* @param json JSON encoding to convert
|
2015-05-17 17:28:18 +02:00
|
|
|
* @return corresponding public key
|
|
|
|
*/
|
|
|
|
struct GNUNET_CRYPTO_rsa_PublicKey *
|
|
|
|
TALER_json_to_rsa_public_key (json_t *json)
|
|
|
|
{
|
|
|
|
const char *enc;
|
|
|
|
char *buf;
|
|
|
|
size_t len;
|
|
|
|
size_t buf_len;
|
|
|
|
struct GNUNET_CRYPTO_rsa_PublicKey *pk;
|
|
|
|
|
|
|
|
buf = NULL;
|
|
|
|
EXITIF (NULL == (enc = json_string_value (json)));
|
|
|
|
len = strlen (enc);
|
|
|
|
buf_len = (len * 5) / 8;
|
|
|
|
buf = GNUNET_malloc (buf_len);
|
2015-06-02 12:52:05 +02:00
|
|
|
EXITIF (GNUNET_OK !=
|
|
|
|
GNUNET_STRINGS_string_to_data (enc,
|
|
|
|
len,
|
|
|
|
buf,
|
2015-05-17 17:28:18 +02:00
|
|
|
buf_len));
|
|
|
|
EXITIF (NULL == (pk = GNUNET_CRYPTO_rsa_public_key_decode (buf,
|
|
|
|
buf_len)));
|
|
|
|
GNUNET_free (buf);
|
|
|
|
return pk;
|
|
|
|
EXITIF_exit:
|
|
|
|
GNUNET_free_non_null (buf);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Convert JSON to RSA signature.
|
|
|
|
*
|
2015-06-12 11:00:04 +02:00
|
|
|
* @param json JSON encoding to convert
|
2015-05-17 17:28:18 +02:00
|
|
|
* @return corresponding signature
|
|
|
|
*/
|
|
|
|
struct GNUNET_CRYPTO_rsa_Signature *
|
|
|
|
TALER_json_to_rsa_signature (json_t *json)
|
|
|
|
{
|
|
|
|
const char *enc;
|
|
|
|
char *buf;
|
|
|
|
size_t len;
|
|
|
|
size_t buf_len;
|
|
|
|
struct GNUNET_CRYPTO_rsa_Signature *sig;
|
|
|
|
|
|
|
|
buf = NULL;
|
|
|
|
EXITIF (NULL == (enc = json_string_value (json)));
|
|
|
|
len = strlen (enc);
|
|
|
|
buf_len = (len * 5) / 8;
|
|
|
|
buf = GNUNET_malloc (buf_len);
|
2015-06-02 12:52:05 +02:00
|
|
|
EXITIF (GNUNET_OK !=
|
|
|
|
GNUNET_STRINGS_string_to_data (enc,
|
|
|
|
len,
|
|
|
|
buf,
|
2015-05-17 17:28:18 +02:00
|
|
|
buf_len));
|
|
|
|
EXITIF (NULL == (sig = GNUNET_CRYPTO_rsa_signature_decode (buf,
|
|
|
|
buf_len)));
|
|
|
|
GNUNET_free (buf);
|
|
|
|
return sig;
|
|
|
|
EXITIF_exit:
|
|
|
|
GNUNET_free_non_null (buf);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2015-03-15 16:39:06 +01:00
|
|
|
/**
|
|
|
|
* Convert RSA signature to JSON.
|
|
|
|
*
|
|
|
|
* @param sig signature to convert
|
|
|
|
* @return corresponding JSON encoding
|
|
|
|
*/
|
|
|
|
json_t *
|
2015-03-27 19:58:40 +01:00
|
|
|
TALER_json_from_rsa_signature (struct GNUNET_CRYPTO_rsa_Signature *sig)
|
2015-03-15 16:39:06 +01:00
|
|
|
{
|
|
|
|
char *buf;
|
|
|
|
size_t buf_len;
|
|
|
|
json_t *ret;
|
|
|
|
|
|
|
|
buf_len = GNUNET_CRYPTO_rsa_signature_encode (sig,
|
|
|
|
&buf);
|
2015-03-27 19:58:40 +01:00
|
|
|
ret = TALER_json_from_data (buf,
|
2015-03-15 16:39:06 +01:00
|
|
|
buf_len);
|
|
|
|
GNUNET_free (buf);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2015-01-08 18:37:20 +01:00
|
|
|
/**
|
|
|
|
* Convert binary data to a JSON string
|
|
|
|
* with the base32crockford encoding.
|
|
|
|
*
|
|
|
|
* @param data binary data
|
|
|
|
* @param size size of @a data in bytes
|
|
|
|
* @return json string that encodes @a data
|
|
|
|
*/
|
|
|
|
json_t *
|
2015-03-28 15:42:07 +01:00
|
|
|
TALER_json_from_data (const void *data,
|
|
|
|
size_t size)
|
2015-01-08 18:37:20 +01:00
|
|
|
{
|
|
|
|
char *buf;
|
|
|
|
json_t *json;
|
2015-01-28 20:31:28 +01:00
|
|
|
|
|
|
|
buf = GNUNET_STRINGS_data_to_string_alloc (data, size);
|
2015-01-08 18:37:20 +01:00
|
|
|
json = json_string (buf);
|
|
|
|
GNUNET_free (buf);
|
|
|
|
return json;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Parse given JSON object to Amount
|
|
|
|
*
|
|
|
|
* @param json the json object representing Amount
|
2015-03-28 15:42:07 +01:00
|
|
|
* @param[out] r_amount where the amount has to be written
|
2015-01-28 20:37:51 +01:00
|
|
|
* @return #GNUNET_OK upon successful parsing; #GNUNET_SYSERR upon error
|
2015-01-08 18:37:20 +01:00
|
|
|
*/
|
|
|
|
int
|
2015-03-27 19:58:40 +01:00
|
|
|
TALER_json_to_amount (json_t *json,
|
2015-01-08 18:37:20 +01:00
|
|
|
struct TALER_Amount *r_amount)
|
|
|
|
{
|
|
|
|
json_int_t value;
|
|
|
|
json_int_t fraction;
|
2015-06-20 22:53:01 +02:00
|
|
|
const char *currency;
|
|
|
|
|
|
|
|
memset (r_amount,
|
|
|
|
0,
|
|
|
|
sizeof (struct TALER_Amount));
|
|
|
|
if (-1 == json_unpack (json,
|
|
|
|
"{s:I, s:I, s:s}",
|
|
|
|
"value", &value,
|
|
|
|
"fraction", &fraction,
|
|
|
|
"currency", ¤cy))
|
|
|
|
{
|
|
|
|
GNUNET_break_op (0);
|
|
|
|
return GNUNET_SYSERR;
|
|
|
|
}
|
|
|
|
if ( (value < 0) ||
|
|
|
|
(fraction < 0) ||
|
|
|
|
(value > UINT64_MAX) ||
|
|
|
|
(fraction > UINT32_MAX) )
|
|
|
|
{
|
|
|
|
GNUNET_break_op (0);
|
|
|
|
return GNUNET_SYSERR;
|
|
|
|
}
|
|
|
|
if (strlen (currency) >= TALER_CURRENCY_LEN)
|
|
|
|
{
|
|
|
|
GNUNET_break_op (0);
|
|
|
|
return GNUNET_SYSERR;
|
|
|
|
}
|
|
|
|
r_amount->value = (uint64_t) value;
|
2015-01-08 18:37:20 +01:00
|
|
|
r_amount->fraction = (uint32_t) fraction;
|
2015-06-20 22:53:01 +02:00
|
|
|
strcpy (r_amount->currency, currency);
|
|
|
|
(void) TALER_amount_normalize (r_amount);
|
2015-01-08 18:37:20 +01:00
|
|
|
return GNUNET_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
2015-07-05 16:55:01 +02:00
|
|
|
* Parse given JSON object to absolute time.
|
2015-01-08 18:37:20 +01:00
|
|
|
*
|
|
|
|
* @param json the json object representing Amount
|
2015-03-28 15:42:07 +01:00
|
|
|
* @param[out] abs where the amount has to be written
|
2015-01-28 20:37:51 +01:00
|
|
|
* @return #GNUNET_OK upon successful parsing; #GNUNET_SYSERR upon error
|
2015-01-08 18:37:20 +01:00
|
|
|
*/
|
|
|
|
int
|
2015-03-27 19:58:40 +01:00
|
|
|
TALER_json_to_abs (json_t *json,
|
2015-01-08 18:37:20 +01:00
|
|
|
struct GNUNET_TIME_Absolute *abs)
|
|
|
|
{
|
2015-06-20 22:53:01 +02:00
|
|
|
const char *val;
|
|
|
|
unsigned long long int tval;
|
2015-01-08 18:37:20 +01:00
|
|
|
|
2015-06-20 22:53:01 +02:00
|
|
|
val = json_string_value (json);
|
|
|
|
if (NULL == val)
|
|
|
|
{
|
|
|
|
GNUNET_break_op (0);
|
|
|
|
return GNUNET_SYSERR;
|
|
|
|
}
|
|
|
|
if ( (0 == strcasecmp (val,
|
|
|
|
"/forever/")) ||
|
|
|
|
(0 == strcasecmp (val,
|
|
|
|
"/never/")) )
|
2015-05-17 17:01:25 +02:00
|
|
|
{
|
|
|
|
*abs = GNUNET_TIME_UNIT_FOREVER_ABS;
|
|
|
|
return GNUNET_OK;
|
|
|
|
}
|
2015-07-05 16:55:01 +02:00
|
|
|
if (1 != sscanf (val,
|
|
|
|
"/Date(%llu)/",
|
|
|
|
&tval))
|
2015-06-20 22:53:01 +02:00
|
|
|
{
|
|
|
|
GNUNET_break_op (0);
|
|
|
|
return GNUNET_SYSERR;
|
|
|
|
}
|
|
|
|
/* Time is in seconds in JSON, but in microseconds in GNUNET_TIME_Absolute */
|
|
|
|
abs->abs_value_us = tval * 1000LL * 1000LL;
|
|
|
|
if ( (abs->abs_value_us) / 1000LL / 1000LL != tval)
|
|
|
|
{
|
|
|
|
/* Integer overflow */
|
|
|
|
GNUNET_break_op (0);
|
|
|
|
return GNUNET_SYSERR;
|
|
|
|
}
|
2015-01-08 18:37:20 +01:00
|
|
|
return GNUNET_OK;
|
|
|
|
}
|
|
|
|
|
2015-06-20 22:53:01 +02:00
|
|
|
|
2015-01-08 18:37:20 +01:00
|
|
|
/**
|
|
|
|
* Parse given JSON object to data
|
|
|
|
*
|
|
|
|
* @param json the json object representing data
|
|
|
|
* @param out the pointer to hold the parsed data.
|
2015-01-28 20:37:51 +01:00
|
|
|
* @param out_size the size of @a out
|
|
|
|
* @return #GNUNET_OK upon successful parsing; #GNUNET_SYSERR upon error
|
2015-01-08 18:37:20 +01:00
|
|
|
*/
|
|
|
|
int
|
2015-03-27 19:58:40 +01:00
|
|
|
TALER_json_to_data (json_t *json,
|
2015-01-08 18:37:20 +01:00
|
|
|
void *out,
|
|
|
|
size_t out_size)
|
|
|
|
{
|
|
|
|
const char *enc;
|
|
|
|
unsigned int len;
|
|
|
|
|
|
|
|
EXITIF (NULL == (enc = json_string_value (json)));
|
|
|
|
len = strlen (enc);
|
2015-05-17 17:10:38 +02:00
|
|
|
EXITIF (((len * 5) / 8) != out_size);
|
2015-01-08 18:37:20 +01:00
|
|
|
EXITIF (GNUNET_OK != GNUNET_STRINGS_string_to_data (enc, len, out, out_size));
|
|
|
|
return GNUNET_OK;
|
|
|
|
EXITIF_exit:
|
|
|
|
return GNUNET_SYSERR;
|
|
|
|
}
|
|
|
|
|
2015-06-21 19:18:31 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Hash a JSON for binary signing.
|
|
|
|
*
|
|
|
|
* @param[in] json some JSON value
|
|
|
|
* @param[out] hc resulting hash code
|
|
|
|
* @return #GNUNET_OK on success, #GNUNET_SYSERR on error
|
|
|
|
*/
|
|
|
|
int
|
|
|
|
TALER_hash_json (json_t *json,
|
|
|
|
struct GNUNET_HashCode *hc)
|
|
|
|
{
|
|
|
|
char *wire_enc;
|
|
|
|
size_t len;
|
|
|
|
|
|
|
|
if (NULL == (wire_enc = json_dumps (json,
|
|
|
|
JSON_COMPACT | JSON_SORT_KEYS)))
|
|
|
|
return GNUNET_SYSERR;
|
|
|
|
len = strlen (wire_enc) + 1;
|
|
|
|
GNUNET_CRYPTO_hash (wire_enc,
|
|
|
|
len,
|
|
|
|
hc);
|
|
|
|
GNUNET_free (wire_enc);
|
|
|
|
return GNUNET_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2015-01-08 18:37:20 +01:00
|
|
|
/* End of util/json.c */
|