@@ -2528,7 +2528,8 @@ where
2528
2528
2529
2529
fn construct_recv_pending_htlc_info (
2530
2530
& self , hop_data : msgs:: OnionHopData , shared_secret : [ u8 ; 32 ] , payment_hash : PaymentHash ,
2531
- amt_msat : u64 , cltv_expiry : u32 , phantom_shared_secret : Option < [ u8 ; 32 ] > , allow_underpay : bool
2531
+ amt_msat : u64 , cltv_expiry : u32 , phantom_shared_secret : Option < [ u8 ; 32 ] > , allow_underpay : bool ,
2532
+ counterparty_skimmed_fee_msat : Option < u64 > ,
2532
2533
) -> Result < PendingHTLCInfo , ReceiveError > {
2533
2534
// final_incorrect_cltv_expiry
2534
2535
if hop_data. outgoing_cltv_value > cltv_expiry {
@@ -2555,7 +2556,10 @@ where
2555
2556
msg : "The final CLTV expiry is too soon to handle" ,
2556
2557
} ) ;
2557
2558
}
2558
- if !allow_underpay && hop_data. amt_to_forward > amt_msat {
2559
+ if ( !allow_underpay && hop_data. amt_to_forward > amt_msat) ||
2560
+ ( allow_underpay && hop_data. amt_to_forward >
2561
+ amt_msat. saturating_add ( counterparty_skimmed_fee_msat. unwrap_or ( 0 ) ) )
2562
+ {
2559
2563
return Err ( ReceiveError {
2560
2564
err_code : 19 ,
2561
2565
err_data : amt_msat. to_be_bytes ( ) . to_vec ( ) ,
@@ -2622,7 +2626,7 @@ where
2622
2626
incoming_amt_msat : Some ( amt_msat) ,
2623
2627
outgoing_amt_msat : hop_data. amt_to_forward ,
2624
2628
outgoing_cltv_value : hop_data. outgoing_cltv_value ,
2625
- skimmed_fee_msat : None ,
2629
+ skimmed_fee_msat : counterparty_skimmed_fee_msat ,
2626
2630
} )
2627
2631
}
2628
2632
@@ -2861,7 +2865,7 @@ where
2861
2865
onion_utils:: Hop :: Receive ( next_hop_data) => {
2862
2866
// OUR PAYMENT!
2863
2867
match self . construct_recv_pending_htlc_info ( next_hop_data, shared_secret, msg. payment_hash ,
2864
- msg. amount_msat , msg. cltv_expiry , None , allow_underpay)
2868
+ msg. amount_msat , msg. cltv_expiry , None , allow_underpay, msg . skimmed_fee_msat )
2865
2869
{
2866
2870
Ok ( info) => {
2867
2871
// Note that we could obviously respond immediately with an update_fulfill_htlc
@@ -3680,7 +3684,7 @@ where
3680
3684
onion_utils:: Hop :: Receive ( hop_data) => {
3681
3685
match self . construct_recv_pending_htlc_info( hop_data,
3682
3686
incoming_shared_secret, payment_hash, outgoing_amt_msat,
3683
- outgoing_cltv_value, Some ( phantom_shared_secret) , false )
3687
+ outgoing_cltv_value, Some ( phantom_shared_secret) , false , None )
3684
3688
{
3685
3689
Ok ( info) => phantom_receives. push( ( prev_short_channel_id, prev_funding_outpoint, prev_user_channel_id, vec![ ( info, prev_htlc_id) ] ) ) ,
3686
3690
Err ( ReceiveError { err_code, err_data, msg } ) => failed_payment!( msg, err_code, err_data, Some ( phantom_shared_secret) )
@@ -9743,6 +9747,50 @@ mod tests {
9743
9747
get_event_msg ! ( nodes[ 1 ] , MessageSendEvent :: SendAcceptChannel , last_random_pk) ;
9744
9748
}
9745
9749
9750
+ #[ test]
9751
+ fn reject_excessively_underpaying_htlcs ( ) {
9752
+ let chanmon_cfg = create_chanmon_cfgs ( 1 ) ;
9753
+ let node_cfg = create_node_cfgs ( 1 , & chanmon_cfg) ;
9754
+ let node_chanmgr = create_node_chanmgrs ( 1 , & node_cfg, & [ None ] ) ;
9755
+ let node = create_network ( 1 , & node_cfg, & node_chanmgr) ;
9756
+ let sender_intended_amt_msat = 100 ;
9757
+ let extra_fee_msat = 10 ;
9758
+ let hop_data = msgs:: OnionHopData {
9759
+ amt_to_forward : 100 ,
9760
+ outgoing_cltv_value : 42 ,
9761
+ format : msgs:: OnionHopDataFormat :: FinalNode {
9762
+ keysend_preimage : None ,
9763
+ payment_metadata : None ,
9764
+ payment_data : Some ( msgs:: FinalOnionHopData {
9765
+ payment_secret : PaymentSecret ( [ 0 ; 32 ] ) , total_msat : sender_intended_amt_msat,
9766
+ } ) ,
9767
+ }
9768
+ } ;
9769
+ // Check that if the amount we received + the penultimate hop extra fee is less than the sender
9770
+ // intended amount, we fail the payment.
9771
+ if let Err ( crate :: ln:: channelmanager:: ReceiveError { err_code, .. } ) =
9772
+ node[ 0 ] . node . construct_recv_pending_htlc_info ( hop_data, [ 0 ; 32 ] , PaymentHash ( [ 0 ; 32 ] ) ,
9773
+ sender_intended_amt_msat - extra_fee_msat - 1 , 42 , None , true , Some ( extra_fee_msat) )
9774
+ {
9775
+ assert_eq ! ( err_code, 19 ) ;
9776
+ } else { panic ! ( ) ; }
9777
+
9778
+ // If amt_received + extra_fee is equal to the sender intended amount, we're fine.
9779
+ let hop_data = msgs:: OnionHopData { // This is the same hop_data as above, OnionHopData doesn't implement Clone
9780
+ amt_to_forward : 100 ,
9781
+ outgoing_cltv_value : 42 ,
9782
+ format : msgs:: OnionHopDataFormat :: FinalNode {
9783
+ keysend_preimage : None ,
9784
+ payment_metadata : None ,
9785
+ payment_data : Some ( msgs:: FinalOnionHopData {
9786
+ payment_secret : PaymentSecret ( [ 0 ; 32 ] ) , total_msat : sender_intended_amt_msat,
9787
+ } ) ,
9788
+ }
9789
+ } ;
9790
+ assert ! ( node[ 0 ] . node. construct_recv_pending_htlc_info( hop_data, [ 0 ; 32 ] , PaymentHash ( [ 0 ; 32 ] ) ,
9791
+ sender_intended_amt_msat - extra_fee_msat, 42 , None , true , Some ( extra_fee_msat) ) . is_ok( ) ) ;
9792
+ }
9793
+
9746
9794
#[ cfg( anchors) ]
9747
9795
#[ test]
9748
9796
fn test_anchors_zero_fee_htlc_tx_fallback ( ) {
0 commit comments