mirror of
https://github.com/ElementsProject/lightning.git
synced 2025-02-22 06:41:44 +01:00
Without knowing what method was called, we can't have useful general logging methods, so go through the pain of adding "const char *method" everywhere, and add: 1. ignore_and_complete - we're done when jsonrpc returned 2. log_broken_and_complete - we're done, but emit BROKEN log. 3. plugin_broken_cb - if this happens, fail the plugin. Signed-off-by: Rusty Russell <rusty@rustcorp.com.au>
1249 lines
39 KiB
C
1249 lines
39 KiB
C
#include "config.h"
|
|
#include <ccan/bitmap/bitmap.h>
|
|
#include <common/amount.h>
|
|
#include <common/bolt11.h>
|
|
#include <common/gossmods_listpeerchannels.h>
|
|
#include <common/json_stream.h>
|
|
#include <plugins/renepay/json.h>
|
|
#include <plugins/renepay/mcf.h>
|
|
#include <plugins/renepay/mods.h>
|
|
#include <plugins/renepay/payplugin.h>
|
|
#include <plugins/renepay/renepayconfig.h>
|
|
#include <plugins/renepay/route.h>
|
|
#include <plugins/renepay/routebuilder.h>
|
|
#include <plugins/renepay/routetracker.h>
|
|
#include <unistd.h>
|
|
|
|
#define INVALID_ID UINT32_MAX
|
|
|
|
#define OP_NULL NULL
|
|
#define OP_CALL (void *)1
|
|
#define OP_IF (void *)2
|
|
|
|
void *payment_virtual_program[];
|
|
|
|
/* Advance the payment virtual machine */
|
|
struct command_result *payment_continue(struct payment *payment)
|
|
{
|
|
assert(payment->exec_state != INVALID_STATE);
|
|
void *op = payment_virtual_program[payment->exec_state++];
|
|
|
|
if (op == OP_NULL) {
|
|
plugin_err(pay_plugin->plugin,
|
|
"payment_continue reached the end of the virtual "
|
|
"machine execution.");
|
|
} else if (op == OP_CALL) {
|
|
const struct payment_modifier *mod =
|
|
(const struct payment_modifier *)
|
|
payment_virtual_program[payment->exec_state++];
|
|
|
|
if (mod == NULL)
|
|
plugin_err(pay_plugin->plugin,
|
|
"payment_continue expected payment_modifier "
|
|
"but NULL found");
|
|
|
|
plugin_log(pay_plugin->plugin, LOG_DBG, "Calling modifier %s",
|
|
mod->name);
|
|
return mod->step_cb(payment);
|
|
} else if (op == OP_IF) {
|
|
const struct payment_condition *cond =
|
|
(const struct payment_condition *)
|
|
payment_virtual_program[payment->exec_state++];
|
|
|
|
if (cond == NULL)
|
|
plugin_err(pay_plugin->plugin,
|
|
"payment_continue expected pointer to "
|
|
"condition but NULL found");
|
|
|
|
plugin_log(pay_plugin->plugin, LOG_DBG,
|
|
"Calling payment condition %s", cond->name);
|
|
|
|
const u64 position_iftrue =
|
|
(intptr_t)payment_virtual_program[payment->exec_state++];
|
|
|
|
if (cond->condition_cb(payment))
|
|
payment->exec_state = position_iftrue;
|
|
|
|
return payment_continue(payment);
|
|
}
|
|
plugin_err(pay_plugin->plugin, "payment_continue op code not defined");
|
|
return NULL;
|
|
}
|
|
|
|
|
|
/* Generic handler for RPC failures that should end up failing the payment. */
|
|
static struct command_result *payment_rpc_failure(struct command *cmd,
|
|
const char *method UNUSED,
|
|
const char *buffer,
|
|
const jsmntok_t *toks,
|
|
struct payment *payment)
|
|
{
|
|
const jsmntok_t *codetok = json_get_member(buffer, toks, "code");
|
|
u32 errcode;
|
|
if (codetok != NULL)
|
|
json_to_u32(buffer, codetok, &errcode);
|
|
else
|
|
errcode = LIGHTNINGD;
|
|
|
|
return payment_fail(
|
|
payment, errcode,
|
|
"Failing a partial payment due to a failed RPC call: %.*s",
|
|
json_tok_full_len(toks), json_tok_full(buffer, toks));
|
|
}
|
|
|
|
/*****************************************************************************
|
|
* previoussuccess
|
|
*
|
|
* Obtain a list of previous sendpay requests and check if
|
|
* the current payment hash has already succeed.
|
|
*/
|
|
|
|
struct success_data {
|
|
u32 parts, created_at, groupid;
|
|
struct amount_msat deliver_msat, sent_msat;
|
|
struct preimage preimage;
|
|
};
|
|
|
|
/* Extracts success data from listsendpays. */
|
|
static bool success_data_from_listsendpays(const char *buf,
|
|
const jsmntok_t *arr,
|
|
struct success_data *success)
|
|
{
|
|
assert(success);
|
|
|
|
size_t i;
|
|
const char *err;
|
|
const jsmntok_t *t;
|
|
assert(arr && arr->type == JSMN_ARRAY);
|
|
|
|
success->parts = 0;
|
|
success->deliver_msat = AMOUNT_MSAT(0);
|
|
success->sent_msat = AMOUNT_MSAT(0);
|
|
|
|
json_for_each_arr(i, t, arr)
|
|
{
|
|
u32 groupid;
|
|
struct amount_msat this_msat, this_sent;
|
|
|
|
const jsmntok_t *status_tok = json_get_member(buf, t, "status");
|
|
if (!status_tok)
|
|
plugin_err(
|
|
pay_plugin->plugin,
|
|
"%s (line %d) missing status token from json.",
|
|
__func__, __LINE__);
|
|
const char *status = json_strdup(tmpctx, buf, status_tok);
|
|
if (!status)
|
|
plugin_err(
|
|
pay_plugin->plugin,
|
|
"%s (line %d) failed to allocate status string.",
|
|
__func__, __LINE__);
|
|
|
|
if (streq(status, "complete")) {
|
|
/* FIXME we assume amount_msat is always present, but
|
|
* according to the documentation this field is
|
|
* optional. How do I interpret if amount_msat is
|
|
* missing? */
|
|
err = json_scan(
|
|
tmpctx, buf, t,
|
|
"{groupid:%"
|
|
",amount_msat:%"
|
|
",amount_sent_msat:%"
|
|
",created_at:%"
|
|
",payment_preimage:%}",
|
|
JSON_SCAN(json_to_u32, &groupid),
|
|
JSON_SCAN(json_to_msat, &this_msat),
|
|
JSON_SCAN(json_to_msat, &this_sent),
|
|
JSON_SCAN(json_to_u32, &success->created_at),
|
|
JSON_SCAN(json_to_preimage, &success->preimage));
|
|
|
|
if (err)
|
|
plugin_err(pay_plugin->plugin,
|
|
"%s (line %d) json_scan of "
|
|
"listsendpay returns the "
|
|
"following error: %s",
|
|
__func__, __LINE__, err);
|
|
success->groupid = groupid;
|
|
/* Now we know the payment completed. */
|
|
if (!amount_msat_add(&success->deliver_msat,
|
|
success->deliver_msat,
|
|
this_msat) ||
|
|
!amount_msat_add(&success->sent_msat,
|
|
success->sent_msat, this_sent))
|
|
plugin_err(pay_plugin->plugin,
|
|
"%s (line %d) amount_msat overflow.",
|
|
__func__, __LINE__);
|
|
|
|
success->parts++;
|
|
}
|
|
}
|
|
|
|
return success->parts > 0;
|
|
}
|
|
|
|
static struct command_result *previoussuccess_done(struct command *cmd,
|
|
const char *method UNUSED,
|
|
const char *buf,
|
|
const jsmntok_t *result,
|
|
struct payment *payment)
|
|
{
|
|
const jsmntok_t *arr = json_get_member(buf, result, "payments");
|
|
if (!arr || arr->type != JSMN_ARRAY) {
|
|
return payment_fail(
|
|
payment, LIGHTNINGD,
|
|
"Unexpected non-array result from listsendpays: %.*s",
|
|
json_tok_full_len(result), json_tok_full(buf, result));
|
|
}
|
|
|
|
struct success_data success;
|
|
if (!success_data_from_listsendpays(buf, arr, &success)) {
|
|
/* There are no success sendpays. */
|
|
return payment_continue(payment);
|
|
}
|
|
|
|
payment->payment_info.start_time.ts.tv_sec = success.created_at;
|
|
payment->payment_info.start_time.ts.tv_nsec = 0;
|
|
payment->total_delivering = success.deliver_msat;
|
|
payment->total_sent = success.sent_msat;
|
|
payment->next_partid = success.parts + 1;
|
|
payment->groupid = success.groupid;
|
|
|
|
payment_note(payment, LOG_DBG,
|
|
"Payment completed by a previous sendpay.");
|
|
return payment_success(payment, &success.preimage);
|
|
}
|
|
|
|
static struct command_result *previoussuccess_cb(struct payment *payment)
|
|
{
|
|
struct command *cmd = payment_command(payment);
|
|
assert(cmd);
|
|
|
|
struct out_req *req = jsonrpc_request_start(
|
|
cmd, "listsendpays", previoussuccess_done,
|
|
payment_rpc_failure, payment);
|
|
|
|
json_add_sha256(req->js, "payment_hash",
|
|
&payment->payment_info.payment_hash);
|
|
json_add_string(req->js, "status", "complete");
|
|
return send_outreq(req);
|
|
}
|
|
|
|
REGISTER_PAYMENT_MODIFIER(previoussuccess, previoussuccess_cb);
|
|
|
|
/*****************************************************************************
|
|
* initial_sanity_checks
|
|
*
|
|
* Some checks on a payment about to start.
|
|
*/
|
|
static struct command_result *initial_sanity_checks_cb(struct payment *payment)
|
|
{
|
|
assert(amount_msat_is_zero(payment->total_sent));
|
|
assert(amount_msat_is_zero(payment->total_delivering));
|
|
assert(!payment->preimage);
|
|
assert(tal_count(payment->cmd_array) == 1);
|
|
|
|
return payment_continue(payment);
|
|
}
|
|
|
|
REGISTER_PAYMENT_MODIFIER(initial_sanity_checks, initial_sanity_checks_cb);
|
|
|
|
/*****************************************************************************
|
|
* selfpay
|
|
*
|
|
* Checks if the payment destination is the sender's node and perform a self
|
|
* payment.
|
|
*/
|
|
|
|
static struct command_result *selfpay_success(struct command *cmd,
|
|
const char *method UNUSED,
|
|
const char *buf,
|
|
const jsmntok_t *tok,
|
|
struct route *route)
|
|
{
|
|
tal_steal(tmpctx, route); // discard this route when tmpctx clears
|
|
struct payment *payment =
|
|
payment_map_get(pay_plugin->payment_map, route->key.payment_hash);
|
|
assert(payment);
|
|
|
|
struct preimage preimage;
|
|
const char *err;
|
|
err = json_scan(tmpctx, buf, tok, "{payment_preimage:%}",
|
|
JSON_SCAN(json_to_preimage, &preimage));
|
|
if (err)
|
|
plugin_err(
|
|
cmd->plugin, "selfpay didn't have payment_preimage: %.*s",
|
|
json_tok_full_len(tok), json_tok_full(buf, tok));
|
|
|
|
|
|
payment_note(payment, LOG_DBG, "Paid with self-pay.");
|
|
return payment_success(payment, &preimage);
|
|
}
|
|
static struct command_result *selfpay_failure(struct command *cmd,
|
|
const char *method UNUSED,
|
|
const char *buf,
|
|
const jsmntok_t *tok,
|
|
struct route *route)
|
|
{
|
|
tal_steal(tmpctx, route); // discard this route when tmpctx clears
|
|
struct payment *payment =
|
|
payment_map_get(pay_plugin->payment_map, route->key.payment_hash);
|
|
assert(payment);
|
|
struct payment_result *result = tal_sendpay_result_from_json(tmpctx, buf, tok);
|
|
if (result == NULL)
|
|
plugin_err(pay_plugin->plugin,
|
|
"Unable to parse sendpay failure: %.*s",
|
|
json_tok_full_len(tok), json_tok_full(buf, tok));
|
|
|
|
return payment_fail(payment, result->code, "%s", result->message);
|
|
}
|
|
|
|
static struct command_result *selfpay_cb(struct payment *payment)
|
|
{
|
|
if (!node_id_eq(&pay_plugin->my_id,
|
|
&payment->payment_info.destination)) {
|
|
return payment_continue(payment);
|
|
}
|
|
|
|
struct command *cmd = payment_command(payment);
|
|
if (!cmd)
|
|
plugin_err(pay_plugin->plugin,
|
|
"Selfpay: cannot get a valid cmd.");
|
|
|
|
struct payment_info *pinfo = &payment->payment_info;
|
|
/* Self-payment routes are not part of the routetracker, we build them
|
|
* on-the-fly here and release them on success or failure. */
|
|
struct route *route =
|
|
new_route(payment, payment->groupid,
|
|
/*partid=*/0, pinfo->payment_hash,
|
|
pinfo->amount, pinfo->amount);
|
|
struct out_req *req;
|
|
req = jsonrpc_request_start(cmd, "sendpay",
|
|
selfpay_success, selfpay_failure, route);
|
|
route->hops = tal_arr(route, struct route_hop, 0);
|
|
json_add_route(req->js, route, payment);
|
|
return send_outreq(req);
|
|
}
|
|
|
|
REGISTER_PAYMENT_MODIFIER(selfpay, selfpay_cb);
|
|
|
|
/*****************************************************************************
|
|
* getmychannels
|
|
*
|
|
* Calls listpeerchannels to get and updated state of the local channels.
|
|
*/
|
|
|
|
static void
|
|
uncertainty_update_from_listpeerchannels(struct uncertainty *uncertainty,
|
|
const struct short_channel_id_dir *scidd,
|
|
struct amount_msat max, bool enabled,
|
|
const char *buf, const jsmntok_t *chantok)
|
|
{
|
|
if (!enabled)
|
|
return;
|
|
|
|
struct amount_msat capacity, min, gap;
|
|
const char *errmsg = json_scan(tmpctx, buf, chantok, "{total_msat:%}",
|
|
JSON_SCAN(json_to_msat, &capacity));
|
|
if (errmsg)
|
|
goto error;
|
|
|
|
if (!uncertainty_add_channel(pay_plugin->uncertainty, scidd->scid,
|
|
capacity)) {
|
|
errmsg = tal_fmt(
|
|
tmpctx,
|
|
"Unable to find/add scid=%s in the uncertainty network",
|
|
fmt_short_channel_id(tmpctx, scidd->scid));
|
|
goto error;
|
|
}
|
|
|
|
if (!amount_msat_scale(&gap, capacity, 0.1) ||
|
|
!amount_msat_sub(&min, max, gap))
|
|
min = AMOUNT_MSAT(0);
|
|
|
|
// FIXME this does not include pending HTLC of ongoing payments!
|
|
/* Allow a gap between min and max so that we don't use up all of our
|
|
* channels' spendable sats and avoid our local error:
|
|
* WIRE_TEMPORARY_CHANNEL_FAILURE: Capacity exceeded - HTLC fee: Xsat
|
|
*
|
|
* */
|
|
if (!uncertainty_set_liquidity(pay_plugin->uncertainty, scidd, min,
|
|
max)) {
|
|
errmsg = tal_fmt(
|
|
tmpctx,
|
|
"Unable to set liquidity to channel scidd=%s in the "
|
|
"uncertainty network.",
|
|
fmt_short_channel_id_dir(tmpctx, scidd));
|
|
goto error;
|
|
}
|
|
return;
|
|
|
|
error:
|
|
plugin_log(
|
|
pay_plugin->plugin, LOG_UNUSUAL,
|
|
"Failed to update local channel %s from listpeerchannels rpc: %s",
|
|
fmt_short_channel_id(tmpctx, scidd->scid),
|
|
errmsg);
|
|
}
|
|
|
|
static void gossmod_cb(struct gossmap_localmods *mods,
|
|
const struct node_id *self,
|
|
const struct node_id *peer,
|
|
const struct short_channel_id_dir *scidd,
|
|
struct amount_msat capacity_msat,
|
|
struct amount_msat htlcmin,
|
|
struct amount_msat htlcmax,
|
|
struct amount_msat spendable,
|
|
struct amount_msat fee_base,
|
|
u32 fee_proportional,
|
|
u16 cltv_delta,
|
|
bool enabled,
|
|
const char *buf,
|
|
const jsmntok_t *chantok,
|
|
struct payment *payment)
|
|
{
|
|
struct amount_msat min, max;
|
|
|
|
if (scidd->dir == node_id_idx(self, peer)) {
|
|
/* local channels can send up to what's spendable */
|
|
min = AMOUNT_MSAT(0);
|
|
max = spendable;
|
|
} else {
|
|
/* remote channels can send up no more than spendable */
|
|
min = htlcmin;
|
|
max = amount_msat_min(spendable, htlcmax);
|
|
}
|
|
|
|
/* FIXME: features? */
|
|
gossmap_local_addchan(mods, self, peer, scidd->scid, capacity_msat,
|
|
NULL);
|
|
gossmap_local_updatechan(mods, scidd,
|
|
&enabled,
|
|
&min, &max,
|
|
&fee_base, &fee_proportional, &cltv_delta);
|
|
|
|
/* Is it disabled? */
|
|
if (!enabled)
|
|
payment_disable_chan(payment, *scidd, LOG_DBG,
|
|
"listpeerchannels says not enabled");
|
|
|
|
/* Also update the uncertainty network by fixing the liquidity of the
|
|
* outgoing channel. If we try to set the liquidity of the incoming
|
|
* channel as well we would have conflicting information because our
|
|
* knowledge model does not take into account channel reserves. */
|
|
if (scidd->dir == node_id_idx(self, peer))
|
|
uncertainty_update_from_listpeerchannels(
|
|
pay_plugin->uncertainty, scidd, max, enabled, buf, chantok);
|
|
}
|
|
|
|
static struct command_result *getmychannels_done(struct command *cmd,
|
|
const char *method UNUSED,
|
|
const char *buf,
|
|
const jsmntok_t *result,
|
|
struct payment *payment)
|
|
{
|
|
// FIXME: should local gossmods be global (ie. member of pay_plugin) or
|
|
// local (ie. member of payment)?
|
|
payment->local_gossmods = gossmods_from_listpeerchannels(
|
|
payment, &pay_plugin->my_id, buf, result, /* zero_rates = */ true,
|
|
gossmod_cb, payment);
|
|
|
|
return payment_continue(payment);
|
|
}
|
|
|
|
static struct command_result *getmychannels_cb(struct payment *payment)
|
|
{
|
|
struct command *cmd = payment_command(payment);
|
|
if (!cmd)
|
|
plugin_err(pay_plugin->plugin,
|
|
"getmychannels_pay_mod: cannot get a valid cmd.");
|
|
|
|
struct out_req *req = jsonrpc_request_start(
|
|
cmd, "listpeerchannels", getmychannels_done,
|
|
payment_rpc_failure, payment);
|
|
return send_outreq(req);
|
|
}
|
|
|
|
REGISTER_PAYMENT_MODIFIER(getmychannels, getmychannels_cb);
|
|
|
|
/*****************************************************************************
|
|
* refreshgossmap
|
|
*
|
|
* Update the gossmap.
|
|
*/
|
|
|
|
static struct command_result *refreshgossmap_cb(struct payment *payment)
|
|
{
|
|
assert(pay_plugin->gossmap); // gossmap must be already initialized
|
|
assert(payment);
|
|
assert(payment->local_gossmods);
|
|
|
|
size_t num_channel_updates_rejected = 0;
|
|
bool gossmap_changed =
|
|
gossmap_refresh(pay_plugin->gossmap, &num_channel_updates_rejected);
|
|
|
|
if (gossmap_changed && num_channel_updates_rejected)
|
|
plugin_log(pay_plugin->plugin, LOG_DBG,
|
|
"gossmap ignored %zu channel updates",
|
|
num_channel_updates_rejected);
|
|
|
|
if (gossmap_changed) {
|
|
gossmap_apply_localmods(pay_plugin->gossmap,
|
|
payment->local_gossmods);
|
|
int skipped_count = uncertainty_update(pay_plugin->uncertainty,
|
|
pay_plugin->gossmap);
|
|
gossmap_remove_localmods(pay_plugin->gossmap,
|
|
payment->local_gossmods);
|
|
if (skipped_count)
|
|
plugin_log(
|
|
pay_plugin->plugin, LOG_UNUSUAL,
|
|
"%s: uncertainty was updated but %d channels have "
|
|
"been ignored.",
|
|
__func__, skipped_count);
|
|
}
|
|
return payment_continue(payment);
|
|
}
|
|
|
|
REGISTER_PAYMENT_MODIFIER(refreshgossmap, refreshgossmap_cb);
|
|
|
|
/*****************************************************************************
|
|
* routehints
|
|
*
|
|
* Use route hints from the invoice to update the local gossmods and uncertainty
|
|
* network.
|
|
*/
|
|
// TODO check how this is done in pay.c
|
|
|
|
static void add_hintchan(struct payment *payment, const struct node_id *src,
|
|
const struct node_id *dst, u16 cltv_expiry_delta,
|
|
const struct short_channel_id scid, u32 fee_base_msat,
|
|
u32 fee_proportional_millionths)
|
|
{
|
|
assert(payment);
|
|
assert(payment->local_gossmods);
|
|
|
|
const char *errmsg;
|
|
const struct chan_extra *ce =
|
|
uncertainty_find_channel(pay_plugin->uncertainty, scid);
|
|
|
|
if (!ce) {
|
|
struct short_channel_id_dir scidd;
|
|
/* We assume any HTLC is allowed */
|
|
struct amount_msat htlc_min = AMOUNT_MSAT(0), htlc_max = MAX_CAPACITY;
|
|
struct amount_msat fee_base = amount_msat(fee_base_msat);
|
|
bool enabled = true;
|
|
scidd.scid = scid;
|
|
scidd.dir = node_id_idx(src, dst);
|
|
|
|
/* This channel is not public, we don't know his capacity
|
|
One possible solution is set the capacity to
|
|
MAX_CAP and the state to [0,MAX_CAP]. Alternatively we could
|
|
the capacity to amount and state to [amount,amount], but that
|
|
wouldn't work if the recepient provides more than one hints
|
|
telling us to partition the payment in multiple routes. */
|
|
ce = uncertainty_add_channel(pay_plugin->uncertainty, scid,
|
|
MAX_CAPACITY);
|
|
if (!ce) {
|
|
errmsg = tal_fmt(tmpctx,
|
|
"Unable to find/add scid=%s in the "
|
|
"local uncertainty network",
|
|
fmt_short_channel_id(tmpctx, scid));
|
|
goto function_error;
|
|
}
|
|
/* FIXME: features? */
|
|
if (!gossmap_local_addchan(payment->local_gossmods, src, dst,
|
|
scid, MAX_CAPACITY, NULL) ||
|
|
!gossmap_local_updatechan(
|
|
payment->local_gossmods, &scidd,
|
|
&enabled, &htlc_min, &htlc_max,
|
|
&fee_base, &fee_proportional_millionths,
|
|
&cltv_expiry_delta)) {
|
|
errmsg = tal_fmt(
|
|
tmpctx,
|
|
"Failed to update scid=%s in the local_gossmods.",
|
|
fmt_short_channel_id(tmpctx, scid));
|
|
goto function_error;
|
|
}
|
|
} else {
|
|
/* The channel is pubic and we already keep track of it in the
|
|
* gossmap and uncertainty network. It would be wrong to assume
|
|
* that this channel has sufficient capacity to forward the
|
|
* entire payment! Doing so leads to knowledge updates in which
|
|
* the known min liquidity is greater than the channel's
|
|
* capacity. */
|
|
}
|
|
|
|
return;
|
|
|
|
function_error:
|
|
plugin_log(pay_plugin->plugin, LOG_UNUSUAL,
|
|
"Failed to update hint channel %s: %s",
|
|
fmt_short_channel_id(tmpctx, scid),
|
|
errmsg);
|
|
}
|
|
|
|
static struct command_result *routehints_done(struct command *cmd UNUSED,
|
|
const char *method UNUSED,
|
|
const char *buf UNUSED,
|
|
const jsmntok_t *result UNUSED,
|
|
struct payment *payment)
|
|
{
|
|
// FIXME are there route hints for B12?
|
|
assert(payment);
|
|
assert(payment->local_gossmods);
|
|
|
|
const struct node_id *destination = &payment->payment_info.destination;
|
|
const struct route_info **routehints = payment->payment_info.routehints;
|
|
assert(routehints);
|
|
const size_t nhints = tal_count(routehints);
|
|
/* Hints are added to the local_gossmods. */
|
|
for (size_t i = 0; i < nhints; i++) {
|
|
/* Each one, presumably, leads to the destination */
|
|
const struct route_info *r = routehints[i];
|
|
const struct node_id *end = destination;
|
|
|
|
for (int j = tal_count(r) - 1; j >= 0; j--) {
|
|
add_hintchan(payment, &r[j].pubkey, end,
|
|
r[j].cltv_expiry_delta,
|
|
r[j].short_channel_id, r[j].fee_base_msat,
|
|
r[j].fee_proportional_millionths);
|
|
end = &r[j].pubkey;
|
|
}
|
|
}
|
|
|
|
/* Add hints to the uncertainty network. */
|
|
gossmap_apply_localmods(pay_plugin->gossmap, payment->local_gossmods);
|
|
int skipped_count =
|
|
uncertainty_update(pay_plugin->uncertainty, pay_plugin->gossmap);
|
|
gossmap_remove_localmods(pay_plugin->gossmap, payment->local_gossmods);
|
|
if (skipped_count)
|
|
plugin_log(pay_plugin->plugin, LOG_UNUSUAL,
|
|
"%s: uncertainty was updated but %d channels have "
|
|
"been ignored.",
|
|
__func__, skipped_count);
|
|
|
|
return payment_continue(payment);
|
|
}
|
|
|
|
static struct command_result *routehints_cb(struct payment *payment)
|
|
{
|
|
struct command *cmd = payment_command(payment);
|
|
assert(cmd);
|
|
struct out_req *req = jsonrpc_request_start(
|
|
cmd, "waitblockheight", routehints_done,
|
|
payment_rpc_failure, payment);
|
|
json_add_num(req->js, "blockheight", 0);
|
|
return send_outreq(req);
|
|
}
|
|
|
|
REGISTER_PAYMENT_MODIFIER(routehints, routehints_cb);
|
|
|
|
/*****************************************************************************
|
|
* compute_routes
|
|
*
|
|
* Compute the payment routes.
|
|
*/
|
|
|
|
static struct command_result *compute_routes_cb(struct payment *payment)
|
|
{
|
|
assert(payment->status == PAYMENT_PENDING);
|
|
struct routetracker *routetracker = payment->routetracker;
|
|
assert(routetracker);
|
|
|
|
if (routetracker->computed_routes &&
|
|
tal_count(routetracker->computed_routes))
|
|
plugin_err(pay_plugin->plugin,
|
|
"%s: no previously computed routes expected.",
|
|
__func__);
|
|
|
|
struct amount_msat feebudget, fees_spent, remaining;
|
|
|
|
/* Total feebudget */
|
|
if (!amount_msat_sub(&feebudget, payment->payment_info.maxspend,
|
|
payment->payment_info.amount))
|
|
plugin_err(pay_plugin->plugin, "%s: fee budget is negative?",
|
|
__func__);
|
|
|
|
/* Fees spent so far */
|
|
if (!amount_msat_sub(&fees_spent, payment->total_sent,
|
|
payment->total_delivering))
|
|
plugin_err(pay_plugin->plugin,
|
|
"%s: total_delivering is greater than total_sent?",
|
|
__func__);
|
|
|
|
/* Remaining fee budget. */
|
|
if (!amount_msat_sub(&feebudget, feebudget, fees_spent))
|
|
feebudget = AMOUNT_MSAT(0);
|
|
|
|
/* How much are we still trying to send? */
|
|
if (!amount_msat_sub(&remaining, payment->payment_info.amount,
|
|
payment->total_delivering) ||
|
|
amount_msat_is_zero(remaining)) {
|
|
plugin_log(pay_plugin->plugin, LOG_UNUSUAL,
|
|
"%s: Payment is pending with full amount already "
|
|
"committed. We skip the computation of new routes.",
|
|
__func__);
|
|
return payment_continue(payment);
|
|
}
|
|
|
|
enum jsonrpc_errcode errcode;
|
|
const char *err_msg = NULL;
|
|
|
|
gossmap_apply_localmods(pay_plugin->gossmap, payment->local_gossmods);
|
|
|
|
/* get_routes returns the answer, we assign it to the computed_routes,
|
|
* that's why we need to tal_free the older array. Maybe it would be
|
|
* better to pass computed_routes as a reference? */
|
|
routetracker->computed_routes = tal_free(routetracker->computed_routes);
|
|
|
|
// TODO: add an algorithm selector here
|
|
/* We let this return an unlikely path, as it's better to try once than
|
|
* simply refuse. Plus, models are not truth! */
|
|
routetracker->computed_routes = get_routes(
|
|
routetracker,
|
|
&payment->payment_info,
|
|
&pay_plugin->my_id,
|
|
&payment->payment_info.destination,
|
|
pay_plugin->gossmap,
|
|
pay_plugin->uncertainty,
|
|
payment->disabledmap,
|
|
remaining,
|
|
feebudget,
|
|
&payment->next_partid,
|
|
payment->groupid,
|
|
&errcode,
|
|
&err_msg);
|
|
/* Otherwise the error message remains a child of the routetracker. */
|
|
err_msg = tal_steal(tmpctx, err_msg);
|
|
|
|
gossmap_remove_localmods(pay_plugin->gossmap, payment->local_gossmods);
|
|
|
|
/* Couldn't feasible route, we stop. */
|
|
if (!routetracker->computed_routes ||
|
|
tal_count(routetracker->computed_routes) == 0) {
|
|
if (err_msg == NULL)
|
|
err_msg = tal_fmt(
|
|
tmpctx, "get_routes returned NULL error message");
|
|
return payment_fail(payment, errcode, "%s", err_msg);
|
|
}
|
|
|
|
return payment_continue(payment);
|
|
}
|
|
|
|
REGISTER_PAYMENT_MODIFIER(compute_routes, compute_routes_cb);
|
|
|
|
/*****************************************************************************
|
|
* send_routes
|
|
*
|
|
* This payment modifier takes the payment routes and starts the payment
|
|
* request calling sendpay.
|
|
*/
|
|
|
|
static struct command_result *send_routes_cb(struct payment *payment)
|
|
{
|
|
assert(payment);
|
|
struct routetracker *routetracker = payment->routetracker;
|
|
assert(routetracker);
|
|
if (!routetracker->computed_routes ||
|
|
tal_count(routetracker->computed_routes) == 0) {
|
|
plugin_log(pay_plugin->plugin, LOG_UNUSUAL,
|
|
"%s: there are no routes to send, skipping.",
|
|
__func__);
|
|
return payment_continue(payment);
|
|
}
|
|
struct command *cmd = payment_command(payment);
|
|
assert(cmd);
|
|
for (size_t i = 0; i < tal_count(routetracker->computed_routes); i++) {
|
|
struct route *route = routetracker->computed_routes[i];
|
|
|
|
route_sendpay_request(cmd, take(route), payment);
|
|
|
|
payment_note(payment, LOG_INFORM,
|
|
"Sent route request: partid=%" PRIu64
|
|
" amount=%s prob=%.3lf fees=%s delay=%u path=%s",
|
|
route->key.partid,
|
|
fmt_amount_msat(tmpctx, route_delivers(route)),
|
|
route->success_prob,
|
|
fmt_amount_msat(tmpctx, route_fees(route)),
|
|
route_delay(route), fmt_route_path(tmpctx, route));
|
|
}
|
|
tal_resize(&routetracker->computed_routes, 0);
|
|
return payment_continue(payment);
|
|
}
|
|
|
|
REGISTER_PAYMENT_MODIFIER(send_routes, send_routes_cb);
|
|
|
|
/*****************************************************************************
|
|
* sleep
|
|
*
|
|
* The payment main thread sleeps for some time.
|
|
*/
|
|
|
|
static struct command_result *sleep_done(struct command *cmd, struct payment *payment)
|
|
{
|
|
struct command_result *ret;
|
|
payment->waitresult_timer = NULL;
|
|
ret = timer_complete(cmd);
|
|
payment_continue(payment);
|
|
return ret;
|
|
}
|
|
|
|
static struct command_result *sleep_cb(struct payment *payment)
|
|
{
|
|
struct command *cmd = payment_command(payment);
|
|
assert(cmd);
|
|
assert(payment->waitresult_timer == NULL);
|
|
payment->waitresult_timer
|
|
= command_timer(cmd,
|
|
time_from_msec(COLLECTOR_TIME_WINDOW_MSEC),
|
|
sleep_done, payment);
|
|
return command_still_pending(cmd);
|
|
}
|
|
|
|
REGISTER_PAYMENT_MODIFIER(sleep, sleep_cb);
|
|
|
|
/*****************************************************************************
|
|
* collect_results
|
|
*/
|
|
|
|
static struct command_result *collect_results_cb(struct payment *payment)
|
|
{
|
|
assert(payment);
|
|
payment->have_results = false;
|
|
payment->retry = false;
|
|
|
|
/* pending sendpay callbacks should be zero */
|
|
if (!routetracker_have_results(payment->routetracker))
|
|
return payment_continue(payment);
|
|
|
|
/* all sendpays have been sent, look for success */
|
|
struct preimage *payment_preimage = NULL;
|
|
enum jsonrpc_errcode final_error = LIGHTNINGD;
|
|
const char *final_msg = NULL;
|
|
|
|
payment_collect_results(payment, &payment_preimage, &final_error, &final_msg);
|
|
|
|
if (payment_preimage) {
|
|
/* If we have the preimage that means one succeed, we
|
|
* inmediately finish the payment. */
|
|
if (!amount_msat_greater_eq(payment->total_delivering,
|
|
payment->payment_info.amount)) {
|
|
plugin_log(
|
|
pay_plugin->plugin, LOG_UNUSUAL,
|
|
"%s: received a success sendpay for this "
|
|
"payment but the total delivering amount %s "
|
|
"is less than the payment amount %s.",
|
|
__func__,
|
|
fmt_amount_msat(tmpctx, payment->total_delivering),
|
|
fmt_amount_msat(tmpctx,
|
|
payment->payment_info.amount));
|
|
}
|
|
return payment_success(payment, take(payment_preimage));
|
|
}
|
|
if (final_msg) {
|
|
/* We received a sendpay result with a final error message, we
|
|
* inmediately finish the payment. */
|
|
return payment_fail(payment, final_error, "%s", final_msg);
|
|
}
|
|
|
|
if (amount_msat_greater_eq(payment->total_delivering,
|
|
payment->payment_info.amount)) {
|
|
/* There are no succeeds but we are still pending delivering the
|
|
* entire payment. We still need to collect more results. */
|
|
payment->have_results = false;
|
|
payment->retry = false;
|
|
} else {
|
|
/* We have some failures so that now we are short of
|
|
* total_delivering, we may retry. */
|
|
payment->have_results = true;
|
|
|
|
// FIXME: we seem to always retry here if we don't fail
|
|
// inmediately. But I am going to leave this variable here,
|
|
// cause we might decide in the future to put some conditions on
|
|
// retries, like a maximum number of retries.
|
|
payment->retry = true;
|
|
}
|
|
|
|
return payment_continue(payment);
|
|
}
|
|
|
|
REGISTER_PAYMENT_MODIFIER(collect_results, collect_results_cb);
|
|
|
|
/*****************************************************************************
|
|
* end
|
|
*
|
|
* The default ending of a payment.
|
|
*/
|
|
static struct command_result *end_done(struct command *cmd UNUSED,
|
|
const char *method UNUSED,
|
|
const char *buf UNUSED,
|
|
const jsmntok_t *result UNUSED,
|
|
struct payment *payment)
|
|
{
|
|
return payment_fail(payment, PAY_STOPPED_RETRYING,
|
|
"Payment execution ended without success.");
|
|
}
|
|
static struct command_result *end_cb(struct payment *payment)
|
|
{
|
|
struct command *cmd = payment_command(payment);
|
|
assert(cmd);
|
|
struct out_req *req =
|
|
jsonrpc_request_start(cmd, "waitblockheight", end_done,
|
|
payment_rpc_failure, payment);
|
|
json_add_num(req->js, "blockheight", 0);
|
|
return send_outreq(req);
|
|
}
|
|
|
|
REGISTER_PAYMENT_MODIFIER(end, end_cb);
|
|
|
|
/*****************************************************************************
|
|
* checktimeout
|
|
*
|
|
* Fail the payment if we have exceeded the timeout.
|
|
*/
|
|
|
|
static struct command_result *checktimeout_cb(struct payment *payment)
|
|
{
|
|
if (time_after(time_now(), payment->payment_info.stop_time)) {
|
|
return payment_fail(payment, PAY_STOPPED_RETRYING, "Timed out");
|
|
}
|
|
return payment_continue(payment);
|
|
}
|
|
|
|
REGISTER_PAYMENT_MODIFIER(checktimeout, checktimeout_cb);
|
|
|
|
/*****************************************************************************
|
|
* pendingsendpays
|
|
*
|
|
* Obtain a list of sendpays, add up the amount of those pending and decide
|
|
* which groupid and partid we should use next. If there is a "complete" sendpay
|
|
* we should return payment_success inmediately.
|
|
*/
|
|
|
|
static struct command_result *pendingsendpays_done(struct command *cmd,
|
|
const char *method UNUSED,
|
|
const char *buf,
|
|
const jsmntok_t *result,
|
|
struct payment *payment)
|
|
{
|
|
size_t i;
|
|
const char *err;
|
|
const jsmntok_t *t, *arr;
|
|
u32 max_group_id = 0;
|
|
|
|
/* Data for pending payments, this will be the one
|
|
* who's result gets replayed if we end up suspending. */
|
|
u32 pending_group_id = INVALID_ID;
|
|
u32 max_pending_partid = 0;
|
|
struct amount_msat pending_sent = AMOUNT_MSAT(0),
|
|
pending_msat = AMOUNT_MSAT(0);
|
|
|
|
arr = json_get_member(buf, result, "payments");
|
|
if (!arr || arr->type != JSMN_ARRAY) {
|
|
return payment_fail(
|
|
payment, LIGHTNINGD,
|
|
"Unexpected non-array result from listsendpays: %.*s",
|
|
json_tok_full_len(result), json_tok_full(buf, result));
|
|
}
|
|
|
|
struct success_data success;
|
|
if (success_data_from_listsendpays(buf, arr, &success)) {
|
|
/* Have success data, hence the payment is complete, we stop. */
|
|
payment->payment_info.start_time.ts.tv_sec = success.created_at;
|
|
payment->payment_info.start_time.ts.tv_nsec = 0;
|
|
payment->total_delivering = success.deliver_msat;
|
|
payment->total_sent = success.sent_msat;
|
|
payment->next_partid = success.parts + 1;
|
|
payment->groupid = success.groupid;
|
|
|
|
payment_note(payment, LOG_DBG,
|
|
"%s: Payment completed before computing the next "
|
|
"round of routes.",
|
|
__func__);
|
|
return payment_success(payment, &success.preimage);
|
|
}
|
|
|
|
// find if there is one pending group
|
|
json_for_each_arr(i, t, arr)
|
|
{
|
|
u32 groupid;
|
|
const char *status;
|
|
|
|
err = json_scan(tmpctx, buf, t,
|
|
"{status:%"
|
|
",groupid:%}",
|
|
JSON_SCAN_TAL(tmpctx, json_strdup, &status),
|
|
JSON_SCAN(json_to_u32, &groupid));
|
|
|
|
if (err)
|
|
plugin_err(pay_plugin->plugin,
|
|
"%s json_scan of listsendpay returns the "
|
|
"following error: %s",
|
|
__func__, err);
|
|
|
|
if (streq(status, "pending")) {
|
|
pending_group_id = groupid;
|
|
break;
|
|
}
|
|
}
|
|
|
|
/* We need two loops to get the highest partid for a groupid that has
|
|
* pending sendpays. */
|
|
json_for_each_arr(i, t, arr)
|
|
{
|
|
u32 partid = 0, groupid;
|
|
struct amount_msat this_msat, this_sent;
|
|
const char *status;
|
|
|
|
// FIXME we assume amount_msat is always present, but according
|
|
// to the documentation this field is optional. How do I
|
|
// interpret if amount_msat is missing?
|
|
err = json_scan(tmpctx, buf, t,
|
|
"{status:%"
|
|
",partid?:%"
|
|
",groupid:%"
|
|
",amount_msat:%"
|
|
",amount_sent_msat:%}",
|
|
JSON_SCAN_TAL(tmpctx, json_strdup, &status),
|
|
JSON_SCAN(json_to_u32, &partid),
|
|
JSON_SCAN(json_to_u32, &groupid),
|
|
JSON_SCAN(json_to_msat, &this_msat),
|
|
JSON_SCAN(json_to_msat, &this_sent));
|
|
|
|
if (err)
|
|
plugin_err(pay_plugin->plugin,
|
|
"%s json_scan of listsendpay returns the "
|
|
"following error: %s",
|
|
__func__, err);
|
|
|
|
/* If we decide to create a new group, we base it on
|
|
* max_group_id */
|
|
if (groupid > max_group_id)
|
|
max_group_id = groupid;
|
|
|
|
if (groupid == pending_group_id && partid > max_pending_partid)
|
|
max_pending_partid = partid;
|
|
|
|
/* status could be completed, pending or failed */
|
|
if (streq(status, "pending")) {
|
|
/* If we have more than one pending group, something
|
|
* went wrong! */
|
|
if (groupid != pending_group_id)
|
|
return payment_fail(
|
|
payment, PAY_STATUS_UNEXPECTED,
|
|
"Multiple pending groups for this "
|
|
"payment.");
|
|
|
|
if (!amount_msat_add(&pending_msat, pending_msat,
|
|
this_msat) ||
|
|
!amount_msat_add(&pending_sent, pending_sent,
|
|
this_sent))
|
|
plugin_err(pay_plugin->plugin,
|
|
"%s (line %d) amount_msat overflow.",
|
|
__func__, __LINE__);
|
|
}
|
|
assert(!streq(status, "complete"));
|
|
}
|
|
|
|
if (pending_group_id != INVALID_ID) {
|
|
/* Continue where we left off? */
|
|
payment->groupid = pending_group_id;
|
|
payment->next_partid = max_pending_partid + 1;
|
|
payment->total_sent = pending_sent;
|
|
payment->total_delivering = pending_msat;
|
|
|
|
plugin_log(pay_plugin->plugin, LOG_DBG,
|
|
"There are pending sendpays to this invoice. "
|
|
"groupid = %" PRIu32 " "
|
|
"delivering = %s, "
|
|
"last_partid = %" PRIu32,
|
|
pending_group_id,
|
|
fmt_amount_msat(tmpctx, payment->total_delivering),
|
|
max_pending_partid);
|
|
} else {
|
|
/* There are no pending nor completed sendpays, get me the last
|
|
* sendpay group. */
|
|
payment->groupid = max_group_id + 1;
|
|
payment->next_partid = 1;
|
|
payment->total_sent = AMOUNT_MSAT(0);
|
|
payment->total_delivering = AMOUNT_MSAT(0);
|
|
}
|
|
|
|
return payment_continue(payment);
|
|
}
|
|
|
|
static struct command_result *pendingsendpays_cb(struct payment *payment)
|
|
{
|
|
struct command *cmd = payment_command(payment);
|
|
assert(cmd);
|
|
|
|
struct out_req *req = jsonrpc_request_start(
|
|
cmd, "listsendpays", pendingsendpays_done,
|
|
payment_rpc_failure, payment);
|
|
|
|
json_add_sha256(req->js, "payment_hash",
|
|
&payment->payment_info.payment_hash);
|
|
return send_outreq(req);
|
|
}
|
|
|
|
REGISTER_PAYMENT_MODIFIER(pendingsendpays, pendingsendpays_cb);
|
|
|
|
/*****************************************************************************
|
|
* knowledgerelax
|
|
*
|
|
* Reduce the knowledge of the network as time goes by.
|
|
*/
|
|
|
|
static struct command_result *knowledgerelax_cb(struct payment *payment)
|
|
{
|
|
const u64 now_sec = time_now().ts.tv_sec;
|
|
enum renepay_errorcode err = uncertainty_relax(
|
|
pay_plugin->uncertainty, now_sec - pay_plugin->last_time);
|
|
if (err)
|
|
plugin_err(pay_plugin->plugin,
|
|
"uncertainty_relax failed with error %s",
|
|
renepay_errorcode_name(err));
|
|
pay_plugin->last_time = now_sec;
|
|
return payment_continue(payment);
|
|
}
|
|
|
|
REGISTER_PAYMENT_MODIFIER(knowledgerelax, knowledgerelax_cb);
|
|
|
|
/*****************************************************************************
|
|
* channelfilter
|
|
*
|
|
* Disable some channels. The possible motivations are:
|
|
* - avoid the overhead of unproductive routes that go through channels with
|
|
* very low max_htlc that would lead us to a payment partition with too
|
|
* many HTCLs,
|
|
* - avoid channels with very small capacity as well, for which the probability
|
|
* of success is always small anyways,
|
|
* - discard channels with very high base fee that would break our cost
|
|
* estimation,
|
|
* - avoid high latency tor nodes.
|
|
* All combined should reduce the size of the network we explore hopefully
|
|
* reducing the runtime of the MCF solver (FIXME: I should measure this
|
|
* eventually).
|
|
* FIXME: shall we set these threshold parameters as plugin options?
|
|
*/
|
|
|
|
static struct command_result *channelfilter_cb(struct payment *payment)
|
|
{
|
|
assert(payment);
|
|
assert(pay_plugin->gossmap);
|
|
const double HTLC_MAX_FRACTION = 0.01; // 1%
|
|
const u64 HTLC_MAX_STOP_MSAT = 1000000000; // 1M sats
|
|
|
|
u64 disabled_count = 0;
|
|
|
|
|
|
u64 htlc_max_threshold = HTLC_MAX_FRACTION * payment->payment_info
|
|
.amount.millisatoshis; /* Raw: a fraction of this amount. */
|
|
/* Don't exclude channels with htlc_max above HTLC_MAX_STOP_MSAT even if
|
|
* that represents a fraction of the payment smaller than
|
|
* HTLC_MAX_FRACTION. */
|
|
htlc_max_threshold = MIN(htlc_max_threshold, HTLC_MAX_STOP_MSAT);
|
|
|
|
gossmap_apply_localmods(pay_plugin->gossmap, payment->local_gossmods);
|
|
for (const struct gossmap_node *node =
|
|
gossmap_first_node(pay_plugin->gossmap);
|
|
node; node = gossmap_next_node(pay_plugin->gossmap, node)) {
|
|
for (size_t i = 0; i < node->num_chans; i++) {
|
|
int dir;
|
|
const struct gossmap_chan *chan = gossmap_nth_chan(
|
|
pay_plugin->gossmap, node, i, &dir);
|
|
const u64 htlc_max =
|
|
fp16_to_u64(chan->half[dir].htlc_max);
|
|
if (htlc_max < htlc_max_threshold) {
|
|
struct short_channel_id_dir scidd = {
|
|
.scid = gossmap_chan_scid(
|
|
pay_plugin->gossmap, chan),
|
|
.dir = dir};
|
|
disabledmap_add_channel(payment->disabledmap,
|
|
scidd);
|
|
disabled_count++;
|
|
}
|
|
}
|
|
}
|
|
gossmap_remove_localmods(pay_plugin->gossmap, payment->local_gossmods);
|
|
// FIXME: prune the network over other parameters, eg. capacity,
|
|
// fees, ...
|
|
plugin_log(pay_plugin->plugin, LOG_DBG,
|
|
"channelfilter: disabling %" PRIu64 " channels.",
|
|
disabled_count);
|
|
return payment_continue(payment);
|
|
}
|
|
|
|
REGISTER_PAYMENT_MODIFIER(channelfilter, channelfilter_cb);
|
|
|
|
/*****************************************************************************
|
|
* alwaystrue
|
|
*
|
|
* A funny payment condition that always returns true.
|
|
*/
|
|
static bool alwaystrue_cb(const struct payment *payment) { return true; }
|
|
|
|
REGISTER_PAYMENT_CONDITION(alwaystrue, alwaystrue_cb);
|
|
|
|
/*****************************************************************************
|
|
* nothaveresults
|
|
*
|
|
* A payment condition that returns true if the payment has not yet
|
|
* collected enough results to decide whether the payment has succeed,
|
|
* failed or need retrying.
|
|
*/
|
|
static bool nothaveresults_cb(const struct payment *payment)
|
|
{
|
|
return !payment->have_results;
|
|
}
|
|
|
|
REGISTER_PAYMENT_CONDITION(nothaveresults, nothaveresults_cb);
|
|
|
|
/*****************************************************************************
|
|
* retry
|
|
*
|
|
* A payment condition that returns true if we should retry the payment.
|
|
*/
|
|
static bool retry_cb(const struct payment *payment) { return payment->retry; }
|
|
|
|
REGISTER_PAYMENT_CONDITION(retry, retry_cb);
|
|
|
|
/*****************************************************************************
|
|
* Virtual machine
|
|
*
|
|
* The plugin API is based on function calls. This makes is difficult to
|
|
* summarize all payment steps into one function, because the workflow
|
|
* is distributed across multiple functions. The default pay plugin
|
|
* implements a "state machine" for each payment attempt/part and that
|
|
* improves a lot the code readability and modularity. Based on that
|
|
* idea renepay has its own state machine for the whole payment. We go
|
|
* one step further by adding not just function calls (or payment
|
|
* modifiers with OP_CALL) but also conditions with OP_IF that allows
|
|
* for instance to have loops. Renepay's "program" is nicely summarized
|
|
* in the following set of instructions:
|
|
*/
|
|
// TODO
|
|
// add shadow route
|
|
// add check pre-approved invoice
|
|
void *payment_virtual_program[] = {
|
|
/*0*/ OP_CALL, &previoussuccess_pay_mod,
|
|
/*2*/ OP_CALL, &selfpay_pay_mod,
|
|
/*4*/ OP_CALL, &knowledgerelax_pay_mod,
|
|
/*6*/ OP_CALL, &getmychannels_pay_mod,
|
|
/*8*/ OP_CALL, &refreshgossmap_pay_mod,
|
|
/*10*/ OP_CALL, &routehints_pay_mod,
|
|
/*12*/OP_CALL, &channelfilter_pay_mod,
|
|
// TODO shadow_additions
|
|
/* do */
|
|
/*14*/ OP_CALL, &pendingsendpays_pay_mod,
|
|
/*16*/ OP_CALL, &checktimeout_pay_mod,
|
|
/*18*/ OP_CALL, &refreshgossmap_pay_mod,
|
|
/*20*/ OP_CALL, &compute_routes_pay_mod,
|
|
/*22*/ OP_CALL, &send_routes_pay_mod,
|
|
/*do*/
|
|
/*24*/ OP_CALL, &sleep_pay_mod,
|
|
/*26*/ OP_CALL, &collect_results_pay_mod,
|
|
/*while*/
|
|
/*28*/ OP_IF, ¬haveresults_pay_cond, (void *)24,
|
|
/* while */
|
|
/*31*/ OP_IF, &retry_pay_cond, (void *)14,
|
|
/*34*/ OP_CALL, &end_pay_mod, /* safety net, default failure if reached */
|
|
/*36*/ NULL};
|