2019-01-15 04:54:27 +01:00
|
|
|
#include "../json_helpers.c"
|
2020-01-26 13:07:50 +01:00
|
|
|
#include <assert.h>
|
2018-12-16 05:44:06 +01:00
|
|
|
#include <ccan/mem/mem.h>
|
2020-11-25 01:17:44 +01:00
|
|
|
#include <ccan/tal/str/str.h>
|
2020-01-26 13:07:50 +01:00
|
|
|
#include <common/json.h>
|
2020-12-02 01:10:04 +01:00
|
|
|
#include <common/setup.h>
|
2018-04-25 12:55:34 +02:00
|
|
|
#include <common/utils.h>
|
2020-01-26 13:07:50 +01:00
|
|
|
#include <inttypes.h>
|
2017-11-01 02:58:44 +01:00
|
|
|
#include <stdio.h>
|
2019-06-21 02:11:41 +02:00
|
|
|
#include <wire/wire.h>
|
2017-11-01 02:58:44 +01:00
|
|
|
|
|
|
|
/* AUTOGENERATED MOCKS START */
|
2020-12-03 10:34:03 +01:00
|
|
|
/* Generated stub for fromwire_tlv */
|
|
|
|
bool fromwire_tlv(const u8 **cursor UNNEEDED, size_t *max UNNEEDED,
|
|
|
|
const struct tlv_record_type *types UNNEEDED, size_t num_types UNNEEDED,
|
|
|
|
void *record UNNEEDED, struct tlv_field **fields UNNEEDED)
|
|
|
|
{ fprintf(stderr, "fromwire_tlv called!\n"); abort(); }
|
|
|
|
/* Generated stub for tlv_fields_valid */
|
2021-06-17 12:54:45 +02:00
|
|
|
bool tlv_fields_valid(const struct tlv_field *fields UNNEEDED, u64 *allow_extra UNNEEDED,
|
|
|
|
size_t *err_index UNNEEDED)
|
2020-12-03 10:34:03 +01:00
|
|
|
{ fprintf(stderr, "tlv_fields_valid called!\n"); abort(); }
|
|
|
|
/* Generated stub for towire_tlv */
|
|
|
|
void towire_tlv(u8 **pptr UNNEEDED,
|
|
|
|
const struct tlv_record_type *types UNNEEDED, size_t num_types UNNEEDED,
|
|
|
|
const void *record UNNEEDED)
|
|
|
|
{ fprintf(stderr, "towire_tlv called!\n"); abort(); }
|
2017-11-01 02:58:44 +01:00
|
|
|
/* AUTOGENERATED MOCKS END */
|
|
|
|
|
2018-01-12 17:46:12 +01:00
|
|
|
|
|
|
|
// issue #577
|
|
|
|
|
|
|
|
static void do_json_tok_bitcoin_amount(const char* val, uint64_t expected)
|
|
|
|
{
|
|
|
|
uint64_t amount;
|
|
|
|
jsmntok_t tok;
|
|
|
|
|
|
|
|
tok.start = 0;
|
|
|
|
tok.end = strlen(val);
|
|
|
|
|
|
|
|
fprintf(stderr, "do_json_tok_bitcoin_amount(\"%s\", %"PRIu64"): ", val, expected);
|
|
|
|
|
2018-12-08 01:38:56 +01:00
|
|
|
assert(json_to_bitcoin_amount(val, &tok, &amount) == true);
|
2018-01-12 17:46:12 +01:00
|
|
|
assert(amount == expected);
|
|
|
|
|
|
|
|
fprintf(stderr, "ok\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static int test_json_tok_bitcoin_amount(void)
|
|
|
|
{
|
|
|
|
do_json_tok_bitcoin_amount("0.00000001", 1);
|
|
|
|
do_json_tok_bitcoin_amount("0.00000007", 7);
|
|
|
|
do_json_tok_bitcoin_amount("0.00000008", 8);
|
|
|
|
do_json_tok_bitcoin_amount("0.00000010", 10);
|
|
|
|
do_json_tok_bitcoin_amount("0.12345678", 12345678);
|
|
|
|
do_json_tok_bitcoin_amount("0.01234567", 1234567);
|
|
|
|
do_json_tok_bitcoin_amount("123.45678900", 12345678900);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2020-02-19 11:11:36 +01:00
|
|
|
static void do_json_tok_millionths(const char *val, bool ok, uint64_t expected)
|
|
|
|
{
|
|
|
|
uint64_t amount;
|
|
|
|
jsmntok_t tok;
|
|
|
|
|
|
|
|
tok.start = 0;
|
|
|
|
tok.end = strlen(val);
|
|
|
|
|
|
|
|
assert(json_to_millionths(val, &tok, &amount) == ok);
|
|
|
|
if (ok)
|
|
|
|
assert(amount == expected);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int test_json_tok_millionths(void)
|
|
|
|
{
|
|
|
|
do_json_tok_millionths("", false, 0);
|
|
|
|
do_json_tok_millionths("0..0", false, 0);
|
|
|
|
do_json_tok_millionths("0.0.", false, 0);
|
|
|
|
do_json_tok_millionths(".", false, 0);
|
|
|
|
do_json_tok_millionths("..", false, 0);
|
|
|
|
|
|
|
|
do_json_tok_millionths("0", true, 0);
|
|
|
|
do_json_tok_millionths(".0", true, 0);
|
|
|
|
do_json_tok_millionths("0.", true, 0);
|
|
|
|
do_json_tok_millionths("100", true, 100 * 1000000);
|
|
|
|
do_json_tok_millionths("100.0", true, 100 * 1000000);
|
|
|
|
do_json_tok_millionths("100.", true, 100 * 1000000);
|
|
|
|
do_json_tok_millionths("100.000001", true, 100 * 1000000 + 1);
|
|
|
|
do_json_tok_millionths("100.0000001", true, 100 * 1000000);
|
|
|
|
do_json_tok_millionths(".000009", true, 9);
|
|
|
|
do_json_tok_millionths(".0000099", true, 9);
|
|
|
|
do_json_tok_millionths("18446744073709.551615", true,
|
|
|
|
18446744073709551615ULL);
|
|
|
|
do_json_tok_millionths("18446744073709.551616", false, 0);
|
|
|
|
do_json_tok_millionths("18446744073709.551625", false, 0);
|
|
|
|
do_json_tok_millionths("18446744073709.551715", false, 0);
|
|
|
|
do_json_tok_millionths("18446744073709.552615", false, 0);
|
|
|
|
do_json_tok_millionths("18446744073709.561615", false, 0);
|
|
|
|
do_json_tok_millionths("18446744073709.651615", false, 0);
|
|
|
|
do_json_tok_millionths("18446744073710.551615", false, 0);
|
|
|
|
do_json_tok_millionths("18446744073809.551615", false, 0);
|
|
|
|
do_json_tok_millionths("18446744074709.551615", false, 0);
|
|
|
|
do_json_tok_millionths("18446744083709.551615", false, 0);
|
|
|
|
do_json_tok_millionths("18446744173709.551615", false, 0);
|
|
|
|
do_json_tok_millionths("18446745073709.551615", false, 0);
|
|
|
|
do_json_tok_millionths("18446754073709.551615", false, 0);
|
|
|
|
do_json_tok_millionths("18446844073709.551615", false, 0);
|
|
|
|
do_json_tok_millionths("18447744073709.551615", false, 0);
|
|
|
|
do_json_tok_millionths("18456744073709.551615", false, 0);
|
|
|
|
do_json_tok_millionths("18546744073709.551615", false, 0);
|
|
|
|
do_json_tok_millionths("19446744073709.551615", false, 0);
|
|
|
|
do_json_tok_millionths("28446744073709.551615", false, 0);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2019-01-17 03:05:26 +01:00
|
|
|
static void test_json_tok_size(void)
|
|
|
|
{
|
2020-08-20 15:40:32 +02:00
|
|
|
jsmntok_t *toks;
|
2019-01-17 03:05:26 +01:00
|
|
|
char *buf;
|
2020-08-20 15:40:32 +02:00
|
|
|
bool ok, complete;
|
|
|
|
jsmn_parser parser;
|
2019-01-17 03:05:26 +01:00
|
|
|
|
|
|
|
buf = "[\"e1\", [\"e2\", \"e3\"]]";
|
2020-08-20 15:40:32 +02:00
|
|
|
toks = toks_alloc(tmpctx);
|
|
|
|
jsmn_init(&parser);
|
|
|
|
ok = json_parse_input(&parser, &toks, buf, strlen(buf), &complete);
|
2019-01-17 03:05:26 +01:00
|
|
|
assert(ok);
|
2020-08-20 15:40:32 +02:00
|
|
|
assert(complete);
|
2019-01-17 03:05:26 +01:00
|
|
|
/* size only counts *direct* children */
|
|
|
|
assert(toks[0].size == 2);
|
|
|
|
assert(toks[2].size == 2);
|
|
|
|
|
|
|
|
buf = "[[\"e1\", \"e2\"], \"e3\"]";
|
2020-08-20 15:40:32 +02:00
|
|
|
toks_reset(toks);
|
|
|
|
jsmn_init(&parser);
|
|
|
|
ok = json_parse_input(&parser, &toks, buf, strlen(buf), &complete);
|
2019-01-17 03:05:26 +01:00
|
|
|
assert(ok);
|
2020-08-20 15:40:32 +02:00
|
|
|
assert(complete);
|
2019-01-17 03:05:26 +01:00
|
|
|
/* size only counts *direct* children */
|
|
|
|
assert(toks[0].size == 2);
|
|
|
|
assert(toks[1].size == 2);
|
|
|
|
|
2020-06-08 19:23:54 +02:00
|
|
|
buf = "{\"e1\" : {\"e2\": 2, \"e3\": 3}}";
|
2020-08-20 15:40:32 +02:00
|
|
|
toks_reset(toks);
|
|
|
|
jsmn_init(&parser);
|
|
|
|
ok = json_parse_input(&parser, &toks, buf, strlen(buf), &complete);
|
2019-01-17 03:05:26 +01:00
|
|
|
assert(ok);
|
2020-08-20 15:40:32 +02:00
|
|
|
assert(complete);
|
2019-01-17 03:05:26 +01:00
|
|
|
/* size only counts *direct* children */
|
|
|
|
assert(toks[0].size == 1);
|
|
|
|
assert(toks[2].size == 2);
|
|
|
|
|
2020-06-08 19:23:54 +02:00
|
|
|
buf = "{\"e1\" : {\"e2\": 2, \"e3\": 3}, \"e4\" : {\"e5\": 5, \"e6\": 6}}";
|
2020-08-20 15:40:32 +02:00
|
|
|
toks_reset(toks);
|
|
|
|
jsmn_init(&parser);
|
|
|
|
ok = json_parse_input(&parser, &toks, buf, strlen(buf), &complete);
|
2019-01-17 03:05:26 +01:00
|
|
|
assert(ok);
|
2020-08-20 15:40:32 +02:00
|
|
|
assert(complete);
|
2019-01-17 03:05:26 +01:00
|
|
|
/* size only counts *direct* children */
|
|
|
|
assert(toks[0].size == 2);
|
|
|
|
assert(toks[2].size == 2);
|
2020-06-08 19:23:54 +02:00
|
|
|
assert(toks[8].size == 2);
|
|
|
|
|
|
|
|
/* This should *not* parse! (used to give toks[0]->size == 3!) */
|
|
|
|
buf = "{ \"\" \"\" \"\" }";
|
2020-08-20 15:40:32 +02:00
|
|
|
toks_reset(toks);
|
|
|
|
jsmn_init(&parser);
|
|
|
|
ok = json_parse_input(&parser, &toks, buf, strlen(buf), &complete);
|
2020-06-08 19:23:54 +02:00
|
|
|
assert(!ok);
|
2019-01-17 03:09:45 +01:00
|
|
|
|
|
|
|
/* This should *not* parse! (used to give toks[0]->size == 2!) */
|
|
|
|
buf = "{ 'satoshi', '546' }";
|
2020-08-20 02:16:55 +02:00
|
|
|
toks = json_parse_simple(tmpctx, buf, strlen(buf));
|
|
|
|
assert(!toks);
|
2019-01-17 03:05:26 +01:00
|
|
|
}
|
|
|
|
|
2020-11-25 01:17:44 +01:00
|
|
|
static void test_json_bad_utf8(void)
|
|
|
|
{
|
|
|
|
const jsmntok_t *toks;
|
|
|
|
char *buf;
|
|
|
|
|
|
|
|
buf = tal_strdup(tmpctx, "{\"1\":\"one\", \"2\":\"two\", \"3\":[\"three\", {\"deeper\": 17}]}");
|
|
|
|
toks = json_parse_simple(tmpctx, buf, strlen(buf));
|
|
|
|
assert(toks);
|
|
|
|
assert(toks->size == 3);
|
|
|
|
|
|
|
|
assert(json_tok_streq(buf, &toks[1], "1"));
|
|
|
|
buf[toks[1].start] = 0xC0;
|
|
|
|
assert(!json_parse_simple(tmpctx, buf, strlen(buf)));
|
|
|
|
buf[toks[1].start] = '1';
|
|
|
|
|
|
|
|
assert(json_tok_streq(buf, &toks[2], "one"));
|
|
|
|
buf[toks[2].start] = 0xC0;
|
|
|
|
assert(!json_parse_simple(tmpctx, buf, strlen(buf)));
|
|
|
|
buf[toks[2].start] = 'o';
|
|
|
|
|
|
|
|
assert(json_tok_streq(buf, &toks[7], "three"));
|
|
|
|
buf[toks[7].start] = 0xC0;
|
|
|
|
assert(!json_parse_simple(tmpctx, buf, strlen(buf)));
|
|
|
|
buf[toks[7].start] = 't';
|
|
|
|
|
|
|
|
assert(json_parse_simple(tmpctx, buf, strlen(buf)));
|
|
|
|
}
|
|
|
|
|
2020-12-02 01:10:04 +01:00
|
|
|
int main(int argc, char *argv[])
|
2018-01-12 17:46:12 +01:00
|
|
|
{
|
2020-12-02 01:10:04 +01:00
|
|
|
common_setup(argv[0]);
|
2018-04-25 12:55:34 +02:00
|
|
|
|
2019-01-17 03:05:26 +01:00
|
|
|
test_json_tok_size();
|
2018-01-12 17:46:12 +01:00
|
|
|
test_json_tok_bitcoin_amount();
|
2020-02-19 11:11:36 +01:00
|
|
|
test_json_tok_millionths();
|
2020-11-25 01:17:44 +01:00
|
|
|
test_json_bad_utf8();
|
2020-12-02 01:10:04 +01:00
|
|
|
|
|
|
|
common_shutdown();
|
2018-01-12 17:46:12 +01:00
|
|
|
}
|