Skip to content

Commit e8e35c5

Browse files
committed
Add test seven
1 parent 4999462 commit e8e35c5

File tree

1 file changed

+83
-50
lines changed

1 file changed

+83
-50
lines changed

src/coin_grinder.rs

+83-50
Original file line numberDiff line numberDiff line change
@@ -45,24 +45,10 @@ fn calc_effective_values<Utxo: WeightedUtxo>(weighted_utxos: &[Utxo], fee_rate:
4545
.collect()
4646
}
4747

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.
5049
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()]
6652
}
6753

6854
fn index_to_utxo_list<Utxo: WeightedUtxo>(
@@ -138,11 +124,16 @@ pub fn select_coins<Utxo: WeightedUtxo>(
138124
});
139125

140126
let lookahead = build_lookahead(w_utxos.clone(), available_value);
141-
142-
//let min_group_weight = w_utxos.clone();
143127
let min_tail_weight = build_min_tail_weight(w_utxos.clone());
144128

129+
for i in &min_tail_weight {
130+
println!("{:?}", i);
131+
}
132+
145133
let total_target = target + change_target;
134+
println!("total_target: {:?}", total_target);
135+
println!("target: {:?}", target);
136+
println!("change_target: {:?}", change_target);
146137

147138
if available_value < total_target {
148139
return None
@@ -166,6 +157,10 @@ pub fn select_coins<Utxo: WeightedUtxo>(
166157

167158
// EXPLORE
168159
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);
169164

170165
amount_sum += eff_value;
171166
weight_sum += u.weight();
@@ -175,12 +170,19 @@ pub fn select_coins<Utxo: WeightedUtxo>(
175170
iteration += 1;
176171

177172
let tail: usize = *selection.last().unwrap();
178-
179173
// no possibility of hitting the total along this branch.
180174
// 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);
181181
if amount_sum + lookahead[tail] < total_target {
182+
println!("amount_sum + lookahead[tail] < total_target");
182183
cut = true;
183184
} else if weight_sum > best_weight_sum {
185+
println!("weight_sum > best_weight_sum");
184186
// check if a better solution could exist. IE there exists a utxo with a better
185187
// weight along the current branch
186188
if w_utxos[tail].1.weight() <= min_tail_weight[tail] {
@@ -197,12 +199,33 @@ pub fn select_coins<Utxo: WeightedUtxo>(
197199
shift = true;
198200
}
199201
} else if amount_sum >= total_target {
202+
println!("amount_sum >= total_target");
200203
shift = true;
201204
if weight_sum < best_weight_sum || weight_sum == best_weight_sum && amount_sum < best_amount_sum {
202205
best_selection = selection.clone();
203206
best_weight_sum = weight_sum;
204207
best_amount_sum = amount_sum;
205208
}
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+
}
206229
}
207230

208231
// check if evaluating a leaf node.
@@ -216,7 +239,6 @@ pub fn select_coins<Utxo: WeightedUtxo>(
216239
amount_sum -= eff_value;
217240
weight_sum -= u.weight();
218241
selection.pop();
219-
220242
shift = true;
221243
}
222244

@@ -316,28 +338,6 @@ mod tests {
316338
}
317339
}
318340

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-
341341
#[test]
342342
fn lookahead() {
343343
let weighted_utxos = vec![
@@ -447,7 +447,7 @@ mod tests {
447447
expected.push("0.33 BTC");
448448
}
449449

450-
assert_coin_select_params(&params, 184, Some(&expected));
450+
assert_coin_select_params(&params, 37, Some(&expected));
451451
}
452452

453453
#[test]
@@ -499,13 +499,13 @@ mod tests {
499499
weighted_utxos: coins
500500
};
501501

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"]));
503503
}
504504

505505
#[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
509509
let mut coins = vec![
510510
"4 BTC/400",
511511
"3 BTC/400",
@@ -521,7 +521,7 @@ mod tests {
521521
}
522522

523523
let params = ParamsStr {
524-
target: "9.9 BTC",
524+
target: "989999999 sats",
525525
change_target: "1000000 sats",
526526
max_weight: "400000",
527527
fee_rate: "5",
@@ -535,6 +535,39 @@ mod tests {
535535
"1 BTC"
536536
];
537537

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));
539572
}
540573
}

0 commit comments

Comments
 (0)