Skip to content

Commit d083c04

Browse files
committed
Drop non-matching custom TLVs when receiving MPP
Upon receiving multiple payment parts with custom TLVs, we fail payments if they have any non-matching or missing even TLVs, and otherwise just drop non-matching TLVs if they're odd.
1 parent 3b30791 commit d083c04

File tree

2 files changed

+169
-1
lines changed

2 files changed

+169
-1
lines changed

lightning/src/ln/outbound_payment.rs

Lines changed: 9 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -504,7 +504,15 @@ impl RecipientOnionFields {
504504
pub(super) fn check_merge(&mut self, further_htlc_fields: &mut Self) -> Result<(), ()> {
505505
if self.payment_secret != further_htlc_fields.payment_secret { return Err(()); }
506506
if self.payment_metadata != further_htlc_fields.payment_metadata { return Err(()); }
507-
// For custom TLVs we should just drop non-matching ones, but not reject the payment.
507+
508+
if let (Some(tlvs), Some(further_tlvs)) = (&mut self.custom_tlvs, &mut further_htlc_fields.custom_tlvs) {
509+
let even_tlvs: Vec<&(u64, Vec<u8>)> = tlvs.iter().filter(|(typ, _)| *typ % 2 == 0).collect();
510+
let further_even_tlvs: Vec<&(u64, Vec<u8>)> = further_tlvs.iter().filter(|(typ, _)| *typ % 2 == 0).collect();
511+
if even_tlvs != further_even_tlvs { return Err(()) }
512+
513+
tlvs.retain(|tlv| further_tlvs.iter().any(|further_tlv| tlv == further_tlv));
514+
further_tlvs.retain(|further_tlv| tlvs.iter().any(|tlv| tlv == further_tlv));
515+
}
508516
Ok(())
509517
}
510518
}

lightning/src/ln/payment_tests.rs

Lines changed: 160 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -3260,6 +3260,166 @@ fn do_test_custom_tlvs(spontaneous: bool) {
32603260
claim_payment(&nodes[0], &[&nodes[1]], our_payment_preimage);
32613261
}
32623262

3263+
#[test]
3264+
fn test_custom_tlvs_consistency() {
3265+
let even_type_1 = 1 << 16;
3266+
let odd_type_1 = (1 << 16)+ 1;
3267+
let even_type_2 = (1 << 16) + 2;
3268+
let odd_type_2 = (1 << 16) + 3;
3269+
let value_1 = || vec![1, 2, 3, 4];
3270+
let differing_value_1 = || vec![1, 2, 3, 5];
3271+
let value_2 = || vec![42u8; 16];
3272+
3273+
// Drop missing odd tlvs
3274+
do_test_custom_tlvs_consistency(
3275+
vec![(odd_type_1, value_1()), (odd_type_2, value_2())],
3276+
vec![(odd_type_1, value_1())],
3277+
Some(vec![(odd_type_1, value_1())]),
3278+
);
3279+
// Drop non-matching odd tlvs
3280+
do_test_custom_tlvs_consistency(
3281+
vec![(odd_type_1, value_1()), (odd_type_2, value_2())],
3282+
vec![(odd_type_1, differing_value_1()), (odd_type_2, value_2())],
3283+
Some(vec![(odd_type_2, value_2())]),
3284+
);
3285+
// Fail missing even tlvs
3286+
do_test_custom_tlvs_consistency(
3287+
vec![(odd_type_1, value_1()), (even_type_2, value_2())],
3288+
vec![(odd_type_1, value_1())],
3289+
None,
3290+
);
3291+
// Fail non-matching even tlvs
3292+
do_test_custom_tlvs_consistency(
3293+
vec![(even_type_1, value_1()), (odd_type_2, value_2())],
3294+
vec![(even_type_1, differing_value_1()), (odd_type_2, value_2())],
3295+
None,
3296+
);
3297+
}
3298+
3299+
fn do_test_custom_tlvs_consistency(first_tlvs: Vec<(u64, Vec<u8>)>, second_tlvs: Vec<(u64, Vec<u8>)>,
3300+
expected_receive_tlvs: Option<Vec<(u64, Vec<u8>)>>) {
3301+
3302+
let chanmon_cfgs = create_chanmon_cfgs(4);
3303+
let node_cfgs = create_node_cfgs(4, &chanmon_cfgs);
3304+
let node_chanmgrs = create_node_chanmgrs(4, &node_cfgs, &[None, None, None, None]);
3305+
let nodes = create_network(4, &node_cfgs, &node_chanmgrs);
3306+
3307+
create_announced_chan_between_nodes_with_value(&nodes, 0, 1, 100_000, 0);
3308+
create_announced_chan_between_nodes_with_value(&nodes, 0, 2, 100_000, 0);
3309+
create_announced_chan_between_nodes_with_value(&nodes, 1, 3, 100_000, 0);
3310+
let chan_2_3 = create_announced_chan_between_nodes_with_value(&nodes, 2, 3, 100_000, 0);
3311+
3312+
let payment_params = PaymentParameters::from_node_id(nodes[3].node.get_our_node_id(), TEST_FINAL_CLTV)
3313+
.with_bolt11_features(nodes[3].node.invoice_features()).unwrap();
3314+
let mut route = get_route!(nodes[0], payment_params, 15_000_000).unwrap();
3315+
assert_eq!(route.paths.len(), 2);
3316+
route.paths.sort_by(|path_a, _| {
3317+
// Sort the path so that the path through nodes[1] comes first
3318+
if path_a.hops[0].pubkey == nodes[1].node.get_our_node_id() {
3319+
core::cmp::Ordering::Less } else { core::cmp::Ordering::Greater }
3320+
});
3321+
3322+
let (our_payment_preimage, our_payment_hash, our_payment_secret) = get_payment_preimage_hash!(&nodes[3]);
3323+
let payment_id = PaymentId([42; 32]);
3324+
let amt_msat = 15_000_000;
3325+
3326+
// Send first part
3327+
let onion_fields = RecipientOnionFields {
3328+
payment_secret: Some(our_payment_secret),
3329+
payment_metadata: None,
3330+
custom_tlvs: Some(first_tlvs)
3331+
};
3332+
let session_privs = nodes[0].node.test_add_new_pending_payment(our_payment_hash,
3333+
onion_fields.clone(), payment_id, &route).unwrap();
3334+
let cur_height = nodes[0].best_block_info().1;
3335+
nodes[0].node.test_send_payment_along_path(&route.paths[0], &our_payment_hash,
3336+
onion_fields.clone(), amt_msat, cur_height, payment_id,
3337+
&None, session_privs[0]).unwrap();
3338+
check_added_monitors!(nodes[0], 1);
3339+
3340+
{
3341+
let mut events = nodes[0].node.get_and_clear_pending_msg_events();
3342+
assert_eq!(events.len(), 1);
3343+
pass_along_path(&nodes[0], &[&nodes[1], &nodes[3]], amt_msat, our_payment_hash, Some(our_payment_secret), events.pop().unwrap(), false, None);
3344+
}
3345+
assert!(nodes[3].node.get_and_clear_pending_events().is_empty());
3346+
3347+
// Send second part
3348+
let onion_fields = RecipientOnionFields {
3349+
payment_secret: Some(our_payment_secret),
3350+
payment_metadata: None,
3351+
custom_tlvs: Some(second_tlvs)
3352+
};
3353+
nodes[0].node.test_send_payment_along_path(&route.paths[1], &our_payment_hash,
3354+
onion_fields.clone(), amt_msat, cur_height, payment_id, &None, session_privs[1]).unwrap();
3355+
check_added_monitors!(nodes[0], 1);
3356+
3357+
{
3358+
let mut events = nodes[0].node.get_and_clear_pending_msg_events();
3359+
assert_eq!(events.len(), 1);
3360+
let payment_event = SendEvent::from_event(events.pop().unwrap());
3361+
3362+
nodes[2].node.handle_update_add_htlc(&nodes[0].node.get_our_node_id(), &payment_event.msgs[0]);
3363+
commitment_signed_dance!(nodes[2], nodes[0], payment_event.commitment_msg, false);
3364+
3365+
expect_pending_htlcs_forwardable!(nodes[2]);
3366+
check_added_monitors!(nodes[2], 1);
3367+
3368+
let mut events = nodes[2].node.get_and_clear_pending_msg_events();
3369+
assert_eq!(events.len(), 1);
3370+
let payment_event = SendEvent::from_event(events.pop().unwrap());
3371+
3372+
nodes[3].node.handle_update_add_htlc(&nodes[2].node.get_our_node_id(), &payment_event.msgs[0]);
3373+
check_added_monitors!(nodes[3], 0);
3374+
commitment_signed_dance!(nodes[3], nodes[2], payment_event.commitment_msg, true, true);
3375+
}
3376+
expect_pending_htlcs_forwardable_ignore!(nodes[3]);
3377+
nodes[3].node.process_pending_htlc_forwards();
3378+
3379+
if let Some(expected_tlvs) = expected_receive_tlvs {
3380+
// Claim and match expected
3381+
let events = nodes[3].node.get_and_clear_pending_events();
3382+
println!("events: {:?}", events);
3383+
assert_eq!(events.len(), 1);
3384+
match events[0] {
3385+
Event::PaymentClaimable { ref purpose, amount_msat, ref onion_fields, .. } => {
3386+
match &purpose {
3387+
PaymentPurpose::InvoicePayment { payment_secret, .. } => {
3388+
assert_eq!(our_payment_secret, *payment_secret);
3389+
assert_eq!(Some(*payment_secret), onion_fields.as_ref().unwrap().payment_secret);
3390+
},
3391+
PaymentPurpose::SpontaneousPayment(payment_preimage) => {
3392+
assert_eq!(our_payment_preimage, *payment_preimage);
3393+
},
3394+
}
3395+
assert_eq!(amount_msat, amt_msat);
3396+
assert_eq!(onion_fields.clone().unwrap().custom_tlvs.unwrap(), expected_tlvs);
3397+
},
3398+
_ => panic!("Unexpected event"),
3399+
}
3400+
3401+
do_claim_payment_along_route(&nodes[0], &[&[&nodes[1], &nodes[3]], &[&nodes[2], &nodes[3]]], false, our_payment_preimage);
3402+
expect_payment_sent(&nodes[0], our_payment_preimage, Some(Some(2000)), true);
3403+
} else {
3404+
// Expect fail back
3405+
let expected_destinations = vec![HTLCDestination::FailedPayment { payment_hash: our_payment_hash }];
3406+
expect_pending_htlcs_forwardable_and_htlc_handling_failed!(nodes[3], expected_destinations);
3407+
check_added_monitors!(nodes[3], 1);
3408+
3409+
let fail_updates_1 = get_htlc_update_msgs!(nodes[3], nodes[2].node.get_our_node_id());
3410+
nodes[2].node.handle_update_fail_htlc(&nodes[3].node.get_our_node_id(), &fail_updates_1.update_fail_htlcs[0]);
3411+
commitment_signed_dance!(nodes[2], nodes[3], fail_updates_1.commitment_signed, false);
3412+
3413+
expect_pending_htlcs_forwardable_and_htlc_handling_failed!(nodes[2], vec![HTLCDestination::NextHopChannel { node_id: Some(nodes[3].node.get_our_node_id()), channel_id: chan_2_3.2 }]);
3414+
check_added_monitors!(nodes[2], 1);
3415+
3416+
let fail_updates_2 = get_htlc_update_msgs!(nodes[2], nodes[0].node.get_our_node_id());
3417+
nodes[0].node.handle_update_fail_htlc(&nodes[2].node.get_our_node_id(), &fail_updates_2.update_fail_htlcs[0]);
3418+
commitment_signed_dance!(nodes[0], nodes[2], fail_updates_2.commitment_signed, false);
3419+
3420+
expect_payment_failed_conditions(&nodes[0], our_payment_hash, true, PaymentFailedConditions::new().mpp_parts_remain());
3421+
}
3422+
}
32633423

32643424
fn do_test_payment_metadata_consistency(do_reload: bool, do_modify: bool) {
32653425
// Check that a payment metadata received on one HTLC that doesn't match the one received on

0 commit comments

Comments
 (0)