@@ -2460,3 +2460,165 @@ fn test_trampoline_forward_rejection() {
2460
2460
expect_payment_failed_conditions ( & nodes[ 0 ] , payment_hash, false , payment_failed_conditions) ;
2461
2461
}
2462
2462
}
2463
+
2464
+ #[ test]
2465
+ fn test_unblinded_trampoline_forward ( ) {
2466
+ // Simulate a payment of A (0) -> B (1) -> C(Trampoline) (2) -> D(Trampoline(receive)) (3)
2467
+ // trampoline hops C -> T0 (4) -> D
2468
+ // make it fail at B, then at C's outer onion, then at C's inner onion
2469
+ const TOTAL_NODE_COUNT : usize = 5 ;
2470
+ let secp_ctx = Secp256k1 :: new ( ) ;
2471
+
2472
+ let chanmon_cfgs = create_chanmon_cfgs ( TOTAL_NODE_COUNT ) ;
2473
+ let node_cfgs = create_node_cfgs ( TOTAL_NODE_COUNT , & chanmon_cfgs) ;
2474
+ let node_chanmgrs = create_node_chanmgrs ( TOTAL_NODE_COUNT , & node_cfgs, & vec ! [ None ; TOTAL_NODE_COUNT ] ) ;
2475
+ let mut nodes = create_network ( TOTAL_NODE_COUNT , & node_cfgs, & node_chanmgrs) ;
2476
+
2477
+ let ( _, _, chan_id_alice_bob, _) = create_announced_chan_between_nodes_with_value ( & nodes, 0 , 1 , 1_000_000 , 0 ) ;
2478
+ let ( _, _, chan_id_bob_carol, _) = create_announced_chan_between_nodes_with_value ( & nodes, 1 , 2 , 1_000_000 , 0 ) ;
2479
+ let ( _, _, _, _) = create_announced_chan_between_nodes_with_value ( & nodes, 2 , 4 , 1_000_000 , 0 ) ;
2480
+ let ( _, _, _, _) = create_announced_chan_between_nodes_with_value ( & nodes, 4 , 3 , 1_000_000 , 0 ) ;
2481
+
2482
+ for i in 0 ..TOTAL_NODE_COUNT { // connect all nodes' blocks
2483
+ connect_blocks ( & nodes[ i] , ( TOTAL_NODE_COUNT as u32 ) * CHAN_CONFIRM_DEPTH + 1 - nodes[ i] . best_block_info ( ) . 1 ) ;
2484
+ }
2485
+
2486
+ let alice_node_id = nodes[ 0 ] . node ( ) . get_our_node_id ( ) ;
2487
+ let bob_node_id = nodes[ 1 ] . node ( ) . get_our_node_id ( ) ;
2488
+ let carol_node_id = nodes[ 2 ] . node ( ) . get_our_node_id ( ) ;
2489
+ let dave_node_id = nodes[ 3 ] . node ( ) . get_our_node_id ( ) ;
2490
+
2491
+ let alice_bob_scid = nodes[ 0 ] . node ( ) . list_channels ( ) . iter ( ) . find ( |c| c. channel_id == chan_id_alice_bob) . unwrap ( ) . short_channel_id . unwrap ( ) ;
2492
+ let bob_carol_scid = nodes[ 1 ] . node ( ) . list_channels ( ) . iter ( ) . find ( |c| c. channel_id == chan_id_bob_carol) . unwrap ( ) . short_channel_id . unwrap ( ) ;
2493
+
2494
+ let amt_msat = 1000 ;
2495
+ let ( payment_preimage, payment_hash, payment_secret) = get_payment_preimage_hash ( & nodes[ 3 ] , Some ( amt_msat) , None ) ;
2496
+
2497
+ let route = Route {
2498
+ paths : vec ! [ Path {
2499
+ hops: vec![
2500
+ // Bob
2501
+ RouteHop {
2502
+ pubkey: bob_node_id,
2503
+ node_features: NodeFeatures :: empty( ) ,
2504
+ short_channel_id: alice_bob_scid,
2505
+ channel_features: ChannelFeatures :: empty( ) ,
2506
+ fee_msat: 1000 , // forwarding fee to Carol
2507
+ cltv_expiry_delta: 48 ,
2508
+ maybe_announced_channel: false ,
2509
+ } ,
2510
+
2511
+ // Carol
2512
+ RouteHop {
2513
+ pubkey: carol_node_id,
2514
+ node_features: NodeFeatures :: empty( ) ,
2515
+ short_channel_id: bob_carol_scid,
2516
+ channel_features: ChannelFeatures :: empty( ) ,
2517
+ fee_msat: 2000 , // fee for the usage of the entire blinded path, including Trampoline
2518
+ cltv_expiry_delta: 48 ,
2519
+ maybe_announced_channel: false ,
2520
+ }
2521
+ ] ,
2522
+ blinded_tail: Some ( BlindedTail {
2523
+ trampoline_hops: vec![
2524
+ // Carol
2525
+ TrampolineHop {
2526
+ pubkey: carol_node_id,
2527
+ node_features: Features :: empty( ) ,
2528
+ fee_msat: amt_msat,
2529
+ cltv_expiry_delta: 176 , // let her cook
2530
+ } ,
2531
+
2532
+ // Dave (recipient)
2533
+ TrampolineHop {
2534
+ pubkey: dave_node_id,
2535
+ node_features: Features :: empty( ) ,
2536
+ fee_msat: 0 , // no need to charge a fee as the recipient
2537
+ cltv_expiry_delta: 24 ,
2538
+ } ,
2539
+ ] ,
2540
+ hops: vec![
2541
+ // Dave's blinded node id
2542
+ BlindedHop {
2543
+ blinded_node_id: pubkey_from_hex( "0295d40514096a8be54859e7dfe947b376eaafea8afe5cb4eb2c13ff857ed0b4be" ) ,
2544
+ encrypted_payload: bytes_from_hex( "0ccf3c8a58deaa603f657ee2a5ed9d604eb5c8ca1e5f801989afa8f3ea6d789bbdde2c7e7a1ef9ca8c38d2c54760febad8446d3f273ddb537569ef56613846ccd3aba78a" ) ,
2545
+ }
2546
+ ] ,
2547
+ blinding_point: alice_node_id,
2548
+ excess_final_cltv_expiry_delta: 39 ,
2549
+ final_value_msat: amt_msat,
2550
+ } )
2551
+ } ] ,
2552
+ route_params : None ,
2553
+ } ;
2554
+
2555
+ nodes[ 0 ] . node . send_payment_with_route ( route. clone ( ) , payment_hash, RecipientOnionFields :: spontaneous_empty ( ) , PaymentId ( payment_hash. 0 ) ) . unwrap ( ) ;
2556
+
2557
+ let replacement_onion = {
2558
+ // create a substitute onion where the last Trampoline hop is an unblinded receive, which we
2559
+ // (deliberately) do not support out of the box, therefore necessitating this workaround
2560
+ let trampoline_secret_key = secret_from_hex ( "0134928f7b7ca6769080d70f16be84c812c741f545b49a34db47ce338a205799" ) ;
2561
+ let prng_seed = secret_from_hex ( "fe02b4b9054302a3ddf4e1e9f7c411d644aebbd295218ab009dca94435f775a9" ) ;
2562
+ let recipient_onion_fields = RecipientOnionFields :: spontaneous_empty ( ) ;
2563
+
2564
+ let blinded_tail = route. paths [ 0 ] . blinded_tail . clone ( ) . unwrap ( ) ;
2565
+ let ( mut trampoline_payloads, outer_total_msat, outer_starting_htlc_offset) = onion_utils:: build_trampoline_onion_payloads ( & blinded_tail, amt_msat, & recipient_onion_fields, 32 , & None ) . unwrap ( ) ;
2566
+
2567
+ // pop the last dummy hop
2568
+ trampoline_payloads. pop ( ) ;
2569
+
2570
+ trampoline_payloads. push ( msgs:: OutboundTrampolinePayload :: Receive {
2571
+ payment_data : Some ( msgs:: FinalOnionHopData {
2572
+ payment_secret,
2573
+ total_msat : amt_msat,
2574
+ } ) ,
2575
+ sender_intended_htlc_amt_msat : amt_msat,
2576
+ cltv_expiry_height : 96 ,
2577
+ } ) ;
2578
+
2579
+ let trampoline_onion_keys = onion_utils:: construct_trampoline_onion_keys ( & secp_ctx, & route. paths [ 0 ] . blinded_tail . as_ref ( ) . unwrap ( ) , & trampoline_secret_key) . unwrap ( ) ;
2580
+ let trampoline_packet = onion_utils:: construct_trampoline_onion_packet (
2581
+ trampoline_payloads,
2582
+ trampoline_onion_keys,
2583
+ prng_seed. secret_bytes ( ) ,
2584
+ & payment_hash,
2585
+ None ,
2586
+ ) . unwrap ( ) ;
2587
+
2588
+ let outer_session_priv = secret_from_hex ( "e52c20461ed7acd46c4e7b591a37610519179482887bd73bf3b94617f8f03677" ) ;
2589
+
2590
+ let ( outer_payloads, _, _) = onion_utils:: build_onion_payloads ( & route. paths [ 0 ] , outer_total_msat, & recipient_onion_fields, outer_starting_htlc_offset, & None , None , Some ( trampoline_packet) ) . unwrap ( ) ;
2591
+ let outer_onion_keys = onion_utils:: construct_onion_keys ( & secp_ctx, & route. clone ( ) . paths [ 0 ] , & outer_session_priv) . unwrap ( ) ;
2592
+ let outer_packet = onion_utils:: construct_onion_packet (
2593
+ outer_payloads,
2594
+ outer_onion_keys,
2595
+ prng_seed. secret_bytes ( ) ,
2596
+ & payment_hash,
2597
+ ) . unwrap ( ) ;
2598
+
2599
+ outer_packet
2600
+ } ;
2601
+
2602
+ check_added_monitors ! ( & nodes[ 0 ] , 1 ) ;
2603
+
2604
+ let mut events = nodes[ 0 ] . node . get_and_clear_pending_msg_events ( ) ;
2605
+ assert_eq ! ( events. len( ) , 1 ) ;
2606
+ let mut first_message_event = remove_first_msg_event_to_node ( & nodes[ 1 ] . node . get_our_node_id ( ) , & mut events) ;
2607
+ let mut update_message = match first_message_event {
2608
+ MessageSendEvent :: UpdateHTLCs { ref mut updates, .. } => {
2609
+ assert_eq ! ( updates. update_add_htlcs. len( ) , 1 ) ;
2610
+ updates. update_add_htlcs . get_mut ( 0 )
2611
+ } ,
2612
+ _ => panic ! ( )
2613
+ } ;
2614
+ update_message. map ( |msg| {
2615
+ msg. onion_routing_packet = replacement_onion. clone ( ) ;
2616
+ } ) ;
2617
+
2618
+ let route: & [ & Node ] = & [ & nodes[ 1 ] , & nodes[ 2 ] , & nodes[ 4 ] , & nodes[ 3 ] ] ;
2619
+ let args = PassAlongPathArgs :: new ( & nodes[ 0 ] , route, amt_msat, payment_hash, first_message_event)
2620
+ . with_payment_secret ( payment_secret) ;
2621
+ do_pass_along_path ( args) ;
2622
+
2623
+ claim_payment ( & nodes[ 0 ] , & [ & nodes[ 1 ] , & nodes[ 2 ] , & nodes[ 4 ] , & nodes[ 3 ] ] , payment_preimage) ;
2624
+ }
0 commit comments