Browse Source

channeld: handle malformed onion properly.

When the next node tells us the onion is malformed, we now actually
report the failcode to lightningd (rather than generating an invalid
error as we do now).

We could generate the onion at this point, but except we don't know
the shared secret; we'd have to plumb that through from the incoming
channeld's HTLC.

Signed-off-by: Rusty Russell <rusty@rustcorp.com.au>
pr-2218
Rusty Russell 6 years ago
committed by Christian Decker
parent
commit
f5ced1ddd5
  1. 1
      CHANGELOG.md
  2. 28
      channeld/channeld.c

1
CHANGELOG.md

@ -40,6 +40,7 @@ changes.
- JSON API: commands are once again read even if one hasn't responded yet (broken in 0.6.2). - JSON API: commands are once again read even if one hasn't responded yet (broken in 0.6.2).
- Protocol: allow lnd to send `update_fee` before `funding_locked`. - Protocol: allow lnd to send `update_fee` before `funding_locked`.
- Protocol: fix limit on how much funder can send (fee was 1000x too small) - Protocol: fix limit on how much funder can send (fee was 1000x too small)
- Protocol: don't send invalid onion errors if peer says onion was bad.
- pylightning: handle multiple simultanous RPC replies reliably. - pylightning: handle multiple simultanous RPC replies reliably.

28
channeld/channeld.c

@ -1240,13 +1240,13 @@ static u8 *got_commitsig_msg(const tal_t *ctx,
} else if (htlc->state == RCVD_REMOVE_COMMIT) { } else if (htlc->state == RCVD_REMOVE_COMMIT) {
if (htlc->r) { if (htlc->r) {
struct fulfilled_htlc *f; struct fulfilled_htlc *f;
assert(!htlc->fail); assert(!htlc->fail && !htlc->failcode);
f = tal_arr_expand(&fulfilled); f = tal_arr_expand(&fulfilled);
f->id = htlc->id; f->id = htlc->id;
f->payment_preimage = *htlc->r; f->payment_preimage = *htlc->r;
} else { } else {
struct failed_htlc *f; struct failed_htlc *f;
assert(htlc->fail); assert(htlc->fail || htlc->failcode);
f = tal(failed, struct failed_htlc); f = tal(failed, struct failed_htlc);
f->id = htlc->id; f->id = htlc->id;
f->failcode = htlc->failcode; f->failcode = htlc->failcode;
@ -1585,7 +1585,6 @@ static void handle_peer_fail_malformed_htlc(struct peer *peer, const u8 *msg)
struct sha256 sha256_of_onion; struct sha256 sha256_of_onion;
u16 failure_code; u16 failure_code;
struct htlc *htlc; struct htlc *htlc;
u8 *fail;
if (!fromwire_update_fail_malformed_htlc(msg, &channel_id, &id, if (!fromwire_update_fail_malformed_htlc(msg, &channel_id, &id,
&sha256_of_onion, &sha256_of_onion,
@ -1602,12 +1601,16 @@ static void handle_peer_fail_malformed_htlc(struct peer *peer, const u8 *msg)
* `update_fail_malformed_htlc`: * `update_fail_malformed_htlc`:
* - MUST fail the channel. * - MUST fail the channel.
*/ */
if (!(failure_code & BADONION)) { /* We only handle these cases. */
if (failure_code != WIRE_INVALID_ONION_VERSION
&& failure_code != WIRE_INVALID_ONION_HMAC
&& failure_code != WIRE_INVALID_ONION_KEY) {
peer_failed(&peer->cs, peer_failed(&peer->cs,
&peer->channel_id, &peer->channel_id,
"Bad update_fail_malformed_htlc failure code %u", "Bad update_fail_malformed_htlc failure code %u",
failure_code); failure_code);
} }
assert(failure_code & BADONION);
e = channel_fail_htlc(peer->channel, LOCAL, id, &htlc); e = channel_fail_htlc(peer->channel, LOCAL, id, &htlc);
switch (e) { switch (e) {
@ -1620,20 +1623,9 @@ static void handle_peer_fail_malformed_htlc(struct peer *peer, const u8 *msg)
* - MAY retry or choose an alternate error response. * - MAY retry or choose an alternate error response.
*/ */
/* BOLT #2: /* This is the only case where we set failcode for a non-local
* * failure; in a way, it is, since we have to report it. */
* - otherwise, a receiving node which has an outgoing HTLC htlc->failcode = failure_code;
* canceled by `update_fail_malformed_htlc`:
*
* - MUST return an error in the `update_fail_htlc` sent to
* the link which originally sent the HTLC, using the
* `failure_code` given and setting the data to
* `sha256_of_onion`.
*/
fail = tal_arr(htlc, u8, 0);
towire_u16(&fail, failure_code);
towire_sha256(&fail, &sha256_of_onion);
htlc->fail = fail;
start_commit_timer(peer); start_commit_timer(peer);
return; return;
case CHANNEL_ERR_NO_SUCH_ID: case CHANNEL_ERR_NO_SUCH_ID:

Loading…
Cancel
Save