@@ -3089,32 +3089,41 @@ mod tests {
3089
3089
3090
3090
#[ derive( PartialEq ) ]
3091
3091
enum HTLCType { NONE , TIMEOUT , SUCCESS }
3092
- #[ derive( PartialEq ) ]
3093
- enum PenaltyType { NONE , HTLC }
3094
- fn test_txn_broadcast ( node : & Node , chan : & ( msgs:: ChannelUpdate , msgs:: ChannelUpdate , [ u8 ; 32 ] , Transaction ) , commitment_tx : Option < Transaction > , revoked_tx : Option < Transaction > , has_htlc_tx : HTLCType , has_penalty_tx : PenaltyType ) -> Vec < Transaction > {
3092
+ /// Tests that the given node has broadcast transactions for the given Channel
3093
+ ///
3094
+ /// First checks that the latest local commitment tx has been broadcast, unless an explicit
3095
+ /// commitment_tx is provided, which may be used to test that a remote commitment tx was
3096
+ /// broadcast and the revoked outputs were claimed.
3097
+ ///
3098
+ /// Next tests that there is (or is not) a transaction that spends the commitment transaction
3099
+ /// that appears to be the type of HTLC transaction specified in has_htlc_tx.
3100
+ ///
3101
+ /// All broadcast transactions must be accounted for in one of the above three types of we'll
3102
+ /// also fail.
3103
+ fn test_txn_broadcast ( node : & Node , chan : & ( msgs:: ChannelUpdate , msgs:: ChannelUpdate , [ u8 ; 32 ] , Transaction ) , commitment_tx : Option < Transaction > , has_htlc_tx : HTLCType ) -> Vec < Transaction > {
3095
3104
let mut node_txn = node. tx_broadcaster . txn_broadcasted . lock ( ) . unwrap ( ) ;
3096
- assert ! ( node_txn. len( ) >= if has_htlc_tx == HTLCType :: NONE { 0 } else { 1 } + if has_penalty_tx == PenaltyType :: NONE { 0 } else { 1 } ) ;
3105
+ assert ! ( node_txn. len( ) >= if commitment_tx . is_some ( ) { 0 } else { 1 } + if has_htlc_tx == HTLCType :: NONE { 0 } else { 1 } ) ;
3097
3106
3098
3107
let mut res = Vec :: with_capacity ( 2 ) ;
3099
-
3100
- if let Some ( explicit_tx) = commitment_tx {
3101
- res. push ( explicit_tx. clone ( ) ) ;
3102
- } else {
3103
- for tx in node_txn. iter ( ) {
3104
- if tx. input . len ( ) == 1 && tx. input [ 0 ] . previous_output . txid == chan. 3 . txid ( ) {
3105
- let mut funding_tx_map = HashMap :: new ( ) ;
3106
- funding_tx_map. insert ( chan. 3 . txid ( ) , chan. 3 . clone ( ) ) ;
3107
- tx. verify ( & funding_tx_map) . unwrap ( ) ;
3108
+ node_txn. retain ( |tx| {
3109
+ if tx. input . len ( ) == 1 && tx. input [ 0 ] . previous_output . txid == chan. 3 . txid ( ) {
3110
+ let mut funding_tx_map = HashMap :: new ( ) ;
3111
+ funding_tx_map. insert ( chan. 3 . txid ( ) , chan. 3 . clone ( ) ) ;
3112
+ tx. verify ( & funding_tx_map) . unwrap ( ) ;
3113
+ if commitment_tx. is_none ( ) {
3108
3114
res. push ( tx. clone ( ) ) ;
3109
3115
}
3110
- }
3111
- }
3112
- if !revoked_tx. is_some ( ) && !( has_penalty_tx == PenaltyType :: HTLC ) {
3113
- assert_eq ! ( res. len( ) , 1 ) ;
3116
+ false
3117
+ } else { true }
3118
+ } ) ;
3119
+ if let Some ( explicit_tx) = commitment_tx {
3120
+ res. push ( explicit_tx. clone ( ) ) ;
3114
3121
}
3115
3122
3123
+ assert_eq ! ( res. len( ) , 1 ) ;
3124
+
3116
3125
if has_htlc_tx != HTLCType :: NONE {
3117
- for tx in node_txn. iter ( ) {
3126
+ node_txn. retain ( |tx| {
3118
3127
if tx. input . len ( ) == 1 && tx. input [ 0 ] . previous_output . txid == res[ 0 ] . txid ( ) {
3119
3128
let mut funding_tx_map = HashMap :: new ( ) ;
3120
3129
funding_tx_map. insert ( res[ 0 ] . txid ( ) , res[ 0 ] . clone ( ) ) ;
@@ -3125,29 +3134,33 @@ mod tests {
3125
3134
assert ! ( tx. lock_time == 0 ) ;
3126
3135
}
3127
3136
res. push ( tx. clone ( ) ) ;
3128
- break ;
3129
- }
3130
- }
3137
+ println ! ( "11" ) ;
3138
+ false
3139
+ } else { true }
3140
+ } ) ;
3131
3141
assert_eq ! ( res. len( ) , 2 ) ;
3132
3142
}
3133
3143
3134
- if has_penalty_tx == PenaltyType :: HTLC {
3135
- let revoked_tx = revoked_tx. unwrap ( ) ;
3136
- for tx in node_txn. iter ( ) {
3137
- if tx. input . len ( ) == 1 && tx. input [ 0 ] . previous_output . txid == revoked_tx. txid ( ) {
3138
- let mut funding_tx_map = HashMap :: new ( ) ;
3139
- funding_tx_map. insert ( revoked_tx. txid ( ) , revoked_tx. clone ( ) ) ;
3140
- tx. verify ( & funding_tx_map) . unwrap ( ) ;
3141
- res. push ( tx. clone ( ) ) ;
3142
- break ;
3143
- }
3144
- }
3145
- assert_eq ! ( res. len( ) , 1 ) ;
3146
- }
3147
- node_txn. clear ( ) ;
3144
+ assert ! ( node_txn. is_empty( ) ) ;
3148
3145
res
3149
3146
}
3150
3147
3148
+ /// Tests that the given node has broadcast a claim transaction against the provided revoked
3149
+ /// HTLC transaction.
3150
+ fn test_revoked_htlc_claim_txn_broadcast ( node : & Node , revoked_tx : Transaction ) {
3151
+ let mut node_txn = node. tx_broadcaster . txn_broadcasted . lock ( ) . unwrap ( ) ;
3152
+ assert_eq ! ( node_txn. len( ) , 1 ) ;
3153
+ node_txn. retain ( |tx| {
3154
+ if tx. input . len ( ) == 1 && tx. input [ 0 ] . previous_output . txid == revoked_tx. txid ( ) {
3155
+ let mut funding_tx_map = HashMap :: new ( ) ;
3156
+ funding_tx_map. insert ( revoked_tx. txid ( ) , revoked_tx. clone ( ) ) ;
3157
+ tx. verify ( & funding_tx_map) . unwrap ( ) ;
3158
+ false
3159
+ } else { true }
3160
+ } ) ;
3161
+ assert ! ( node_txn. is_empty( ) ) ;
3162
+ }
3163
+
3151
3164
fn check_preimage_claim ( node : & Node , prev_txn : & Vec < Transaction > ) -> Vec < Transaction > {
3152
3165
let mut node_txn = node. tx_broadcaster . txn_broadcasted . lock ( ) . unwrap ( ) ;
3153
3166
@@ -3221,10 +3234,10 @@ mod tests {
3221
3234
// Simple case with no pending HTLCs:
3222
3235
nodes[ 1 ] . node . peer_disconnected ( & nodes[ 0 ] . node . get_our_node_id ( ) , true ) ;
3223
3236
{
3224
- let mut node_txn = test_txn_broadcast ( & nodes[ 1 ] , & chan_1, None , None , HTLCType :: NONE , PenaltyType :: NONE ) ;
3237
+ let mut node_txn = test_txn_broadcast ( & nodes[ 1 ] , & chan_1, None , HTLCType :: NONE ) ;
3225
3238
let header = BlockHeader { version : 0x20000000 , prev_blockhash : Default :: default ( ) , merkle_root : Default :: default ( ) , time : 42 , bits : 42 , nonce : 42 } ;
3226
3239
nodes[ 0 ] . chain_monitor . block_connected_with_filtering ( & Block { header, txdata : vec ! [ node_txn. drain( ..) . next( ) . unwrap( ) ] } , 1 ) ;
3227
- test_txn_broadcast ( & nodes[ 0 ] , & chan_1, None , None , HTLCType :: NONE , PenaltyType :: NONE ) ;
3240
+ test_txn_broadcast ( & nodes[ 0 ] , & chan_1, None , HTLCType :: NONE ) ;
3228
3241
}
3229
3242
get_announce_close_broadcast_events ( & nodes, 0 , 1 ) ;
3230
3243
assert_eq ! ( nodes[ 0 ] . node. list_channels( ) . len( ) , 0 ) ;
@@ -3236,10 +3249,10 @@ mod tests {
3236
3249
// Simple case of one pending HTLC to HTLC-Timeout
3237
3250
nodes[ 1 ] . node . peer_disconnected ( & nodes[ 2 ] . node . get_our_node_id ( ) , true ) ;
3238
3251
{
3239
- let mut node_txn = test_txn_broadcast ( & nodes[ 1 ] , & chan_2, None , None , HTLCType :: TIMEOUT , PenaltyType :: NONE ) ;
3252
+ let mut node_txn = test_txn_broadcast ( & nodes[ 1 ] , & chan_2, None , HTLCType :: TIMEOUT ) ;
3240
3253
let header = BlockHeader { version : 0x20000000 , prev_blockhash : Default :: default ( ) , merkle_root : Default :: default ( ) , time : 42 , bits : 42 , nonce : 42 } ;
3241
3254
nodes[ 2 ] . chain_monitor . block_connected_with_filtering ( & Block { header, txdata : vec ! [ node_txn. drain( ..) . next( ) . unwrap( ) ] } , 1 ) ;
3242
- test_txn_broadcast ( & nodes[ 2 ] , & chan_2, None , None , HTLCType :: NONE , PenaltyType :: NONE ) ;
3255
+ test_txn_broadcast ( & nodes[ 2 ] , & chan_2, None , HTLCType :: NONE ) ;
3243
3256
}
3244
3257
get_announce_close_broadcast_events ( & nodes, 1 , 2 ) ;
3245
3258
assert_eq ! ( nodes[ 1 ] . node. list_channels( ) . len( ) , 0 ) ;
@@ -3273,7 +3286,7 @@ mod tests {
3273
3286
// HTLC-Timeout and a nodes[3] claim against it (+ its own announces)
3274
3287
nodes[ 2 ] . node . peer_disconnected ( & nodes[ 3 ] . node . get_our_node_id ( ) , true ) ;
3275
3288
{
3276
- let node_txn = test_txn_broadcast ( & nodes[ 2 ] , & chan_3, None , None , HTLCType :: TIMEOUT , PenaltyType :: NONE ) ;
3289
+ let node_txn = test_txn_broadcast ( & nodes[ 2 ] , & chan_3, None , HTLCType :: TIMEOUT ) ;
3277
3290
3278
3291
// Claim the payment on nodes[3], giving it knowledge of the preimage
3279
3292
claim_funds ! ( nodes[ 3 ] , nodes[ 2 ] , payment_preimage_1) ;
@@ -3298,7 +3311,7 @@ mod tests {
3298
3311
nodes[ 3 ] . chain_monitor . block_connected_checked ( & header, i, & Vec :: new ( ) [ ..] , & [ 0 ; 0 ] ) ;
3299
3312
}
3300
3313
3301
- let node_txn = test_txn_broadcast ( & nodes[ 3 ] , & chan_4, None , None , HTLCType :: TIMEOUT , PenaltyType :: NONE ) ;
3314
+ let node_txn = test_txn_broadcast ( & nodes[ 3 ] , & chan_4, None , HTLCType :: TIMEOUT ) ;
3302
3315
3303
3316
// Claim the payment on nodes[4], giving it knowledge of the preimage
3304
3317
claim_funds ! ( nodes[ 4 ] , nodes[ 3 ] , payment_preimage_2) ;
@@ -3310,7 +3323,7 @@ mod tests {
3310
3323
nodes[ 4 ] . chain_monitor . block_connected_checked ( & header, i, & Vec :: new ( ) [ ..] , & [ 0 ; 0 ] ) ;
3311
3324
}
3312
3325
3313
- test_txn_broadcast ( & nodes[ 4 ] , & chan_4, None , None , HTLCType :: SUCCESS , PenaltyType :: NONE ) ;
3326
+ test_txn_broadcast ( & nodes[ 4 ] , & chan_4, None , HTLCType :: SUCCESS ) ;
3314
3327
3315
3328
header = BlockHeader { version : 0x20000000 , prev_blockhash : header. bitcoin_hash ( ) , merkle_root : Default :: default ( ) , time : 42 , bits : 42 , nonce : 42 } ;
3316
3329
nodes[ 4 ] . chain_monitor . block_connected_with_filtering ( & Block { header, txdata : vec ! [ node_txn[ 0 ] . clone( ) ] } , TEST_FINAL_CLTV - 5 ) ;
@@ -3345,13 +3358,13 @@ mod tests {
3345
3358
node_txn[ 0 ] . verify ( & funding_tx_map) . unwrap ( ) ;
3346
3359
node_txn. swap_remove ( 0 ) ;
3347
3360
}
3348
- test_txn_broadcast ( & nodes[ 1 ] , & chan_5, None , None , HTLCType :: NONE , PenaltyType :: NONE ) ;
3361
+ test_txn_broadcast ( & nodes[ 1 ] , & chan_5, None , HTLCType :: NONE ) ;
3349
3362
3350
3363
nodes[ 0 ] . chain_monitor . block_connected_with_filtering ( & Block { header, txdata : vec ! [ revoked_local_txn[ 0 ] . clone( ) ] } , 1 ) ;
3351
- let node_txn = test_txn_broadcast ( & nodes[ 0 ] , & chan_5, Some ( revoked_local_txn[ 0 ] . clone ( ) ) , None , HTLCType :: TIMEOUT , PenaltyType :: NONE ) ;
3364
+ let node_txn = test_txn_broadcast ( & nodes[ 0 ] , & chan_5, Some ( revoked_local_txn[ 0 ] . clone ( ) ) , HTLCType :: TIMEOUT ) ;
3352
3365
header = BlockHeader { version : 0x20000000 , prev_blockhash : header. bitcoin_hash ( ) , merkle_root : Default :: default ( ) , time : 42 , bits : 42 , nonce : 42 } ;
3353
3366
nodes[ 1 ] . chain_monitor . block_connected_with_filtering ( & Block { header, txdata : vec ! [ node_txn[ 1 ] . clone( ) ] } , 1 ) ;
3354
- test_txn_broadcast ( & nodes[ 1 ] , & chan_5 , None , Some ( node_txn[ 1 ] . clone ( ) ) , HTLCType :: NONE , PenaltyType :: HTLC ) ;
3367
+ test_revoked_htlc_claim_txn_broadcast ( & nodes[ 1 ] , node_txn[ 1 ] . clone ( ) ) ;
3355
3368
}
3356
3369
get_announce_close_broadcast_events ( & nodes, 0 , 1 ) ;
3357
3370
assert_eq ! ( nodes[ 0 ] . node. list_channels( ) . len( ) , 0 ) ;
0 commit comments