@@ -45,24 +45,10 @@ fn calc_effective_values<Utxo: WeightedUtxo>(weighted_utxos: &[Utxo], fee_rate:
45
45
. collect ( )
46
46
}
47
47
48
- // A Vector of Weights that is the same size as utxo_pool. Provides a lookup
49
- // to derermine the smallest possible Weight a UTXO can be after a given index.
48
+ // Provides a lookup to derermine the minimum UTXO weight after a given index.
50
49
fn build_min_tail_weight < Utxo : WeightedUtxo > ( weighted_utxos : Vec < ( Amount , & Utxo ) > ) -> Vec < Weight > {
51
- let mut min_group_weight: Vec < Weight > = vec ! [ ] ;
52
- let mut min = Weight :: MAX ;
53
-
54
- for ( _, u) in & weighted_utxos {
55
- min_group_weight. push ( min) ;
56
-
57
- // weight is used instead of satisfaction_weight to mirror core.
58
- let weight = u. weight ( ) ;
59
-
60
- if weight < min {
61
- min = weight;
62
- }
63
- }
64
-
65
- min_group_weight. into_iter ( ) . rev ( ) . collect ( )
50
+ let min_element = weighted_utxos. iter ( ) . min_by ( |a, b| a. 1 . weight ( ) . cmp ( & b. 1 . weight ( ) ) ) . unwrap ( ) ;
51
+ vec ! [ min_element. 1 . weight( ) ; weighted_utxos. len( ) ]
66
52
}
67
53
68
54
fn index_to_utxo_list < Utxo : WeightedUtxo > (
@@ -138,11 +124,16 @@ pub fn select_coins<Utxo: WeightedUtxo>(
138
124
} ) ;
139
125
140
126
let lookahead = build_lookahead ( w_utxos. clone ( ) , available_value) ;
141
-
142
- //let min_group_weight = w_utxos.clone();
143
127
let min_tail_weight = build_min_tail_weight ( w_utxos. clone ( ) ) ;
144
128
129
+ for i in & min_tail_weight {
130
+ println ! ( "{:?}" , i) ;
131
+ }
132
+
145
133
let total_target = target + change_target;
134
+ println ! ( "total_target: {:?}" , total_target) ;
135
+ println ! ( "target: {:?}" , target) ;
136
+ println ! ( "change_target: {:?}" , change_target) ;
146
137
147
138
if available_value < total_target {
148
139
return None
@@ -166,6 +157,10 @@ pub fn select_coins<Utxo: WeightedUtxo>(
166
157
167
158
// EXPLORE
168
159
let ( eff_value, u) = w_utxos[ next_utxo_index] ;
160
+ println ! ( "" ) ;
161
+ println ! ( "* amt: {:?}" , eff_value) ;
162
+ println ! ( "* amount_sum: {:?}" , amount_sum) ;
163
+ println ! ( "* weight_sum: {:?}" , weight_sum) ;
169
164
170
165
amount_sum += eff_value;
171
166
weight_sum += u. weight ( ) ;
@@ -175,12 +170,19 @@ pub fn select_coins<Utxo: WeightedUtxo>(
175
170
iteration += 1 ;
176
171
177
172
let tail: usize = * selection. last ( ) . unwrap ( ) ;
178
-
179
173
// no possibility of hitting the total along this branch.
180
174
// CUT
175
+ println ! ( "weight sum: {:?}" , weight_sum. to_wu( ) ) ;
176
+ println ! ( "min tail weight: {:?}" , min_tail_weight[ tail] . to_wu( ) ) ;
177
+ println ! ( "total target {:?}: " , total_target) ;
178
+ println ! ( "amount sum: {:?}" , amount_sum) ;
179
+ println ! ( "tail: {:?}" , w_utxos[ tail] . 0 . to_sat( ) ) ;
180
+ println ! ( "best_weight_sum: {:?}" , best_weight_sum) ;
181
181
if amount_sum + lookahead[ tail] < total_target {
182
+ println ! ( "amount_sum + lookahead[tail] < total_target" ) ;
182
183
cut = true ;
183
184
} else if weight_sum > best_weight_sum {
185
+ println ! ( "weight_sum > best_weight_sum" ) ;
184
186
// check if a better solution could exist. IE there exists a utxo with a better
185
187
// weight along the current branch
186
188
if w_utxos[ tail] . 1 . weight ( ) <= min_tail_weight[ tail] {
@@ -197,12 +199,33 @@ pub fn select_coins<Utxo: WeightedUtxo>(
197
199
shift = true ;
198
200
}
199
201
} else if amount_sum >= total_target {
202
+ println ! ( "amount_sum >= total_target" ) ;
200
203
shift = true ;
201
204
if weight_sum < best_weight_sum || weight_sum == best_weight_sum && amount_sum < best_amount_sum {
202
205
best_selection = selection. clone ( ) ;
203
206
best_weight_sum = weight_sum;
204
207
best_amount_sum = amount_sum;
205
208
}
209
+ } else if !best_selection. is_empty ( ) && weight_sum + min_tail_weight[ tail] * ( ( total_target. to_sat ( ) - amount_sum. to_sat ( ) + w_utxos[ tail] . 0 . to_sat ( ) - 1 ) / w_utxos[ tail] . 0 . to_sat ( ) ) > best_weight_sum {
210
+ //println!("num: {:?}", total_target.to_sat() - amount_sum.to_sat() + w_utxos[tail].0.to_sat() - 1);
211
+ //println!("denom: {:?}", w_utxos[tail].0.to_sat());
212
+ //println!("eval: {:?}", min_tail_weight[tail].to_wu() * ((total_target.to_sat() - amount_sum.to_sat() + w_utxos[tail].0.to_sat() - 1)/ w_utxos[tail].0.to_sat()));
213
+ //println!("eval: {:?}", weight_sum + min_tail_weight[tail] * ((total_target.to_sat() - amount_sum.to_sat() + w_utxos[tail].0.to_sat() - 1)/ w_utxos[tail].0.to_sat()));
214
+
215
+ println ! ( "weight sum: {:?}" , weight_sum. to_wu( ) ) ;
216
+ println ! ( "min tail weight: {:?}" , min_tail_weight[ tail] . to_wu( ) ) ;
217
+ println ! ( "total target {:?}: " , total_target) ;
218
+ println ! ( "amount sum: {:?}" , amount_sum) ;
219
+ println ! ( "tail: {:?}" , w_utxos[ tail] . 0 . to_sat( ) ) ;
220
+ println ! ( "best_weight_sum: {:?}" , best_weight_sum) ;
221
+
222
+ if w_utxos[ tail] . 1 . weight ( ) <= min_tail_weight[ tail] {
223
+ println ! ( "cut" ) ;
224
+ cut = true ;
225
+ } else {
226
+ println ! ( "shift" ) ;
227
+ shift = true ;
228
+ }
206
229
}
207
230
208
231
// check if evaluating a leaf node.
@@ -216,7 +239,6 @@ pub fn select_coins<Utxo: WeightedUtxo>(
216
239
amount_sum -= eff_value;
217
240
weight_sum -= u. weight ( ) ;
218
241
selection. pop ( ) ;
219
-
220
242
shift = true ;
221
243
}
222
244
@@ -316,28 +338,6 @@ mod tests {
316
338
}
317
339
}
318
340
319
- #[ test]
320
- fn min_tail_weight ( ) {
321
- let weighted_utxos = vec ! [
322
- "10 sats/8" ,
323
- "7 sats/4" ,
324
- "5 sats/4" ,
325
- "4 sats/8"
326
- ] ;
327
-
328
- let utxos: Vec < _ > = build_utxos ( weighted_utxos) ;
329
- let eff_values: Vec < ( Amount , & Utxo ) > = calc_effective_values ( & utxos, FeeRate :: ZERO ) ;
330
- let min_tail_weight = build_min_tail_weight ( eff_values. clone ( ) ) ;
331
-
332
- let expect: Vec < Weight > = [
333
- 4u64 ,
334
- 4u64 ,
335
- 8u64 ,
336
- 18446744073709551615u64
337
- ] . iter ( ) . map ( |w| Weight :: from_wu ( * w) ) . collect ( ) ;
338
- assert_eq ! ( min_tail_weight, expect) ;
339
- }
340
-
341
341
#[ test]
342
342
fn lookahead ( ) {
343
343
let weighted_utxos = vec ! [
@@ -447,7 +447,7 @@ mod tests {
447
447
expected. push ( "0.33 BTC" ) ;
448
448
}
449
449
450
- assert_coin_select_params ( & params, 184 , Some ( & expected) ) ;
450
+ assert_coin_select_params ( & params, 37 , Some ( & expected) ) ;
451
451
}
452
452
453
453
#[ test]
@@ -499,13 +499,13 @@ mod tests {
499
499
weighted_utxos : coins
500
500
} ;
501
501
502
- assert_coin_select_params ( & params, 213 , Some ( & [ "14 BTC" , "13 BTC" , "4 BTC" ] ) ) ;
502
+ assert_coin_select_params ( & params, 92 , Some ( & [ "14 BTC" , "13 BTC" , "4 BTC" ] ) ) ;
503
503
}
504
504
505
505
#[ test]
506
- // 6) Test that the lightest solution among many clones is found
507
- // https://github.com/bitcoin/bitcoin/blob/43e71f74988b2ad87e4bfc0e1b5c921ab86ec176/src/wallet/test/coinselector_tests.cpp#L1244
508
- fn lightest_amount_many_clones ( ) {
506
+ fn lightest_amoung_many_clones ( ) {
507
+ // 6) Test that the lightest solution among many clones is found
508
+ // https://github.com/bitcoin/bitcoin/blob/43e71f74988b2ad87e4bfc0e1b5c921ab86ec176/src/wallet/test/coinselector_tests.cpp#L1244
509
509
let mut coins = vec ! [
510
510
"4 BTC/400" ,
511
511
"3 BTC/400" ,
@@ -521,7 +521,7 @@ mod tests {
521
521
}
522
522
523
523
let params = ParamsStr {
524
- target : "9.9 BTC " ,
524
+ target : "989999999 sats " ,
525
525
change_target : "1000000 sats" ,
526
526
max_weight : "400000" ,
527
527
fee_rate : "5" ,
@@ -535,6 +535,39 @@ mod tests {
535
535
"1 BTC"
536
536
] ;
537
537
538
- assert_coin_select_params ( & params, 31 , Some ( & expected) ) ;
538
+ assert_coin_select_params ( & params, 38 , Some ( & expected) ) ;
539
+ }
540
+
541
+ #[ test]
542
+ fn skip_tiny_inputs ( ) {
543
+ // 7) Test that lots of tiny UTXOs can be skipped if they are too heavy while there are enough funds in lookahead
544
+ // https://github.com/bitcoin/bitcoin/blob/43e71f74988b2ad87e4bfc0e1b5c921ab86ec176/src/wallet/test/coinselector_tests.cpp#L1153
545
+ let mut coins = vec ! [
546
+ "1.8 BTC/10000" ,
547
+ "1 BTC/4000" ,
548
+ "1 BTC/4000"
549
+ ] ;
550
+ let mut tiny_coins = vec ! [ ] ;
551
+ for i in 0 ..100 {
552
+ tiny_coins. push ( 0.01 * 100000000 as f64 + i as f64 ) ;
553
+ }
554
+ let tiny_coins: Vec < String > = tiny_coins. iter ( ) . map ( |a| format ! ( "{} sats/440" , a) ) . collect ( ) ;
555
+ let mut tiny_coins: Vec < & str > = tiny_coins. iter ( ) . map ( |s| s as & str ) . collect ( ) ;
556
+ coins. append ( & mut tiny_coins) ;
557
+
558
+ let params = ParamsStr {
559
+ target : "1.9 BTC" ,
560
+ change_target : "1000000 sats" ,
561
+ max_weight : "400000" ,
562
+ fee_rate : "5" ,
563
+ weighted_utxos : coins
564
+ } ;
565
+
566
+ let expected = vec ! [
567
+ "1 BTC" ,
568
+ "1 BTC"
569
+ ] ;
570
+
571
+ assert_coin_select_params ( & params, 7 , Some ( & expected) ) ;
539
572
}
540
573
}
0 commit comments