2018-10-19 03:17:48 +02:00
|
|
|
#include "../json_escaped.c"
|
|
|
|
#include "../jsonrpc.c"
|
|
|
|
#include "../json.c"
|
|
|
|
|
|
|
|
/* AUTOGENERATED MOCKS START */
|
|
|
|
/* Generated stub for db_begin_transaction_ */
|
|
|
|
void db_begin_transaction_(struct db *db UNNEEDED, const char *location UNNEEDED)
|
|
|
|
{ fprintf(stderr, "db_begin_transaction_ called!\n"); abort(); }
|
|
|
|
/* Generated stub for db_commit_transaction */
|
|
|
|
void db_commit_transaction(struct db *db UNNEEDED)
|
|
|
|
{ fprintf(stderr, "db_commit_transaction called!\n"); abort(); }
|
|
|
|
/* Generated stub for fatal */
|
|
|
|
void fatal(const char *fmt UNNEEDED, ...)
|
|
|
|
{ fprintf(stderr, "fatal called!\n"); abort(); }
|
|
|
|
/* Generated stub for feerate_from_style */
|
|
|
|
u32 feerate_from_style(u32 feerate UNNEEDED, enum feerate_style style UNNEEDED)
|
|
|
|
{ fprintf(stderr, "feerate_from_style called!\n"); abort(); }
|
|
|
|
/* Generated stub for feerate_name */
|
|
|
|
const char *feerate_name(enum feerate feerate UNNEEDED)
|
|
|
|
{ fprintf(stderr, "feerate_name called!\n"); abort(); }
|
|
|
|
/* Generated stub for fmt_wireaddr_without_port */
|
|
|
|
char *fmt_wireaddr_without_port(const tal_t *ctx UNNEEDED, const struct wireaddr *a UNNEEDED)
|
|
|
|
{ fprintf(stderr, "fmt_wireaddr_without_port called!\n"); abort(); }
|
|
|
|
/* Generated stub for get_block_height */
|
|
|
|
u32 get_block_height(const struct chain_topology *topo UNNEEDED)
|
|
|
|
{ fprintf(stderr, "get_block_height called!\n"); abort(); }
|
|
|
|
/* Generated stub for get_chainparams */
|
|
|
|
const struct chainparams *get_chainparams(const struct lightningd *ld UNNEEDED)
|
|
|
|
{ fprintf(stderr, "get_chainparams called!\n"); abort(); }
|
|
|
|
/* Generated stub for json_feerate_estimate */
|
|
|
|
bool json_feerate_estimate(struct command *cmd UNNEEDED,
|
|
|
|
u32 **feerate_per_kw UNNEEDED, enum feerate feerate UNNEEDED)
|
|
|
|
{ fprintf(stderr, "json_feerate_estimate called!\n"); abort(); }
|
|
|
|
/* Generated stub for log_ */
|
|
|
|
void log_(struct log *log UNNEEDED, enum log_level level UNNEEDED, const char *fmt UNNEEDED, ...)
|
|
|
|
|
|
|
|
{ fprintf(stderr, "log_ called!\n"); abort(); }
|
|
|
|
/* Generated stub for log_io */
|
|
|
|
void log_io(struct log *log UNNEEDED, enum log_level dir UNNEEDED, const char *comment UNNEEDED,
|
|
|
|
const void *data UNNEEDED, size_t len UNNEEDED)
|
|
|
|
{ fprintf(stderr, "log_io called!\n"); abort(); }
|
|
|
|
/* Generated stub for log_prefix */
|
|
|
|
const char *log_prefix(const struct log *log UNNEEDED)
|
|
|
|
{ fprintf(stderr, "log_prefix called!\n"); abort(); }
|
|
|
|
/* Generated stub for new_log */
|
|
|
|
struct log *new_log(const tal_t *ctx UNNEEDED, struct log_book *record UNNEEDED, const char *fmt UNNEEDED, ...)
|
|
|
|
{ fprintf(stderr, "new_log called!\n"); abort(); }
|
|
|
|
/* Generated stub for param */
|
|
|
|
bool param(struct command *cmd UNNEEDED, const char *buffer UNNEEDED,
|
|
|
|
const jsmntok_t params[] UNNEEDED, ...)
|
|
|
|
{ fprintf(stderr, "param called!\n"); abort(); }
|
|
|
|
/* Generated stub for version */
|
|
|
|
const char *version(void)
|
|
|
|
{ fprintf(stderr, "version called!\n"); abort(); }
|
2018-10-20 04:32:29 +02:00
|
|
|
/* Generated stub for wallet_total_forward_fees */
|
|
|
|
u64 wallet_total_forward_fees(struct wallet *w UNNEEDED)
|
|
|
|
{ fprintf(stderr, "wallet_total_forward_fees called!\n"); abort(); }
|
2018-10-19 03:17:48 +02:00
|
|
|
/* AUTOGENERATED MOCKS END */
|
|
|
|
|
|
|
|
bool deprecated_apis;
|
|
|
|
|
|
|
|
static int test_json_filter(void)
|
|
|
|
{
|
2018-10-19 03:17:48 +02:00
|
|
|
struct command *cmd = talz(NULL, struct command);
|
2018-10-19 03:17:48 +02:00
|
|
|
struct json_connection *jcon = talz(cmd, struct json_connection);
|
2018-10-19 03:17:49 +02:00
|
|
|
struct json_stream *result = json_stream_success(cmd);
|
2018-10-19 03:17:48 +02:00
|
|
|
jsmntok_t *toks;
|
|
|
|
const jsmntok_t *x;
|
|
|
|
bool valid;
|
|
|
|
int i;
|
|
|
|
char *badstr = tal_arr(result, char, 256);
|
|
|
|
const char *str;
|
|
|
|
|
2018-10-19 03:17:48 +02:00
|
|
|
/* We need to initialize membuf so we can gather results. */
|
|
|
|
cmd->jcon = jcon;
|
|
|
|
jcon->lock = io_lock_new(jcon);
|
|
|
|
membuf_init(&jcon->outbuf,
|
|
|
|
tal_arr(cmd, char, 64), 64, membuf_tal_realloc);
|
|
|
|
|
2018-10-19 03:17:48 +02:00
|
|
|
/* Fill with junk, and nul-terminate (256 -> 0) */
|
|
|
|
for (i = 1; i < 257; i++)
|
|
|
|
badstr[i-1] = i;
|
|
|
|
|
|
|
|
json_object_start(result, NULL);
|
|
|
|
json_add_string(result, "x", badstr);
|
|
|
|
json_object_end(result);
|
|
|
|
|
|
|
|
/* Parse back in, make sure nothing crazy. */
|
2018-10-19 03:17:48 +02:00
|
|
|
str = tal_strndup(cmd, membuf_elems(&jcon->outbuf),
|
|
|
|
membuf_num_elems(&jcon->outbuf));
|
2018-10-19 03:17:48 +02:00
|
|
|
|
|
|
|
toks = json_parse_input(str, strlen(str), &valid);
|
|
|
|
assert(valid);
|
|
|
|
assert(toks);
|
|
|
|
|
|
|
|
assert(toks[0].type == JSMN_OBJECT);
|
|
|
|
x = json_get_member(str, toks, "x");
|
|
|
|
assert(x);
|
|
|
|
assert(x->type == JSMN_STRING);
|
|
|
|
|
|
|
|
/* There are 7 one-letter escapes, and (32-5) \uXXXX escapes. */
|
|
|
|
assert((x->end - x->start) == 255 + 7*1 + (32-5)*5);
|
|
|
|
/* No control characters. */
|
|
|
|
for (i = x->start; i < x->end; i++) {
|
|
|
|
assert((unsigned)str[i] >= ' ');
|
|
|
|
assert((unsigned)str[i] != 127);
|
|
|
|
}
|
2018-10-19 03:17:48 +02:00
|
|
|
tal_free(cmd);
|
2018-10-19 03:17:48 +02:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void test_json_escape(void)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = 1; i < 256; i++) {
|
|
|
|
char badstr[2];
|
2018-10-19 03:17:48 +02:00
|
|
|
struct command *cmd = talz(NULL, struct command);
|
2018-10-19 03:17:48 +02:00
|
|
|
struct json_connection *jcon = talz(cmd, struct json_connection);
|
2018-10-19 03:17:49 +02:00
|
|
|
struct json_stream *result = json_stream_success(cmd);
|
2018-10-19 03:17:48 +02:00
|
|
|
struct json_escaped *esc;
|
|
|
|
|
|
|
|
badstr[0] = i;
|
|
|
|
badstr[1] = 0;
|
|
|
|
|
2018-10-19 03:17:48 +02:00
|
|
|
/* We need to initialize membuf so we can gather results. */
|
|
|
|
cmd->jcon = jcon;
|
|
|
|
jcon->lock = io_lock_new(jcon);
|
|
|
|
membuf_init(&jcon->outbuf,
|
|
|
|
tal_arr(cmd, char, 64), 64, membuf_tal_realloc);
|
|
|
|
|
2018-10-19 03:17:48 +02:00
|
|
|
json_object_start(result, NULL);
|
|
|
|
esc = json_escape(NULL, badstr);
|
|
|
|
json_add_escaped_string(result, "x", take(esc));
|
|
|
|
json_object_end(result);
|
|
|
|
|
2018-10-19 03:17:48 +02:00
|
|
|
const char *str = tal_strndup(cmd, membuf_elems(&jcon->outbuf),
|
|
|
|
membuf_num_elems(&jcon->outbuf));
|
2018-10-19 03:17:48 +02:00
|
|
|
if (i == '\\' || i == '"'
|
|
|
|
|| i == '\n' || i == '\r' || i == '\b'
|
|
|
|
|| i == '\t' || i == '\f')
|
|
|
|
assert(strstarts(str, "\n{\n \"x\": \"\\"));
|
|
|
|
else if (i < 32 || i == 127) {
|
|
|
|
assert(strstarts(str, "\n{\n \"x\": \"\\u00"));
|
|
|
|
} else {
|
|
|
|
char expect[] = "\n{\n \"x\": \"?\"\n}";
|
|
|
|
expect[11] = i;
|
|
|
|
assert(streq(str, expect));
|
|
|
|
}
|
2018-10-19 03:17:48 +02:00
|
|
|
tal_free(cmd);
|
2018-10-19 03:17:48 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void test_json_partial(void)
|
|
|
|
{
|
|
|
|
const tal_t *ctx = tal(NULL, char);
|
|
|
|
|
|
|
|
assert(streq(json_partial_escape(ctx, "\\")->s, "\\\\"));
|
|
|
|
assert(streq(json_partial_escape(ctx, "\\\\")->s, "\\\\"));
|
|
|
|
assert(streq(json_partial_escape(ctx, "\\\\\\")->s, "\\\\\\\\"));
|
|
|
|
assert(streq(json_partial_escape(ctx, "\\\\\\\\")->s, "\\\\\\\\"));
|
|
|
|
assert(streq(json_partial_escape(ctx, "\\n")->s, "\\n"));
|
|
|
|
assert(streq(json_partial_escape(ctx, "\n")->s, "\\n"));
|
|
|
|
assert(streq(json_partial_escape(ctx, "\\\"")->s, "\\\""));
|
|
|
|
assert(streq(json_partial_escape(ctx, "\"")->s, "\\\""));
|
|
|
|
assert(streq(json_partial_escape(ctx, "\\t")->s, "\\t"));
|
|
|
|
assert(streq(json_partial_escape(ctx, "\t")->s, "\\t"));
|
|
|
|
assert(streq(json_partial_escape(ctx, "\\b")->s, "\\b"));
|
|
|
|
assert(streq(json_partial_escape(ctx, "\b")->s, "\\b"));
|
|
|
|
assert(streq(json_partial_escape(ctx, "\\r")->s, "\\r"));
|
|
|
|
assert(streq(json_partial_escape(ctx, "\r")->s, "\\r"));
|
|
|
|
assert(streq(json_partial_escape(ctx, "\\f")->s, "\\f"));
|
|
|
|
assert(streq(json_partial_escape(ctx, "\f")->s, "\\f"));
|
|
|
|
/* You're allowed to escape / according to json.org. */
|
|
|
|
assert(streq(json_partial_escape(ctx, "\\/")->s, "\\/"));
|
|
|
|
assert(streq(json_partial_escape(ctx, "/")->s, "/"));
|
|
|
|
|
|
|
|
assert(streq(json_partial_escape(ctx, "\\u0FFF")->s, "\\u0FFF"));
|
|
|
|
assert(streq(json_partial_escape(ctx, "\\u0FFFx")->s, "\\u0FFFx"));
|
|
|
|
|
|
|
|
/* Unknown escapes should be escaped. */
|
|
|
|
assert(streq(json_partial_escape(ctx, "\\x")->s, "\\\\x"));
|
|
|
|
tal_free(ctx);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Test that we can segment and parse a stream of json objects correctly. */
|
|
|
|
static void test_json_stream(void)
|
|
|
|
{
|
|
|
|
bool valid;
|
|
|
|
char *input, *talstr;
|
|
|
|
jsmntok_t *toks;
|
|
|
|
|
|
|
|
/* Multiple full messages in a single buffer (happens when buffer
|
|
|
|
* boundary coincides with message boundary, or read returned after
|
|
|
|
* timeout. */
|
|
|
|
input = "{\"x\":\"x\"}{\"y\":\"y\"}";
|
|
|
|
talstr = tal_strndup(NULL, input, strlen(input));
|
|
|
|
toks = json_parse_input(talstr, strlen(talstr), &valid);
|
|
|
|
assert(toks);
|
|
|
|
assert(tal_count(toks) == 4);
|
|
|
|
assert(toks[0].start == 0 && toks[0].end == 9);
|
|
|
|
assert(valid);
|
|
|
|
tal_free(talstr);
|
|
|
|
|
|
|
|
/* Multiple messages, and the last one is partial, far more likely than
|
|
|
|
* accidentally getting the boundaries to match. */
|
|
|
|
input = "{\"x\":\"x\"}{\"y\":\"y\"}{\"z\":\"z";
|
|
|
|
talstr = tal_strndup(NULL, input, strlen(input));
|
|
|
|
toks = json_parse_input(talstr, strlen(talstr), &valid);
|
|
|
|
assert(toks);
|
|
|
|
assert(tal_count(toks) == 4);
|
|
|
|
assert(toks[0].start == 0 && toks[0].end == 9);
|
|
|
|
assert(valid);
|
|
|
|
tal_free(talstr);
|
|
|
|
}
|
|
|
|
|
|
|
|
int main(void)
|
|
|
|
{
|
|
|
|
setup_locale();
|
|
|
|
|
|
|
|
test_json_filter();
|
|
|
|
test_json_escape();
|
|
|
|
test_json_partial();
|
|
|
|
test_json_stream();
|
|
|
|
assert(!taken_any());
|
|
|
|
take_cleanup();
|
|
|
|
}
|