Skip to content

Commit 694ef1e

Browse files
Fix bug where we encode flags field into all updates on htlc fail
Failing an HTLC with onion error channel_disabled requires encoding a 'flags' field into the failure packet. However, we were encoding this 'flags' field for all failures packets that were failing on update_add_htlc with an update (error 0x1000 UPDATE). Discovered in the course of adding phantom payment failure tests, which also added testing for this bug
1 parent 26fe879 commit 694ef1e

File tree

2 files changed

+46
-1
lines changed

2 files changed

+46
-1
lines changed

lightning/src/ln/channelmanager.rs

Lines changed: 3 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -4504,7 +4504,9 @@ impl<Signer: Sign, M: Deref, T: Deref, K: Deref, F: Deref, L: Deref> ChannelMana
45044504
onion_utils::build_first_hop_failure_packet(incoming_shared_secret, error_code, &{
45054505
let mut res = Vec::with_capacity(8 + 128);
45064506
// TODO: underspecified, follow https://github.com/lightningnetwork/lightning-rfc/issues/791
4507-
res.extend_from_slice(&byte_utils::be16_to_array(0));
4507+
if error_code == 0x1000 | 20 {
4508+
res.extend_from_slice(&byte_utils::be16_to_array(0));
4509+
}
45084510
res.extend_from_slice(&upd.encode_with_len()[..]);
45094511
res
45104512
}[..])

lightning/src/ln/onion_route_tests.rs

Lines changed: 43 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -899,6 +899,49 @@ fn test_phantom_failure_too_low_recv_amt() {
899899
expect_payment_failed_conditions!(nodes[0], payment_hash, true, fail_conditions);
900900
}
901901

902+
#[test]
903+
fn test_phantom_dust_exposure_failure() {
904+
// Set the max dust exposure to the dust limit.
905+
let max_dust_exposure = 546;
906+
let mut receiver_config = UserConfig::default();
907+
receiver_config.channel_options.max_dust_htlc_exposure_msat = max_dust_exposure;
908+
receiver_config.channel_options.announced_channel = true;
909+
910+
let chanmon_cfgs = create_chanmon_cfgs(2);
911+
let node_cfgs = create_node_cfgs(2, &chanmon_cfgs);
912+
let node_chanmgrs = create_node_chanmgrs(2, &node_cfgs, &[None, Some(receiver_config)]);
913+
let nodes = create_network(2, &node_cfgs, &node_chanmgrs);
914+
915+
let channel = create_announced_chan_between_nodes(&nodes, 0, 1, InitFeatures::known(), InitFeatures::known());
916+
917+
// Get the route with an amount exceeding the dust exposure threshold of nodes[1].
918+
let (_, payment_hash, payment_secret) = get_payment_preimage_hash!(nodes[1], Some(max_dust_exposure + 1));
919+
let (mut route, _) = get_phantom_route!(nodes, max_dust_exposure + 1, channel);
920+
921+
// Route the HTLC through to the destination.
922+
nodes[0].node.send_payment(&route, payment_hash.clone(), &Some(payment_secret)).unwrap();
923+
check_added_monitors!(nodes[0], 1);
924+
let update_0 = get_htlc_update_msgs!(nodes[0], nodes[1].node.get_our_node_id());
925+
let mut update_add = update_0.update_add_htlcs[0].clone();
926+
927+
nodes[1].node.handle_update_add_htlc(&nodes[0].node.get_our_node_id(), &update_add);
928+
commitment_signed_dance!(nodes[1], nodes[0], &update_0.commitment_signed, false, true);
929+
930+
let update_1 = get_htlc_update_msgs!(nodes[1], nodes[0].node.get_our_node_id());
931+
assert!(update_1.update_fail_htlcs.len() == 1);
932+
let fail_msg = update_1.update_fail_htlcs[0].clone();
933+
nodes[0].node.handle_update_fail_htlc(&nodes[1].node.get_our_node_id(), &fail_msg);
934+
commitment_signed_dance!(nodes[0], nodes[1], update_1.commitment_signed, false);
935+
936+
// Ensure the payment fails with the expected error.
937+
let mut error_data = channel.1.encode_with_len();
938+
let mut fail_conditions = PaymentFailedConditions::new()
939+
.blamed_scid(channel.0.contents.short_channel_id)
940+
.blamed_chan_closed(false)
941+
.expected_htlc_error_data(0x1000 | 7, &error_data);
942+
expect_payment_failed_conditions!(nodes[0], payment_hash, false, fail_conditions);
943+
}
944+
902945
#[test]
903946
fn test_phantom_failure_reject_payment() {
904947
// Test that the user can successfully fail back a phantom node payment.

0 commit comments

Comments
 (0)