@@ -143,7 +143,6 @@ pub fn select_coins_bnb(
143
143
let mut iteration = 0 ;
144
144
let mut index = 0 ;
145
145
let mut backtrack;
146
- let mut backtrack_subtree;
147
146
148
147
let mut value = Amount :: ZERO ;
149
148
@@ -180,59 +179,14 @@ pub fn select_coins_bnb(
180
179
}
181
180
182
181
while iteration < ITERATION_LIMIT {
183
- // There are two conditions for backtracking:
184
- //
185
- // 1: Not enough value to make it to target.
186
- // This condition happens before reaching a leaf node.
187
- // Looking for a leaf node condition should not make a difference.
188
- // This backtrack removes more than one node and instead starts
189
- // the exploration of a new subtree.
190
- //
191
- // From:
192
- // o
193
- // / \
194
- // 4
195
- // \
196
- // 3
197
- // \
198
- // 2
199
- // /
200
- // 1
201
- // To:
202
- // o
203
- // \
204
- // 4
205
- // /
206
- // 3
207
- //
208
- // 2: value meets or exceeded target.
209
- // In this condition, we only backtrack one node
210
- //
211
- // From:
212
- // o
213
- // /
214
- // 4
215
- // /
216
- // 3
217
- //
218
- // To:
219
- // o
220
- // /
221
- // 4
222
- // \
223
- // 3
224
- // Set initial loop state
225
182
backtrack = false ;
226
- backtrack_subtree = false ;
227
-
228
183
// * not enough value to make it to the target.
229
- // Therefore, explore a new new subtree.
230
184
//
231
185
// unchecked_add is used here for performance. Before entering the search loop, all
232
186
// utxos are summed and checked for overflow. Since there was no overflow then, any
233
187
// subset of addition will not overflow.
234
188
if available_value. unchecked_add ( value) < target {
235
- backtrack_subtree = true ;
189
+ backtrack = true ;
236
190
}
237
191
// This optimization provides an upper bound on the amount of waste that is acceptable.
238
192
// Since value is lost when we create a change output due to increasing the size of the
@@ -273,47 +227,28 @@ pub fn select_coins_bnb(
273
227
274
228
current_waste = current_waste. checked_sub ( waste) ?;
275
229
}
276
-
277
230
// * Backtrack one node
278
231
if backtrack {
279
- let last_index = index_selection. pop ( ) . unwrap ( ) ;
280
- let ( eff_value, utxo_waste, _) = w_utxos[ last_index] ;
281
-
282
- current_waste = current_waste. checked_sub ( utxo_waste) ?;
283
- value = value. checked_sub ( eff_value) ?;
284
- index -= 1 ;
285
- assert_eq ! ( index, last_index) ;
286
- }
287
- // * Backtrack to new tree
288
- else if backtrack_subtree {
289
- // No new subtree left to explore.
290
232
if index_selection. is_empty ( ) {
291
233
return index_to_utxo_list ( best_selection, w_utxos) ;
292
234
}
293
235
294
- // Anchor the new subtree at the next available index.
295
- //
296
- // if our index selection is: [0,1,2,3]
297
- // then copy the head 0 into index.
298
- // At the end of this loop, index is incremented to 1.
299
- // Therefore, we will be starting our next search tree at index 1.
300
- index = index_selection[ 0 ] ;
301
-
302
- // Reset waste counter since we are starting a new search branch.
303
- current_waste = SignedAmount :: ZERO ;
236
+ loop {
237
+ index -= 1 ;
304
238
305
- // The available value of the next iteration. This should never overflow
306
- // since the value is always less than the last available_value calculation.
307
- available_value = w_utxos [ index + 1 .. ] . iter ( ) . map ( | & ( v , _ , _ ) | v ) . sum ( ) ;
239
+ if index <= * index_selection . last ( ) . unwrap ( ) {
240
+ break ;
241
+ }
308
242
309
- // If the new subtree does not have enough value, we are done searching.
310
- if available_value < target {
311
- return index_to_utxo_list ( best_selection, w_utxos) ;
312
- }
243
+ let ( eff_value, _, _) = w_utxos[ index] ;
244
+ available_value += eff_value;
245
+ } ;
313
246
314
- // Start a new selection and add the root of the new subtree to the index selection.
315
- index_selection. clear ( ) ;
316
- value = Amount :: ZERO ;
247
+ assert_eq ! ( index, * index_selection. last( ) . unwrap( ) ) ;
248
+ let ( eff_value, utxo_waste, _) = w_utxos[ index] ;
249
+ current_waste = current_waste. checked_sub ( utxo_waste) ?;
250
+ value = value. checked_sub ( eff_value) ?;
251
+ index_selection. pop ( ) . unwrap ( ) ;
317
252
}
318
253
// * Add next node to the inclusion branch.
319
254
else {
@@ -392,7 +327,7 @@ mod tests {
392
327
. collect ( )
393
328
}
394
329
395
- fn create_weighted_utxos_from_values ( fee : Amount , values : Vec < Amount > ) -> Vec < WeightedUtxo > {
330
+ fn create_weighted_utxos_from_values ( values : Vec < Amount > ) -> Vec < WeightedUtxo > {
396
331
values
397
332
. iter ( )
398
333
. map ( |amt| WeightedUtxo {
@@ -737,7 +672,31 @@ mod tests {
737
672
}
738
673
739
674
#[ test]
740
- fn select_coins_bnb_extended_set ( ) {
675
+ fn select_coins_bnb_set_size_five ( ) {
676
+ let target = Amount :: from_str ( "6 cBTC" ) . unwrap ( ) ;
677
+ let cost_of_change = Amount :: ZERO ;
678
+ let vals = vec ! [
679
+ Amount :: from_str( "3 cBTC" ) . unwrap( ) ,
680
+ Amount :: from_str( "2.9 cBTC" ) . unwrap( ) ,
681
+ Amount :: from_str( "2 cBTC" ) . unwrap( ) ,
682
+ Amount :: from_str( "1.9 cBTC" ) . unwrap( ) ,
683
+ Amount :: from_str( "1 cBTC" ) . unwrap( ) ,
684
+ ] ;
685
+
686
+ let weighted_utxos = create_weighted_utxos_from_values ( vals) ;
687
+ let list: Vec < _ > =
688
+ select_coins_bnb ( target, cost_of_change, FeeRate :: ZERO , FeeRate :: ZERO , & weighted_utxos)
689
+ . unwrap ( )
690
+ . collect ( ) ;
691
+
692
+ assert_eq ! ( list. len( ) , 3 ) ;
693
+ assert_eq ! ( list[ 0 ] . utxo. value, Amount :: from_str( "3 cBTC" ) . unwrap( ) ) ;
694
+ assert_eq ! ( list[ 1 ] . utxo. value, Amount :: from_str( "2 cBTC" ) . unwrap( ) ) ;
695
+ assert_eq ! ( list[ 2 ] . utxo. value, Amount :: from_str( "1 cBTC" ) . unwrap( ) ) ;
696
+ }
697
+
698
+ #[ test]
699
+ fn select_coins_bnb_set_size_seven ( ) {
741
700
let target = Amount :: from_str ( "18 cBTC" ) . unwrap ( ) ;
742
701
let cost_of_change = Amount :: from_str ( "50 sats" ) . unwrap ( ) ;
743
702
let vals = vec ! [
@@ -748,17 +707,18 @@ mod tests {
748
707
Amount :: from_str( "3 cBTC" ) . unwrap( ) ,
749
708
Amount :: from_str( "2 cBTC" ) . unwrap( ) ,
750
709
Amount :: from_str( "1 cBTC" ) . unwrap( ) + Amount :: from_str( "5 sats" ) . unwrap( ) ,
751
- ] ;
710
+ ] ;
752
711
753
- let weighted_utxos = create_weighted_utxos_from_values ( Amount :: ZERO , vals) ;
712
+ let weighted_utxos = create_weighted_utxos_from_values ( vals) ;
754
713
let list: Vec < _ > =
755
714
select_coins_bnb ( target, cost_of_change, FeeRate :: ZERO , FeeRate :: ZERO , & weighted_utxos)
756
715
. unwrap ( )
757
716
. collect ( ) ;
758
717
759
- assert_eq ! ( list. len( ) , 3 ) ;
760
- assert_eq ! ( list[ 0 ] . utxo. value, Amount :: from_str( "10 cBTC" ) . unwrap( ) ) ;
761
- assert_eq ! ( list[ 1 ] . utxo. value, Amount :: from_str( "6 cBTC" ) . unwrap( ) ) ;
762
- assert_eq ! ( list[ 2 ] . utxo. value, Amount :: from_str( "2 cBTC" ) . unwrap( ) ) ;
763
- }
718
+ assert_eq ! ( list. len( ) , 3 ) ;
719
+ assert_eq ! ( list[ 0 ] . utxo. value, Amount :: from_str( "10 cBTC" ) . unwrap( ) ) ;
720
+ assert_eq ! ( list[ 1 ] . utxo. value, Amount :: from_str( "6 cBTC" ) . unwrap( ) ) ;
721
+ assert_eq ! ( list[ 2 ] . utxo. value, Amount :: from_str( "2 cBTC" ) . unwrap( ) ) ;
722
+ }
723
+
764
724
}
0 commit comments