#include "config.h" #include #include #include #include #include #include #include #include #include #include #include char *fail_msg = NULL; bool failed = false; static bool check_fail(void) { if (!failed) return false; failed = false; return true; } struct command *cmd; void command_fail(struct command *cmd, int code, const char *fmt, ...) { failed = true; va_list ap; va_start(ap, fmt); fail_msg = tal_vfmt(cmd, fmt, ap); va_end(ap); } void command_fail_detailed(struct command *cmd, int code, const struct json_result *data, const char *fmt, ...) { failed = true; va_list ap; va_start(ap, fmt); fail_msg = tal_vfmt(cmd, fmt, ap); fail_msg = tal_fmt(cmd, "%s data: %s", fail_msg, json_result_string(data)); va_end(ap); } /* AUTOGENERATED MOCKS START */ /* 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 json_tok_newaddr */ bool json_tok_newaddr(const char *buffer UNNEEDED, const jsmntok_t *tok UNNEEDED, bool *is_p2wpkh UNNEEDED) { fprintf(stderr, "json_tok_newaddr called!\n"); abort(); } /* Generated stub for json_tok_wtx */ bool json_tok_wtx(struct wallet_tx * tx UNNEEDED, const char * buffer UNNEEDED, const jsmntok_t * sattok UNNEEDED, u64 max UNNEEDED) { fprintf(stderr, "json_tok_wtx called!\n"); abort(); } /* AUTOGENERATED MOCKS END */ struct json { jsmntok_t *toks; char *buffer; }; static void convert_quotes(char *first) { while (*first != '\0') { if (*first == '\'') *first = '"'; first++; } } static struct json *json_parse(const tal_t * ctx, const char *str) { struct json *j = tal(ctx, struct json); j->buffer = tal_strdup(j, str); convert_quotes(j->buffer); j->toks = tal_arr(j, jsmntok_t, 50); assert(j->toks); jsmn_parser parser; again: jsmn_init(&parser); int ret = jsmn_parse(&parser, j->buffer, strlen(j->buffer), j->toks, tal_count(j->toks)); if (ret == JSMN_ERROR_NOMEM) { tal_resize(&j->toks, tal_count(j->toks) * 2); goto again; } if (ret <= 0) { assert(0); } return j; } static void zero_params(void) { struct json *j = json_parse(cmd, "{}"); assert(param(cmd, j->buffer, j->toks, NULL)); j = json_parse(cmd, "[]"); assert(param(cmd, j->buffer, j->toks, NULL)); } struct sanity { char *str; bool failed; int ival; double dval; char *fail_str; }; struct sanity buffers[] = { // pass {"['42', '3.15']", false, 42, 3.15, NULL}, {"{ 'u64' : '42', 'double' : '3.15' }", false, 42, 3.15, NULL}, // fail {"{'u64':'42', 'double':'3.15', 'extra':'stuff'}", true, 0, 0, "unknown parameter"}, {"['42', '3.15', 'stuff']", true, 0, 0, "too many"}, {"['42', '3.15', 'null']", true, 0, 0, "too many"}, // not enough {"{'u64':'42'}", true, 0, 0, "missing required"}, {"['42']", true, 0, 0, "missing required"}, // fail wrong type {"{'u64':'hello', 'double':'3.15'}", true, 0, 0, "\"u64\": \"hello\""}, {"['3.15', '3.15', 'stuff']", true, 0, 0, "integer"}, }; static void stest(const struct json *j, struct sanity *b) { u64 ival; double dval; if (!param(cmd, j->buffer, j->toks, p_req("u64", json_tok_u64, &ival), p_req("double", json_tok_double, &dval), NULL)) { assert(check_fail()); assert(b->failed == true); assert(strstr(fail_msg, b->fail_str)); } else { assert(!check_fail()); assert(b->failed == false); assert(ival == 42); assert(dval > 3.1499 && b->dval < 3.1501); } } static void sanity(void) { for (int i = 0; i < ARRAY_SIZE(buffers); ++i) { struct json *j = json_parse(cmd, buffers[i].str); assert(j->toks->type == JSMN_OBJECT || j->toks->type == JSMN_ARRAY); stest(j, &buffers[i]); } } /* * Make sure toks are passed through correctly, and also make sure * optional missing toks are set to NULL. */ static void tok_tok(void) { { unsigned int n; const jsmntok_t *tok = NULL; struct json *j = json_parse(cmd, "{ 'satoshi', '546' }"); assert(param(cmd, j->buffer, j->toks, p_req_tal("satoshi", json_tok_tok, &tok), NULL)); assert(tok); assert(json_to_number(j->buffer, tok, &n)); assert(n == 546); } // again with missing optional parameter { /* make sure it is *not* NULL */ const jsmntok_t *tok = (const jsmntok_t *) 65535; struct json *j = json_parse(cmd, "{}"); assert(param(cmd, j->buffer, j->toks, p_opt_tal("satoshi", json_tok_tok, &tok), NULL)); /* make sure it *is* NULL */ assert(tok == NULL); } } /* check for valid but duplicate json name-value pairs */ static void dup_names(void) { struct json *j = json_parse(cmd, "{ 'u64' : '42', 'u64' : '43', 'double' : '3.15' }"); u64 i; double d; assert(!param(cmd, j->buffer, j->toks, p_req("u64", json_tok_u64, &i), p_req("double", json_tok_double, &d), NULL)); } static void null_params(void) { uint64_t *ints = tal_arr(cmd, uint64_t, 5); uint64_t **intptrs = tal_arr(cmd, uint64_t *, 2); /* no null params */ struct json *j = json_parse(cmd, "[ '10', '11', '12', '13', '14', '15', '16']"); for (int i = 0; i < tal_count(ints) - 1; ++i) ints[i] = i; assert(param(cmd, j->buffer, j->toks, p_req("0", json_tok_u64, &ints[0]), p_req("1", json_tok_u64, &ints[1]), p_req("2", json_tok_u64, &ints[2]), p_req("3", json_tok_u64, &ints[3]), p_opt_def("4", json_tok_u64, &ints[4], 999), p_opt("5", json_tok_u64, &intptrs[0]), p_opt("6", json_tok_u64, &intptrs[1]), NULL)); for (int i = 0; i < tal_count(ints); ++i) assert(ints[i] == i + 10); for (int i = 0; i < tal_count(intptrs); ++i) assert(*intptrs[i] == i + 10 + tal_count(ints)); /* missing at end */ for (int i = 0; i < tal_count(ints); ++i) ints[i] = 42; for (int i = 0; i < tal_count(intptrs); ++i) intptrs[i] = (void *)42; j = json_parse(cmd, "[ '10', '11', '12', '13', '14']"); assert(param(cmd, j->buffer, j->toks, p_req("0", json_tok_u64, &ints[0]), p_req("1", json_tok_u64, &ints[1]), p_req("2", json_tok_u64, &ints[2]), p_req("3", json_tok_u64, &ints[3]), p_opt("4", json_tok_u64, &intptrs[0]), p_opt("5", json_tok_u64, &intptrs[1]), p_opt_def("6", json_tok_u64, &ints[4], 888), NULL)); assert(*intptrs[0] == 14); assert(intptrs[1] == NULL); assert(ints[4] == 888); } #if DEVELOPER /* * Check to make sure there are no programming mistakes. */ static void bad_programmer(void) { u64 ival; u64 ival2; double dval; struct json *j = json_parse(cmd, "[ '25', '546', '26' ]"); /* check for repeated names */ assert(!param(cmd, j->buffer, j->toks, p_req("repeat", json_tok_u64, &ival), p_req("double", json_tok_double, &dval), p_req("repeat", json_tok_u64, &ival2), NULL)); assert(check_fail()); assert(strstr(fail_msg, "developer error")); assert(!param(cmd, j->buffer, j->toks, p_req("repeat", json_tok_u64, &ival), p_req("double", json_tok_double, &dval), p_req("repeat", json_tok_u64, &ival), NULL)); assert(check_fail()); assert(strstr(fail_msg, "developer error")); assert(!param(cmd, j->buffer, j->toks, p_req("u64", json_tok_u64, &ival), p_req("repeat", json_tok_double, &dval), p_req("repeat", json_tok_double, &dval), NULL)); assert(check_fail()); assert(strstr(fail_msg, "developer error")); /* check for repeated arguments */ assert(!param(cmd, j->buffer, j->toks, p_req("u64", json_tok_u64, &ival), p_req("repeated-arg", json_tok_u64, &ival), NULL)); assert(check_fail()); assert(strstr(fail_msg, "developer error")); assert(!param(cmd, j->buffer, j->toks, p_req("u64", (param_cb) NULL, &ival), NULL)); assert(check_fail()); assert(strstr(fail_msg, "developer error")); /* Add required param after optional */ j = json_parse(cmd, "[ '25', '546', '26', '1.1' ]"); unsigned int *msatoshi; double riskfactor; assert(!param(cmd, j->buffer, j->toks, p_req("u64", json_tok_u64, &ival), p_req("double", json_tok_double, &dval), p_opt_def_tal("msatoshi", json_tok_number, &msatoshi, 100), p_req("riskfactor", json_tok_double, &riskfactor), NULL)); assert(*msatoshi); assert(*msatoshi == 100); assert(check_fail()); assert(strstr(fail_msg, "developer error")); } #endif static void add_members(struct param **params, struct json_result *obj, struct json_result *arr, unsigned int **ints) { for (int i = 0; i < tal_count(ints); ++i) { char *name = tal_fmt(tmpctx, "%i", i); json_add_num(obj, name, i); json_add_num(arr, NULL, i); param_add(params, name, true, NULL, typesafe_cb_preargs(bool, void **, json_tok_number, &ints[i], struct command *, const char *, const char *, const jsmntok_t *), &ints[i], 0); } } /* * A roundabout way of initializing an array of ints to: * ints[0] = 0, ints[1] = 1, ... ints[499] = 499 */ static void five_hundred_params(void) { struct param *params = tal_arr(NULL, struct param, 0); unsigned int **ints = tal_arr(params, unsigned int*, 500); struct json_result *obj = new_json_result(params); struct json_result *arr = new_json_result(params); json_object_start(obj, NULL); json_array_start(arr, NULL); add_members(¶ms, obj, arr, ints); json_object_end(obj); json_array_end(arr); /* first test object version */ struct json *j = json_parse(params, obj->s); assert(param_arr(cmd, j->buffer, j->toks, params)); for (int i = 0; i < tal_count(ints); ++i) { assert(ints[i]); assert(*ints[i] == i); *ints[i] = 65535; } /* now test array */ j = json_parse(params, arr->s); assert(param_arr(cmd, j->buffer, j->toks, params)); for (int i = 0; i < tal_count(ints); ++i) { assert(*ints[i] == i); } tal_free(params); } static void sendpay(void) { struct json *j = json_parse(cmd, "[ 'A', '123', 'hello there' '547']"); const jsmntok_t *routetok, *note; u64 *msatoshi; unsigned *cltv; if (!param(cmd, j->buffer, j->toks, p_req_tal("route", json_tok_tok, &routetok), p_req_tal("cltv", json_tok_number, &cltv), p_opt_tal("note", json_tok_tok, ¬e), p_opt("msatoshi", json_tok_u64, &msatoshi), NULL)) assert(false); assert(note); assert(!strncmp("hello there", j->buffer + note->start, note->end - note->start)); assert(msatoshi); assert(*msatoshi == 547); } static void sendpay_nulltok(void) { struct json *j = json_parse(cmd, "[ 'A', '123']"); const jsmntok_t *routetok, *note = (void *) 65535; u64 *msatoshi; unsigned *cltv; if (!param(cmd, j->buffer, j->toks, p_req_tal("route", json_tok_tok, &routetok), p_req_tal("cltv", json_tok_number, &cltv), p_opt_tal("note", json_tok_tok, ¬e), p_opt("msatoshi", json_tok_u64, &msatoshi), NULL)) assert(false); assert(note == NULL); assert(msatoshi == NULL); } static bool json_tok_msat(struct command *cmd, const char *name, const char *buffer, const jsmntok_t * tok, u64 **msatoshi_val) { if (json_tok_streq(buffer, tok, "any")) { *msatoshi_val = NULL; return true; } *msatoshi_val = tal(cmd, u64); if (json_tok_u64(buffer, tok, *msatoshi_val) && *msatoshi_val != 0) return true; command_fail(cmd, JSONRPC2_INVALID_PARAMS, "'%s' should be a positive number or 'any', not '%.*s'", name, tok->end - tok->start, buffer + tok->start); return false; } /* * New version of json_tok_label conforming to advanced style. This can eventually * replace the existing json_tok_label. */ static bool json_tok_label_x(struct command *cmd, const char *name, const char *buffer, const jsmntok_t *tok, struct json_escaped **label) { if ((*label = json_tok_escaped_string(cmd, buffer, tok))) return true; /* Allow literal numbers */ if (tok->type != JSMN_PRIMITIVE) goto fail; for (int i = tok->start; i < tok->end; i++) if (!cisdigit(buffer[i])) goto fail; if ((*label = json_escaped_string_(cmd, buffer + tok->start, tok->end - tok->start))) return true; fail: command_fail(cmd, JSONRPC2_INVALID_PARAMS, "'%s' should be a string or number, not '%.*s'", name, tok->end - tok->start, buffer + tok->start); return false; } static void advanced(void) { { struct json *j = json_parse(cmd, "[ 'lightning', 24, 'tok', 543 ]"); struct json_escaped *label; u64 *msat; u64 *msat_opt1, *msat_opt2; const jsmntok_t *tok; assert(param(cmd, j->buffer, j->toks, p_req_tal("description", json_tok_label_x, &label), p_req_tal("msat", json_tok_msat, &msat), p_req_tal("tok", json_tok_tok, &tok), p_opt_tal("msat_opt1", json_tok_msat, &msat_opt1), p_opt_tal("msat_opt2", json_tok_msat, &msat_opt2), NULL)); assert(label != NULL); assert(!strcmp(label->s, "lightning")); assert(*msat == 24); assert(tok); assert(msat_opt1); assert(*msat_opt1 == 543); assert(msat_opt2 == NULL); } { struct json *j = json_parse(cmd, "[ 3 'foo' ]"); struct json_escaped *label, *foo; assert(param(cmd, j->buffer, j->toks, p_req_tal("label", json_tok_label_x, &label), p_opt_tal("foo", json_tok_label_x, &foo), NULL)); assert(!strcmp(label->s, "3")); assert(!strcmp(foo->s, "foo")); } { u64 *msat; u64 *msat2; struct json *j = json_parse(cmd, "[ 3 ]"); assert(param(cmd, j->buffer, j->toks, p_opt_def_tal("msat", json_tok_msat, &msat, 23), p_opt_def_tal("msat2", json_tok_msat, &msat2, 53), NULL)); assert(*msat == 3); assert(msat2); assert(*msat2 == 53); } } static void advanced_fail(void) { { struct json *j = json_parse(cmd, "[ 'anyx' ]"); u64 *msat; assert(!param(cmd, j->buffer, j->toks, p_req_tal("msat", json_tok_msat, &msat), NULL)); assert(check_fail()); assert(strstr(fail_msg, "'msat' should be a positive" " number or 'any', not 'anyx'")); } } int main(void) { setup_locale(); setup_tmpctx(); cmd = tal(tmpctx, struct command); fail_msg = tal_arr(cmd, char, 10000); zero_params(); sanity(); tok_tok(); null_params(); #if DEVELOPER bad_programmer(); #endif dup_names(); five_hundred_params(); sendpay(); sendpay_nulltok(); advanced(); advanced_fail(); tal_free(tmpctx); printf("run-params ok\n"); }