2021-12-04 12:23:56 +01:00
|
|
|
#include "config.h"
|
2019-04-10 06:58:24 +02:00
|
|
|
#include <ccan/array_size/array_size.h>
|
2020-09-08 06:57:53 +02:00
|
|
|
#include <ccan/asort/asort.h>
|
2021-09-16 07:00:42 +02:00
|
|
|
#include <ccan/cast/cast.h>
|
2022-01-03 19:45:35 +01:00
|
|
|
#include <ccan/json_escape/json_escape.h>
|
2016-09-06 09:17:49 +02:00
|
|
|
#include <ccan/str/hex/hex.h>
|
|
|
|
#include <ccan/tal/str/str.h>
|
2021-01-07 19:47:47 +01:00
|
|
|
#include <common/bolt11_json.h>
|
2020-12-14 02:21:48 +01:00
|
|
|
#include <common/bolt12_merkle.h>
|
2020-01-31 06:40:26 +01:00
|
|
|
#include <common/configdir.h>
|
2018-12-08 01:39:28 +01:00
|
|
|
#include <common/json_command.h>
|
2019-01-15 05:00:27 +01:00
|
|
|
#include <common/json_helpers.h>
|
2021-09-16 07:00:42 +02:00
|
|
|
#include <common/json_tok.h>
|
2021-06-17 16:00:17 +02:00
|
|
|
#include <common/onion.h>
|
2019-04-10 06:58:24 +02:00
|
|
|
#include <common/overflows.h>
|
2018-12-08 01:39:28 +01:00
|
|
|
#include <common/param.h>
|
2020-08-11 07:05:56 +02:00
|
|
|
#include <common/random_select.h>
|
2020-01-28 02:30:00 +01:00
|
|
|
#include <common/timeout.h>
|
2021-09-16 07:00:42 +02:00
|
|
|
#include <common/type_to_string.h>
|
2022-01-03 19:45:35 +01:00
|
|
|
#include <db/exec.h>
|
2017-11-22 01:25:39 +01:00
|
|
|
#include <errno.h>
|
2020-08-25 03:55:38 +02:00
|
|
|
#include <hsmd/hsmd_wiregen.h>
|
json-invoice: add routeboost, warnings.
We split json_invoice(), as it now needs to round-trip to the gossipd,
and uniqueness checks need to happen *after* gossipd replies to avoid
a race.
For every candidate channel gossipd gives us, we check that it's in
state NORMAL (not shutting down, not still waiting for lockin), that
it's connected, and that it has capacity. We then choose one with
probability weighted by excess capacity, so larger channels are more
likely.
As a side effect of this, we can tell if an invoice is unpayble (no
channels have sufficient incoming capacity) or difficuly (no *online*
channels have sufficient capacity), so we add those warnings.
Signed-off-by: Rusty Russell <rusty@rustcorp.com.au>
2018-09-27 07:29:17 +02:00
|
|
|
#include <lightningd/channel.h>
|
2021-12-04 12:23:56 +01:00
|
|
|
#include <lightningd/invoice.h>
|
2019-07-21 13:39:15 +02:00
|
|
|
#include <lightningd/notification.h>
|
2019-04-11 02:10:27 +02:00
|
|
|
#include <lightningd/plugin_hook.h>
|
2020-09-08 06:57:53 +02:00
|
|
|
#include <lightningd/routehint.h>
|
2016-09-06 09:17:49 +02:00
|
|
|
#include <sodium/randombytes.h>
|
2017-11-22 01:25:39 +01:00
|
|
|
#include <wire/wire_sync.h>
|
2016-09-06 09:17:49 +02:00
|
|
|
|
2018-02-20 02:07:30 +01:00
|
|
|
static const char *invoice_status_str(const struct invoice_details *inv)
|
2018-01-17 04:04:29 +01:00
|
|
|
{
|
|
|
|
if (inv->state == PAID)
|
|
|
|
return "paid";
|
2018-01-23 02:02:10 +01:00
|
|
|
if (inv->state == EXPIRED)
|
2018-01-17 04:04:29 +01:00
|
|
|
return "expired";
|
|
|
|
return "unpaid";
|
|
|
|
}
|
|
|
|
|
2018-10-19 03:17:49 +02:00
|
|
|
static void json_add_invoice(struct json_stream *response,
|
2018-07-29 04:08:29 +02:00
|
|
|
const struct invoice_details *inv)
|
2018-01-13 12:14:13 +01:00
|
|
|
{
|
2018-03-26 02:08:16 +02:00
|
|
|
json_add_escaped_string(response, "label", inv->label);
|
2022-01-18 15:13:44 +01:00
|
|
|
if (inv->invstring)
|
|
|
|
json_add_invstring(response, inv->invstring);
|
2019-08-10 10:39:04 +02:00
|
|
|
json_add_sha256(response, "payment_hash", &inv->rhash);
|
2019-02-21 04:45:55 +01:00
|
|
|
if (inv->msat)
|
2019-05-20 07:07:40 +02:00
|
|
|
json_add_amount_msat_compat(response, *inv->msat,
|
|
|
|
"msatoshi", "amount_msat");
|
2018-07-29 04:08:29 +02:00
|
|
|
json_add_string(response, "status", invoice_status_str(inv));
|
2018-01-13 12:19:33 +01:00
|
|
|
if (inv->state == PAID) {
|
2018-01-13 12:14:13 +01:00
|
|
|
json_add_u64(response, "pay_index", inv->pay_index);
|
2019-05-20 07:07:40 +02:00
|
|
|
json_add_amount_msat_compat(response, inv->received,
|
|
|
|
"msatoshi_received",
|
|
|
|
"amount_received_msat");
|
2018-01-19 10:53:24 +01:00
|
|
|
json_add_u64(response, "paid_at", inv->paid_timestamp);
|
2019-11-26 02:48:03 +01:00
|
|
|
json_add_preimage(response, "payment_preimage", &inv->r);
|
2018-01-13 12:19:33 +01:00
|
|
|
}
|
2018-07-24 20:56:21 +02:00
|
|
|
if (inv->description)
|
|
|
|
json_add_string(response, "description", inv->description);
|
|
|
|
|
2018-01-19 10:53:24 +01:00
|
|
|
json_add_u64(response, "expires_at", inv->expiry_time);
|
2021-07-02 02:11:35 +02:00
|
|
|
if (inv->local_offer_id) {
|
|
|
|
char *fail;
|
|
|
|
struct tlv_invoice *tinv;
|
|
|
|
|
2020-12-14 02:18:24 +01:00
|
|
|
json_add_sha256(response, "local_offer_id", inv->local_offer_id);
|
2021-07-02 02:11:35 +02:00
|
|
|
|
|
|
|
/* Everyone loves seeing their own payer notes!
|
|
|
|
* Well: they will. Trust me. */
|
|
|
|
tinv = invoice_decode(tmpctx,
|
|
|
|
inv->invstring, strlen(inv->invstring),
|
|
|
|
NULL, NULL, &fail);
|
|
|
|
if (tinv && tinv->payer_note)
|
|
|
|
json_add_stringn(response, "payer_note",
|
|
|
|
tinv->payer_note,
|
|
|
|
tal_bytelen(tinv->payer_note));
|
|
|
|
}
|
2018-01-13 12:14:13 +01:00
|
|
|
}
|
|
|
|
|
2018-12-16 05:52:06 +01:00
|
|
|
static struct command_result *tell_waiter(struct command *cmd,
|
|
|
|
const struct invoice *inv)
|
2016-09-06 09:17:49 +02:00
|
|
|
{
|
2018-10-19 03:17:49 +02:00
|
|
|
struct json_stream *response;
|
2018-07-27 12:57:02 +02:00
|
|
|
const struct invoice_details *details;
|
2016-09-06 09:17:49 +02:00
|
|
|
|
2018-07-27 12:57:02 +02:00
|
|
|
details = wallet_invoice_details(cmd, cmd->ld->wallet, *inv);
|
2018-10-19 03:17:48 +02:00
|
|
|
if (details->state == PAID) {
|
|
|
|
response = json_stream_success(cmd);
|
|
|
|
json_add_invoice(response, details);
|
2018-12-16 05:52:06 +01:00
|
|
|
return command_success(cmd, response);
|
2018-10-19 03:17:48 +02:00
|
|
|
} else {
|
2020-01-26 13:52:29 +01:00
|
|
|
response = json_stream_fail(cmd, INVOICE_EXPIRED_DURING_WAIT,
|
2018-10-19 03:17:48 +02:00
|
|
|
"invoice expired during wait");
|
|
|
|
json_add_invoice(response, details);
|
2019-06-12 02:38:54 +02:00
|
|
|
json_object_end(response);
|
2018-12-16 05:52:06 +01:00
|
|
|
return command_failed(cmd, response);
|
2018-05-24 23:40:18 +02:00
|
|
|
}
|
2016-09-06 09:17:49 +02:00
|
|
|
}
|
2018-10-19 03:17:48 +02:00
|
|
|
|
2018-01-14 15:15:30 +01:00
|
|
|
static void tell_waiter_deleted(struct command *cmd)
|
2017-12-27 13:55:22 +01:00
|
|
|
{
|
2018-12-16 05:53:06 +01:00
|
|
|
was_pending(command_fail(cmd, LIGHTNINGD,
|
|
|
|
"Invoice deleted during wait"));
|
2017-12-27 13:55:22 +01:00
|
|
|
}
|
2018-01-14 15:15:30 +01:00
|
|
|
static void wait_on_invoice(const struct invoice *invoice, void *cmd)
|
2016-09-06 09:17:49 +02:00
|
|
|
{
|
2018-01-14 15:15:30 +01:00
|
|
|
if (invoice)
|
|
|
|
tell_waiter((struct command *) cmd, invoice);
|
|
|
|
else
|
|
|
|
tell_waiter_deleted((struct command *) cmd);
|
2016-09-06 09:17:49 +02:00
|
|
|
}
|
2020-01-28 02:30:00 +01:00
|
|
|
static void wait_timed_out(struct command *cmd)
|
|
|
|
{
|
|
|
|
was_pending(command_fail(cmd, INVOICE_WAIT_TIMED_OUT,
|
|
|
|
"Timed out while waiting "
|
|
|
|
"for invoice to be paid"));
|
|
|
|
}
|
2016-11-11 00:02:04 +01:00
|
|
|
|
2019-11-23 01:19:23 +01:00
|
|
|
/* We derive invoice secret using 1-way function from payment_preimage
|
|
|
|
* (just a different one from the payment_hash!) */
|
|
|
|
static void invoice_secret(const struct preimage *payment_preimage,
|
|
|
|
struct secret *payment_secret)
|
|
|
|
{
|
|
|
|
struct preimage modified;
|
|
|
|
struct sha256 secret;
|
|
|
|
|
|
|
|
modified = *payment_preimage;
|
|
|
|
modified.r[0] ^= 1;
|
|
|
|
|
|
|
|
sha256(&secret, modified.r,
|
|
|
|
ARRAY_SIZE(modified.r) * sizeof(*modified.r));
|
|
|
|
BUILD_ASSERT(sizeof(secret.u.u8) == sizeof(payment_secret->data));
|
|
|
|
memcpy(payment_secret->data, secret.u.u8, sizeof(secret.u.u8));
|
|
|
|
}
|
|
|
|
|
2020-12-16 04:18:42 +01:00
|
|
|
/* FIXME: This is a hack. The real secret should be a signature of some
|
|
|
|
* onion key, using the payer_id */
|
|
|
|
static void invoice_secret_bolt12(struct lightningd *ld,
|
|
|
|
const char *invstring,
|
|
|
|
struct secret *payment_secret)
|
|
|
|
{
|
|
|
|
char *fail;
|
|
|
|
struct tlv_invoice *inv;
|
|
|
|
struct sha256 merkle;
|
|
|
|
|
|
|
|
inv = invoice_decode(tmpctx, invstring, strlen(invstring),
|
|
|
|
NULL, NULL, &fail);
|
|
|
|
if (!inv) {
|
|
|
|
log_broken(ld->log, "Unable to decode our invoice %s",
|
|
|
|
invstring);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
merkle_tlv(inv->fields, &merkle);
|
|
|
|
BUILD_ASSERT(sizeof(*payment_secret) == sizeof(merkle));
|
|
|
|
memcpy(payment_secret, &merkle, sizeof(merkle));
|
|
|
|
}
|
|
|
|
|
2019-04-11 02:10:27 +02:00
|
|
|
struct invoice_payment_hook_payload {
|
|
|
|
struct lightningd *ld;
|
|
|
|
/* Set to NULL if it is deleted while waiting for plugin */
|
2019-12-12 00:55:45 +01:00
|
|
|
struct htlc_set *set;
|
2019-04-11 02:10:27 +02:00
|
|
|
/* What invoice it's trying to pay. */
|
2019-06-12 02:38:54 +02:00
|
|
|
const struct json_escape *label;
|
2019-04-11 02:10:27 +02:00
|
|
|
/* Amount it's offering. */
|
|
|
|
struct amount_msat msat;
|
|
|
|
/* Preimage we'll give it if succeeds. */
|
|
|
|
struct preimage preimage;
|
|
|
|
/* FIXME: Include raw payload! */
|
|
|
|
};
|
|
|
|
|
2021-06-17 16:00:17 +02:00
|
|
|
#ifdef DEVELOPER
|
|
|
|
static void invoice_payment_add_tlvs(struct json_stream *stream,
|
|
|
|
struct htlc_set *hset)
|
|
|
|
{
|
|
|
|
struct htlc_in *hin;
|
|
|
|
struct tlv_tlv_payload *tlvs;
|
|
|
|
assert(tal_count(hset->htlcs) > 0);
|
|
|
|
|
|
|
|
/* Pick the first HTLC as representative for the entire set. */
|
|
|
|
hin = hset->htlcs[0];
|
|
|
|
|
|
|
|
if (hin->payload->type != ONION_TLV_PAYLOAD)
|
|
|
|
return;
|
|
|
|
tlvs = hin->payload->tlv;
|
|
|
|
|
|
|
|
json_array_start(stream, "extratlvs");
|
|
|
|
|
|
|
|
for (size_t i = 0; i < tal_count(tlvs->fields); i++) {
|
|
|
|
struct tlv_field *field = &tlvs->fields[i];
|
|
|
|
/* If we have metadata attached it is not an extra TLV field. */
|
|
|
|
if (field->meta == NULL) {
|
|
|
|
json_object_start(stream, NULL);
|
|
|
|
json_add_u64(stream, "type", field->numtype);
|
|
|
|
json_add_num(stream, "length", field->length);
|
|
|
|
json_add_hex_talarr(stream, "value", field->value);
|
|
|
|
json_object_end(stream);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
json_array_end(stream);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2019-04-11 02:10:27 +02:00
|
|
|
static void
|
|
|
|
invoice_payment_serialize(struct invoice_payment_hook_payload *payload,
|
2021-06-02 17:33:23 +02:00
|
|
|
struct json_stream *stream,
|
|
|
|
struct plugin *plugin)
|
2019-04-11 02:10:27 +02:00
|
|
|
{
|
|
|
|
json_object_start(stream, "payment");
|
|
|
|
json_add_escaped_string(stream, "label", payload->label);
|
2019-11-26 02:33:19 +01:00
|
|
|
json_add_preimage(stream, "preimage", &payload->preimage);
|
2019-04-11 02:10:27 +02:00
|
|
|
json_add_string(stream, "msat",
|
|
|
|
type_to_string(tmpctx, struct amount_msat,
|
|
|
|
&payload->msat));
|
2021-06-17 16:00:17 +02:00
|
|
|
#ifdef DEVELOPER
|
|
|
|
invoice_payment_add_tlvs(stream, payload->set);
|
|
|
|
#endif
|
2019-04-11 02:10:27 +02:00
|
|
|
json_object_end(stream); /* .payment */
|
|
|
|
}
|
|
|
|
|
2019-12-12 00:55:45 +01:00
|
|
|
/* Set times out or HTLC deleted? Remove set ptr from payload so we
|
|
|
|
* know to ignore plugin return */
|
|
|
|
static void invoice_payload_remove_set(struct htlc_set *set,
|
2019-04-16 02:10:59 +02:00
|
|
|
struct invoice_payment_hook_payload *payload)
|
|
|
|
{
|
2019-12-12 00:55:45 +01:00
|
|
|
assert(payload->set == set);
|
|
|
|
payload->set = NULL;
|
2019-04-16 02:10:59 +02:00
|
|
|
}
|
|
|
|
|
2020-02-21 06:08:39 +01:00
|
|
|
static const u8 *hook_gives_failmsg(const tal_t *ctx,
|
2020-04-15 12:19:46 +02:00
|
|
|
struct lightningd *ld,
|
2020-02-21 06:08:39 +01:00
|
|
|
const struct htlc_in *hin,
|
|
|
|
const char *buffer,
|
|
|
|
const jsmntok_t *toks)
|
2019-04-11 02:10:27 +02:00
|
|
|
{
|
2020-01-31 06:40:26 +01:00
|
|
|
const jsmntok_t *resulttok;
|
2019-04-16 02:16:22 +02:00
|
|
|
const jsmntok_t *t;
|
2019-04-11 02:10:27 +02:00
|
|
|
unsigned int val;
|
|
|
|
|
2019-04-16 02:10:59 +02:00
|
|
|
/* No plugin registered on hook at all? */
|
|
|
|
if (!buffer)
|
2020-02-21 06:08:39 +01:00
|
|
|
return NULL;
|
2019-04-16 02:10:59 +02:00
|
|
|
|
2020-01-31 06:40:26 +01:00
|
|
|
resulttok = json_get_member(buffer, toks, "result");
|
|
|
|
if (resulttok) {
|
|
|
|
if (json_tok_streq(buffer, resulttok, "continue")) {
|
2020-02-21 06:08:39 +01:00
|
|
|
return NULL;
|
2020-01-31 06:40:26 +01:00
|
|
|
} else if (json_tok_streq(buffer, resulttok, "reject")) {
|
2020-04-15 12:19:46 +02:00
|
|
|
return failmsg_incorrect_or_unknown(ctx, ld, hin);
|
2020-01-31 06:40:26 +01:00
|
|
|
} else
|
|
|
|
fatal("Invalid invoice_payment hook result: %.*s",
|
|
|
|
toks[0].end - toks[0].start, buffer);
|
|
|
|
}
|
|
|
|
|
2020-02-21 06:10:45 +01:00
|
|
|
t = json_get_member(buffer, toks, "failure_message");
|
|
|
|
if (t) {
|
|
|
|
const u8 *failmsg = json_tok_bin_from_hex(ctx, buffer, t);
|
|
|
|
if (!failmsg)
|
|
|
|
fatal("Invalid invoice_payment_hook failure_message: %.*s",
|
|
|
|
toks[0].end - toks[1].start, buffer);
|
|
|
|
return failmsg;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!deprecated_apis)
|
|
|
|
return NULL;
|
|
|
|
|
2019-04-16 02:16:22 +02:00
|
|
|
t = json_get_member(buffer, toks, "failure_code");
|
2020-02-21 06:10:45 +01:00
|
|
|
if (!t) {
|
2020-01-31 06:40:26 +01:00
|
|
|
static bool warned = false;
|
|
|
|
if (!warned) {
|
|
|
|
warned = true;
|
2020-04-15 12:19:46 +02:00
|
|
|
log_unusual(ld->log,
|
2020-01-31 06:40:26 +01:00
|
|
|
"Plugin did not return object with "
|
2020-02-21 06:10:45 +01:00
|
|
|
"'result' or 'failure_message' fields. "
|
2020-01-31 06:40:26 +01:00
|
|
|
"This is now deprecated and you should "
|
|
|
|
"return {'result': 'continue' } or "
|
|
|
|
"{'result': 'reject'} or "
|
2020-02-21 06:10:45 +01:00
|
|
|
"{'failure_message'... instead.");
|
2020-01-31 06:40:26 +01:00
|
|
|
}
|
2020-04-15 12:19:46 +02:00
|
|
|
return failmsg_incorrect_or_unknown(ctx, ld, hin);
|
2020-01-31 06:40:26 +01:00
|
|
|
}
|
2019-04-16 02:10:59 +02:00
|
|
|
|
2019-04-11 02:10:27 +02:00
|
|
|
if (!json_to_number(buffer, t, &val))
|
|
|
|
fatal("Invalid invoice_payment_hook failure_code: %.*s",
|
|
|
|
toks[0].end - toks[1].start, buffer);
|
|
|
|
|
2020-02-21 06:08:39 +01:00
|
|
|
if (val == WIRE_TEMPORARY_NODE_FAILURE)
|
|
|
|
return towire_temporary_node_failure(ctx);
|
|
|
|
if (val != WIRE_INCORRECT_OR_UNKNOWN_PAYMENT_DETAILS)
|
|
|
|
log_broken(hin->key.channel->log,
|
|
|
|
"invoice_payment hook returned failcode %u,"
|
|
|
|
" changing to incorrect_or_unknown_payment_details",
|
|
|
|
val);
|
|
|
|
|
2020-04-15 12:19:46 +02:00
|
|
|
return failmsg_incorrect_or_unknown(ctx, ld, hin);
|
2019-04-11 02:10:27 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2020-11-25 01:18:49 +01:00
|
|
|
invoice_payment_hooks_done(struct invoice_payment_hook_payload *payload STEALS)
|
2019-04-11 02:10:27 +02:00
|
|
|
{
|
|
|
|
struct invoice invoice;
|
2020-11-25 01:18:49 +01:00
|
|
|
struct lightningd *ld = payload->ld;
|
2019-07-21 13:39:15 +02:00
|
|
|
|
2019-12-12 00:55:45 +01:00
|
|
|
tal_del_destructor2(payload->set, invoice_payload_remove_set, payload);
|
2019-04-11 02:10:27 +02:00
|
|
|
/* We want to free this, whatever happens. */
|
|
|
|
tal_steal(tmpctx, payload);
|
|
|
|
|
|
|
|
/* If invoice gets paid meanwhile (plugin responds out-of-order?) then
|
|
|
|
* we can also fail */
|
|
|
|
if (!wallet_invoice_find_by_label(ld->wallet, &invoice, payload->label)) {
|
2020-02-21 06:08:39 +01:00
|
|
|
htlc_set_fail(payload->set, take(failmsg_incorrect_or_unknown(
|
2020-04-15 12:19:46 +02:00
|
|
|
NULL, ld, payload->set->htlcs[0])));
|
2019-04-11 02:10:27 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-12-02 03:30:59 +01:00
|
|
|
/* Paid or expired in the meantime. */
|
|
|
|
if (!wallet_invoice_resolve(ld->wallet, invoice, payload->msat)) {
|
|
|
|
htlc_set_fail(payload->set, take(failmsg_incorrect_or_unknown(
|
|
|
|
NULL, ld, payload->set->htlcs[0])));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2019-12-12 00:55:45 +01:00
|
|
|
log_info(ld->log, "Resolved invoice '%s' with amount %s in %zu htlcs",
|
|
|
|
payload->label->s,
|
|
|
|
type_to_string(tmpctx, struct amount_msat, &payload->msat),
|
|
|
|
tal_count(payload->set->htlcs));
|
|
|
|
htlc_set_fulfill(payload->set, &payload->preimage);
|
2021-03-17 13:22:17 +01:00
|
|
|
|
|
|
|
notify_invoice_payment(ld, payload->msat, payload->preimage,
|
|
|
|
payload->label);
|
2019-04-11 02:10:27 +02:00
|
|
|
}
|
|
|
|
|
2020-11-25 01:18:49 +01:00
|
|
|
static bool
|
|
|
|
invoice_payment_deserialize(struct invoice_payment_hook_payload *payload,
|
|
|
|
const char *buffer,
|
|
|
|
const jsmntok_t *toks)
|
|
|
|
{
|
|
|
|
struct lightningd *ld = payload->ld;
|
|
|
|
const u8 *failmsg;
|
|
|
|
|
|
|
|
/* If peer dies or something, this can happen. */
|
|
|
|
if (!payload->set) {
|
|
|
|
log_debug(ld->log, "invoice '%s' paying htlc_in has gone!",
|
|
|
|
payload->label->s);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Did we have a hook result? */
|
|
|
|
failmsg = hook_gives_failmsg(NULL, ld,
|
|
|
|
payload->set->htlcs[0], buffer, toks);
|
|
|
|
if (failmsg) {
|
|
|
|
htlc_set_fail(payload->set, take(failmsg));
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
REGISTER_PLUGIN_HOOK(invoice_payment,
|
|
|
|
invoice_payment_deserialize,
|
|
|
|
invoice_payment_hooks_done,
|
|
|
|
invoice_payment_serialize,
|
|
|
|
struct invoice_payment_hook_payload *);
|
2019-04-11 02:10:27 +02:00
|
|
|
|
2019-12-12 00:55:45 +01:00
|
|
|
const struct invoice_details *
|
|
|
|
invoice_check_payment(const tal_t *ctx,
|
|
|
|
struct lightningd *ld,
|
|
|
|
const struct sha256 *payment_hash,
|
|
|
|
const struct amount_msat msat,
|
|
|
|
const struct secret *payment_secret)
|
2019-04-11 02:08:55 +02:00
|
|
|
{
|
|
|
|
struct invoice invoice;
|
|
|
|
const struct invoice_details *details;
|
|
|
|
|
2019-12-12 00:55:45 +01:00
|
|
|
/* BOLT #4:
|
|
|
|
* - if the payment hash has already been paid:
|
|
|
|
* - MAY treat the payment hash as unknown.
|
|
|
|
* - MAY succeed in accepting the HTLC.
|
|
|
|
*...
|
|
|
|
* - if the payment hash is unknown:
|
|
|
|
* - MUST fail the HTLC.
|
|
|
|
* - MUST return an `incorrect_or_unknown_payment_details` error.
|
|
|
|
*/
|
2020-07-20 07:49:52 +02:00
|
|
|
if (!wallet_invoice_find_unpaid(ld->wallet, &invoice, payment_hash)) {
|
|
|
|
log_debug(ld->log, "Unknown paid invoice %s",
|
|
|
|
type_to_string(tmpctx, struct sha256, payment_hash));
|
|
|
|
if (wallet_invoice_find_by_rhash(ld->wallet, &invoice, payment_hash)) {
|
|
|
|
log_debug(ld->log, "ALREADY paid invoice %s",
|
|
|
|
type_to_string(tmpctx, struct sha256, payment_hash));
|
|
|
|
}
|
2019-12-12 00:55:45 +01:00
|
|
|
return NULL;
|
2020-07-20 07:49:52 +02:00
|
|
|
}
|
2019-04-11 02:08:55 +02:00
|
|
|
|
2019-12-12 00:55:45 +01:00
|
|
|
details = wallet_invoice_details(ctx, ld->wallet, invoice);
|
2019-11-23 01:19:23 +01:00
|
|
|
|
2020-01-31 02:40:36 +01:00
|
|
|
/* BOLT #4:
|
2019-12-12 00:55:45 +01:00
|
|
|
* - if the `payment_secret` doesn't match the expected value for that
|
|
|
|
* `payment_hash`, or the `payment_secret` is required and is not
|
|
|
|
* present:
|
|
|
|
* - MUST fail the HTLC.
|
2019-11-23 01:19:23 +01:00
|
|
|
*/
|
|
|
|
if (feature_is_set(details->features, COMPULSORY_FEATURE(OPT_VAR_ONION))
|
2020-04-07 09:10:30 +02:00
|
|
|
&& !payment_secret) {
|
|
|
|
log_debug(ld->log, "Attept to pay %s without secret",
|
|
|
|
type_to_string(tmpctx, struct sha256, &details->rhash));
|
2019-12-12 00:55:45 +01:00
|
|
|
return tal_free(details);
|
2020-04-07 09:10:30 +02:00
|
|
|
}
|
2019-11-23 01:19:23 +01:00
|
|
|
|
|
|
|
if (payment_secret) {
|
|
|
|
struct secret expected;
|
|
|
|
|
2020-12-16 04:18:42 +01:00
|
|
|
if (details->invstring && strstarts(details->invstring, "lni1"))
|
|
|
|
invoice_secret_bolt12(ld, details->invstring, &expected);
|
|
|
|
else
|
2021-01-13 09:58:38 +01:00
|
|
|
invoice_secret(&details->r, &expected);
|
2020-04-07 09:10:30 +02:00
|
|
|
if (!secret_eq_consttime(payment_secret, &expected)) {
|
|
|
|
log_debug(ld->log, "Attept to pay %s with wrong secret",
|
|
|
|
type_to_string(tmpctx, struct sha256,
|
|
|
|
&details->rhash));
|
2019-12-12 00:55:45 +01:00
|
|
|
return tal_free(details);
|
2020-04-07 09:10:30 +02:00
|
|
|
}
|
2019-11-23 01:19:23 +01:00
|
|
|
}
|
|
|
|
|
2019-04-11 02:08:55 +02:00
|
|
|
/* BOLT #4:
|
|
|
|
*
|
|
|
|
* An _intermediate hop_ MUST NOT, but the _final node_:
|
|
|
|
*...
|
|
|
|
* - if the amount paid is less than the amount expected:
|
|
|
|
* - MUST fail the HTLC.
|
|
|
|
*/
|
|
|
|
if (details->msat != NULL) {
|
|
|
|
struct amount_msat twice;
|
|
|
|
|
2020-07-20 07:49:52 +02:00
|
|
|
if (amount_msat_less(msat, *details->msat)) {
|
|
|
|
log_debug(ld->log, "Attept to pay %s with amount %s < %s",
|
|
|
|
type_to_string(tmpctx, struct sha256,
|
|
|
|
&details->rhash),
|
|
|
|
type_to_string(tmpctx, struct amount_msat, &msat),
|
|
|
|
type_to_string(tmpctx, struct amount_msat, details->msat));
|
2019-12-12 00:55:45 +01:00
|
|
|
return tal_free(details);
|
2020-07-20 07:49:52 +02:00
|
|
|
}
|
2019-04-11 02:08:55 +02:00
|
|
|
|
|
|
|
if (amount_msat_add(&twice, *details->msat, *details->msat)
|
|
|
|
&& amount_msat_greater(msat, twice)) {
|
2020-07-20 07:49:52 +02:00
|
|
|
log_debug(ld->log, "Attept to pay %s with amount %s > %s",
|
|
|
|
type_to_string(tmpctx, struct sha256,
|
|
|
|
&details->rhash),
|
2021-06-28 06:22:03 +02:00
|
|
|
type_to_string(tmpctx, struct amount_msat, &msat),
|
2020-07-20 07:49:52 +02:00
|
|
|
type_to_string(tmpctx, struct amount_msat, &twice));
|
2019-04-11 02:08:55 +02:00
|
|
|
/* BOLT #4:
|
|
|
|
*
|
2019-12-12 00:55:45 +01:00
|
|
|
* - if the amount paid is more than twice the amount
|
|
|
|
* expected:
|
|
|
|
* - SHOULD fail the HTLC.
|
2019-04-11 02:08:55 +02:00
|
|
|
*/
|
2019-12-12 00:55:45 +01:00
|
|
|
return tal_free(details);
|
2019-04-11 02:08:55 +02:00
|
|
|
}
|
|
|
|
}
|
2019-12-12 00:55:45 +01:00
|
|
|
return details;
|
|
|
|
}
|
|
|
|
|
|
|
|
void invoice_try_pay(struct lightningd *ld,
|
2019-12-12 00:55:45 +01:00
|
|
|
struct htlc_set *set,
|
|
|
|
const struct invoice_details *details)
|
2019-12-12 00:55:45 +01:00
|
|
|
{
|
|
|
|
struct invoice_payment_hook_payload *payload;
|
|
|
|
|
2020-04-15 12:20:41 +02:00
|
|
|
payload = tal(NULL, struct invoice_payment_hook_payload);
|
2019-04-11 02:10:27 +02:00
|
|
|
payload->ld = ld;
|
|
|
|
payload->label = tal_steal(payload, details->label);
|
2019-12-12 00:55:45 +01:00
|
|
|
payload->msat = set->so_far;
|
2019-04-11 02:10:27 +02:00
|
|
|
payload->preimage = details->r;
|
2019-12-12 00:55:45 +01:00
|
|
|
payload->set = set;
|
|
|
|
tal_add_destructor2(set, invoice_payload_remove_set, payload);
|
2019-04-11 02:10:27 +02:00
|
|
|
|
2020-04-15 12:20:41 +02:00
|
|
|
plugin_hook_call_invoice_payment(ld, payload);
|
2019-04-11 02:08:55 +02:00
|
|
|
}
|
|
|
|
|
2017-11-22 01:25:39 +01:00
|
|
|
static bool hsm_sign_b11(const u5 *u5bytes,
|
|
|
|
const u8 *hrpu8,
|
|
|
|
secp256k1_ecdsa_recoverable_signature *rsig,
|
|
|
|
struct lightningd *ld)
|
|
|
|
{
|
2020-08-25 03:55:38 +02:00
|
|
|
u8 *msg = towire_hsmd_sign_invoice(NULL, u5bytes, hrpu8);
|
2017-11-22 01:25:39 +01:00
|
|
|
|
|
|
|
if (!wire_sync_write(ld->hsm_fd, take(msg)))
|
|
|
|
fatal("Could not write to HSM: %s", strerror(errno));
|
|
|
|
|
2018-07-09 13:17:59 +02:00
|
|
|
msg = wire_sync_read(tmpctx, ld->hsm_fd);
|
2020-08-25 03:55:38 +02:00
|
|
|
if (!fromwire_hsmd_sign_invoice_reply(msg, rsig))
|
2017-11-22 01:25:39 +01:00
|
|
|
fatal("HSM gave bad sign_invoice_reply %s",
|
|
|
|
tal_hex(msg, msg));
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2020-12-14 02:21:48 +01:00
|
|
|
static void hsm_sign_b12_invoice(struct lightningd *ld,
|
|
|
|
struct tlv_invoice *invoice)
|
|
|
|
{
|
|
|
|
struct sha256 merkle;
|
|
|
|
u8 *msg;
|
|
|
|
|
|
|
|
assert(!invoice->signature);
|
|
|
|
|
|
|
|
merkle_tlv(invoice->fields, &merkle);
|
2020-12-16 04:13:37 +01:00
|
|
|
msg = towire_hsmd_sign_bolt12(NULL, "invoice", "signature", &merkle, NULL);
|
2020-12-14 02:21:48 +01:00
|
|
|
|
|
|
|
if (!wire_sync_write(ld->hsm_fd, take(msg)))
|
|
|
|
fatal("Could not write to HSM: %s", strerror(errno));
|
|
|
|
|
|
|
|
msg = wire_sync_read(tmpctx, ld->hsm_fd);
|
|
|
|
invoice->signature = tal(invoice, struct bip340sig);
|
|
|
|
if (!fromwire_hsmd_sign_bolt12_reply(msg, invoice->signature))
|
|
|
|
fatal("HSM gave bad sign_invoice_reply %s",
|
|
|
|
tal_hex(msg, msg));
|
|
|
|
}
|
|
|
|
|
2018-12-16 05:52:06 +01:00
|
|
|
static struct command_result *parse_fallback(struct command *cmd,
|
|
|
|
const char *buffer,
|
|
|
|
const jsmntok_t *fallback,
|
|
|
|
const u8 **fallback_script)
|
2018-04-05 07:13:51 +02:00
|
|
|
|
|
|
|
{
|
|
|
|
enum address_parse_result fallback_parse;
|
|
|
|
|
|
|
|
fallback_parse
|
2019-10-06 14:12:41 +02:00
|
|
|
= json_to_address_scriptpubkey(cmd,
|
2019-10-15 12:58:30 +02:00
|
|
|
chainparams,
|
2019-10-06 14:12:41 +02:00
|
|
|
buffer, fallback,
|
|
|
|
fallback_script);
|
2018-04-05 07:13:51 +02:00
|
|
|
if (fallback_parse == ADDRESS_PARSE_UNRECOGNIZED) {
|
2018-12-16 05:52:06 +01:00
|
|
|
return command_fail(cmd, LIGHTNINGD,
|
|
|
|
"Fallback address not valid");
|
2018-04-05 07:13:51 +02:00
|
|
|
} else if (fallback_parse == ADDRESS_PARSE_WRONG_NETWORK) {
|
2018-12-16 05:52:06 +01:00
|
|
|
return command_fail(cmd, LIGHTNINGD,
|
|
|
|
"Fallback address does not match our network %s",
|
2019-10-15 12:58:30 +02:00
|
|
|
chainparams->network_name);
|
2018-04-05 07:13:51 +02:00
|
|
|
}
|
2018-12-16 05:52:06 +01:00
|
|
|
return NULL;
|
2018-04-05 07:13:51 +02:00
|
|
|
}
|
|
|
|
|
2019-04-09 17:35:28 +02:00
|
|
|
/*
|
|
|
|
* From array of incoming channels [inchan], find suitable ones for
|
|
|
|
* a payment-to-us of [amount_needed], using criteria:
|
|
|
|
* 1. Channel's peer is known, in state CHANNELD_NORMAL and is online.
|
|
|
|
* 2. Channel's peer capacity to pay us is sufficient.
|
|
|
|
*
|
|
|
|
* Then use weighted reservoir sampling, which makes probing channel balances
|
|
|
|
* harder, to choose one channel from the set of suitable channels. It favors
|
|
|
|
* channels that have less balance on our side as fraction of their capacity.
|
|
|
|
*/
|
json-invoice: add routeboost, warnings.
We split json_invoice(), as it now needs to round-trip to the gossipd,
and uniqueness checks need to happen *after* gossipd replies to avoid
a race.
For every candidate channel gossipd gives us, we check that it's in
state NORMAL (not shutting down, not still waiting for lockin), that
it's connected, and that it has capacity. We then choose one with
probability weighted by excess capacity, so larger channels are more
likely.
As a side effect of this, we can tell if an invoice is unpayble (no
channels have sufficient incoming capacity) or difficuly (no *online*
channels have sufficient capacity), so we add those warnings.
Signed-off-by: Rusty Russell <rusty@rustcorp.com.au>
2018-09-27 07:29:17 +02:00
|
|
|
static struct route_info **select_inchan(const tal_t *ctx,
|
|
|
|
struct lightningd *ld,
|
2019-04-09 17:35:28 +02:00
|
|
|
struct amount_msat amount_needed,
|
2020-09-08 06:57:53 +02:00
|
|
|
const struct routehint_candidate
|
|
|
|
*candidates)
|
json-invoice: add routeboost, warnings.
We split json_invoice(), as it now needs to round-trip to the gossipd,
and uniqueness checks need to happen *after* gossipd replies to avoid
a race.
For every candidate channel gossipd gives us, we check that it's in
state NORMAL (not shutting down, not still waiting for lockin), that
it's connected, and that it has capacity. We then choose one with
probability weighted by excess capacity, so larger channels are more
likely.
As a side effect of this, we can tell if an invoice is unpayble (no
channels have sufficient incoming capacity) or difficuly (no *online*
channels have sufficient capacity), so we add those warnings.
Signed-off-by: Rusty Russell <rusty@rustcorp.com.au>
2018-09-27 07:29:17 +02:00
|
|
|
{
|
2019-04-09 17:35:28 +02:00
|
|
|
/* BOLT11 struct wants an array of arrays (can provide multiple routes) */
|
2020-08-11 07:05:56 +02:00
|
|
|
struct route_info **r = NULL;
|
|
|
|
double total_weight = 0.0;
|
json-invoice: add routeboost, warnings.
We split json_invoice(), as it now needs to round-trip to the gossipd,
and uniqueness checks need to happen *after* gossipd replies to avoid
a race.
For every candidate channel gossipd gives us, we check that it's in
state NORMAL (not shutting down, not still waiting for lockin), that
it's connected, and that it has capacity. We then choose one with
probability weighted by excess capacity, so larger channels are more
likely.
As a side effect of this, we can tell if an invoice is unpayble (no
channels have sufficient incoming capacity) or difficuly (no *online*
channels have sufficient capacity), so we add those warnings.
Signed-off-by: Rusty Russell <rusty@rustcorp.com.au>
2018-09-27 07:29:17 +02:00
|
|
|
|
2019-04-09 17:35:28 +02:00
|
|
|
/* Collect suitable channels and assign each a weight. */
|
2020-09-08 06:57:53 +02:00
|
|
|
for (size_t i = 0; i < tal_count(candidates); i++) {
|
|
|
|
struct amount_msat excess, capacity;
|
2019-04-09 17:35:28 +02:00
|
|
|
struct amount_sat cumulative_reserve;
|
|
|
|
double excess_frac;
|
json-invoice: add routeboost, warnings.
We split json_invoice(), as it now needs to round-trip to the gossipd,
and uniqueness checks need to happen *after* gossipd replies to avoid
a race.
For every candidate channel gossipd gives us, we check that it's in
state NORMAL (not shutting down, not still waiting for lockin), that
it's connected, and that it has capacity. We then choose one with
probability weighted by excess capacity, so larger channels are more
likely.
As a side effect of this, we can tell if an invoice is unpayble (no
channels have sufficient incoming capacity) or difficuly (no *online*
channels have sufficient capacity), so we add those warnings.
Signed-off-by: Rusty Russell <rusty@rustcorp.com.au>
2018-09-27 07:29:17 +02:00
|
|
|
|
2020-09-08 06:57:53 +02:00
|
|
|
/* Does the peer have sufficient balance to pay us,
|
|
|
|
* even after having taken into account their reserve? */
|
|
|
|
if (!amount_msat_sub(&excess, candidates[i].capacity,
|
|
|
|
amount_needed))
|
2019-12-17 07:11:08 +01:00
|
|
|
continue;
|
|
|
|
|
2019-04-09 17:35:28 +02:00
|
|
|
/* Channel balance as seen by our node:
|
|
|
|
|
|
|
|
|<----------------- capacity ----------------->|
|
|
|
|
. .
|
|
|
|
. |<------------------ their_msat -------------------->|
|
|
|
|
. | . |
|
|
|
|
. |<----- capacity_to_pay_us ----->|<- their_reserve ->|
|
|
|
|
. | | |
|
|
|
|
. |<- amount_needed --><- excess ->| |
|
|
|
|
. | | |
|
|
|
|
|-------|-------------|--------------------------------|-------------------|
|
|
|
|
0 ^ ^ ^ funding
|
|
|
|
our_reserve our_msat */
|
|
|
|
|
|
|
|
/* Find capacity and calculate its excess fraction */
|
|
|
|
if (!amount_sat_add(&cumulative_reserve,
|
2020-09-08 06:57:53 +02:00
|
|
|
candidates[i].c->our_config.channel_reserve,
|
|
|
|
candidates[i].c->channel_info.their_config.channel_reserve)
|
2021-10-13 05:45:36 +02:00
|
|
|
|| !amount_sat_to_msat(&capacity, candidates[i].c->funding_sats)
|
2019-04-09 17:35:28 +02:00
|
|
|
|| !amount_msat_sub_sat(&capacity, capacity, cumulative_reserve)) {
|
|
|
|
log_broken(ld->log, "Channel %s capacity overflow!",
|
2020-09-08 06:57:53 +02:00
|
|
|
type_to_string(tmpctx, struct short_channel_id, candidates[i].c->scid));
|
2019-04-09 17:35:28 +02:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2020-08-11 07:05:56 +02:00
|
|
|
/* We don't want a 0 probability if 0 excess; it might be the
|
|
|
|
* only one! So bump it by 1 msat */
|
|
|
|
if (!amount_msat_add(&excess, excess, AMOUNT_MSAT(1))) {
|
|
|
|
log_broken(ld->log, "Channel %s excess overflow!",
|
2020-09-08 06:57:53 +02:00
|
|
|
type_to_string(tmpctx,
|
|
|
|
struct short_channel_id,
|
|
|
|
candidates[i].c->scid));
|
2020-08-11 07:05:56 +02:00
|
|
|
continue;
|
|
|
|
}
|
2020-08-05 05:56:32 +02:00
|
|
|
excess_frac = amount_msat_ratio(excess, capacity);
|
2019-04-09 17:35:28 +02:00
|
|
|
|
2020-08-11 07:05:56 +02:00
|
|
|
if (random_select(excess_frac, &total_weight)) {
|
|
|
|
tal_free(r);
|
|
|
|
r = tal_arr(ctx, struct route_info *, 1);
|
2020-09-08 06:57:53 +02:00
|
|
|
r[0] = tal_dup(r, struct route_info, candidates[i].r);
|
2020-08-11 07:05:56 +02:00
|
|
|
}
|
2019-04-09 17:35:28 +02:00
|
|
|
}
|
|
|
|
|
2020-08-11 07:05:56 +02:00
|
|
|
return r;
|
json-invoice: add routeboost, warnings.
We split json_invoice(), as it now needs to round-trip to the gossipd,
and uniqueness checks need to happen *after* gossipd replies to avoid
a race.
For every candidate channel gossipd gives us, we check that it's in
state NORMAL (not shutting down, not still waiting for lockin), that
it's connected, and that it has capacity. We then choose one with
probability weighted by excess capacity, so larger channels are more
likely.
As a side effect of this, we can tell if an invoice is unpayble (no
channels have sufficient incoming capacity) or difficuly (no *online*
channels have sufficient capacity), so we add those warnings.
Signed-off-by: Rusty Russell <rusty@rustcorp.com.au>
2018-09-27 07:29:17 +02:00
|
|
|
}
|
|
|
|
|
2020-09-08 06:57:53 +02:00
|
|
|
static int cmp_rr_number(const struct routehint_candidate *a,
|
|
|
|
const struct routehint_candidate *b,
|
|
|
|
void *unused)
|
|
|
|
{
|
|
|
|
/* They're unique, so can't be equal */
|
|
|
|
if (a->c->rr_number > b->c->rr_number)
|
|
|
|
return 1;
|
|
|
|
assert(a->c->rr_number < b->c->rr_number);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2020-08-05 15:50:38 +02:00
|
|
|
/** select_inchan_mpp
|
|
|
|
*
|
|
|
|
* @brief fallback in case select_inchan cannot find a *single*
|
|
|
|
* channel capable of accepting the payment as a whole.
|
|
|
|
* Also the main routehint-selector if we are completely unpublished
|
|
|
|
* (i.e. all our channels are unpublished), since if we are completely
|
|
|
|
* unpublished then the payer cannot fall back to just directly routing
|
|
|
|
* to us.
|
|
|
|
*/
|
|
|
|
static struct route_info **select_inchan_mpp(const tal_t *ctx,
|
|
|
|
struct lightningd *ld,
|
|
|
|
struct amount_msat amount_needed,
|
2020-09-08 06:57:53 +02:00
|
|
|
struct routehint_candidate
|
2021-06-14 23:07:38 +02:00
|
|
|
*candidates)
|
2020-08-05 15:50:38 +02:00
|
|
|
{
|
|
|
|
/* The total amount we have gathered for incoming channels. */
|
|
|
|
struct amount_msat gathered;
|
|
|
|
/* Routehint array. */
|
|
|
|
struct route_info **routehints;
|
|
|
|
|
|
|
|
gathered = AMOUNT_MSAT(0);
|
|
|
|
routehints = tal_arr(ctx, struct route_info *, 0);
|
|
|
|
|
2020-09-08 06:57:53 +02:00
|
|
|
/* Sort by rr_number, so we get fresh channels. */
|
|
|
|
asort(candidates, tal_count(candidates), cmp_rr_number, NULL);
|
|
|
|
for (size_t i = 0; i < tal_count(candidates); i++) {
|
|
|
|
if (amount_msat_greater_eq(gathered, amount_needed))
|
|
|
|
break;
|
2020-08-05 15:50:38 +02:00
|
|
|
|
|
|
|
/* Add to current routehints set. */
|
2020-09-08 06:57:53 +02:00
|
|
|
if (!amount_msat_add(&gathered, gathered, candidates[i].capacity)) {
|
2020-08-05 15:50:38 +02:00
|
|
|
log_broken(ld->log,
|
|
|
|
"Gathered channel capacity overflow: "
|
|
|
|
"%s + %s",
|
|
|
|
type_to_string(tmpctx, struct amount_msat, &gathered),
|
2020-09-08 06:57:53 +02:00
|
|
|
type_to_string(tmpctx, struct amount_msat,
|
|
|
|
&candidates[i].capacity));
|
2020-08-05 15:50:38 +02:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
tal_arr_expand(&routehints,
|
2020-09-08 06:57:53 +02:00
|
|
|
tal_dup(routehints, struct route_info,
|
|
|
|
candidates[i].r));
|
|
|
|
/* Put to the back of the round-robin list */
|
|
|
|
candidates[i].c->rr_number = ld->rr_counter++;
|
2020-08-05 15:50:38 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return routehints;
|
|
|
|
}
|
|
|
|
|
json-invoice: add routeboost, warnings.
We split json_invoice(), as it now needs to round-trip to the gossipd,
and uniqueness checks need to happen *after* gossipd replies to avoid
a race.
For every candidate channel gossipd gives us, we check that it's in
state NORMAL (not shutting down, not still waiting for lockin), that
it's connected, and that it has capacity. We then choose one with
probability weighted by excess capacity, so larger channels are more
likely.
As a side effect of this, we can tell if an invoice is unpayble (no
channels have sufficient incoming capacity) or difficuly (no *online*
channels have sufficient capacity), so we add those warnings.
Signed-off-by: Rusty Russell <rusty@rustcorp.com.au>
2018-09-27 07:29:17 +02:00
|
|
|
/* Encapsulating struct while we wait for gossipd to give us incoming channels */
|
2019-12-17 07:11:08 +01:00
|
|
|
struct chanhints {
|
|
|
|
bool expose_all_private;
|
|
|
|
struct short_channel_id *hints;
|
|
|
|
};
|
|
|
|
|
json-invoice: add routeboost, warnings.
We split json_invoice(), as it now needs to round-trip to the gossipd,
and uniqueness checks need to happen *after* gossipd replies to avoid
a race.
For every candidate channel gossipd gives us, we check that it's in
state NORMAL (not shutting down, not still waiting for lockin), that
it's connected, and that it has capacity. We then choose one with
probability weighted by excess capacity, so larger channels are more
likely.
As a side effect of this, we can tell if an invoice is unpayble (no
channels have sufficient incoming capacity) or difficuly (no *online*
channels have sufficient capacity), so we add those warnings.
Signed-off-by: Rusty Russell <rusty@rustcorp.com.au>
2018-09-27 07:29:17 +02:00
|
|
|
struct invoice_info {
|
|
|
|
struct command *cmd;
|
|
|
|
struct preimage payment_preimage;
|
|
|
|
struct bolt11 *b11;
|
2019-06-12 02:38:54 +02:00
|
|
|
struct json_escape *label;
|
2019-12-17 07:11:08 +01:00
|
|
|
struct chanhints *chanhints;
|
json-invoice: add routeboost, warnings.
We split json_invoice(), as it now needs to round-trip to the gossipd,
and uniqueness checks need to happen *after* gossipd replies to avoid
a race.
For every candidate channel gossipd gives us, we check that it's in
state NORMAL (not shutting down, not still waiting for lockin), that
it's connected, and that it has capacity. We then choose one with
probability weighted by excess capacity, so larger channels are more
likely.
As a side effect of this, we can tell if an invoice is unpayble (no
channels have sufficient incoming capacity) or difficuly (no *online*
channels have sufficient capacity), so we add those warnings.
Signed-off-by: Rusty Russell <rusty@rustcorp.com.au>
2018-09-27 07:29:17 +02:00
|
|
|
};
|
|
|
|
|
2021-06-14 23:07:38 +02:00
|
|
|
/* Add routehints based on listincoming results: NULL means success. */
|
|
|
|
static struct command_result *
|
|
|
|
add_routehints(struct invoice_info *info,
|
|
|
|
const char *buffer,
|
|
|
|
const jsmntok_t *toks,
|
|
|
|
bool *warning_mpp,
|
|
|
|
bool *warning_capacity,
|
|
|
|
bool *warning_deadends,
|
|
|
|
bool *warning_offline,
|
|
|
|
bool *warning_private_unused)
|
2016-09-06 09:17:49 +02:00
|
|
|
{
|
2019-12-17 07:11:08 +01:00
|
|
|
const struct chanhints *chanhints = info->chanhints;
|
2020-08-05 15:50:38 +02:00
|
|
|
bool node_unpublished;
|
2021-06-14 23:07:38 +02:00
|
|
|
struct amount_msat avail_capacity, deadend_capacity, offline_capacity,
|
|
|
|
private_capacity;
|
|
|
|
struct routehint_candidate *candidates;
|
|
|
|
struct amount_msat total, needed;
|
2020-08-05 15:50:38 +02:00
|
|
|
|
2021-06-14 23:07:38 +02:00
|
|
|
/* Dev code can force routes. */
|
|
|
|
if (tal_count(info->b11->routes) != 0) {
|
|
|
|
*warning_mpp = *warning_capacity = *warning_deadends
|
|
|
|
= *warning_offline = *warning_private_unused
|
|
|
|
= false;
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
candidates = routehint_candidates(tmpctx, info->cmd->ld,
|
|
|
|
buffer, toks,
|
|
|
|
chanhints ? &chanhints->expose_all_private : NULL,
|
2020-09-08 06:57:53 +02:00
|
|
|
chanhints ? chanhints->hints : NULL,
|
|
|
|
&node_unpublished,
|
2021-06-14 23:07:38 +02:00
|
|
|
&avail_capacity,
|
|
|
|
&private_capacity,
|
|
|
|
&deadend_capacity,
|
|
|
|
&offline_capacity);
|
2019-12-17 07:11:08 +01:00
|
|
|
|
2020-09-08 06:57:53 +02:00
|
|
|
/* If they told us to use scids and we couldn't, fail. */
|
|
|
|
if (tal_count(candidates) == 0
|
|
|
|
&& chanhints && tal_count(chanhints->hints) != 0) {
|
2021-06-14 23:07:38 +02:00
|
|
|
return command_fail(info->cmd,
|
|
|
|
INVOICE_HINTS_GAVE_NO_ROUTES,
|
|
|
|
"None of those hints were suitable local channels");
|
2019-12-17 07:11:08 +01:00
|
|
|
}
|
|
|
|
|
2021-06-14 23:07:38 +02:00
|
|
|
needed = info->b11->msat ? *info->b11->msat : AMOUNT_MSAT(1);
|
|
|
|
|
|
|
|
/* If we are not completely unpublished, try with reservoir
|
|
|
|
* sampling first.
|
|
|
|
*
|
|
|
|
* Why do we not do this if we are completely unpublished?
|
|
|
|
* Because it is possible that multiple invoices will, by
|
|
|
|
* chance, select the same channel as routehint.
|
|
|
|
* This single channel might not be able to accept all the
|
|
|
|
* incoming payments on all the invoices generated.
|
|
|
|
* If we were published, that is fine because the payer can
|
|
|
|
* fall back to just attempting to route directly.
|
|
|
|
* But if we were unpublished, the only way for the payer to
|
|
|
|
* reach us would be via the routehints we provide, so we
|
|
|
|
* should make an effort to avoid overlapping incoming
|
|
|
|
* channels, which is done by select_inchan_mpp.
|
|
|
|
*/
|
|
|
|
if (!node_unpublished)
|
|
|
|
info->b11->routes = select_inchan(info->b11,
|
|
|
|
info->cmd->ld,
|
|
|
|
needed,
|
|
|
|
candidates);
|
|
|
|
|
|
|
|
/* If we are completely unpublished, or if the above reservoir
|
|
|
|
* sampling fails, select channels by round-robin. */
|
2020-08-05 15:50:38 +02:00
|
|
|
if (tal_count(info->b11->routes) == 0) {
|
2021-06-14 23:07:38 +02:00
|
|
|
info->b11->routes = select_inchan_mpp(info->b11,
|
|
|
|
info->cmd->ld,
|
|
|
|
needed,
|
|
|
|
candidates);
|
|
|
|
*warning_mpp = (tal_count(info->b11->routes) > 1);
|
|
|
|
} else {
|
|
|
|
*warning_mpp = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
log_debug(info->cmd->ld->log, "needed = %s, avail_capacity = %s, private_capacity = %s, offline_capacity = %s, deadend_capacity = %s",
|
|
|
|
type_to_string(tmpctx, struct amount_msat, &needed),
|
|
|
|
type_to_string(tmpctx, struct amount_msat, &avail_capacity),
|
|
|
|
type_to_string(tmpctx, struct amount_msat, &private_capacity),
|
|
|
|
type_to_string(tmpctx, struct amount_msat, &offline_capacity),
|
|
|
|
type_to_string(tmpctx, struct amount_msat, &deadend_capacity));
|
|
|
|
|
|
|
|
if (!amount_msat_add(&total, avail_capacity, offline_capacity)
|
|
|
|
|| !amount_msat_add(&total, total, deadend_capacity)
|
|
|
|
|| !amount_msat_add(&total, total, private_capacity))
|
|
|
|
fatal("Cannot add %s + %s + %s + %s",
|
|
|
|
type_to_string(tmpctx, struct amount_msat,
|
|
|
|
&avail_capacity),
|
|
|
|
type_to_string(tmpctx, struct amount_msat,
|
|
|
|
&offline_capacity),
|
|
|
|
type_to_string(tmpctx, struct amount_msat,
|
|
|
|
&deadend_capacity),
|
|
|
|
type_to_string(tmpctx, struct amount_msat,
|
|
|
|
&private_capacity));
|
|
|
|
|
|
|
|
/* If we literally didn't have capacity at all, warn. */
|
|
|
|
*warning_capacity = amount_msat_greater_eq(needed, total);
|
|
|
|
|
|
|
|
/* We only warn about these if we didn't have capacity and
|
|
|
|
* they would have helped. */
|
|
|
|
*warning_offline = false;
|
|
|
|
*warning_deadends = false;
|
|
|
|
*warning_private_unused = false;
|
|
|
|
if (amount_msat_greater(needed, avail_capacity)) {
|
|
|
|
struct amount_msat tot;
|
|
|
|
|
|
|
|
/* We didn't get enough: would offline have helped? */
|
|
|
|
if (!amount_msat_add(&tot, avail_capacity, offline_capacity))
|
|
|
|
abort();
|
|
|
|
if (amount_msat_greater_eq(tot, needed)) {
|
|
|
|
*warning_offline = true;
|
|
|
|
goto done;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Hmm, what about deadends? */
|
|
|
|
if (!amount_msat_add(&tot, tot, deadend_capacity))
|
|
|
|
abort();
|
|
|
|
if (amount_msat_greater_eq(tot, needed)) {
|
|
|
|
*warning_deadends = true;
|
|
|
|
goto done;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* What about private channels? */
|
|
|
|
if (!amount_msat_add(&tot, tot, private_capacity))
|
|
|
|
abort();
|
|
|
|
if (amount_msat_greater_eq(tot, needed)) {
|
|
|
|
*warning_private_unused = true;
|
|
|
|
goto done;
|
2020-08-05 15:50:38 +02:00
|
|
|
}
|
|
|
|
}
|
json-invoice: add routeboost, warnings.
We split json_invoice(), as it now needs to round-trip to the gossipd,
and uniqueness checks need to happen *after* gossipd replies to avoid
a race.
For every candidate channel gossipd gives us, we check that it's in
state NORMAL (not shutting down, not still waiting for lockin), that
it's connected, and that it has capacity. We then choose one with
probability weighted by excess capacity, so larger channels are more
likely.
As a side effect of this, we can tell if an invoice is unpayble (no
channels have sufficient incoming capacity) or difficuly (no *online*
channels have sufficient capacity), so we add those warnings.
Signed-off-by: Rusty Russell <rusty@rustcorp.com.au>
2018-09-27 07:29:17 +02:00
|
|
|
|
2021-06-14 23:07:38 +02:00
|
|
|
done:
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct command_result *
|
|
|
|
invoice_complete(struct invoice_info *info,
|
|
|
|
bool warning_no_listincoming,
|
|
|
|
bool warning_mpp,
|
|
|
|
bool warning_capacity,
|
|
|
|
bool warning_deadends,
|
|
|
|
bool warning_offline,
|
|
|
|
bool warning_private_unused)
|
|
|
|
{
|
|
|
|
struct json_stream *response;
|
|
|
|
struct invoice invoice;
|
|
|
|
char *b11enc;
|
|
|
|
const struct invoice_details *details;
|
2021-07-12 01:04:33 +02:00
|
|
|
struct secret payment_secret;
|
2021-06-14 23:07:38 +02:00
|
|
|
struct wallet *wallet = info->cmd->ld->wallet;
|
|
|
|
|
json-invoice: add routeboost, warnings.
We split json_invoice(), as it now needs to round-trip to the gossipd,
and uniqueness checks need to happen *after* gossipd replies to avoid
a race.
For every candidate channel gossipd gives us, we check that it's in
state NORMAL (not shutting down, not still waiting for lockin), that
it's connected, and that it has capacity. We then choose one with
probability weighted by excess capacity, so larger channels are more
likely.
As a side effect of this, we can tell if an invoice is unpayble (no
channels have sufficient incoming capacity) or difficuly (no *online*
channels have sufficient capacity), so we add those warnings.
Signed-off-by: Rusty Russell <rusty@rustcorp.com.au>
2018-09-27 07:29:17 +02:00
|
|
|
b11enc = bolt11_encode(info, info->b11, false,
|
|
|
|
hsm_sign_b11, info->cmd->ld);
|
|
|
|
|
|
|
|
/* Check duplicate preimage (unlikely unless they specified it!) */
|
|
|
|
if (wallet_invoice_find_by_rhash(wallet,
|
|
|
|
&invoice, &info->b11->payment_hash)) {
|
2021-06-14 23:07:38 +02:00
|
|
|
return command_fail(info->cmd,
|
|
|
|
INVOICE_PREIMAGE_ALREADY_EXISTS,
|
|
|
|
"preimage already used");
|
json-invoice: add routeboost, warnings.
We split json_invoice(), as it now needs to round-trip to the gossipd,
and uniqueness checks need to happen *after* gossipd replies to avoid
a race.
For every candidate channel gossipd gives us, we check that it's in
state NORMAL (not shutting down, not still waiting for lockin), that
it's connected, and that it has capacity. We then choose one with
probability weighted by excess capacity, so larger channels are more
likely.
As a side effect of this, we can tell if an invoice is unpayble (no
channels have sufficient incoming capacity) or difficuly (no *online*
channels have sufficient capacity), so we add those warnings.
Signed-off-by: Rusty Russell <rusty@rustcorp.com.au>
2018-09-27 07:29:17 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!wallet_invoice_create(wallet,
|
|
|
|
&invoice,
|
2019-02-21 03:38:35 +01:00
|
|
|
info->b11->msat,
|
json-invoice: add routeboost, warnings.
We split json_invoice(), as it now needs to round-trip to the gossipd,
and uniqueness checks need to happen *after* gossipd replies to avoid
a race.
For every candidate channel gossipd gives us, we check that it's in
state NORMAL (not shutting down, not still waiting for lockin), that
it's connected, and that it has capacity. We then choose one with
probability weighted by excess capacity, so larger channels are more
likely.
As a side effect of this, we can tell if an invoice is unpayble (no
channels have sufficient incoming capacity) or difficuly (no *online*
channels have sufficient capacity), so we add those warnings.
Signed-off-by: Rusty Russell <rusty@rustcorp.com.au>
2018-09-27 07:29:17 +02:00
|
|
|
info->label,
|
|
|
|
info->b11->expiry,
|
|
|
|
b11enc,
|
|
|
|
info->b11->description,
|
2019-11-23 01:19:23 +01:00
|
|
|
info->b11->features,
|
json-invoice: add routeboost, warnings.
We split json_invoice(), as it now needs to round-trip to the gossipd,
and uniqueness checks need to happen *after* gossipd replies to avoid
a race.
For every candidate channel gossipd gives us, we check that it's in
state NORMAL (not shutting down, not still waiting for lockin), that
it's connected, and that it has capacity. We then choose one with
probability weighted by excess capacity, so larger channels are more
likely.
As a side effect of this, we can tell if an invoice is unpayble (no
channels have sufficient incoming capacity) or difficuly (no *online*
channels have sufficient capacity), so we add those warnings.
Signed-off-by: Rusty Russell <rusty@rustcorp.com.au>
2018-09-27 07:29:17 +02:00
|
|
|
&info->payment_preimage,
|
2020-12-14 02:18:24 +01:00
|
|
|
&info->b11->payment_hash,
|
|
|
|
NULL)) {
|
2021-06-14 23:07:38 +02:00
|
|
|
return command_fail(info->cmd, INVOICE_LABEL_ALREADY_EXISTS,
|
|
|
|
"Duplicate label '%s'",
|
|
|
|
info->label->s);
|
json-invoice: add routeboost, warnings.
We split json_invoice(), as it now needs to round-trip to the gossipd,
and uniqueness checks need to happen *after* gossipd replies to avoid
a race.
For every candidate channel gossipd gives us, we check that it's in
state NORMAL (not shutting down, not still waiting for lockin), that
it's connected, and that it has capacity. We then choose one with
probability weighted by excess capacity, so larger channels are more
likely.
As a side effect of this, we can tell if an invoice is unpayble (no
channels have sufficient incoming capacity) or difficuly (no *online*
channels have sufficient capacity), so we add those warnings.
Signed-off-by: Rusty Russell <rusty@rustcorp.com.au>
2018-09-27 07:29:17 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Get details */
|
|
|
|
details = wallet_invoice_details(info, wallet, invoice);
|
|
|
|
|
2018-10-19 03:17:48 +02:00
|
|
|
response = json_stream_success(info->cmd);
|
2019-08-10 10:39:04 +02:00
|
|
|
json_add_sha256(response, "payment_hash", &details->rhash);
|
json-invoice: add routeboost, warnings.
We split json_invoice(), as it now needs to round-trip to the gossipd,
and uniqueness checks need to happen *after* gossipd replies to avoid
a race.
For every candidate channel gossipd gives us, we check that it's in
state NORMAL (not shutting down, not still waiting for lockin), that
it's connected, and that it has capacity. We then choose one with
probability weighted by excess capacity, so larger channels are more
likely.
As a side effect of this, we can tell if an invoice is unpayble (no
channels have sufficient incoming capacity) or difficuly (no *online*
channels have sufficient capacity), so we add those warnings.
Signed-off-by: Rusty Russell <rusty@rustcorp.com.au>
2018-09-27 07:29:17 +02:00
|
|
|
json_add_u64(response, "expires_at", details->expiry_time);
|
2020-12-14 02:23:07 +01:00
|
|
|
json_add_string(response, "bolt11", details->invstring);
|
2021-07-12 01:04:33 +02:00
|
|
|
invoice_secret(&details->r, &payment_secret);
|
|
|
|
json_add_secret(response, "payment_secret", &payment_secret);
|
json-invoice: add routeboost, warnings.
We split json_invoice(), as it now needs to round-trip to the gossipd,
and uniqueness checks need to happen *after* gossipd replies to avoid
a race.
For every candidate channel gossipd gives us, we check that it's in
state NORMAL (not shutting down, not still waiting for lockin), that
it's connected, and that it has capacity. We then choose one with
probability weighted by excess capacity, so larger channels are more
likely.
As a side effect of this, we can tell if an invoice is unpayble (no
channels have sufficient incoming capacity) or difficuly (no *online*
channels have sufficient capacity), so we add those warnings.
Signed-off-by: Rusty Russell <rusty@rustcorp.com.au>
2018-09-27 07:29:17 +02:00
|
|
|
|
2020-04-21 03:04:01 +02:00
|
|
|
notify_invoice_creation(info->cmd->ld, info->b11->msat,
|
|
|
|
info->payment_preimage, info->label);
|
|
|
|
|
2021-06-14 23:07:38 +02:00
|
|
|
if (warning_no_listincoming)
|
|
|
|
json_add_string(response, "warning_listincoming",
|
|
|
|
"No listincoming command available, cannot add routehints to invoice");
|
2020-08-05 15:50:38 +02:00
|
|
|
if (warning_mpp)
|
|
|
|
json_add_string(response, "warning_mpp",
|
2021-09-08 02:07:14 +02:00
|
|
|
"The invoice is only payable by MPP-capable payers.");
|
2021-06-14 23:07:38 +02:00
|
|
|
if (warning_capacity)
|
|
|
|
json_add_string(response, "warning_capacity",
|
|
|
|
"Insufficient incoming channel capacity to pay invoice");
|
|
|
|
|
|
|
|
if (warning_deadends)
|
|
|
|
json_add_string(response, "warning_deadends",
|
|
|
|
"Insufficient incoming capacity, once dead-end peers were excluded");
|
|
|
|
|
|
|
|
if (warning_offline)
|
|
|
|
json_add_string(response, "warning_offline",
|
|
|
|
"Insufficient incoming capacity, once offline peers were excluded");
|
2020-08-05 15:50:38 +02:00
|
|
|
|
2021-06-14 23:07:38 +02:00
|
|
|
if (warning_private_unused)
|
|
|
|
json_add_string(response, "warning_private_unused",
|
|
|
|
"Insufficient incoming capacity, once private channels were excluded (try exposeprivatechannels=true?)");
|
|
|
|
|
|
|
|
return command_success(info->cmd, response);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Return from "listincoming". */
|
|
|
|
static void listincoming_done(const char *buffer,
|
|
|
|
const jsmntok_t *toks,
|
|
|
|
const jsmntok_t *idtok UNUSED,
|
|
|
|
struct invoice_info *info)
|
|
|
|
{
|
|
|
|
struct lightningd *ld = info->cmd->ld;
|
|
|
|
struct command_result *ret;
|
|
|
|
bool warning_mpp, warning_capacity, warning_deadends, warning_offline, warning_private_unused;
|
|
|
|
|
|
|
|
ret = add_routehints(info, buffer, toks,
|
|
|
|
&warning_mpp,
|
|
|
|
&warning_capacity,
|
|
|
|
&warning_deadends,
|
|
|
|
&warning_offline,
|
|
|
|
&warning_private_unused);
|
|
|
|
if (ret)
|
|
|
|
return;
|
|
|
|
|
|
|
|
/* We're actually outside a db transaction here: spooky! */
|
|
|
|
db_begin_transaction(ld->wallet->db);
|
|
|
|
invoice_complete(info,
|
|
|
|
false,
|
|
|
|
warning_mpp,
|
|
|
|
warning_capacity,
|
|
|
|
warning_deadends,
|
|
|
|
warning_offline,
|
|
|
|
warning_private_unused);
|
|
|
|
db_commit_transaction(ld->wallet->db);
|
json-invoice: add routeboost, warnings.
We split json_invoice(), as it now needs to round-trip to the gossipd,
and uniqueness checks need to happen *after* gossipd replies to avoid
a race.
For every candidate channel gossipd gives us, we check that it's in
state NORMAL (not shutting down, not still waiting for lockin), that
it's connected, and that it has capacity. We then choose one with
probability weighted by excess capacity, so larger channels are more
likely.
As a side effect of this, we can tell if an invoice is unpayble (no
channels have sufficient incoming capacity) or difficuly (no *online*
channels have sufficient capacity), so we add those warnings.
Signed-off-by: Rusty Russell <rusty@rustcorp.com.au>
2018-09-27 07:29:17 +02:00
|
|
|
}
|
|
|
|
|
2019-01-15 05:00:27 +01:00
|
|
|
#if DEVELOPER
|
|
|
|
/* Since this is a dev-only option, we will crash if dev-routes is not
|
|
|
|
* an array-of-arrays-of-correct-items. */
|
|
|
|
static struct route_info *unpack_route(const tal_t *ctx,
|
|
|
|
const char *buffer,
|
|
|
|
const jsmntok_t *routetok)
|
|
|
|
{
|
2019-01-17 03:12:13 +01:00
|
|
|
const jsmntok_t *t;
|
|
|
|
size_t i;
|
|
|
|
struct route_info *route = tal_arr(ctx, struct route_info, routetok->size);
|
2019-01-15 05:00:27 +01:00
|
|
|
|
2019-01-17 03:12:13 +01:00
|
|
|
json_for_each_arr(i, t, routetok) {
|
2019-01-15 05:00:27 +01:00
|
|
|
const jsmntok_t *pubkey, *fee_base, *fee_prop, *scid, *cltv;
|
2019-01-17 03:12:13 +01:00
|
|
|
struct route_info *r = &route[i];
|
2019-01-15 05:00:27 +01:00
|
|
|
u32 cltv_u32;
|
|
|
|
|
|
|
|
pubkey = json_get_member(buffer, t, "id");
|
|
|
|
scid = json_get_member(buffer, t, "short_channel_id");
|
|
|
|
fee_base = json_get_member(buffer, t, "fee_base_msat");
|
|
|
|
fee_prop = json_get_member(buffer, t,
|
|
|
|
"fee_proportional_millionths");
|
|
|
|
cltv = json_get_member(buffer, t, "cltv_expiry_delta");
|
|
|
|
|
2019-04-08 11:58:32 +02:00
|
|
|
if (!json_to_node_id(buffer, pubkey, &r->pubkey)
|
2019-01-15 05:00:27 +01:00
|
|
|
|| !json_to_short_channel_id(buffer, scid,
|
2019-09-06 08:41:41 +02:00
|
|
|
&r->short_channel_id)
|
2019-01-17 03:12:13 +01:00
|
|
|
|| !json_to_number(buffer, fee_base, &r->fee_base_msat)
|
2019-01-15 05:00:27 +01:00
|
|
|
|| !json_to_number(buffer, fee_prop,
|
2019-01-17 03:12:13 +01:00
|
|
|
&r->fee_proportional_millionths)
|
2019-01-15 05:00:27 +01:00
|
|
|
|| !json_to_number(buffer, cltv, &cltv_u32))
|
|
|
|
abort();
|
|
|
|
/* We don't have a json_to_u16 */
|
2019-01-17 03:12:13 +01:00
|
|
|
r->cltv_expiry_delta = cltv_u32;
|
2019-01-15 05:00:27 +01:00
|
|
|
}
|
|
|
|
return route;
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct route_info **unpack_routes(const tal_t *ctx,
|
|
|
|
const char *buffer,
|
|
|
|
const jsmntok_t *routestok)
|
|
|
|
{
|
|
|
|
struct route_info **routes;
|
2019-01-17 03:12:13 +01:00
|
|
|
const jsmntok_t *t;
|
|
|
|
size_t i;
|
2019-01-15 05:00:27 +01:00
|
|
|
|
|
|
|
if (!routestok)
|
|
|
|
return NULL;
|
|
|
|
|
2019-01-17 03:12:13 +01:00
|
|
|
routes = tal_arr(ctx, struct route_info *, routestok->size);
|
|
|
|
json_for_each_arr(i, t, routestok)
|
|
|
|
routes[i] = unpack_route(routes, buffer, t);
|
|
|
|
|
2019-01-15 05:00:27 +01:00
|
|
|
return routes;
|
|
|
|
}
|
|
|
|
#endif /* DEVELOPER */
|
|
|
|
|
2020-08-25 03:05:46 +02:00
|
|
|
static struct command_result *param_positive_msat_or_any(struct command *cmd,
|
|
|
|
const char *name,
|
|
|
|
const char *buffer,
|
|
|
|
const jsmntok_t *tok,
|
|
|
|
struct amount_msat **msat)
|
2019-02-21 01:45:57 +01:00
|
|
|
{
|
|
|
|
if (json_tok_streq(buffer, tok, "any")) {
|
|
|
|
*msat = NULL;
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
*msat = tal(cmd, struct amount_msat);
|
2020-08-25 03:05:46 +02:00
|
|
|
if (parse_amount_msat(*msat, buffer + tok->start, tok->end - tok->start)
|
|
|
|
&& !amount_msat_eq(**msat, AMOUNT_MSAT(0)))
|
2019-02-21 01:45:57 +01:00
|
|
|
return NULL;
|
|
|
|
|
2020-08-25 23:20:50 +02:00
|
|
|
return command_fail_badparam(cmd, name, buffer, tok,
|
|
|
|
"should be positive msat or 'any'");
|
2019-02-21 01:45:57 +01:00
|
|
|
}
|
|
|
|
|
2019-04-10 06:58:24 +02:00
|
|
|
/* Parse time with optional suffix, return seconds */
|
|
|
|
static struct command_result *param_time(struct command *cmd, const char *name,
|
|
|
|
const char *buffer,
|
|
|
|
const jsmntok_t *tok,
|
|
|
|
uint64_t **secs)
|
|
|
|
{
|
|
|
|
/* We need to manipulate this, so make copy */
|
|
|
|
jsmntok_t timetok = *tok;
|
|
|
|
u64 mul;
|
|
|
|
char s;
|
|
|
|
struct {
|
|
|
|
char suffix;
|
|
|
|
u64 mul;
|
|
|
|
} suffixes[] = {
|
|
|
|
{ 's', 1 },
|
|
|
|
{ 'm', 60 },
|
|
|
|
{ 'h', 60*60 },
|
|
|
|
{ 'd', 24*60*60 },
|
|
|
|
{ 'w', 7*24*60*60 } };
|
|
|
|
|
|
|
|
mul = 1;
|
|
|
|
if (timetok.end == timetok.start)
|
|
|
|
s = '\0';
|
|
|
|
else
|
|
|
|
s = buffer[timetok.end - 1];
|
|
|
|
for (size_t i = 0; i < ARRAY_SIZE(suffixes); i++) {
|
|
|
|
if (s == suffixes[i].suffix) {
|
|
|
|
mul = suffixes[i].mul;
|
|
|
|
timetok.end--;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
*secs = tal(cmd, uint64_t);
|
|
|
|
if (json_to_u64(buffer, &timetok, *secs)) {
|
|
|
|
if (mul_overflows_u64(**secs, mul)) {
|
2020-08-25 23:20:50 +02:00
|
|
|
return command_fail_badparam(cmd, name, buffer, tok,
|
|
|
|
"value too large");
|
2019-04-10 06:58:24 +02:00
|
|
|
}
|
|
|
|
**secs *= mul;
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2020-08-25 23:20:50 +02:00
|
|
|
return command_fail_badparam(cmd, name, buffer, tok,
|
|
|
|
"should be a number with optional {s,m,h,d,w} suffix");
|
2019-04-10 06:58:24 +02:00
|
|
|
}
|
|
|
|
|
2019-12-17 07:18:13 +01:00
|
|
|
static struct command_result *param_chanhints(struct command *cmd,
|
|
|
|
const char *name,
|
|
|
|
const char *buffer,
|
|
|
|
const jsmntok_t *tok,
|
|
|
|
struct chanhints **chanhints)
|
|
|
|
{
|
|
|
|
bool boolhint;
|
|
|
|
|
|
|
|
*chanhints = tal(cmd, struct chanhints);
|
|
|
|
|
|
|
|
/* Could be simply "true" or "false" */
|
|
|
|
if (json_to_bool(buffer, tok, &boolhint)) {
|
|
|
|
(*chanhints)->expose_all_private = boolhint;
|
2021-06-14 23:07:38 +02:00
|
|
|
(*chanhints)->hints = NULL;
|
2019-12-17 07:18:13 +01:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2021-06-14 23:07:38 +02:00
|
|
|
(*chanhints)->expose_all_private = true;
|
2019-12-17 07:18:13 +01:00
|
|
|
/* Could be a single short_channel_id or an array */
|
|
|
|
if (tok->type == JSMN_ARRAY) {
|
|
|
|
size_t i;
|
|
|
|
const jsmntok_t *t;
|
|
|
|
|
|
|
|
(*chanhints)->hints
|
|
|
|
= tal_arr(*chanhints, struct short_channel_id,
|
|
|
|
tok->size);
|
|
|
|
json_for_each_arr(i, t, tok) {
|
|
|
|
if (!json_to_short_channel_id(buffer, t,
|
|
|
|
&(*chanhints)->hints[i])) {
|
2020-08-25 23:20:50 +02:00
|
|
|
return command_fail_badparam(cmd, name, buffer, t,
|
|
|
|
"should be a short channel id");
|
2019-12-17 07:18:13 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Otherwise should be a short_channel_id */
|
|
|
|
return param_short_channel_id(cmd, name, buffer, tok,
|
|
|
|
&(*chanhints)->hints);
|
|
|
|
}
|
|
|
|
|
2020-12-14 02:21:48 +01:00
|
|
|
static struct command_result *param_preimage(struct command *cmd,
|
|
|
|
const char *name,
|
|
|
|
const char *buffer,
|
|
|
|
const jsmntok_t *tok,
|
|
|
|
struct preimage **preimage)
|
|
|
|
{
|
|
|
|
*preimage = tal(cmd, struct preimage);
|
|
|
|
if (!hex_decode(buffer + tok->start, tok->end - tok->start,
|
|
|
|
*preimage, sizeof(**preimage)))
|
|
|
|
return command_fail_badparam(cmd, "preimage",
|
|
|
|
buffer, tok,
|
|
|
|
"should be 64 hex digits");
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2018-12-16 05:52:06 +01:00
|
|
|
static struct command_result *json_invoice(struct command *cmd,
|
|
|
|
const char *buffer,
|
|
|
|
const jsmntok_t *obj UNNEEDED,
|
|
|
|
const jsmntok_t *params)
|
json-invoice: add routeboost, warnings.
We split json_invoice(), as it now needs to round-trip to the gossipd,
and uniqueness checks need to happen *after* gossipd replies to avoid
a race.
For every candidate channel gossipd gives us, we check that it's in
state NORMAL (not shutting down, not still waiting for lockin), that
it's connected, and that it has capacity. We then choose one with
probability weighted by excess capacity, so larger channels are more
likely.
As a side effect of this, we can tell if an invoice is unpayble (no
channels have sufficient incoming capacity) or difficuly (no *online*
channels have sufficient capacity), so we add those warnings.
Signed-off-by: Rusty Russell <rusty@rustcorp.com.au>
2018-09-27 07:29:17 +02:00
|
|
|
{
|
2018-08-29 22:44:04 +02:00
|
|
|
const jsmntok_t *fallbacks;
|
2019-02-21 01:45:57 +01:00
|
|
|
struct amount_msat *msatoshi_val;
|
json-invoice: add routeboost, warnings.
We split json_invoice(), as it now needs to round-trip to the gossipd,
and uniqueness checks need to happen *after* gossipd replies to avoid
a race.
For every candidate channel gossipd gives us, we check that it's in
state NORMAL (not shutting down, not still waiting for lockin), that
it's connected, and that it has capacity. We then choose one with
probability weighted by excess capacity, so larger channels are more
likely.
As a side effect of this, we can tell if an invoice is unpayble (no
channels have sufficient incoming capacity) or difficuly (no *online*
channels have sufficient capacity), so we add those warnings.
Signed-off-by: Rusty Russell <rusty@rustcorp.com.au>
2018-09-27 07:29:17 +02:00
|
|
|
struct invoice_info *info;
|
2018-02-17 12:48:22 +01:00
|
|
|
const char *desc_val;
|
2018-04-05 07:13:51 +02:00
|
|
|
const u8 **fallback_scripts = NULL;
|
2018-08-13 18:16:41 +02:00
|
|
|
u64 *expiry;
|
json-invoice: add routeboost, warnings.
We split json_invoice(), as it now needs to round-trip to the gossipd,
and uniqueness checks need to happen *after* gossipd replies to avoid
a race.
For every candidate channel gossipd gives us, we check that it's in
state NORMAL (not shutting down, not still waiting for lockin), that
it's connected, and that it has capacity. We then choose one with
probability weighted by excess capacity, so larger channels are more
likely.
As a side effect of this, we can tell if an invoice is unpayble (no
channels have sufficient incoming capacity) or difficuly (no *online*
channels have sufficient capacity), so we add those warnings.
Signed-off-by: Rusty Russell <rusty@rustcorp.com.au>
2018-09-27 07:29:17 +02:00
|
|
|
struct sha256 rhash;
|
2019-11-23 01:19:23 +01:00
|
|
|
struct secret payment_secret;
|
2020-12-14 02:21:48 +01:00
|
|
|
struct preimage *preimage;
|
2021-01-08 05:43:50 +01:00
|
|
|
u32 *cltv;
|
2021-06-14 23:07:38 +02:00
|
|
|
struct jsonrpc_request *req;
|
|
|
|
struct plugin *plugin;
|
2019-01-15 05:00:27 +01:00
|
|
|
#if DEVELOPER
|
|
|
|
const jsmntok_t *routes;
|
|
|
|
#endif
|
json-invoice: add routeboost, warnings.
We split json_invoice(), as it now needs to round-trip to the gossipd,
and uniqueness checks need to happen *after* gossipd replies to avoid
a race.
For every candidate channel gossipd gives us, we check that it's in
state NORMAL (not shutting down, not still waiting for lockin), that
it's connected, and that it has capacity. We then choose one with
probability weighted by excess capacity, so larger channels are more
likely.
As a side effect of this, we can tell if an invoice is unpayble (no
channels have sufficient incoming capacity) or difficuly (no *online*
channels have sufficient capacity), so we add those warnings.
Signed-off-by: Rusty Russell <rusty@rustcorp.com.au>
2018-09-27 07:29:17 +02:00
|
|
|
|
|
|
|
info = tal(cmd, struct invoice_info);
|
|
|
|
info->cmd = cmd;
|
2016-09-06 09:17:49 +02:00
|
|
|
|
2018-07-16 22:48:38 +02:00
|
|
|
if (!param(cmd, buffer, params,
|
2020-08-25 03:05:46 +02:00
|
|
|
p_req("msatoshi", param_positive_msat_or_any, &msatoshi_val),
|
2018-12-16 05:50:06 +01:00
|
|
|
p_req("label", param_label, &info->label),
|
|
|
|
p_req("description", param_escaped_string, &desc_val),
|
2019-04-10 06:58:24 +02:00
|
|
|
p_opt_def("expiry", param_time, &expiry, 3600*24*7),
|
2018-12-16 05:50:06 +01:00
|
|
|
p_opt("fallbacks", param_array, &fallbacks),
|
2020-12-14 02:21:48 +01:00
|
|
|
p_opt("preimage", param_preimage, &preimage),
|
2019-12-17 07:18:13 +01:00
|
|
|
p_opt("exposeprivatechannels", param_chanhints,
|
|
|
|
&info->chanhints),
|
2021-01-08 05:43:50 +01:00
|
|
|
p_opt_def("cltv", param_number, &cltv,
|
|
|
|
cmd->ld->config.cltv_final),
|
2019-01-15 05:00:27 +01:00
|
|
|
#if DEVELOPER
|
|
|
|
p_opt("dev-routes", param_array, &routes),
|
|
|
|
#endif
|
2018-07-16 22:48:38 +02:00
|
|
|
NULL))
|
2018-12-16 05:52:06 +01:00
|
|
|
return command_param_failed();
|
2016-09-06 09:17:49 +02:00
|
|
|
|
json-invoice: add routeboost, warnings.
We split json_invoice(), as it now needs to round-trip to the gossipd,
and uniqueness checks need to happen *after* gossipd replies to avoid
a race.
For every candidate channel gossipd gives us, we check that it's in
state NORMAL (not shutting down, not still waiting for lockin), that
it's connected, and that it has capacity. We then choose one with
probability weighted by excess capacity, so larger channels are more
likely.
As a side effect of this, we can tell if an invoice is unpayble (no
channels have sufficient incoming capacity) or difficuly (no *online*
channels have sufficient capacity), so we add those warnings.
Signed-off-by: Rusty Russell <rusty@rustcorp.com.au>
2018-09-27 07:29:17 +02:00
|
|
|
if (strlen(info->label->s) > INVOICE_MAX_LABEL_LEN) {
|
2018-12-16 05:52:06 +01:00
|
|
|
return command_fail(cmd, JSONRPC2_INVALID_PARAMS,
|
|
|
|
"Label '%s' over %u bytes", info->label->s,
|
|
|
|
INVOICE_MAX_LABEL_LEN);
|
2016-09-06 09:17:49 +02:00
|
|
|
}
|
2018-03-26 02:08:47 +02:00
|
|
|
|
2021-09-20 14:00:03 +02:00
|
|
|
if (strlen(desc_val) > BOLT11_FIELD_BYTE_LIMIT) {
|
2018-12-16 05:52:06 +01:00
|
|
|
return command_fail(cmd, JSONRPC2_INVALID_PARAMS,
|
|
|
|
"Descriptions greater than %d bytes "
|
|
|
|
"not yet supported "
|
|
|
|
"(description length %zu)",
|
|
|
|
BOLT11_FIELD_BYTE_LIMIT,
|
|
|
|
strlen(desc_val));
|
2018-02-17 12:48:22 +01:00
|
|
|
}
|
2017-11-21 07:26:15 +01:00
|
|
|
|
2018-04-05 07:13:51 +02:00
|
|
|
if (fallbacks) {
|
2019-01-17 03:12:13 +01:00
|
|
|
size_t i;
|
|
|
|
const jsmntok_t *t;
|
2018-04-05 07:13:51 +02:00
|
|
|
|
2019-01-17 03:12:13 +01:00
|
|
|
fallback_scripts = tal_arr(cmd, const u8 *, fallbacks->size);
|
|
|
|
json_for_each_arr(i, t, fallbacks) {
|
2018-12-16 05:52:06 +01:00
|
|
|
struct command_result *r;
|
2019-01-17 03:12:13 +01:00
|
|
|
|
|
|
|
r = parse_fallback(cmd, buffer, t, &fallback_scripts[i]);
|
2018-12-16 05:52:06 +01:00
|
|
|
if (r)
|
|
|
|
return r;
|
2018-04-05 07:13:51 +02:00
|
|
|
}
|
2018-02-15 00:45:04 +01:00
|
|
|
}
|
|
|
|
|
2020-12-14 02:21:48 +01:00
|
|
|
if (preimage)
|
|
|
|
info->payment_preimage = *preimage;
|
|
|
|
else
|
2018-04-22 15:42:23 +02:00
|
|
|
/* Generate random secret preimage. */
|
json-invoice: add routeboost, warnings.
We split json_invoice(), as it now needs to round-trip to the gossipd,
and uniqueness checks need to happen *after* gossipd replies to avoid
a race.
For every candidate channel gossipd gives us, we check that it's in
state NORMAL (not shutting down, not still waiting for lockin), that
it's connected, and that it has capacity. We then choose one with
probability weighted by excess capacity, so larger channels are more
likely.
As a side effect of this, we can tell if an invoice is unpayble (no
channels have sufficient incoming capacity) or difficuly (no *online*
channels have sufficient capacity), so we add those warnings.
Signed-off-by: Rusty Russell <rusty@rustcorp.com.au>
2018-09-27 07:29:17 +02:00
|
|
|
randombytes_buf(&info->payment_preimage,
|
|
|
|
sizeof(info->payment_preimage));
|
2018-04-22 15:42:23 +02:00
|
|
|
/* Generate preimage hash. */
|
json-invoice: add routeboost, warnings.
We split json_invoice(), as it now needs to round-trip to the gossipd,
and uniqueness checks need to happen *after* gossipd replies to avoid
a race.
For every candidate channel gossipd gives us, we check that it's in
state NORMAL (not shutting down, not still waiting for lockin), that
it's connected, and that it has capacity. We then choose one with
probability weighted by excess capacity, so larger channels are more
likely.
As a side effect of this, we can tell if an invoice is unpayble (no
channels have sufficient incoming capacity) or difficuly (no *online*
channels have sufficient capacity), so we add those warnings.
Signed-off-by: Rusty Russell <rusty@rustcorp.com.au>
2018-09-27 07:29:17 +02:00
|
|
|
sha256(&rhash, &info->payment_preimage, sizeof(info->payment_preimage));
|
2019-11-23 01:19:23 +01:00
|
|
|
/* Generate payment secret. */
|
|
|
|
invoice_secret(&info->payment_preimage, &payment_secret);
|
2018-04-25 01:04:33 +02:00
|
|
|
|
2019-02-21 03:38:35 +01:00
|
|
|
info->b11 = new_bolt11(info, msatoshi_val);
|
2019-01-31 00:42:06 +01:00
|
|
|
info->b11->chain = chainparams;
|
json-invoice: add routeboost, warnings.
We split json_invoice(), as it now needs to round-trip to the gossipd,
and uniqueness checks need to happen *after* gossipd replies to avoid
a race.
For every candidate channel gossipd gives us, we check that it's in
state NORMAL (not shutting down, not still waiting for lockin), that
it's connected, and that it has capacity. We then choose one with
probability weighted by excess capacity, so larger channels are more
likely.
As a side effect of this, we can tell if an invoice is unpayble (no
channels have sufficient incoming capacity) or difficuly (no *online*
channels have sufficient capacity), so we add those warnings.
Signed-off-by: Rusty Russell <rusty@rustcorp.com.au>
2018-09-27 07:29:17 +02:00
|
|
|
info->b11->timestamp = time_now().ts.tv_sec;
|
|
|
|
info->b11->payment_hash = rhash;
|
|
|
|
info->b11->receiver_id = cmd->ld->id;
|
2021-01-08 05:43:50 +01:00
|
|
|
info->b11->min_final_cltv_expiry = *cltv;
|
json-invoice: add routeboost, warnings.
We split json_invoice(), as it now needs to round-trip to the gossipd,
and uniqueness checks need to happen *after* gossipd replies to avoid
a race.
For every candidate channel gossipd gives us, we check that it's in
state NORMAL (not shutting down, not still waiting for lockin), that
it's connected, and that it has capacity. We then choose one with
probability weighted by excess capacity, so larger channels are more
likely.
As a side effect of this, we can tell if an invoice is unpayble (no
channels have sufficient incoming capacity) or difficuly (no *online*
channels have sufficient capacity), so we add those warnings.
Signed-off-by: Rusty Russell <rusty@rustcorp.com.au>
2018-09-27 07:29:17 +02:00
|
|
|
info->b11->expiry = *expiry;
|
|
|
|
info->b11->description = tal_steal(info->b11, desc_val);
|
|
|
|
info->b11->description_hash = NULL;
|
2019-11-23 01:19:23 +01:00
|
|
|
info->b11->payment_secret = tal_dup(info->b11, struct secret,
|
|
|
|
&payment_secret);
|
2020-04-02 06:04:47 +02:00
|
|
|
info->b11->features = tal_dup_talarr(info->b11, u8,
|
2020-04-03 02:03:59 +02:00
|
|
|
cmd->ld->our_features
|
2020-04-02 06:04:47 +02:00
|
|
|
->bits[BOLT11_FEATURE]);
|
2018-02-28 17:26:58 +01:00
|
|
|
|
2019-01-15 05:00:27 +01:00
|
|
|
#if DEVELOPER
|
|
|
|
info->b11->routes = unpack_routes(info->b11, buffer, routes);
|
2020-08-05 15:50:38 +02:00
|
|
|
#else
|
|
|
|
info->b11->routes = NULL;
|
2019-01-15 05:00:27 +01:00
|
|
|
#endif
|
json-invoice: add routeboost, warnings.
We split json_invoice(), as it now needs to round-trip to the gossipd,
and uniqueness checks need to happen *after* gossipd replies to avoid
a race.
For every candidate channel gossipd gives us, we check that it's in
state NORMAL (not shutting down, not still waiting for lockin), that
it's connected, and that it has capacity. We then choose one with
probability weighted by excess capacity, so larger channels are more
likely.
As a side effect of this, we can tell if an invoice is unpayble (no
channels have sufficient incoming capacity) or difficuly (no *online*
channels have sufficient capacity), so we add those warnings.
Signed-off-by: Rusty Russell <rusty@rustcorp.com.au>
2018-09-27 07:29:17 +02:00
|
|
|
if (fallback_scripts)
|
|
|
|
info->b11->fallbacks = tal_steal(info->b11, fallback_scripts);
|
2018-02-28 17:26:58 +01:00
|
|
|
|
2021-06-14 23:07:38 +02:00
|
|
|
req = jsonrpc_request_start(info, "listincoming",
|
|
|
|
cmd->ld->log,
|
|
|
|
NULL, listincoming_done,
|
|
|
|
info);
|
|
|
|
jsonrpc_request_end(req);
|
|
|
|
|
|
|
|
plugin = find_plugin_for_command(cmd->ld, "listincoming");
|
|
|
|
if (plugin) {
|
|
|
|
plugin_request_send(plugin, req);
|
|
|
|
return command_still_pending(cmd);
|
|
|
|
}
|
2016-09-06 09:17:49 +02:00
|
|
|
|
2021-06-14 23:07:38 +02:00
|
|
|
/* We can't generate routehints without listincoming. */
|
|
|
|
return invoice_complete(info, true,
|
|
|
|
false, false, false, false, false);
|
2016-09-06 09:17:49 +02:00
|
|
|
}
|
|
|
|
|
2017-01-04 04:38:15 +01:00
|
|
|
static const struct json_command invoice_command = {
|
2019-05-22 16:08:16 +02:00
|
|
|
"invoice",
|
|
|
|
"payment",
|
|
|
|
json_invoice,
|
|
|
|
"Create an invoice for {msatoshi} with {label} "
|
|
|
|
"and {description} with optional {expiry} seconds "
|
2019-07-25 06:37:16 +02:00
|
|
|
"(default 1 week), optional {fallbacks} address list"
|
2019-05-22 16:08:16 +02:00
|
|
|
"(default empty list) and optional {preimage} "
|
|
|
|
"(default autogenerated)"};
|
2017-01-04 04:38:15 +01:00
|
|
|
AUTODATA(json_command, &invoice_command);
|
2016-09-06 09:17:49 +02:00
|
|
|
|
2018-10-19 03:17:49 +02:00
|
|
|
static void json_add_invoices(struct json_stream *response,
|
2018-01-14 15:15:30 +01:00
|
|
|
struct wallet *wallet,
|
2021-01-05 18:55:01 +01:00
|
|
|
const struct json_escape *label,
|
2021-07-02 02:11:35 +02:00
|
|
|
const struct sha256 *payment_hash,
|
|
|
|
const struct sha256 *local_offer_id)
|
2016-09-06 09:17:49 +02:00
|
|
|
{
|
2018-02-24 13:11:14 +01:00
|
|
|
struct invoice_iterator it;
|
2018-07-27 12:57:02 +02:00
|
|
|
const struct invoice_details *details;
|
2021-01-05 18:55:01 +01:00
|
|
|
struct invoice invoice;
|
2016-09-06 09:17:49 +02:00
|
|
|
|
2018-06-14 06:50:47 +02:00
|
|
|
/* Don't iterate entire db if we're just after one. */
|
|
|
|
if (label) {
|
|
|
|
if (wallet_invoice_find_by_label(wallet, &invoice, label)) {
|
2021-01-05 18:55:01 +01:00
|
|
|
details =
|
|
|
|
wallet_invoice_details(response, wallet, invoice);
|
2019-06-12 02:38:54 +02:00
|
|
|
json_object_start(response, NULL);
|
2018-07-27 12:57:02 +02:00
|
|
|
json_add_invoice(response, details);
|
2019-06-12 02:38:54 +02:00
|
|
|
json_object_end(response);
|
2018-06-14 06:50:47 +02:00
|
|
|
}
|
2021-01-05 18:55:01 +01:00
|
|
|
} else if (payment_hash != NULL) {
|
|
|
|
if (wallet_invoice_find_by_rhash(wallet, &invoice,
|
|
|
|
payment_hash)) {
|
|
|
|
json_object_start(response, NULL);
|
|
|
|
json_add_invoice(
|
|
|
|
response,
|
|
|
|
wallet_invoice_details(response, wallet, invoice));
|
|
|
|
json_object_end(response);
|
|
|
|
}
|
2018-06-14 06:50:47 +02:00
|
|
|
|
2021-01-05 18:55:01 +01:00
|
|
|
} else {
|
|
|
|
memset(&it, 0, sizeof(it));
|
|
|
|
while (wallet_invoice_iterate(wallet, &it)) {
|
|
|
|
details = wallet_invoice_iterator_deref(response,
|
|
|
|
wallet, &it);
|
2021-07-02 02:11:35 +02:00
|
|
|
/* FIXME: db can filter this better! */
|
|
|
|
if (local_offer_id) {
|
|
|
|
if (!details->local_offer_id
|
|
|
|
|| !sha256_eq(local_offer_id,
|
|
|
|
details->local_offer_id))
|
|
|
|
continue;
|
|
|
|
}
|
2021-01-05 18:55:01 +01:00
|
|
|
json_object_start(response, NULL);
|
|
|
|
json_add_invoice(response, details);
|
|
|
|
json_object_end(response);
|
|
|
|
}
|
2016-09-06 09:17:49 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-12-16 05:52:06 +01:00
|
|
|
static struct command_result *json_listinvoices(struct command *cmd,
|
|
|
|
const char *buffer,
|
|
|
|
const jsmntok_t *obj UNNEEDED,
|
|
|
|
const jsmntok_t *params)
|
2016-09-06 09:17:49 +02:00
|
|
|
{
|
2019-06-12 02:38:54 +02:00
|
|
|
struct json_escape *label;
|
2018-10-19 03:17:49 +02:00
|
|
|
struct json_stream *response;
|
2018-01-14 15:15:30 +01:00
|
|
|
struct wallet *wallet = cmd->ld->wallet;
|
2021-01-05 18:55:01 +01:00
|
|
|
const char *invstring;
|
2021-07-02 02:11:35 +02:00
|
|
|
struct sha256 *payment_hash, *offer_id;
|
2021-01-05 18:55:01 +01:00
|
|
|
char *fail;
|
|
|
|
|
2018-07-16 22:48:38 +02:00
|
|
|
if (!param(cmd, buffer, params,
|
2018-12-16 05:50:06 +01:00
|
|
|
p_opt("label", param_label, &label),
|
2021-01-05 18:55:01 +01:00
|
|
|
p_opt("invstring", param_string, &invstring),
|
|
|
|
p_opt("payment_hash", param_sha256, &payment_hash),
|
2021-07-02 02:11:35 +02:00
|
|
|
p_opt("offer_id", param_sha256, &offer_id),
|
2018-07-16 22:48:38 +02:00
|
|
|
NULL))
|
2018-12-16 05:52:06 +01:00
|
|
|
return command_param_failed();
|
2021-01-05 18:55:01 +01:00
|
|
|
|
2021-07-02 02:11:35 +02:00
|
|
|
/* Yeah, I wasn't sure about this style either. It's curt though! */
|
|
|
|
if (!!label + !!invstring + !!payment_hash + !!offer_id > 1) {
|
2021-01-05 18:55:01 +01:00
|
|
|
return command_fail(cmd, JSONRPC2_INVALID_PARAMS,
|
|
|
|
"Can only specify one of"
|
2021-07-02 02:11:35 +02:00
|
|
|
" {label}, {invstring}, {payment_hash}"
|
|
|
|
" or {offer_id}");
|
2021-01-05 18:55:01 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Extract the payment_hash from the invoice. */
|
|
|
|
if (invstring != NULL) {
|
2021-07-01 06:23:25 +02:00
|
|
|
struct bolt11 *b11;
|
2021-01-05 18:55:01 +01:00
|
|
|
b11 = bolt11_decode(cmd, invstring, cmd->ld->our_features, NULL,
|
|
|
|
NULL, &fail);
|
2021-07-01 06:23:25 +02:00
|
|
|
if (b11)
|
|
|
|
payment_hash = &b11->payment_hash;
|
|
|
|
else {
|
|
|
|
struct tlv_invoice *b12
|
|
|
|
= invoice_decode(tmpctx, invstring,
|
|
|
|
strlen(invstring),
|
|
|
|
cmd->ld->our_features, NULL,
|
|
|
|
&fail);
|
|
|
|
if (!b12 || !b12->payment_hash) {
|
|
|
|
return command_fail(cmd, JSONRPC2_INVALID_PARAMS,
|
|
|
|
"Invalid invstring");
|
|
|
|
}
|
|
|
|
payment_hash = b12->payment_hash;
|
|
|
|
}
|
2021-01-05 18:55:01 +01:00
|
|
|
}
|
|
|
|
|
2018-10-19 03:17:48 +02:00
|
|
|
response = json_stream_success(cmd);
|
2018-07-29 04:08:29 +02:00
|
|
|
json_array_start(response, "invoices");
|
2021-07-02 02:11:35 +02:00
|
|
|
json_add_invoices(response, wallet, label, payment_hash, offer_id);
|
2016-09-06 09:17:49 +02:00
|
|
|
json_array_end(response);
|
2018-12-16 05:52:06 +01:00
|
|
|
return command_success(cmd, response);
|
2016-09-06 09:17:49 +02:00
|
|
|
}
|
|
|
|
|
2018-01-16 21:29:32 +01:00
|
|
|
static const struct json_command listinvoices_command = {
|
2018-01-16 21:28:46 +01:00
|
|
|
"listinvoices",
|
2019-05-22 16:08:16 +02:00
|
|
|
"payment",
|
2018-01-16 21:28:46 +01:00
|
|
|
json_listinvoices,
|
2021-07-02 02:11:35 +02:00
|
|
|
"Show invoice matching {label}, {invstring}, {payment_hash} or {offerid} (or all, if "
|
2021-01-05 18:55:01 +01:00
|
|
|
"no query parameter specified)"
|
2018-01-16 21:28:46 +01:00
|
|
|
};
|
|
|
|
AUTODATA(json_command, &listinvoices_command);
|
|
|
|
|
2018-12-16 05:52:06 +01:00
|
|
|
static struct command_result *json_delinvoice(struct command *cmd,
|
|
|
|
const char *buffer,
|
|
|
|
const jsmntok_t *obj UNNEEDED,
|
|
|
|
const jsmntok_t *params)
|
2016-09-06 09:17:49 +02:00
|
|
|
{
|
2018-02-23 02:04:47 +01:00
|
|
|
struct invoice i;
|
2018-07-27 12:57:02 +02:00
|
|
|
const struct invoice_details *details;
|
2018-10-19 03:17:49 +02:00
|
|
|
struct json_stream *response;
|
2018-03-26 02:08:16 +02:00
|
|
|
const char *status, *actual_status;
|
2019-06-12 02:38:54 +02:00
|
|
|
struct json_escape *label;
|
2018-01-14 15:15:30 +01:00
|
|
|
struct wallet *wallet = cmd->ld->wallet;
|
2016-09-06 09:17:49 +02:00
|
|
|
|
2018-07-16 22:48:38 +02:00
|
|
|
if (!param(cmd, buffer, params,
|
2018-12-16 05:50:06 +01:00
|
|
|
p_req("label", param_label, &label),
|
|
|
|
p_req("status", param_string, &status),
|
2018-07-16 22:48:38 +02:00
|
|
|
NULL))
|
2018-12-16 05:52:06 +01:00
|
|
|
return command_param_failed();
|
2016-09-06 09:17:49 +02:00
|
|
|
|
2018-02-23 02:04:47 +01:00
|
|
|
if (!wallet_invoice_find_by_label(wallet, &i, label)) {
|
2020-07-19 11:04:03 +02:00
|
|
|
return command_fail(cmd, INVOICE_NOT_FOUND, "Unknown invoice");
|
2016-09-06 09:17:49 +02:00
|
|
|
}
|
2018-07-27 12:57:02 +02:00
|
|
|
|
|
|
|
details = wallet_invoice_details(cmd, cmd->ld->wallet, i);
|
2017-10-05 23:29:56 +02:00
|
|
|
|
2018-01-18 04:57:16 +01:00
|
|
|
/* This is time-sensitive, so only call once; otherwise error msg
|
|
|
|
* might not make sense if it changed! */
|
2018-07-27 12:57:02 +02:00
|
|
|
actual_status = invoice_status_str(details);
|
2018-01-18 04:57:16 +01:00
|
|
|
if (!streq(actual_status, status)) {
|
2020-07-19 11:04:03 +02:00
|
|
|
struct json_stream *js;
|
|
|
|
js = json_stream_fail(cmd, INVOICE_STATUS_UNEXPECTED,
|
|
|
|
tal_fmt(tmpctx,
|
|
|
|
"Invoice status is %s not %s",
|
|
|
|
actual_status, status));
|
|
|
|
json_add_string(js, "current_status", actual_status);
|
|
|
|
json_add_string(js, "expected_status", status);
|
|
|
|
json_object_end(js);
|
|
|
|
return command_failed(cmd, js);
|
2018-01-18 04:57:16 +01:00
|
|
|
}
|
|
|
|
|
2018-01-18 04:55:43 +01:00
|
|
|
if (!wallet_invoice_delete(wallet, i)) {
|
|
|
|
log_broken(cmd->ld->log,
|
|
|
|
"Error attempting to remove invoice %"PRIu64,
|
2018-02-23 02:04:47 +01:00
|
|
|
i.id);
|
2020-07-19 11:04:03 +02:00
|
|
|
/* FIXME: allocate a generic DATABASE_ERROR code. */
|
2018-12-16 05:52:06 +01:00
|
|
|
return command_fail(cmd, LIGHTNINGD, "Database error");
|
2017-12-27 12:51:58 +01:00
|
|
|
}
|
|
|
|
|
2018-10-19 03:17:48 +02:00
|
|
|
response = json_stream_success(cmd);
|
|
|
|
json_add_invoice(response, details);
|
2018-12-16 05:52:06 +01:00
|
|
|
return command_success(cmd, response);
|
2016-09-06 09:17:49 +02:00
|
|
|
}
|
|
|
|
|
2017-01-04 04:38:15 +01:00
|
|
|
static const struct json_command delinvoice_command = {
|
2016-09-06 09:17:49 +02:00
|
|
|
"delinvoice",
|
2019-05-22 16:08:16 +02:00
|
|
|
"payment",
|
2016-09-06 09:17:49 +02:00
|
|
|
json_delinvoice,
|
2018-01-18 04:57:16 +01:00
|
|
|
"Delete unpaid invoice {label} with {status}",
|
2016-09-06 09:17:49 +02:00
|
|
|
};
|
2017-01-04 04:38:15 +01:00
|
|
|
AUTODATA(json_command, &delinvoice_command);
|
2016-09-06 09:17:49 +02:00
|
|
|
|
2018-12-16 05:52:06 +01:00
|
|
|
static struct command_result *json_delexpiredinvoice(struct command *cmd,
|
|
|
|
const char *buffer,
|
|
|
|
const jsmntok_t *obj UNNEEDED,
|
|
|
|
const jsmntok_t *params)
|
2018-02-26 13:37:53 +01:00
|
|
|
{
|
2018-08-13 18:16:41 +02:00
|
|
|
u64 *maxexpirytime;
|
2018-02-26 13:37:53 +01:00
|
|
|
|
2018-07-16 22:48:38 +02:00
|
|
|
if (!param(cmd, buffer, params,
|
2018-12-16 05:50:06 +01:00
|
|
|
p_opt_def("maxexpirytime", param_u64, &maxexpirytime,
|
2018-08-13 18:16:41 +02:00
|
|
|
time_now().ts.tv_sec),
|
2018-07-16 22:48:38 +02:00
|
|
|
NULL))
|
2018-12-16 05:52:06 +01:00
|
|
|
return command_param_failed();
|
2018-02-26 13:37:53 +01:00
|
|
|
|
2018-08-13 18:16:41 +02:00
|
|
|
wallet_invoice_delete_expired(cmd->ld->wallet, *maxexpirytime);
|
2018-02-26 13:37:53 +01:00
|
|
|
|
2019-06-12 02:38:54 +02:00
|
|
|
return command_success(cmd, json_stream_success(cmd));
|
2018-02-26 13:37:53 +01:00
|
|
|
}
|
|
|
|
static const struct json_command delexpiredinvoice_command = {
|
|
|
|
"delexpiredinvoice",
|
2019-05-22 16:08:16 +02:00
|
|
|
"payment",
|
2018-02-26 13:37:53 +01:00
|
|
|
json_delexpiredinvoice,
|
|
|
|
"Delete all expired invoices that expired as of given {maxexpirytime} (a UNIX epoch time), or all expired invoices if not specified"
|
|
|
|
};
|
|
|
|
AUTODATA(json_command, &delexpiredinvoice_command);
|
|
|
|
|
2018-12-16 05:52:06 +01:00
|
|
|
static struct command_result *json_waitanyinvoice(struct command *cmd,
|
|
|
|
const char *buffer,
|
|
|
|
const jsmntok_t *obj UNNEEDED,
|
|
|
|
const jsmntok_t *params)
|
2016-09-06 09:17:49 +02:00
|
|
|
{
|
2018-08-13 18:16:41 +02:00
|
|
|
u64 *pay_index;
|
2020-01-28 02:30:00 +01:00
|
|
|
u64 *timeout;
|
2017-12-26 14:47:27 +01:00
|
|
|
struct wallet *wallet = cmd->ld->wallet;
|
2016-09-06 09:17:49 +02:00
|
|
|
|
2018-07-16 22:48:38 +02:00
|
|
|
if (!param(cmd, buffer, params,
|
2018-12-16 05:50:06 +01:00
|
|
|
p_opt_def("lastpay_index", param_u64, &pay_index, 0),
|
2020-01-28 02:30:00 +01:00
|
|
|
p_opt("timeout", ¶m_u64, &timeout),
|
2018-07-16 22:48:38 +02:00
|
|
|
NULL))
|
2018-12-16 05:52:06 +01:00
|
|
|
return command_param_failed();
|
2017-12-26 14:47:27 +01:00
|
|
|
|
2020-01-28 02:30:00 +01:00
|
|
|
/*~ We allocate the timeout and the wallet-waitanyinvoice
|
|
|
|
* in the cmd context, so whichever one manages to complete
|
|
|
|
* the command first (and destroy the cmd context)
|
|
|
|
* auto-cancels the other, is not tal amazing?
|
|
|
|
*/
|
|
|
|
if (timeout)
|
|
|
|
(void) new_reltimer(cmd->ld->timers, cmd,
|
|
|
|
time_from_sec(*timeout),
|
|
|
|
&wait_timed_out, cmd);
|
|
|
|
|
2018-01-14 15:15:30 +01:00
|
|
|
/* Set command as pending. We do not know if
|
|
|
|
* wallet_invoice_waitany will return immediately
|
|
|
|
* or not, so indicating pending is safest. */
|
2018-12-16 05:53:06 +01:00
|
|
|
fixme_ignore(command_still_pending(cmd));
|
2018-01-14 15:15:30 +01:00
|
|
|
|
|
|
|
/* Find next paid invoice. */
|
2018-08-13 18:16:41 +02:00
|
|
|
wallet_invoice_waitany(cmd, wallet, *pay_index,
|
2018-01-14 15:15:30 +01:00
|
|
|
&wait_on_invoice, (void*) cmd);
|
2018-12-16 05:52:06 +01:00
|
|
|
|
2018-12-17 04:52:08 +01:00
|
|
|
return command_its_complicated("wallet_invoice_waitany might complete"
|
|
|
|
" immediately, but we also call it as a"
|
|
|
|
" callback so plumbing through the return"
|
|
|
|
" is non-trivial.");
|
2016-09-06 09:17:49 +02:00
|
|
|
}
|
|
|
|
|
2020-08-11 07:05:56 +02:00
|
|
|
|
2017-01-17 22:09:09 +01:00
|
|
|
static const struct json_command waitanyinvoice_command = {
|
|
|
|
"waitanyinvoice",
|
2019-05-22 16:08:16 +02:00
|
|
|
"payment",
|
2017-01-17 22:09:09 +01:00
|
|
|
json_waitanyinvoice,
|
2020-01-28 02:30:00 +01:00
|
|
|
"Wait for the next invoice to be paid, after {lastpay_index} (if supplied). "
|
|
|
|
"If {timeout} seconds is reached while waiting, fail with an error."
|
2016-09-06 09:17:49 +02:00
|
|
|
};
|
2017-01-17 22:09:09 +01:00
|
|
|
AUTODATA(json_command, &waitanyinvoice_command);
|
2017-01-10 12:18:40 +01:00
|
|
|
|
|
|
|
/* Wait for an incoming payment matching the `label` in the JSON
|
|
|
|
* command. This will either return immediately if the payment has
|
|
|
|
* already been received or it may add the `cmd` to the list of
|
|
|
|
* waiters, if the payment is still pending.
|
|
|
|
*/
|
2018-12-16 05:52:06 +01:00
|
|
|
static struct command_result *json_waitinvoice(struct command *cmd,
|
|
|
|
const char *buffer,
|
|
|
|
const jsmntok_t *obj UNNEEDED,
|
|
|
|
const jsmntok_t *params)
|
2017-01-10 12:18:40 +01:00
|
|
|
{
|
2018-02-23 02:04:47 +01:00
|
|
|
struct invoice i;
|
2018-07-27 12:57:02 +02:00
|
|
|
const struct invoice_details *details;
|
2018-01-14 15:15:30 +01:00
|
|
|
struct wallet *wallet = cmd->ld->wallet;
|
2019-06-12 02:38:54 +02:00
|
|
|
struct json_escape *label;
|
2017-01-10 12:18:40 +01:00
|
|
|
|
2018-07-16 22:48:38 +02:00
|
|
|
if (!param(cmd, buffer, params,
|
2018-12-16 05:50:06 +01:00
|
|
|
p_req("label", param_label, &label),
|
2018-07-16 22:48:38 +02:00
|
|
|
NULL))
|
2018-12-16 05:52:06 +01:00
|
|
|
return command_param_failed();
|
2017-01-10 12:18:40 +01:00
|
|
|
|
2018-02-23 02:04:47 +01:00
|
|
|
if (!wallet_invoice_find_by_label(wallet, &i, label)) {
|
2018-12-16 05:52:06 +01:00
|
|
|
return command_fail(cmd, LIGHTNINGD, "Label not found");
|
2018-02-20 02:07:30 +01:00
|
|
|
}
|
2018-07-27 12:57:02 +02:00
|
|
|
details = wallet_invoice_details(cmd, cmd->ld->wallet, i);
|
2018-02-20 02:07:30 +01:00
|
|
|
|
2018-02-23 02:04:47 +01:00
|
|
|
/* If paid or expired return immediately */
|
2018-07-27 12:57:02 +02:00
|
|
|
if (details->state == PAID || details->state == EXPIRED) {
|
2018-12-16 05:52:06 +01:00
|
|
|
return tell_waiter(cmd, &i);
|
2017-10-03 18:31:07 +02:00
|
|
|
} else {
|
|
|
|
/* There is an unpaid one matching, let's wait... */
|
2018-12-16 05:53:06 +01:00
|
|
|
fixme_ignore(command_still_pending(cmd));
|
2018-01-14 15:15:30 +01:00
|
|
|
wallet_invoice_waitone(cmd, wallet, i,
|
|
|
|
&wait_on_invoice, (void *) cmd);
|
2018-12-17 04:52:08 +01:00
|
|
|
return command_its_complicated("wallet_invoice_waitone might"
|
|
|
|
" complete immediately");
|
2017-01-10 12:18:40 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-01-17 22:09:09 +01:00
|
|
|
static const struct json_command waitinvoice_command = {
|
|
|
|
"waitinvoice",
|
2019-05-22 16:08:16 +02:00
|
|
|
"payment",
|
2017-01-17 22:09:09 +01:00
|
|
|
json_waitinvoice,
|
2018-01-23 02:02:10 +01:00
|
|
|
"Wait for an incoming payment matching the invoice with {label}, or if the invoice expires"
|
2017-01-10 12:18:40 +01:00
|
|
|
};
|
2017-01-17 22:09:09 +01:00
|
|
|
AUTODATA(json_command, &waitinvoice_command);
|
2017-11-22 01:25:39 +01:00
|
|
|
|
2018-12-16 05:52:06 +01:00
|
|
|
static struct command_result *json_decodepay(struct command *cmd,
|
|
|
|
const char *buffer,
|
|
|
|
const jsmntok_t *obj UNNEEDED,
|
|
|
|
const jsmntok_t *params)
|
2017-11-22 01:25:39 +01:00
|
|
|
{
|
|
|
|
struct bolt11 *b11;
|
2018-10-19 03:17:49 +02:00
|
|
|
struct json_stream *response;
|
2018-10-19 03:17:48 +02:00
|
|
|
const char *str, *desc;
|
|
|
|
char *fail;
|
2017-11-22 01:25:39 +01:00
|
|
|
|
2018-07-16 22:48:38 +02:00
|
|
|
if (!param(cmd, buffer, params,
|
2018-12-16 05:50:06 +01:00
|
|
|
p_req("bolt11", param_string, &str),
|
|
|
|
p_opt("description", param_string, &desc),
|
2018-07-16 22:48:38 +02:00
|
|
|
NULL))
|
2018-12-16 05:52:06 +01:00
|
|
|
return command_param_failed();
|
2017-11-22 01:25:39 +01:00
|
|
|
|
2020-09-23 03:07:19 +02:00
|
|
|
b11 = bolt11_decode(cmd, str, cmd->ld->our_features, desc, NULL,
|
|
|
|
&fail);
|
2017-11-22 01:25:39 +01:00
|
|
|
|
|
|
|
if (!b11) {
|
2018-12-16 05:52:06 +01:00
|
|
|
return command_fail(cmd, LIGHTNINGD, "Invalid bolt11: %s", fail);
|
2017-11-22 01:25:39 +01:00
|
|
|
}
|
|
|
|
|
2018-10-19 03:17:48 +02:00
|
|
|
response = json_stream_success(cmd);
|
2021-01-07 19:47:47 +01:00
|
|
|
json_add_bolt11(response, b11);
|
2018-12-16 05:52:06 +01:00
|
|
|
return command_success(cmd, response);
|
2017-11-22 01:25:39 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
static const struct json_command decodepay_command = {
|
|
|
|
"decodepay",
|
2019-05-22 16:08:16 +02:00
|
|
|
"payment",
|
2017-11-22 01:25:39 +01:00
|
|
|
json_decodepay,
|
2018-01-22 09:55:07 +01:00
|
|
|
"Decode {bolt11}, using {description} if necessary"
|
2017-11-22 01:25:39 +01:00
|
|
|
};
|
|
|
|
AUTODATA(json_command, &decodepay_command);
|
2020-12-14 02:21:48 +01:00
|
|
|
|
|
|
|
/* If we fail because it exists, we also return the clashing invoice */
|
|
|
|
static struct command_result *fail_exists(struct command *cmd,
|
|
|
|
const struct json_escape *label)
|
|
|
|
{
|
|
|
|
struct json_stream *data;
|
|
|
|
struct invoice invoice;
|
|
|
|
struct wallet *wallet = cmd->ld->wallet;
|
|
|
|
|
|
|
|
data = json_stream_fail(cmd, INVOICE_LABEL_ALREADY_EXISTS,
|
|
|
|
"Duplicate label");
|
|
|
|
if (!wallet_invoice_find_by_label(wallet, &invoice, label))
|
|
|
|
fatal("Duplicate invoice %s not found any more?",
|
|
|
|
label->s);
|
|
|
|
|
|
|
|
json_add_invoice(data, wallet_invoice_details(cmd, wallet, invoice));
|
|
|
|
json_object_end(data);
|
|
|
|
|
|
|
|
return command_failed(cmd, data);
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct command_result *json_createinvoice(struct command *cmd,
|
|
|
|
const char *buffer,
|
|
|
|
const jsmntok_t *obj UNNEEDED,
|
|
|
|
const jsmntok_t *params)
|
|
|
|
{
|
|
|
|
const char *invstring;
|
|
|
|
struct json_escape *label;
|
|
|
|
struct preimage *preimage;
|
|
|
|
struct invoice invoice;
|
|
|
|
struct sha256 payment_hash;
|
|
|
|
struct json_stream *response;
|
|
|
|
struct bolt11 *b11;
|
|
|
|
struct sha256 hash;
|
|
|
|
u5 *sig;
|
|
|
|
bool have_n;
|
|
|
|
char *fail;
|
|
|
|
|
|
|
|
if (!param(cmd, buffer, params,
|
|
|
|
p_req("invstring", param_string, &invstring),
|
|
|
|
p_req("label", param_label, &label),
|
2020-12-15 00:43:42 +01:00
|
|
|
p_req("preimage", param_preimage, &preimage),
|
2020-12-14 02:21:48 +01:00
|
|
|
NULL))
|
|
|
|
return command_param_failed();
|
|
|
|
|
|
|
|
sha256(&payment_hash, preimage, sizeof(*preimage));
|
|
|
|
b11 = bolt11_decode_nosig(cmd, invstring, cmd->ld->our_features,
|
|
|
|
NULL, chainparams, &hash, &sig, &have_n,
|
|
|
|
&fail);
|
|
|
|
if (b11) {
|
|
|
|
/* This adds the signature */
|
|
|
|
char *b11enc = bolt11_encode(cmd, b11, have_n,
|
|
|
|
hsm_sign_b11, cmd->ld);
|
|
|
|
|
|
|
|
if (!b11->description)
|
|
|
|
return command_fail(cmd, JSONRPC2_INVALID_PARAMS,
|
|
|
|
"Missing description in invoice");
|
|
|
|
|
|
|
|
if (!b11->expiry)
|
|
|
|
return command_fail(cmd, JSONRPC2_INVALID_PARAMS,
|
|
|
|
"Missing expiry in invoice");
|
|
|
|
|
2020-12-15 00:43:42 +01:00
|
|
|
if (!sha256_eq(&payment_hash, &b11->payment_hash))
|
|
|
|
return command_fail(cmd, JSONRPC2_INVALID_PARAMS,
|
|
|
|
"Incorrect preimage");
|
|
|
|
|
2020-12-14 02:21:48 +01:00
|
|
|
if (!wallet_invoice_create(cmd->ld->wallet,
|
|
|
|
&invoice,
|
|
|
|
b11->msat,
|
|
|
|
label,
|
|
|
|
b11->expiry,
|
|
|
|
b11enc,
|
|
|
|
b11->description,
|
|
|
|
b11->features,
|
|
|
|
preimage,
|
|
|
|
&payment_hash,
|
|
|
|
NULL))
|
|
|
|
return fail_exists(cmd, label);
|
|
|
|
|
|
|
|
notify_invoice_creation(cmd->ld, b11->msat, *preimage, label);
|
|
|
|
} else {
|
|
|
|
struct tlv_invoice *inv;
|
|
|
|
struct sha256 *local_offer_id;
|
|
|
|
|
|
|
|
inv = invoice_decode_nosig(cmd, invstring, strlen(invstring),
|
|
|
|
cmd->ld->our_features, chainparams,
|
|
|
|
&fail);
|
|
|
|
if (inv) {
|
|
|
|
char *b12enc;
|
|
|
|
struct amount_msat msat;
|
|
|
|
const char *desc;
|
|
|
|
u32 expiry;
|
|
|
|
enum offer_status status;
|
|
|
|
|
|
|
|
if (inv->signature)
|
|
|
|
return command_fail(cmd, JSONRPC2_INVALID_PARAMS,
|
|
|
|
"invoice already signed");
|
|
|
|
hsm_sign_b12_invoice(cmd->ld, inv);
|
|
|
|
b12enc = invoice_encode(cmd, inv);
|
|
|
|
|
|
|
|
if (inv->offer_id
|
|
|
|
&& wallet_offer_find(tmpctx, cmd->ld->wallet,
|
|
|
|
inv->offer_id, NULL, &status)) {
|
|
|
|
if (!offer_status_active(status))
|
|
|
|
return command_fail(cmd, INVOICE_OFFER_INACTIVE,
|
|
|
|
"offer not active");
|
|
|
|
local_offer_id = inv->offer_id;
|
|
|
|
} else
|
|
|
|
local_offer_id = NULL;
|
|
|
|
|
|
|
|
if (inv->amount)
|
|
|
|
msat = amount_msat(*inv->amount);
|
|
|
|
|
|
|
|
if (inv->relative_expiry)
|
|
|
|
expiry = *inv->relative_expiry;
|
|
|
|
else
|
2020-12-15 00:43:42 +01:00
|
|
|
expiry = BOLT12_DEFAULT_REL_EXPIRY;
|
|
|
|
|
|
|
|
if (!inv->payment_hash)
|
|
|
|
return command_fail(cmd, JSONRPC2_INVALID_PARAMS,
|
|
|
|
"Missing payment_hash in invoice");
|
|
|
|
if (!sha256_eq(&payment_hash, inv->payment_hash))
|
|
|
|
return command_fail(cmd, JSONRPC2_INVALID_PARAMS,
|
|
|
|
"Incorrect preimage");
|
2020-12-14 02:21:48 +01:00
|
|
|
|
|
|
|
if (!inv->description)
|
|
|
|
return command_fail(cmd, JSONRPC2_INVALID_PARAMS,
|
|
|
|
"Missing description in invoice");
|
|
|
|
desc = tal_strndup(cmd,
|
|
|
|
cast_signed(char *, inv->description),
|
|
|
|
tal_bytelen(inv->description));
|
|
|
|
|
|
|
|
if (!wallet_invoice_create(cmd->ld->wallet,
|
|
|
|
&invoice,
|
|
|
|
inv->amount ? &msat : NULL,
|
|
|
|
label,
|
|
|
|
expiry,
|
|
|
|
b12enc,
|
|
|
|
desc,
|
|
|
|
inv->features,
|
|
|
|
preimage,
|
|
|
|
&payment_hash,
|
|
|
|
local_offer_id))
|
|
|
|
return fail_exists(cmd, label);
|
|
|
|
|
|
|
|
notify_invoice_creation(cmd->ld,
|
|
|
|
inv->amount ? &msat : NULL,
|
|
|
|
*preimage, label);
|
|
|
|
} else
|
|
|
|
return command_fail(cmd, JSONRPC2_INVALID_PARAMS,
|
|
|
|
"Unparsable invoice '%s': %s",
|
|
|
|
invstring, fail);
|
|
|
|
}
|
|
|
|
|
|
|
|
response = json_stream_success(cmd);
|
|
|
|
json_add_invoice(response,
|
|
|
|
wallet_invoice_details(cmd, cmd->ld->wallet, invoice));
|
|
|
|
return command_success(cmd, response);
|
|
|
|
}
|
|
|
|
|
|
|
|
static const struct json_command createinvoice_command = {
|
|
|
|
"createinvoice",
|
|
|
|
"payment",
|
|
|
|
json_createinvoice,
|
|
|
|
"Lowlevel command to sign and create invoice {invstring}, resolved with {preimage}, using unique {label}."
|
|
|
|
};
|
|
|
|
|
2021-01-09 22:48:05 +01:00
|
|
|
AUTODATA(json_command, &createinvoice_command);
|