Skip to content

Commit

Permalink
pay: listpays groups by payment_hash and groupid
Browse files Browse the repository at this point in the history
Fixes ElementsProject#4482
Fixes ElementsProject#4481

Changelog-Added: pay: Payment attempts are now grouped by the pay command that initiated them
Changelog-Fixed: pay: `listpays` returns payments orderd by their creation date
Changelog-Fixed: pay: `listpays` no longer groups attempts from multiple attempts to pay an invoice
  • Loading branch information
cdecker committed May 31, 2021
1 parent 36d2c0a commit a40f6bd
Show file tree
Hide file tree
Showing 2 changed files with 45 additions and 13 deletions.
57 changes: 45 additions & 12 deletions plugins/pay.c
Original file line number Diff line number Diff line change
Expand Up @@ -1656,6 +1656,13 @@ static bool attempt_ongoing(const struct sha256 *payment_hash)
return false;
}

/* A unique key for each payment attempt, even if the same invoice was
* attempted multiple times. */
struct pay_sort_key {
const struct sha256 *payment_hash;
u64 groupid;
};

/* We consolidate multi-part payments into a single entry. */
struct pay_mpp {
/* payment_hash from the invoice and lookup key */
Expand Down Expand Up @@ -1684,21 +1691,31 @@ struct pay_mpp {

/* The destination of the payment, if specified. */
const jsmntok_t *destination;

/* Which sendpay group is this? Necessary for invoices that have been
* attempted multiple times. */
struct pay_sort_key sortkey;
};

static const struct sha256 *pay_mpp_key(const struct pay_mpp *pm)
static const struct pay_sort_key *pay_mpp_key(const struct pay_mpp *pm)
{
return pm->payment_hash;
return &pm->sortkey;
}

static size_t pay_mpp_hash(const struct sha256 *payment_hash)
static size_t pay_mpp_hash(const struct pay_sort_key *key)
{
return siphash24(siphash_seed(), payment_hash, sizeof(struct sha256));
struct siphash24_ctx ctx;
siphash24_init(&ctx, siphash_seed());
siphash24_update(&ctx, key->payment_hash, sizeof(struct sha256));
siphash24_update(&ctx, &key->groupid, sizeof(u64));
return siphash24_done(&ctx);
}

static bool pay_mpp_eq(const struct pay_mpp *pm, const struct sha256 *payment_hash)
static bool pay_mpp_eq(const struct pay_mpp *pm, const struct pay_sort_key *key)
{
return memcmp(pm->payment_hash, payment_hash, sizeof(struct sha256)) == 0;
return memcmp(pm->sortkey.payment_hash, key->payment_hash,
sizeof(struct sha256)) == 0 &&
pm->sortkey.groupid == key->groupid;
}

HTABLE_DEFINE_TYPE(struct pay_mpp, pay_mpp_key, pay_mpp_hash, pay_mpp_eq,
Expand Down Expand Up @@ -1794,8 +1811,8 @@ static struct command_result *listsendpays_done(struct command *cmd,
const jsmntok_t *t, *arr;
struct json_stream *ret;
struct pay_map pay_map;
struct pay_map_iter it;
struct pay_mpp *pm;
struct pay_sort_key *order = tal_arr(tmpctx, struct pay_sort_key, 0);

pay_map_init(&pay_map);

Expand All @@ -1805,10 +1822,12 @@ static struct command_result *listsendpays_done(struct command *cmd,
"Unexpected non-array result from listsendpays");

json_for_each_arr(i, t, arr) {
const jsmntok_t *status, *invstrtok, *hashtok, *createdtok;
const jsmntok_t *status, *invstrtok, *hashtok, *createdtok, *grouptok;
const char *invstr = invstring;
struct sha256 payment_hash;
u32 created_at;
u64 groupid;
struct pay_sort_key key;

invstrtok = json_get_member(buf, t, "bolt11");
if (!invstrtok)
Expand All @@ -1818,12 +1837,21 @@ static struct command_result *listsendpays_done(struct command *cmd,
assert(hashtok != NULL);
assert(createdtok != NULL);

grouptok = json_get_member(buf, t, "groupid");
if (grouptok != NULL)
json_to_u64(buf, grouptok, &groupid);
else
groupid = 0;

json_to_sha256(buf, hashtok, &payment_hash);
json_to_u32(buf, createdtok, &created_at);
if (invstrtok)
invstr = json_strdup(cmd, buf, invstrtok);

pm = pay_map_get(&pay_map, &payment_hash);
key.payment_hash = &payment_hash;
key.groupid = groupid;

pm = pay_map_get(&pay_map, &key);
if (!pm) {
pm = tal(cmd, struct pay_mpp);
pm->payment_hash = tal_dup(pm, struct sha256, &payment_hash);
Expand All @@ -1836,7 +1864,12 @@ static struct command_result *listsendpays_done(struct command *cmd,
pm->num_nonfailed_parts = 0;
pm->status = NULL;
pm->timestamp = created_at;
pm->sortkey.payment_hash = pm->payment_hash;
pm->sortkey.groupid = groupid;
pay_map_add(&pay_map, pm);
// First time we see the groupid we add it to the order
// array, so we can retrieve them in the correct order.
tal_arr_expand(&order, pm->sortkey);
}

status = json_get_member(buf, t, "status");
Expand Down Expand Up @@ -1868,9 +1901,9 @@ static struct command_result *listsendpays_done(struct command *cmd,
ret = jsonrpc_stream_success(cmd);
json_array_start(ret, "pays");

for (pm = pay_map_first(&pay_map, &it);
pm;
pm = pay_map_next(&pay_map, &it)) {
for (size_t i = 0; i < tal_count(order); i++) {
pm = pay_map_get(&pay_map, &order[i]);
assert(pm != NULL);
add_new_entry(ret, buf, pm);
}
pay_map_clear(&pay_map);
Expand Down
1 change: 0 additions & 1 deletion tests/test_pay.py
Original file line number Diff line number Diff line change
Expand Up @@ -4280,7 +4280,6 @@ def test_routehint_tous(node_factory, bitcoind):
l2.rpc.pay(inv)


@pytest.mark.xfail(strict=True)
def test_sendpay_grouping(node_factory, bitcoind):
"""Paying an invoice multiple times, listpays should list them individually
"""
Expand Down

0 comments on commit a40f6bd

Please sign in to comment.