From bbae7f1e5be6817aea88e64c6d8247b5acff2d5f Mon Sep 17 00:00:00 2001 From: yancy Date: Fri, 29 Dec 2023 17:24:10 +0100 Subject: [PATCH] Move branch and bound to a seperate module --- src/branch_and_bound.rs | 538 ++++++++++++++++++++++++++++++++++++++++ src/lib.rs | 536 +-------------------------------------- 2 files changed, 540 insertions(+), 534 deletions(-) create mode 100644 src/branch_and_bound.rs diff --git a/src/branch_and_bound.rs b/src/branch_and_bound.rs new file mode 100644 index 0000000..dcbaa6a --- /dev/null +++ b/src/branch_and_bound.rs @@ -0,0 +1,538 @@ +use bitcoin::Amount; + +use crate::WeightedUtxo; +use bitcoin::amount::CheckedSum; +use bitcoin::blockdata::transaction::effective_value; +use bitcoin::FeeRate; + +fn index_to_utxo_list( + index_list: Option>, + weighted_utxos: &mut [WeightedUtxo], +) -> Option> { + index_list.map(|i_list| i_list.iter().map(|i: &usize| weighted_utxos[*i].clone()).collect()) +} + +/// Select coins bnb performs a depth first branch and bound search on binary tree. +/// +/// See also core: +/// +/// Returns a collection of `WeightedUtxo` that meet or exceed the target `Amount` when summed. +/// The collection returned seeks to minimize the waste, which is the difference between the target +/// `Amount` and sum of the `WeightedUtxo` collection. If no match can be found, an empty +/// collection is returned wrapped by the option type. However, if an un-expected error was +/// encountered, `None` is returned. +/// +/// # Returns +/// * Some(Vec) where Vec is not empty on match. +/// * Some(vec![]) if no match could be fouund. +/// * None if some un-expected behavior occurred such as an overflow. +/// +/// # Arguments +/// * `target` - Target spend `Amount` +/// * `cost_of_change` - The `Amount` needed to produce a change output +/// * `fee_rate` - `FeeRate` used to calculate each effective_value output value +/// * `weighted_utxos` - The candidate Weighted UTXOs from which to choose a selection from + +// This search can be thought of as exploring a binary tree where the left branch is the inclusion +// of a node and the right branch is the exclusion. For example, if the utxo set consist of a +// list of utxos: [4,3,2,1], and the target is 5, the selection process works as follows: +// +// Start at 4 and try including 4 in the total the first loop. We therefore have a tree with only +// one root node that is less than the total, so the next iteration occurs. The second iteration +// examines a tree where 4 is the root and the left branch is 3. +// o +// / +// 4 +// / +// 3 +// +// At this point, the total is determined to be 7 which exceeds the target of 5. We therefore +// remove 3 from the left branch and it becomes the right branch since 3 is now excluded +// (backtrack). +// o +// / +// 4 +// / \ +// 3 +// +// We next try including 2 on the left branch of 3 (add 2 to the inclusion branch). +// o +// / +// 4 +// / \ +// 3 +// / +// 2 +// +// The sum is now 6, since the sum of the right branch totals 6. Once again, we find the total +// exceeds 5, so we explore the exclusion branch of 2. +// o +// / +// 4 +// / \ +// 3 +// / \ +// 2 +// +// Finally, we add 1 to the inclusion branch. This ends our depth first search by matching two +// conditions, it is both the leaf node (end of the list) and matches our search criteria of +// matching 5. Both 4 and 1 are on the left inclusion branch. We therefore record our solution +// and backtrack to next try the exclusion branch of our root node 4. +// o +// / \ +// 4 +// / \ +// 3 +// / \ +// 2 +// / +// 1 +// +// We try excluding 4 now +// o +// / \ +// 4 +// / \ +// 3 +// +// 3 is less than our target, so we next add 2 to our inclusion branch +// o +// / \ +// 4 +// / \ +// 3 +// / +// 2 +// +// We now stop our search again noticing that 3 and 2 equals our target as 5, and since this +// solution was found last, then [3, 2] overwrites the previously found solution [4, 1]. We next +// backtrack and exclude our root node of this sub tree 3. Since our new sub tree starting at 2 +// doesn't have enough value left to meet the target, we conclude our search at [3, 2]. +pub fn select_coins_bnb( + target: Amount, + cost_of_change: Amount, + fee_rate: FeeRate, + weighted_utxos: &mut [WeightedUtxo], +) -> Option> { + // Total_Tries in Core: + // https://github.com/bitcoin/bitcoin/blob/1d9da8da309d1dbf9aef15eb8dc43b4a2dc3d309/src/wallet/coinselection.cpp#L74 + const ITERATION_LIMIT: i32 = 100_000; + + let mut iteration = 0; + let mut index = 0; + let mut backtrack; + let mut backtrack_subtree; + + let mut value = Amount::ZERO; + let mut waste = Amount::MAX_MONEY; + + let mut index_selection: Vec = vec![]; + let mut best_selection: Option> = None; + let mut utxo_candidate_amounts: Vec = vec![]; + + for u in &mut *weighted_utxos { + let effective_value = effective_value(fee_rate, u.satisfaction_weight, u.utxo.value)?; + + // Discard negative effective values. + let amount = match effective_value.to_unsigned() { + Ok(amt) => amt, + Err(_) => continue, + }; + + utxo_candidate_amounts.push(amount); + } + + let mut available_value = utxo_candidate_amounts.iter().cloned().checked_sum()?; + + if available_value < target { + return Some(Vec::new()); + } + + utxo_candidate_amounts.sort(); + utxo_candidate_amounts.reverse(); + + while iteration < ITERATION_LIMIT { + // There are two conditions for backtracking: + // + // 1_ Not enough value to make it to target. + // This condition happens before reaching a leaf node. + // Looking for a leaf node condition should not make a difference. + // This backtrack removes more than one node and instead starts + // the exploration of a new subtree. + // + // From: + // o + // / \ + // 4 + // / \ + // 3 + // / \ + // 2 + // / + // 1 + // + // To: + // o + // / \ + // 4 + // / \ + // 3 + // + // + // 2_ value meets or exceeded target. + // In this condition, we only backtrack one node + // + // From: + // o + // / + // 4 + // / + // 3 + // + // To: + // o + // / + // 4 + // / \ + // 3 + + // Set initial loop state + backtrack = false; + backtrack_subtree = false; + + // * not enough value to make it to the target. + // Therefore, explore a new new subtree. + if available_value + value < target { + backtrack_subtree = true; + } + // This optimization provides an upper bound on the amount of waste that is acceptable. + // Since value is lost when we create a change output due to increasing the size of the + // transaction by an output, we accept solutions that may be larger than the target as + // if they are exactly equal to the target and consider the overage waste or a throw + // away amount. However we do not consider values greater than value + cost_of_change. + // + // This effectively creates a range of possible solution where; + // range = (target, target + cost_of_change] + // + // That is, the range includes solutions that exactly equal the target up to but not + // including values greater than target + cost_of_change. + else if value > target + cost_of_change { + backtrack = true; + } + // * value meets or exceeds the target. + // Record the solution and the waste then continue. + // + // Check if index_selection is better than the previous known best, and + // update best_selection accordingly. + else if value >= target { + backtrack = true; + + let current_waste = value - target; + + if current_waste <= waste { + best_selection = Some(index_selection.clone()); + waste = current_waste; + } + } + + // * Backtrack + if backtrack { + let last_index = index_selection.pop().unwrap(); + value -= utxo_candidate_amounts[last_index]; + index -= 1; + assert_eq!(index, last_index); + } + // * Backtrack to new tree + else if backtrack_subtree { + // No new subtree left to explore. + if index_selection.is_empty() { + return index_to_utxo_list(best_selection, weighted_utxos); + } + + // Anchor the new subtree at the next available index. + // The next iteration, the index will be incremented by one. + index = index_selection.remove(0); + + // The available value of the next iteration. + available_value = utxo_candidate_amounts[index + 1..].iter().cloned().checked_sum()?; + + // If the new subtree does not have enough value, we are done searching. + if available_value < target { + return index_to_utxo_list(best_selection, weighted_utxos); + } + + // Start a new selection and add the root of the new subtree to the index selection. + index_selection.clear(); + value = Amount::ZERO; + } + // * Add next node to the inclusion branch. + else { + let utxo_value = utxo_candidate_amounts[index]; + + index_selection.push(index); + value += utxo_value; + available_value -= utxo_value; + } + + index += 1; + iteration += 1; + } + + index_to_utxo_list(best_selection, weighted_utxos) +} + +#[cfg(test)] +mod tests { + use super::*; + use crate::WeightedUtxo; + use bitcoin::Amount; + use bitcoin::ScriptBuf; + use bitcoin::TxOut; + use bitcoin::Weight; + use core::str::FromStr; + + fn create_weighted_utxos() -> Vec { + let amts = [ + Amount::from_str("1 cBTC").unwrap(), + Amount::from_str("2 cBTC").unwrap(), + Amount::from_str("3 cBTC").unwrap(), + Amount::from_str("4 cBTC").unwrap(), + ]; + + amts.iter() + .map(|amt| WeightedUtxo { + satisfaction_weight: Weight::ZERO, + utxo: TxOut { value: *amt, script_pubkey: ScriptBuf::new() }, + }) + .collect() + } + + fn expected_list( + index_list: Vec, + weighted_utxos: &mut [WeightedUtxo], + ) -> Vec { + index_to_utxo_list(Some(index_list), weighted_utxos).unwrap() + } + + #[test] + fn select_coins_bnb_one() { + let target = Amount::from_str("1 cBTC").unwrap(); + let mut weighted_utxos = create_weighted_utxos(); + let expected_i_list = vec![3]; + + let list = + select_coins_bnb(target, Amount::ZERO, FeeRate::ZERO, &mut weighted_utxos).unwrap(); + assert_eq!(list, expected_list(expected_i_list, &mut weighted_utxos)); + } + + #[test] + fn select_coins_bnb_two() { + let target = Amount::from_str("2 cBTC").unwrap(); + let mut weighted_utxos = create_weighted_utxos(); + let expected_i_list = vec![2]; + + let list = + select_coins_bnb(target, Amount::ZERO, FeeRate::ZERO, &mut weighted_utxos).unwrap(); + assert_eq!(list, expected_list(expected_i_list, &mut weighted_utxos)); + } + + #[test] + fn select_coins_bnb_three() { + let target = Amount::from_str("3 cBTC").unwrap(); + let mut weighted_utxos = create_weighted_utxos(); + let expected_i_list = vec![2, 3]; + + let list = + select_coins_bnb(target, Amount::ZERO, FeeRate::ZERO, &mut weighted_utxos).unwrap(); + assert_eq!(list, expected_list(expected_i_list, &mut weighted_utxos)); + } + + #[test] + fn select_coins_bnb_four() { + let target = Amount::from_str("4 cBTC").unwrap(); + let mut weighted_utxos = create_weighted_utxos(); + let expected_i_list = vec![1, 3]; + + let list = + select_coins_bnb(target, Amount::ZERO, FeeRate::ZERO, &mut weighted_utxos).unwrap(); + assert_eq!(list, expected_list(expected_i_list, &mut weighted_utxos)); + } + + #[test] + fn select_coins_bnb_five() { + let target = Amount::from_str("5 cBTC").unwrap(); + let mut weighted_utxos = create_weighted_utxos(); + let expected_i_list = vec![1, 2]; + + let list = + select_coins_bnb(target, Amount::ZERO, FeeRate::ZERO, &mut weighted_utxos).unwrap(); + assert_eq!(list, expected_list(expected_i_list, &mut weighted_utxos)); + } + + #[test] + fn select_coins_bnb_six() { + let target = Amount::from_str("6 cBTC").unwrap(); + let mut weighted_utxos = create_weighted_utxos(); + let expected_i_list = vec![1, 2, 3]; + + let list = + select_coins_bnb(target, Amount::ZERO, FeeRate::ZERO, &mut weighted_utxos).unwrap(); + assert_eq!(list, expected_list(expected_i_list, &mut weighted_utxos)); + } + + #[test] + fn select_coins_bnb_seven() { + let target = Amount::from_str("7 cBTC").unwrap(); + let mut weighted_utxos = create_weighted_utxos(); + let expected_i_list = vec![0, 2, 3]; + + let list = + select_coins_bnb(target, Amount::ZERO, FeeRate::ZERO, &mut weighted_utxos).unwrap(); + assert_eq!(list, expected_list(expected_i_list, &mut weighted_utxos)); + } + + #[test] + fn select_coins_bnb_eight() { + let target = Amount::from_str("8 cBTC").unwrap(); + let mut weighted_utxos = create_weighted_utxos(); + let expected_i_list = vec![0, 1, 3]; + + let list = + select_coins_bnb(target, Amount::ZERO, FeeRate::ZERO, &mut weighted_utxos).unwrap(); + assert_eq!(list, expected_list(expected_i_list, &mut weighted_utxos)); + } + + #[test] + fn select_coins_bnb_nine() { + let target = Amount::from_str("9 cBTC").unwrap(); + let mut weighted_utxos = create_weighted_utxos(); + let expected_i_list = vec![0, 1, 2]; + + let list = + select_coins_bnb(target, Amount::ZERO, FeeRate::ZERO, &mut weighted_utxos).unwrap(); + assert_eq!(list, expected_list(expected_i_list, &mut weighted_utxos)); + } + + #[test] + fn select_coins_bnb_ten() { + let target = Amount::from_str("10 cBTC").unwrap(); + let mut weighted_utxos = create_weighted_utxos(); + let expected_i_list = vec![0, 1, 2, 3]; + + let list = + select_coins_bnb(target, Amount::ZERO, FeeRate::ZERO, &mut weighted_utxos).unwrap(); + assert_eq!(list, expected_list(expected_i_list, &mut weighted_utxos)); + } + + #[test] + fn select_coins_bnb_cost_of_change() { + let target = Amount::from_str("1 cBTC").unwrap(); + + // Since cost of change here is one, we accept any solution + // between 1 and 2. Range = (1, 2] + let cost_of_change = target; + let expected_i_list = vec![0]; + + let mut weighted_utxos = vec![WeightedUtxo { + satisfaction_weight: Weight::ZERO, + utxo: TxOut { + value: Amount::from_str("1.5 cBTC").unwrap(), + script_pubkey: ScriptBuf::new(), + }, + }]; + + let list = + select_coins_bnb(target, cost_of_change, FeeRate::ZERO, &mut weighted_utxos.clone()) + .unwrap(); + assert_eq!(list, expected_list(expected_i_list, &mut weighted_utxos)); + + let index_list = select_coins_bnb(target, Amount::ZERO, FeeRate::ZERO, &mut weighted_utxos); + assert_eq!(index_list, None); + } + + #[test] + fn select_coins_bnb_effective_value() { + let target = Amount::from_str("1 cBTC").unwrap(); + let fee_rate = FeeRate::from_sat_per_kwu(10); + let satisfaction_weight = Weight::from_wu(204); + + let weighted_utxos = vec![WeightedUtxo { + satisfaction_weight, + utxo: TxOut { + // This would be a match using value, however since effective_value is used + // the effective_value is calculated, this will fall short of the target. + value: Amount::from_str("1 cBTC").unwrap(), + script_pubkey: ScriptBuf::new(), + }, + }]; + + let index_list = + select_coins_bnb(target, Amount::ZERO, fee_rate, &mut weighted_utxos.clone()).unwrap(); + assert_eq!(index_list, Vec::new()); + } + + #[test] + fn select_coins_bnb_negative_effective_value() { + let target = Amount::from_str("1 cBTC").unwrap(); + let fee_rate = FeeRate::from_sat_per_kwu(10); + let satisfaction_weight = Weight::from_wu(204); + let expected_i_list = vec![0]; + + // Since cost of change here is one, we accept any solution + // between 1 and 2. Range = (1, 2] + let cost_of_change = target; + + let mut weighted_utxos = vec![ + WeightedUtxo { + satisfaction_weight: Weight::ZERO, + utxo: TxOut { + value: Amount::from_str("1.5 cBTC").unwrap(), + script_pubkey: ScriptBuf::new(), + }, + }, + WeightedUtxo { + satisfaction_weight, + utxo: TxOut { + // If this had no fee, a 1 sat utxo would be included since + // there would be less waste. However, since there is a weight + // and fee to spend it, the effective value is negative, so + // it will not be included. + value: Amount::from_str("1 sat").unwrap(), + script_pubkey: ScriptBuf::new(), + }, + }, + ]; + + let list = select_coins_bnb(target, cost_of_change, fee_rate, &mut weighted_utxos.clone()) + .unwrap(); + assert_eq!(list, expected_list(expected_i_list, &mut weighted_utxos)); + } + + #[test] + fn select_coins_bnb_target_greater_than_value() { + let target = Amount::from_str("11 cBTC").unwrap(); + let mut weighted_utxos = create_weighted_utxos(); + let list = + select_coins_bnb(target, Amount::ZERO, FeeRate::ZERO, &mut weighted_utxos).unwrap(); + assert_eq!(list, Vec::new()); + } + + #[test] + fn select_coins_bnb_utxo_pool_sum_overflow() { + let target = Amount::from_str("1 cBTC").unwrap(); + let mut weighted_utxos = vec![ + WeightedUtxo { + satisfaction_weight: Weight::ZERO, + utxo: TxOut { value: Amount::MAX, script_pubkey: ScriptBuf::new() }, + }, + WeightedUtxo { + satisfaction_weight: Weight::ZERO, + utxo: TxOut { value: Amount::MAX, script_pubkey: ScriptBuf::new() }, + }, + ]; + + let list = select_coins_bnb(target, Amount::ZERO, FeeRate::ZERO, &mut weighted_utxos); + + assert!(list.is_none()); + } +} diff --git a/src/lib.rs b/src/lib.rs index 63c0358..c700cbf 100644 --- a/src/lib.rs +++ b/src/lib.rs @@ -16,15 +16,15 @@ #[cfg(bench)] extern crate test; +mod branch_and_bound; mod single_random_draw; -use bitcoin::amount::CheckedSum; -use bitcoin::blockdata::transaction::effective_value; use bitcoin::Amount; use bitcoin::FeeRate; use bitcoin::TxOut; use bitcoin::Weight; +use crate::branch_and_bound::select_coins_bnb; use crate::single_random_draw::select_coins_srd; use rand::thread_rng; @@ -73,538 +73,6 @@ pub fn select_coins( } } -fn index_to_utxo_list( - index_list: Option>, - weighted_utxos: &mut [WeightedUtxo], -) -> Option> { - index_list.map(|i_list| i_list.iter().map(|i: &usize| weighted_utxos[*i].clone()).collect()) -} - -/// Select coins bnb performs a depth first branch and bound search on binary tree. -/// -/// See also core: -/// -/// Returns a collection of `WeightedUtxo` that meet or exceed the target `Amount` when summed. -/// The collection returned seeks to minimize the waste, which is the difference between the target -/// `Amount` and sum of the `WeightedUtxo` collection. If no match can be found, an empty -/// collection is returned wrapped by the option type. However, if an un-expected error was -/// encountered, `None` is returned. -/// -/// # Returns -/// * Some(Vec) where Vec is not empty on match. -/// * Some(vec![]) if no match could be fouund. -/// * None if some un-expected behavior occurred such as an overflow. -/// -/// # Arguments -/// * `target` - Target spend `Amount` -/// * `cost_of_change` - The `Amount` needed to produce a change output -/// * `fee_rate` - `FeeRate` used to calculate each effective_value output value -/// * `weighted_utxos` - The candidate Weighted UTXOs from which to choose a selection from - -// This search can be thought of as exploring a binary tree where the left branch is the inclusion -// of a node and the right branch is the exclusion. For example, if the utxo set consist of a -// list of utxos: [4,3,2,1], and the target is 5, the selection process works as follows: -// -// Start at 4 and try including 4 in the total the first loop. We therefore have a tree with only -// one root node that is less than the total, so the next iteration occurs. The second iteration -// examines a tree where 4 is the root and the left branch is 3. -// o -// / -// 4 -// / -// 3 -// -// At this point, the total is determined to be 7 which exceeds the target of 5. We therefore -// remove 3 from the left branch and it becomes the right branch since 3 is now excluded -// (backtrack). -// o -// / -// 4 -// / \ -// 3 -// -// We next try including 2 on the left branch of 3 (add 2 to the inclusion branch). -// o -// / -// 4 -// / \ -// 3 -// / -// 2 -// -// The sum is now 6, since the sum of the right branch totals 6. Once again, we find the total -// exceeds 5, so we explore the exclusion branch of 2. -// o -// / -// 4 -// / \ -// 3 -// / \ -// 2 -// -// Finally, we add 1 to the inclusion branch. This ends our depth first search by matching two -// conditions, it is both the leaf node (end of the list) and matches our search criteria of -// matching 5. Both 4 and 1 are on the left inclusion branch. We therefore record our solution -// and backtrack to next try the exclusion branch of our root node 4. -// o -// / \ -// 4 -// / \ -// 3 -// / \ -// 2 -// / -// 1 -// -// We try excluding 4 now -// o -// / \ -// 4 -// / \ -// 3 -// -// 3 is less than our target, so we next add 2 to our inclusion branch -// o -// / \ -// 4 -// / \ -// 3 -// / -// 2 -// -// We now stop our search again noticing that 3 and 2 equals our target as 5, and since this -// solution was found last, then [3, 2] overwrites the previously found solution [4, 1]. We next -// backtrack and exclude our root node of this sub tree 3. Since our new sub tree starting at 2 -// doesn't have enough value left to meet the target, we conclude our search at [3, 2]. -pub fn select_coins_bnb( - target: Amount, - cost_of_change: Amount, - fee_rate: FeeRate, - weighted_utxos: &mut [WeightedUtxo], -) -> Option> { - // Total_Tries in Core: - // https://github.com/bitcoin/bitcoin/blob/1d9da8da309d1dbf9aef15eb8dc43b4a2dc3d309/src/wallet/coinselection.cpp#L74 - const ITERATION_LIMIT: i32 = 100_000; - - let mut iteration = 0; - let mut index = 0; - let mut backtrack; - let mut backtrack_subtree; - - let mut value = Amount::ZERO; - let mut waste = Amount::MAX_MONEY; - - let mut index_selection: Vec = vec![]; - let mut best_selection: Option> = None; - let mut utxo_candidate_amounts: Vec = vec![]; - - for u in &mut *weighted_utxos { - let effective_value = effective_value(fee_rate, u.satisfaction_weight, u.utxo.value)?; - - // Discard negative effective values. - let amount = match effective_value.to_unsigned() { - Ok(amt) => amt, - Err(_) => continue, - }; - - utxo_candidate_amounts.push(amount); - } - - let mut available_value = utxo_candidate_amounts.iter().cloned().checked_sum()?; - - if available_value < target { - return Some(Vec::new()); - } - - utxo_candidate_amounts.sort(); - utxo_candidate_amounts.reverse(); - - while iteration < ITERATION_LIMIT { - // There are two conditions for backtracking: - // - // 1_ Not enough value to make it to target. - // This condition happens before reaching a leaf node. - // Looking for a leaf node condition should not make a difference. - // This backtrack removes more than one node and instead starts - // the exploration of a new subtree. - // - // From: - // o - // / \ - // 4 - // / \ - // 3 - // / \ - // 2 - // / - // 1 - // - // To: - // o - // / \ - // 4 - // / \ - // 3 - // - // - // 2_ value meets or exceeded target. - // In this condition, we only backtrack one node - // - // From: - // o - // / - // 4 - // / - // 3 - // - // To: - // o - // / - // 4 - // / \ - // 3 - - // Set initial loop state - backtrack = false; - backtrack_subtree = false; - - // * not enough value to make it to the target. - // Therefore, explore a new new subtree. - if available_value + value < target { - backtrack_subtree = true; - } - // This optimization provides an upper bound on the amount of waste that is acceptable. - // Since value is lost when we create a change output due to increasing the size of the - // transaction by an output, we accept solutions that may be larger than the target as - // if they are exactly equal to the target and consider the overage waste or a throw - // away amount. However we do not consider values greater than value + cost_of_change. - // - // This effectively creates a range of possible solution where; - // range = (target, target + cost_of_change] - // - // That is, the range includes solutions that exactly equal the target up to but not - // including values greater than target + cost_of_change. - else if value > target + cost_of_change { - backtrack = true; - } - // * value meets or exceeds the target. - // Record the solution and the waste then continue. - // - // Check if index_selection is better than the previous known best, and - // update best_selection accordingly. - else if value >= target { - backtrack = true; - - let current_waste = value - target; - - if current_waste <= waste { - best_selection = Some(index_selection.clone()); - waste = current_waste; - } - } - - // * Backtrack - if backtrack { - let last_index = index_selection.pop().unwrap(); - value -= utxo_candidate_amounts[last_index]; - index -= 1; - assert_eq!(index, last_index); - } - // * Backtrack to new tree - else if backtrack_subtree { - // No new subtree left to explore. - if index_selection.is_empty() { - return index_to_utxo_list(best_selection, weighted_utxos); - } - - // Anchor the new subtree at the next available index. - // The next iteration, the index will be incremented by one. - index = index_selection.remove(0); - - // The available value of the next iteration. - available_value = utxo_candidate_amounts[index + 1..].iter().cloned().checked_sum()?; - - // If the new subtree does not have enough value, we are done searching. - if available_value < target { - return index_to_utxo_list(best_selection, weighted_utxos); - } - - // Start a new selection and add the root of the new subtree to the index selection. - index_selection.clear(); - value = Amount::ZERO; - } - // * Add next node to the inclusion branch. - else { - let utxo_value = utxo_candidate_amounts[index]; - - index_selection.push(index); - value += utxo_value; - available_value -= utxo_value; - } - - index += 1; - iteration += 1; - } - - index_to_utxo_list(best_selection, weighted_utxos) -} - -#[cfg(test)] -mod tests { - use super::*; - use crate::WeightedUtxo; - use bitcoin::Amount; - use bitcoin::ScriptBuf; - use bitcoin::TxOut; - use bitcoin::Weight; - use core::str::FromStr; - - fn create_weighted_utxos() -> Vec { - let amts = [ - Amount::from_str("1 cBTC").unwrap(), - Amount::from_str("2 cBTC").unwrap(), - Amount::from_str("3 cBTC").unwrap(), - Amount::from_str("4 cBTC").unwrap(), - ]; - - amts.iter() - .map(|amt| WeightedUtxo { - satisfaction_weight: Weight::ZERO, - utxo: TxOut { value: *amt, script_pubkey: ScriptBuf::new() }, - }) - .collect() - } - - fn expected_list( - index_list: Vec, - weighted_utxos: &mut [WeightedUtxo], - ) -> Vec { - index_to_utxo_list(Some(index_list), weighted_utxos).unwrap() - } - - #[test] - fn one() { - let target = Amount::from_str("1 cBTC").unwrap(); - let mut weighted_utxos = create_weighted_utxos(); - let expected_i_list = vec![3]; - - let list = - select_coins_bnb(target, Amount::ZERO, FeeRate::ZERO, &mut weighted_utxos).unwrap(); - assert_eq!(list, expected_list(expected_i_list, &mut weighted_utxos)); - } - - #[test] - fn two() { - let target = Amount::from_str("2 cBTC").unwrap(); - let mut weighted_utxos = create_weighted_utxos(); - let expected_i_list = vec![2]; - - let list = - select_coins_bnb(target, Amount::ZERO, FeeRate::ZERO, &mut weighted_utxos).unwrap(); - assert_eq!(list, expected_list(expected_i_list, &mut weighted_utxos)); - } - - #[test] - fn three() { - let target = Amount::from_str("3 cBTC").unwrap(); - let mut weighted_utxos = create_weighted_utxos(); - let expected_i_list = vec![2, 3]; - - let list = - select_coins_bnb(target, Amount::ZERO, FeeRate::ZERO, &mut weighted_utxos).unwrap(); - assert_eq!(list, expected_list(expected_i_list, &mut weighted_utxos)); - } - - #[test] - fn four() { - let target = Amount::from_str("4 cBTC").unwrap(); - let mut weighted_utxos = create_weighted_utxos(); - let expected_i_list = vec![1, 3]; - - let list = - select_coins_bnb(target, Amount::ZERO, FeeRate::ZERO, &mut weighted_utxos).unwrap(); - assert_eq!(list, expected_list(expected_i_list, &mut weighted_utxos)); - } - - #[test] - fn five() { - let target = Amount::from_str("5 cBTC").unwrap(); - let mut weighted_utxos = create_weighted_utxos(); - let expected_i_list = vec![1, 2]; - - let list = - select_coins_bnb(target, Amount::ZERO, FeeRate::ZERO, &mut weighted_utxos).unwrap(); - assert_eq!(list, expected_list(expected_i_list, &mut weighted_utxos)); - } - - #[test] - fn six() { - let target = Amount::from_str("6 cBTC").unwrap(); - let mut weighted_utxos = create_weighted_utxos(); - let expected_i_list = vec![1, 2, 3]; - - let list = - select_coins_bnb(target, Amount::ZERO, FeeRate::ZERO, &mut weighted_utxos).unwrap(); - assert_eq!(list, expected_list(expected_i_list, &mut weighted_utxos)); - } - - #[test] - fn seven() { - let target = Amount::from_str("7 cBTC").unwrap(); - let mut weighted_utxos = create_weighted_utxos(); - let expected_i_list = vec![0, 2, 3]; - - let list = - select_coins_bnb(target, Amount::ZERO, FeeRate::ZERO, &mut weighted_utxos).unwrap(); - assert_eq!(list, expected_list(expected_i_list, &mut weighted_utxos)); - } - - #[test] - fn eight() { - let target = Amount::from_str("8 cBTC").unwrap(); - let mut weighted_utxos = create_weighted_utxos(); - let expected_i_list = vec![0, 1, 3]; - - let list = - select_coins_bnb(target, Amount::ZERO, FeeRate::ZERO, &mut weighted_utxos).unwrap(); - assert_eq!(list, expected_list(expected_i_list, &mut weighted_utxos)); - } - - #[test] - fn nine() { - let target = Amount::from_str("9 cBTC").unwrap(); - let mut weighted_utxos = create_weighted_utxos(); - let expected_i_list = vec![0, 1, 2]; - - let list = - select_coins_bnb(target, Amount::ZERO, FeeRate::ZERO, &mut weighted_utxos).unwrap(); - assert_eq!(list, expected_list(expected_i_list, &mut weighted_utxos)); - } - - #[test] - fn ten() { - let target = Amount::from_str("10 cBTC").unwrap(); - let mut weighted_utxos = create_weighted_utxos(); - let expected_i_list = vec![0, 1, 2, 3]; - - let list = - select_coins_bnb(target, Amount::ZERO, FeeRate::ZERO, &mut weighted_utxos).unwrap(); - assert_eq!(list, expected_list(expected_i_list, &mut weighted_utxos)); - } - - #[test] - fn cost_of_change() { - let target = Amount::from_str("1 cBTC").unwrap(); - - // Since cost of change here is one, we accept any solution - // between 1 and 2. Range = (1, 2] - let cost_of_change = target; - let expected_i_list = vec![0]; - - let mut weighted_utxos = vec![WeightedUtxo { - satisfaction_weight: Weight::ZERO, - utxo: TxOut { - value: Amount::from_str("1.5 cBTC").unwrap(), - script_pubkey: ScriptBuf::new(), - }, - }]; - - let list = - select_coins_bnb(target, cost_of_change, FeeRate::ZERO, &mut weighted_utxos.clone()) - .unwrap(); - assert_eq!(list, expected_list(expected_i_list, &mut weighted_utxos)); - - let index_list = select_coins_bnb(target, Amount::ZERO, FeeRate::ZERO, &mut weighted_utxos); - assert_eq!(index_list, None); - } - - #[test] - fn effective_value() { - let target = Amount::from_str("1 cBTC").unwrap(); - let fee_rate = FeeRate::from_sat_per_kwu(10); - let satisfaction_weight = Weight::from_wu(204); - - let weighted_utxos = vec![WeightedUtxo { - satisfaction_weight, - utxo: TxOut { - // This would be a match using value, however since effective_value is used - // the effective_value is calculated, this will fall short of the target. - value: Amount::from_str("1 cBTC").unwrap(), - script_pubkey: ScriptBuf::new(), - }, - }]; - - let index_list = - select_coins_bnb(target, Amount::ZERO, fee_rate, &mut weighted_utxos.clone()).unwrap(); - assert_eq!(index_list, Vec::new()); - } - - #[test] - fn skip_effective_negative_effective_value() { - let target = Amount::from_str("1 cBTC").unwrap(); - let fee_rate = FeeRate::from_sat_per_kwu(10); - let satisfaction_weight = Weight::from_wu(204); - let expected_i_list = vec![0]; - - // Since cost of change here is one, we accept any solution - // between 1 and 2. Range = (1, 2] - let cost_of_change = target; - - let mut weighted_utxos = vec![ - WeightedUtxo { - satisfaction_weight: Weight::ZERO, - utxo: TxOut { - value: Amount::from_str("1.5 cBTC").unwrap(), - script_pubkey: ScriptBuf::new(), - }, - }, - WeightedUtxo { - satisfaction_weight, - utxo: TxOut { - // If this had no fee, a 1 sat utxo would be included since - // there would be less waste. However, since there is a weight - // and fee to spend it, the effective value is negative, so - // it will not be included. - value: Amount::from_str("1 sat").unwrap(), - script_pubkey: ScriptBuf::new(), - }, - }, - ]; - - let list = select_coins_bnb(target, cost_of_change, fee_rate, &mut weighted_utxos.clone()) - .unwrap(); - assert_eq!(list, expected_list(expected_i_list, &mut weighted_utxos)); - } - - #[test] - fn target_greater_than_value() { - let target = Amount::from_str("11 cBTC").unwrap(); - let mut weighted_utxos = create_weighted_utxos(); - let list = - select_coins_bnb(target, Amount::ZERO, FeeRate::ZERO, &mut weighted_utxos).unwrap(); - assert_eq!(list, Vec::new()); - } - - #[test] - fn utxo_pool_sum_overflow() { - let target = Amount::from_str("1 cBTC").unwrap(); - let mut weighted_utxos = vec![ - WeightedUtxo { - satisfaction_weight: Weight::ZERO, - utxo: TxOut { value: Amount::MAX, script_pubkey: ScriptBuf::new() }, - }, - WeightedUtxo { - satisfaction_weight: Weight::ZERO, - utxo: TxOut { value: Amount::MAX, script_pubkey: ScriptBuf::new() }, - }, - ]; - - let list = select_coins_bnb(target, Amount::ZERO, FeeRate::ZERO, &mut weighted_utxos); - - assert!(list.is_none()); - } -} - #[cfg(bench)] mod benches { use crate::select_coins_bnb;