2020-08-31 11:26:36 +02:00
|
|
|
// Magical Bitcoin Library
|
|
|
|
// Written in 2020 by
|
|
|
|
// Alekos Filini <alekos.filini@gmail.com>
|
|
|
|
//
|
|
|
|
// Copyright (c) 2020 Magical Bitcoin
|
|
|
|
//
|
|
|
|
// Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
|
|
// of this software and associated documentation files (the "Software"), to deal
|
|
|
|
// in the Software without restriction, including without limitation the rights
|
|
|
|
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
|
|
// copies of the Software, and to permit persons to whom the Software is
|
|
|
|
// furnished to do so, subject to the following conditions:
|
|
|
|
//
|
|
|
|
// The above copyright notice and this permission notice shall be included in all
|
|
|
|
// copies or substantial portions of the Software.
|
|
|
|
//
|
|
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
|
|
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
|
|
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
|
|
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
|
|
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
|
|
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
|
|
// SOFTWARE.
|
|
|
|
|
2020-09-04 11:44:49 +02:00
|
|
|
//! Coin selection
|
|
|
|
//!
|
|
|
|
//! This module provides the trait [`CoinSelectionAlgorithm`] that can be implemented to
|
|
|
|
//! define custom coin selection algorithms.
|
|
|
|
//!
|
|
|
|
//! The coin selection algorithm is not globally part of a [`Wallet`](super::Wallet), instead it
|
|
|
|
//! is selected whenever a [`Wallet::create_tx`](super::Wallet::create_tx) call is made, through
|
|
|
|
//! the use of the [`TxBuilder`] structure, specifically with
|
|
|
|
//! [`TxBuilder::coin_selection`](super::tx_builder::TxBuilder::coin_selection) method.
|
|
|
|
//!
|
|
|
|
//! The [`DefaultCoinSelectionAlgorithm`] selects the default coin selection algorithm that
|
|
|
|
//! [`TxBuilder`] uses, if it's not explicitly overridden.
|
|
|
|
//!
|
|
|
|
//! [`TxBuilder`]: super::tx_builder::TxBuilder
|
|
|
|
//!
|
|
|
|
//! ## Example
|
|
|
|
//!
|
|
|
|
//! ```no_run
|
|
|
|
//! # use std::str::FromStr;
|
|
|
|
//! # use bitcoin::*;
|
2020-09-14 14:25:38 +02:00
|
|
|
//! # use bdk::wallet::coin_selection::*;
|
2020-10-14 15:21:22 +02:00
|
|
|
//! # use bdk::database::Database;
|
2020-09-14 14:25:38 +02:00
|
|
|
//! # use bdk::*;
|
2020-09-04 11:44:49 +02:00
|
|
|
//! #[derive(Debug)]
|
|
|
|
//! struct AlwaysSpendEverything;
|
|
|
|
//!
|
2020-10-14 15:21:22 +02:00
|
|
|
//! impl<D: Database> CoinSelectionAlgorithm<D> for AlwaysSpendEverything {
|
2020-09-04 11:44:49 +02:00
|
|
|
//! fn coin_select(
|
|
|
|
//! &self,
|
2020-10-14 15:21:22 +02:00
|
|
|
//! database: &D,
|
2020-10-26 14:08:57 -04:00
|
|
|
//! required_utxos: Vec<(UTXO, usize)>,
|
2020-10-26 14:12:46 -04:00
|
|
|
//! optional_utxos: Vec<(UTXO, usize)>,
|
2020-09-04 11:44:49 +02:00
|
|
|
//! fee_rate: FeeRate,
|
|
|
|
//! amount_needed: u64,
|
|
|
|
//! fee_amount: f32,
|
2020-09-14 14:25:38 +02:00
|
|
|
//! ) -> Result<CoinSelectionResult, bdk::Error> {
|
2020-10-14 14:03:12 +11:00
|
|
|
//! let mut selected_amount = 0;
|
2020-10-14 15:21:22 +02:00
|
|
|
//! let mut additional_weight = 0;
|
2020-10-26 14:08:57 -04:00
|
|
|
//! let all_utxos_selected = required_utxos
|
2020-10-26 14:12:46 -04:00
|
|
|
//! .into_iter().chain(optional_utxos)
|
2020-10-14 15:21:22 +02:00
|
|
|
//! .scan((&mut selected_amount, &mut additional_weight), |(selected_amount, additional_weight), (utxo, weight)| {
|
|
|
|
//! let txin = TxIn {
|
|
|
|
//! previous_output: utxo.outpoint,
|
|
|
|
//! ..Default::default()
|
|
|
|
//! };
|
|
|
|
//!
|
2020-10-14 14:03:12 +11:00
|
|
|
//! **selected_amount += utxo.txout.value;
|
2020-11-08 15:46:27 +01:00
|
|
|
//! **additional_weight += TXIN_BASE_WEIGHT + weight;
|
2020-10-14 15:21:22 +02:00
|
|
|
//!
|
2020-10-14 14:03:12 +11:00
|
|
|
//! Some((
|
2020-10-14 15:21:22 +02:00
|
|
|
//! txin,
|
2020-09-04 11:44:49 +02:00
|
|
|
//! utxo.txout.script_pubkey,
|
2020-10-14 14:03:12 +11:00
|
|
|
//! ))
|
2020-09-04 11:44:49 +02:00
|
|
|
//! })
|
|
|
|
//! .collect::<Vec<_>>();
|
|
|
|
//! let additional_fees = additional_weight as f32 * fee_rate.as_sat_vb() / 4.0;
|
|
|
|
//!
|
|
|
|
//! if (fee_amount + additional_fees).ceil() as u64 + amount_needed > selected_amount {
|
2020-09-14 14:25:38 +02:00
|
|
|
//! return Err(bdk::Error::InsufficientFunds);
|
2020-09-04 11:44:49 +02:00
|
|
|
//! }
|
|
|
|
//!
|
|
|
|
//! Ok(CoinSelectionResult {
|
|
|
|
//! txin: all_utxos_selected,
|
|
|
|
//! selected_amount,
|
|
|
|
//! fee_amount: fee_amount + additional_fees,
|
|
|
|
//! })
|
|
|
|
//! }
|
|
|
|
//! }
|
|
|
|
//!
|
2020-09-14 14:25:38 +02:00
|
|
|
//! # let wallet: OfflineWallet<_> = Wallet::new_offline("", None, Network::Testnet, bdk::database::MemoryDatabase::default())?;
|
2020-09-04 11:44:49 +02:00
|
|
|
//! // create wallet, sync, ...
|
|
|
|
//!
|
|
|
|
//! let to_address = Address::from_str("2N4eQYCbKUHCCTUjBJeHcJp9ok6J2GZsTDt").unwrap();
|
|
|
|
//! let (psbt, details) = wallet.create_tx(
|
2020-09-04 15:45:11 +02:00
|
|
|
//! TxBuilder::with_recipients(vec![(to_address.script_pubkey(), 50_000)])
|
2020-09-04 11:44:49 +02:00
|
|
|
//! .coin_selection(AlwaysSpendEverything),
|
|
|
|
//! )?;
|
|
|
|
//!
|
|
|
|
//! // inspect, sign, broadcast, ...
|
|
|
|
//!
|
2020-09-14 14:25:38 +02:00
|
|
|
//! # Ok::<(), bdk::Error>(())
|
2020-09-04 11:44:49 +02:00
|
|
|
//! ```
|
|
|
|
|
2020-08-06 16:56:41 +02:00
|
|
|
use bitcoin::{Script, TxIn};
|
|
|
|
|
2020-10-14 15:21:22 +02:00
|
|
|
use crate::database::Database;
|
2020-08-06 16:56:41 +02:00
|
|
|
use crate::error::Error;
|
2020-08-31 10:49:44 +02:00
|
|
|
use crate::types::{FeeRate, UTXO};
|
2020-08-06 16:56:41 +02:00
|
|
|
|
2020-10-31 16:24:59 +01:00
|
|
|
use rand::seq::SliceRandom;
|
2020-10-31 16:27:33 +01:00
|
|
|
#[cfg(not(test))]
|
|
|
|
use rand::thread_rng;
|
|
|
|
#[cfg(test)]
|
|
|
|
use rand::{rngs::StdRng, SeedableRng};
|
2020-10-31 16:24:59 +01:00
|
|
|
|
2020-09-04 11:44:49 +02:00
|
|
|
/// Default coin selection algorithm used by [`TxBuilder`](super::tx_builder::TxBuilder) if not
|
|
|
|
/// overridden
|
2020-10-26 14:16:25 -04:00
|
|
|
pub type DefaultCoinSelectionAlgorithm = LargestFirstCoinSelection;
|
2020-08-06 16:56:41 +02:00
|
|
|
|
2020-09-04 11:44:49 +02:00
|
|
|
/// Result of a successful coin selection
|
2020-08-06 16:56:41 +02:00
|
|
|
#[derive(Debug)]
|
|
|
|
pub struct CoinSelectionResult {
|
2020-09-04 11:44:49 +02:00
|
|
|
/// List of inputs to use, with the respective previous script_pubkey
|
2020-08-06 16:56:41 +02:00
|
|
|
pub txin: Vec<(TxIn, Script)>,
|
2020-09-04 11:44:49 +02:00
|
|
|
/// Sum of the selected inputs' value
|
2020-08-31 10:49:44 +02:00
|
|
|
pub selected_amount: u64,
|
2020-09-04 11:44:49 +02:00
|
|
|
/// Total fee amount in satoshi
|
2020-08-06 16:56:41 +02:00
|
|
|
pub fee_amount: f32,
|
|
|
|
}
|
|
|
|
|
2020-09-04 11:44:49 +02:00
|
|
|
/// Trait for generalized coin selection algorithms
|
|
|
|
///
|
|
|
|
/// This trait can be implemented to make the [`Wallet`](super::Wallet) use a customized coin
|
|
|
|
/// selection algorithm when it creates transactions.
|
|
|
|
///
|
|
|
|
/// For an example see [this module](crate::wallet::coin_selection)'s documentation.
|
2020-10-14 15:21:22 +02:00
|
|
|
pub trait CoinSelectionAlgorithm<D: Database>: std::fmt::Debug {
|
2020-09-04 11:44:49 +02:00
|
|
|
/// Perform the coin selection
|
|
|
|
///
|
2020-10-14 15:21:22 +02:00
|
|
|
/// - `database`: a reference to the wallet's database that can be used to lookup additional
|
|
|
|
/// details for a specific UTXO
|
2020-10-26 14:08:57 -04:00
|
|
|
/// - `required_utxos`: the utxos that must be spent regardless of `amount_needed` with their
|
2020-10-14 15:21:22 +02:00
|
|
|
/// weight cost
|
2020-10-26 14:12:46 -04:00
|
|
|
/// - `optional_utxos`: the remaining available utxos to satisfy `amount_needed` with their
|
2020-10-14 15:21:22 +02:00
|
|
|
/// weight cost
|
2020-09-04 11:44:49 +02:00
|
|
|
/// - `fee_rate`: fee rate to use
|
|
|
|
/// - `amount_needed`: the amount in satoshi to select
|
2020-10-14 15:21:22 +02:00
|
|
|
/// - `fee_amount`: the amount of fees in satoshi already accumulated from adding outputs and
|
|
|
|
/// the transaction's header
|
2020-08-06 16:56:41 +02:00
|
|
|
fn coin_select(
|
|
|
|
&self,
|
2020-10-14 15:21:22 +02:00
|
|
|
database: &D,
|
2020-10-26 14:08:57 -04:00
|
|
|
required_utxos: Vec<(UTXO, usize)>,
|
2020-10-26 14:12:46 -04:00
|
|
|
optional_utxos: Vec<(UTXO, usize)>,
|
2020-08-31 10:49:44 +02:00
|
|
|
fee_rate: FeeRate,
|
|
|
|
amount_needed: u64,
|
2020-08-06 16:56:41 +02:00
|
|
|
fee_amount: f32,
|
|
|
|
) -> Result<CoinSelectionResult, Error>;
|
|
|
|
}
|
|
|
|
|
2020-09-04 11:44:49 +02:00
|
|
|
/// Simple and dumb coin selection
|
|
|
|
///
|
|
|
|
/// This coin selection algorithm sorts the available UTXOs by value and then picks them starting
|
|
|
|
/// from the largest ones until the required amount is reached.
|
2020-08-06 16:56:41 +02:00
|
|
|
#[derive(Debug, Default)]
|
2020-10-26 14:16:25 -04:00
|
|
|
pub struct LargestFirstCoinSelection;
|
2020-08-06 16:56:41 +02:00
|
|
|
|
2020-10-26 14:16:25 -04:00
|
|
|
impl<D: Database> CoinSelectionAlgorithm<D> for LargestFirstCoinSelection {
|
2020-08-06 16:56:41 +02:00
|
|
|
fn coin_select(
|
|
|
|
&self,
|
2020-10-14 15:21:22 +02:00
|
|
|
_database: &D,
|
2020-10-26 14:08:57 -04:00
|
|
|
required_utxos: Vec<(UTXO, usize)>,
|
2020-10-26 14:12:46 -04:00
|
|
|
mut optional_utxos: Vec<(UTXO, usize)>,
|
2020-08-31 10:49:44 +02:00
|
|
|
fee_rate: FeeRate,
|
2020-10-14 15:21:22 +02:00
|
|
|
amount_needed: u64,
|
2020-08-06 16:56:41 +02:00
|
|
|
mut fee_amount: f32,
|
|
|
|
) -> Result<CoinSelectionResult, Error> {
|
2020-08-31 10:49:44 +02:00
|
|
|
let calc_fee_bytes = |wu| (wu as f32) * fee_rate.as_sat_vb() / 4.0;
|
2020-08-06 16:56:41 +02:00
|
|
|
|
|
|
|
log::debug!(
|
2020-10-14 15:21:22 +02:00
|
|
|
"amount_needed = `{}`, fee_amount = `{}`, fee_rate = `{:?}`",
|
|
|
|
amount_needed,
|
2020-08-06 16:56:41 +02:00
|
|
|
fee_amount,
|
|
|
|
fee_rate
|
|
|
|
);
|
|
|
|
|
2020-10-26 14:12:46 -04:00
|
|
|
// We put the "required UTXOs" first and make sure the optional UTXOs are sorted,
|
|
|
|
// initially smallest to largest, before being reversed with `.rev()`.
|
2020-10-14 14:03:12 +11:00
|
|
|
let utxos = {
|
2020-10-26 14:12:46 -04:00
|
|
|
optional_utxos.sort_unstable_by_key(|(utxo, _)| utxo.txout.value);
|
2020-10-26 14:08:57 -04:00
|
|
|
required_utxos
|
2020-10-14 14:03:12 +11:00
|
|
|
.into_iter()
|
|
|
|
.map(|utxo| (true, utxo))
|
2020-10-26 14:12:46 -04:00
|
|
|
.chain(optional_utxos.into_iter().rev().map(|utxo| (false, utxo)))
|
2020-10-14 14:03:12 +11:00
|
|
|
};
|
|
|
|
|
|
|
|
// Keep including inputs until we've got enough.
|
|
|
|
// Store the total input value in selected_amount and the total fee being paid in fee_amount
|
|
|
|
let mut selected_amount = 0;
|
|
|
|
let txin = utxos
|
|
|
|
.scan(
|
|
|
|
(&mut selected_amount, &mut fee_amount),
|
2020-10-14 15:21:22 +02:00
|
|
|
|(selected_amount, fee_amount), (must_use, (utxo, weight))| {
|
|
|
|
if must_use || **selected_amount < amount_needed + (fee_amount.ceil() as u64) {
|
2020-10-14 14:03:12 +11:00
|
|
|
let new_in = TxIn {
|
|
|
|
previous_output: utxo.outpoint,
|
|
|
|
script_sig: Script::default(),
|
|
|
|
sequence: 0, // Let the caller choose the right nSequence
|
|
|
|
witness: vec![],
|
|
|
|
};
|
|
|
|
|
2020-11-08 15:46:27 +01:00
|
|
|
**fee_amount += calc_fee_bytes(TXIN_BASE_WEIGHT + weight);
|
2020-10-14 14:03:12 +11:00
|
|
|
**selected_amount += utxo.txout.value;
|
|
|
|
|
|
|
|
log::debug!(
|
|
|
|
"Selected {}, updated fee_amount = `{}`",
|
|
|
|
new_in.previous_output,
|
|
|
|
fee_amount
|
|
|
|
);
|
|
|
|
|
|
|
|
Some((new_in, utxo.txout.script_pubkey))
|
|
|
|
} else {
|
|
|
|
None
|
|
|
|
}
|
|
|
|
},
|
|
|
|
)
|
|
|
|
.collect::<Vec<_>>();
|
|
|
|
|
2020-10-14 15:21:22 +02:00
|
|
|
if selected_amount < amount_needed + (fee_amount.ceil() as u64) {
|
2020-10-14 14:03:12 +11:00
|
|
|
return Err(Error::InsufficientFunds);
|
2020-08-06 16:56:41 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
Ok(CoinSelectionResult {
|
|
|
|
txin,
|
|
|
|
fee_amount,
|
2020-08-31 10:49:44 +02:00
|
|
|
selected_amount,
|
2020-08-06 16:56:41 +02:00
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-11-08 15:46:27 +01:00
|
|
|
// Base weight of a Txin, not counting the weight needed for satisfaying it.
|
|
|
|
// prev_txid (32 bytes) + prev_vout (4 bytes) + sequence (4 bytes) + script_len (1 bytes)
|
|
|
|
pub const TXIN_BASE_WEIGHT: usize = (32 + 4 + 4 + 1) * 4;
|
|
|
|
|
2020-10-31 16:24:59 +01:00
|
|
|
#[derive(Debug, Clone)]
|
|
|
|
// Adds fee information to an UTXO.
|
|
|
|
struct OutputGroup {
|
|
|
|
utxo: UTXO,
|
|
|
|
// weight needed to satisfy the UTXO, as described in `Descriptor::max_satisfaction_weight`
|
|
|
|
satisfaction_weight: usize,
|
|
|
|
// Amount of fees for spending a certain utxo, calculated using a certain FeeRate
|
|
|
|
fee: f32,
|
|
|
|
// The effective value of the UTXO, i.e., the utxo value minus the fee for spending it
|
|
|
|
effective_value: i64,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl OutputGroup {
|
|
|
|
fn new(utxo: UTXO, satisfaction_weight: usize, fee_rate: FeeRate) -> Self {
|
|
|
|
let fee = (TXIN_BASE_WEIGHT + satisfaction_weight) as f32 / 4.0 * fee_rate.as_sat_vb();
|
|
|
|
let effective_value = utxo.txout.value as i64 - fee.ceil() as i64;
|
|
|
|
OutputGroup {
|
|
|
|
utxo,
|
|
|
|
satisfaction_weight,
|
|
|
|
effective_value,
|
|
|
|
fee,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Branch and bound coin selection. Code adapted from Bitcoin Core's implementation and from Mark
|
|
|
|
/// Erhardt Master's Thesis (http://murch.one/wp-content/uploads/2016/11/erhardt2016coinselection.pdf)
|
|
|
|
#[derive(Debug)]
|
|
|
|
pub struct BranchAndBoundCoinSelection {
|
|
|
|
size_of_change: u64,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Default for BranchAndBoundCoinSelection {
|
|
|
|
fn default() -> Self {
|
|
|
|
Self {
|
|
|
|
// P2WPKH cost of change -> value (8 bytes) + script len (1 bytes) + script (22 bytes)
|
|
|
|
size_of_change: 8 + 1 + 22,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl BranchAndBoundCoinSelection {
|
|
|
|
pub fn new(size_of_change: u64) -> Self {
|
|
|
|
Self { size_of_change }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const BNB_TOTAL_TRIES: usize = 100_000;
|
|
|
|
|
|
|
|
impl<D: Database> CoinSelectionAlgorithm<D> for BranchAndBoundCoinSelection {
|
|
|
|
fn coin_select(
|
|
|
|
&self,
|
|
|
|
_database: &D,
|
|
|
|
required_utxos: Vec<(UTXO, usize)>,
|
|
|
|
optional_utxos: Vec<(UTXO, usize)>,
|
|
|
|
fee_rate: FeeRate,
|
|
|
|
amount_needed: u64,
|
|
|
|
fee_amount: f32,
|
|
|
|
) -> Result<CoinSelectionResult, Error> {
|
|
|
|
// Mapping every (UTXO, usize) to an output group
|
|
|
|
let required_utxos: Vec<OutputGroup> = required_utxos
|
|
|
|
.into_iter()
|
|
|
|
.map(|u| OutputGroup::new(u.0, u.1, fee_rate))
|
|
|
|
.collect();
|
|
|
|
|
|
|
|
// Mapping every (UTXO, usize) to an output group.
|
|
|
|
// Filtering UTXOs with an effective_value < 0, as the fee paid for
|
|
|
|
// adding them is more than their value
|
|
|
|
let optional_utxos: Vec<OutputGroup> = optional_utxos
|
|
|
|
.into_iter()
|
|
|
|
.map(|u| OutputGroup::new(u.0, u.1, fee_rate))
|
|
|
|
.filter(|u| u.effective_value > 0)
|
|
|
|
.collect();
|
|
|
|
|
|
|
|
let curr_value = required_utxos
|
|
|
|
.iter()
|
|
|
|
.fold(0, |acc, x| acc + x.effective_value as u64);
|
|
|
|
|
|
|
|
let curr_available_value = optional_utxos
|
|
|
|
.iter()
|
|
|
|
.fold(0, |acc, x| acc + x.effective_value as u64);
|
|
|
|
|
|
|
|
let actual_target = fee_amount.ceil() as u64 + amount_needed;
|
|
|
|
let cost_of_change = self.size_of_change as f32 * fee_rate.as_sat_vb();
|
|
|
|
|
|
|
|
if curr_available_value + curr_value < actual_target {
|
|
|
|
return Err(Error::InsufficientFunds);
|
|
|
|
}
|
|
|
|
|
|
|
|
Ok(self
|
|
|
|
.bnb(
|
|
|
|
required_utxos.clone(),
|
|
|
|
optional_utxos.clone(),
|
|
|
|
curr_value,
|
|
|
|
curr_available_value,
|
|
|
|
actual_target,
|
|
|
|
fee_amount,
|
|
|
|
cost_of_change,
|
|
|
|
)
|
|
|
|
.unwrap_or_else(|_| {
|
|
|
|
self.single_random_draw(
|
|
|
|
required_utxos,
|
|
|
|
optional_utxos,
|
|
|
|
curr_value,
|
|
|
|
actual_target,
|
|
|
|
fee_amount,
|
|
|
|
)
|
|
|
|
}))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl BranchAndBoundCoinSelection {
|
|
|
|
// TODO: make this more Rust-onic :)
|
|
|
|
// (And perhpaps refactor with less arguments?)
|
|
|
|
#[allow(clippy::too_many_arguments)]
|
|
|
|
fn bnb(
|
|
|
|
&self,
|
|
|
|
required_utxos: Vec<OutputGroup>,
|
|
|
|
mut optional_utxos: Vec<OutputGroup>,
|
|
|
|
mut curr_value: u64,
|
|
|
|
mut curr_available_value: u64,
|
|
|
|
actual_target: u64,
|
|
|
|
fee_amount: f32,
|
|
|
|
cost_of_change: f32,
|
|
|
|
) -> Result<CoinSelectionResult, Error> {
|
|
|
|
// current_selection[i] will contain true if we are using optional_utxos[i],
|
|
|
|
// false otherwise. Note that current_selection.len() could be less than
|
|
|
|
// optional_utxos.len(), it just means that we still haven't decided if we should keep
|
|
|
|
// certain optional_utxos or not.
|
|
|
|
let mut current_selection: Vec<bool> = Vec::with_capacity(optional_utxos.len());
|
|
|
|
|
|
|
|
// Sort the utxo_pool
|
|
|
|
optional_utxos.sort_unstable_by_key(|a| a.effective_value);
|
|
|
|
optional_utxos.reverse();
|
|
|
|
|
|
|
|
// Contains the best selection we found
|
|
|
|
let mut best_selection = Vec::new();
|
|
|
|
let mut best_selection_value = None;
|
|
|
|
|
|
|
|
// Depth First search loop for choosing the UTXOs
|
|
|
|
for _ in 0..BNB_TOTAL_TRIES {
|
|
|
|
// Conditions for starting a backtrack
|
|
|
|
let mut backtrack = false;
|
|
|
|
// Cannot possibly reach target with the amount remaining in the curr_available_value,
|
|
|
|
// or the selected value is out of range.
|
|
|
|
// Go back and try other branch
|
|
|
|
if curr_value + curr_available_value < actual_target
|
|
|
|
|| curr_value > actual_target + cost_of_change as u64
|
|
|
|
{
|
|
|
|
backtrack = true;
|
|
|
|
} else if curr_value >= actual_target {
|
|
|
|
// Selected value is within range, there's no point in going forward. Start
|
|
|
|
// backtracking
|
|
|
|
backtrack = true;
|
|
|
|
|
|
|
|
// If we found a solution better than the previous one, or if there wasn't previous
|
|
|
|
// solution, update the best solution
|
|
|
|
if best_selection_value.is_none() || curr_value < best_selection_value.unwrap() {
|
|
|
|
best_selection = current_selection.clone();
|
|
|
|
best_selection_value = Some(curr_value);
|
|
|
|
}
|
|
|
|
|
|
|
|
// If we found a perfect match, break here
|
|
|
|
if curr_value == actual_target {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Backtracking, moving backwards
|
|
|
|
if backtrack {
|
|
|
|
// Walk backwards to find the last included UTXO that still needs to have its omission branch traversed.
|
|
|
|
while let Some(false) = current_selection.last() {
|
|
|
|
current_selection.pop();
|
|
|
|
curr_available_value +=
|
|
|
|
optional_utxos[current_selection.len()].effective_value as u64;
|
|
|
|
}
|
|
|
|
|
|
|
|
if current_selection.last_mut().is_none() {
|
|
|
|
// We have walked back to the first utxo and no branch is untraversed. All solutions searched
|
|
|
|
// If best selection is empty, then there's no exact match
|
|
|
|
if best_selection.is_empty() {
|
|
|
|
return Err(Error::BnBNoExactMatch);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if let Some(c) = current_selection.last_mut() {
|
|
|
|
// Output was included on previous iterations, try excluding now.
|
|
|
|
*c = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
let utxo = &optional_utxos[current_selection.len() - 1];
|
|
|
|
curr_value -= utxo.effective_value as u64;
|
|
|
|
} else {
|
|
|
|
// Moving forwards, continuing down this branch
|
|
|
|
let utxo = &optional_utxos[current_selection.len()];
|
|
|
|
|
|
|
|
// Remove this utxo from the curr_available_value utxo amount
|
|
|
|
curr_available_value -= utxo.effective_value as u64;
|
|
|
|
|
|
|
|
// Inclusion branch first (Largest First Exploration)
|
|
|
|
current_selection.push(true);
|
|
|
|
curr_value += utxo.effective_value as u64;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check for solution
|
|
|
|
if best_selection.is_empty() {
|
|
|
|
return Err(Error::BnBTotalTriesExceeded);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Set output set
|
|
|
|
let selected_utxos = optional_utxos
|
|
|
|
.into_iter()
|
|
|
|
.zip(best_selection)
|
|
|
|
.filter_map(|(optional, is_in_best)| if is_in_best { Some(optional) } else { None })
|
|
|
|
.collect();
|
|
|
|
|
|
|
|
Ok(BranchAndBoundCoinSelection::calculate_cs_result(
|
|
|
|
selected_utxos,
|
|
|
|
required_utxos,
|
|
|
|
fee_amount,
|
|
|
|
))
|
|
|
|
}
|
|
|
|
|
|
|
|
fn single_random_draw(
|
|
|
|
&self,
|
|
|
|
required_utxos: Vec<OutputGroup>,
|
|
|
|
mut optional_utxos: Vec<OutputGroup>,
|
|
|
|
curr_value: u64,
|
|
|
|
actual_target: u64,
|
|
|
|
fee_amount: f32,
|
|
|
|
) -> CoinSelectionResult {
|
|
|
|
#[cfg(not(test))]
|
|
|
|
optional_utxos.shuffle(&mut thread_rng());
|
|
|
|
#[cfg(test)]
|
|
|
|
{
|
|
|
|
let seed = [0; 32];
|
|
|
|
let mut rng: StdRng = SeedableRng::from_seed(seed);
|
|
|
|
optional_utxos.shuffle(&mut rng);
|
|
|
|
}
|
|
|
|
|
|
|
|
let selected_utxos = optional_utxos
|
|
|
|
.into_iter()
|
|
|
|
.scan(curr_value, |curr_value, utxo| {
|
|
|
|
if *curr_value >= actual_target {
|
|
|
|
None
|
|
|
|
} else {
|
|
|
|
*curr_value += utxo.effective_value as u64;
|
|
|
|
Some(utxo)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
.collect::<Vec<_>>();
|
|
|
|
|
|
|
|
BranchAndBoundCoinSelection::calculate_cs_result(selected_utxos, required_utxos, fee_amount)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn calculate_cs_result(
|
|
|
|
selected_utxos: Vec<OutputGroup>,
|
|
|
|
required_utxos: Vec<OutputGroup>,
|
|
|
|
fee_amount: f32,
|
|
|
|
) -> CoinSelectionResult {
|
|
|
|
let (txin, fee_amount, selected_amount) =
|
|
|
|
selected_utxos.into_iter().chain(required_utxos).fold(
|
|
|
|
(vec![], fee_amount, 0),
|
|
|
|
|(mut txin, mut fee_amount, mut selected_amount), output_group| {
|
|
|
|
selected_amount += output_group.utxo.txout.value;
|
|
|
|
fee_amount += output_group.fee;
|
|
|
|
txin.push((
|
|
|
|
TxIn {
|
|
|
|
previous_output: output_group.utxo.outpoint,
|
|
|
|
..Default::default()
|
|
|
|
},
|
|
|
|
output_group.utxo.txout.script_pubkey,
|
|
|
|
));
|
|
|
|
(txin, fee_amount, selected_amount)
|
|
|
|
},
|
|
|
|
);
|
|
|
|
CoinSelectionResult {
|
|
|
|
txin,
|
|
|
|
fee_amount,
|
|
|
|
selected_amount,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-08-06 16:56:41 +02:00
|
|
|
#[cfg(test)]
|
|
|
|
mod test {
|
|
|
|
use std::str::FromStr;
|
|
|
|
|
2020-10-31 16:27:33 +01:00
|
|
|
use bitcoin::consensus::encode::serialize;
|
2020-08-06 16:56:41 +02:00
|
|
|
use bitcoin::{OutPoint, Script, TxOut};
|
|
|
|
|
|
|
|
use super::*;
|
2020-10-14 15:21:22 +02:00
|
|
|
use crate::database::MemoryDatabase;
|
2020-08-06 16:56:41 +02:00
|
|
|
use crate::types::*;
|
|
|
|
|
2020-10-31 16:27:33 +01:00
|
|
|
use rand::rngs::StdRng;
|
|
|
|
use rand::seq::SliceRandom;
|
|
|
|
use rand::{Rng, SeedableRng};
|
|
|
|
|
2020-08-06 16:56:41 +02:00
|
|
|
const P2WPKH_WITNESS_SIZE: usize = 73 + 33 + 2;
|
|
|
|
|
2020-10-14 15:21:22 +02:00
|
|
|
fn get_test_utxos() -> Vec<(UTXO, usize)> {
|
2020-08-06 16:56:41 +02:00
|
|
|
vec![
|
2020-10-14 15:21:22 +02:00
|
|
|
(
|
|
|
|
UTXO {
|
|
|
|
outpoint: OutPoint::from_str(
|
|
|
|
"ebd9813ecebc57ff8f30797de7c205e3c7498ca950ea4341ee51a685ff2fa30a:0",
|
|
|
|
)
|
|
|
|
.unwrap(),
|
|
|
|
txout: TxOut {
|
|
|
|
value: 100_000,
|
|
|
|
script_pubkey: Script::new(),
|
|
|
|
},
|
|
|
|
is_internal: false,
|
2020-08-06 16:56:41 +02:00
|
|
|
},
|
2020-10-14 15:21:22 +02:00
|
|
|
P2WPKH_WITNESS_SIZE,
|
|
|
|
),
|
|
|
|
(
|
|
|
|
UTXO {
|
|
|
|
outpoint: OutPoint::from_str(
|
|
|
|
"65d92ddff6b6dc72c89624a6491997714b90f6004f928d875bc0fd53f264fa85:0",
|
|
|
|
)
|
|
|
|
.unwrap(),
|
|
|
|
txout: TxOut {
|
|
|
|
value: 200_000,
|
|
|
|
script_pubkey: Script::new(),
|
|
|
|
},
|
|
|
|
is_internal: true,
|
2020-08-06 16:56:41 +02:00
|
|
|
},
|
2020-10-14 15:21:22 +02:00
|
|
|
P2WPKH_WITNESS_SIZE,
|
|
|
|
),
|
2020-08-06 16:56:41 +02:00
|
|
|
]
|
|
|
|
}
|
|
|
|
|
2020-10-31 16:27:33 +01:00
|
|
|
fn generate_random_utxos(rng: &mut StdRng, utxos_number: usize) -> Vec<(UTXO, usize)> {
|
|
|
|
let mut res = Vec::new();
|
|
|
|
for _ in 0..utxos_number {
|
|
|
|
res.push((
|
|
|
|
UTXO {
|
|
|
|
outpoint: OutPoint::from_str(
|
|
|
|
"ebd9813ecebc57ff8f30797de7c205e3c7498ca950ea4341ee51a685ff2fa30a:0",
|
|
|
|
)
|
|
|
|
.unwrap(),
|
|
|
|
txout: TxOut {
|
|
|
|
value: rng.gen_range(0, 200000000),
|
|
|
|
script_pubkey: Script::new(),
|
|
|
|
},
|
|
|
|
is_internal: false,
|
|
|
|
},
|
|
|
|
P2WPKH_WITNESS_SIZE,
|
|
|
|
));
|
|
|
|
}
|
|
|
|
res
|
|
|
|
}
|
|
|
|
|
|
|
|
fn generate_same_value_utxos(utxos_value: u64, utxos_number: usize) -> Vec<(UTXO, usize)> {
|
|
|
|
let utxo = (
|
|
|
|
UTXO {
|
|
|
|
outpoint: OutPoint::from_str(
|
|
|
|
"ebd9813ecebc57ff8f30797de7c205e3c7498ca950ea4341ee51a685ff2fa30a:0",
|
|
|
|
)
|
|
|
|
.unwrap(),
|
|
|
|
txout: TxOut {
|
|
|
|
value: utxos_value,
|
|
|
|
script_pubkey: Script::new(),
|
|
|
|
},
|
|
|
|
is_internal: false,
|
|
|
|
},
|
|
|
|
P2WPKH_WITNESS_SIZE,
|
|
|
|
);
|
|
|
|
vec![utxo; utxos_number]
|
|
|
|
}
|
|
|
|
|
|
|
|
fn sum_random_utxos(mut rng: &mut StdRng, utxos: &mut Vec<(UTXO, usize)>) -> u64 {
|
|
|
|
let utxos_picked_len = rng.gen_range(2, utxos.len() / 2);
|
|
|
|
utxos.shuffle(&mut rng);
|
|
|
|
utxos[..utxos_picked_len]
|
|
|
|
.iter()
|
|
|
|
.fold(0, |acc, x| acc + x.0.txout.value)
|
|
|
|
}
|
|
|
|
|
2020-08-06 16:56:41 +02:00
|
|
|
#[test]
|
2020-10-26 14:16:25 -04:00
|
|
|
fn test_largest_first_coin_selection_success() {
|
2020-08-06 16:56:41 +02:00
|
|
|
let utxos = get_test_utxos();
|
2020-10-14 15:21:22 +02:00
|
|
|
let database = MemoryDatabase::default();
|
2020-08-06 16:56:41 +02:00
|
|
|
|
2020-10-26 14:16:25 -04:00
|
|
|
let result = LargestFirstCoinSelection::default()
|
2020-08-31 10:49:44 +02:00
|
|
|
.coin_select(
|
2020-10-14 15:21:22 +02:00
|
|
|
&database,
|
2020-08-31 10:49:44 +02:00
|
|
|
utxos,
|
2020-10-14 15:21:22 +02:00
|
|
|
vec![],
|
2020-08-31 10:49:44 +02:00
|
|
|
FeeRate::from_sat_per_vb(1.0),
|
|
|
|
250_000,
|
|
|
|
50.0,
|
|
|
|
)
|
2020-08-06 16:56:41 +02:00
|
|
|
.unwrap();
|
|
|
|
|
|
|
|
assert_eq!(result.txin.len(), 2);
|
2020-08-31 10:49:44 +02:00
|
|
|
assert_eq!(result.selected_amount, 300_000);
|
2020-08-06 16:56:41 +02:00
|
|
|
assert_eq!(result.fee_amount, 186.0);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
2020-10-26 14:16:25 -04:00
|
|
|
fn test_largest_first_coin_selection_use_all() {
|
2020-08-06 16:56:41 +02:00
|
|
|
let utxos = get_test_utxos();
|
2020-10-14 15:21:22 +02:00
|
|
|
let database = MemoryDatabase::default();
|
2020-08-06 16:56:41 +02:00
|
|
|
|
2020-10-26 14:16:25 -04:00
|
|
|
let result = LargestFirstCoinSelection::default()
|
2020-08-31 10:49:44 +02:00
|
|
|
.coin_select(
|
2020-10-14 15:21:22 +02:00
|
|
|
&database,
|
2020-08-31 10:49:44 +02:00
|
|
|
utxos,
|
2020-10-14 14:03:12 +11:00
|
|
|
vec![],
|
2020-08-31 10:49:44 +02:00
|
|
|
FeeRate::from_sat_per_vb(1.0),
|
|
|
|
20_000,
|
|
|
|
50.0,
|
|
|
|
)
|
2020-08-06 16:56:41 +02:00
|
|
|
.unwrap();
|
|
|
|
|
|
|
|
assert_eq!(result.txin.len(), 2);
|
2020-08-31 10:49:44 +02:00
|
|
|
assert_eq!(result.selected_amount, 300_000);
|
2020-08-06 16:56:41 +02:00
|
|
|
assert_eq!(result.fee_amount, 186.0);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
2020-10-26 14:16:25 -04:00
|
|
|
fn test_largest_first_coin_selection_use_only_necessary() {
|
2020-08-06 16:56:41 +02:00
|
|
|
let utxos = get_test_utxos();
|
2020-10-14 15:21:22 +02:00
|
|
|
let database = MemoryDatabase::default();
|
2020-08-06 16:56:41 +02:00
|
|
|
|
2020-10-26 14:16:25 -04:00
|
|
|
let result = LargestFirstCoinSelection::default()
|
2020-08-31 10:49:44 +02:00
|
|
|
.coin_select(
|
2020-10-14 15:21:22 +02:00
|
|
|
&database,
|
2020-10-14 14:03:12 +11:00
|
|
|
vec![],
|
2020-08-31 10:49:44 +02:00
|
|
|
utxos,
|
|
|
|
FeeRate::from_sat_per_vb(1.0),
|
|
|
|
20_000,
|
|
|
|
50.0,
|
|
|
|
)
|
2020-08-06 16:56:41 +02:00
|
|
|
.unwrap();
|
|
|
|
|
|
|
|
assert_eq!(result.txin.len(), 1);
|
2020-08-31 10:49:44 +02:00
|
|
|
assert_eq!(result.selected_amount, 200_000);
|
2020-08-06 16:56:41 +02:00
|
|
|
assert_eq!(result.fee_amount, 118.0);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
#[should_panic(expected = "InsufficientFunds")]
|
2020-10-26 14:16:25 -04:00
|
|
|
fn test_largest_first_coin_selection_insufficient_funds() {
|
2020-08-06 16:56:41 +02:00
|
|
|
let utxos = get_test_utxos();
|
2020-10-14 15:21:22 +02:00
|
|
|
let database = MemoryDatabase::default();
|
2020-08-06 16:56:41 +02:00
|
|
|
|
2020-10-26 14:16:25 -04:00
|
|
|
LargestFirstCoinSelection::default()
|
2020-08-31 10:49:44 +02:00
|
|
|
.coin_select(
|
2020-10-14 15:21:22 +02:00
|
|
|
&database,
|
2020-10-14 14:03:12 +11:00
|
|
|
vec![],
|
2020-08-31 10:49:44 +02:00
|
|
|
utxos,
|
|
|
|
FeeRate::from_sat_per_vb(1.0),
|
|
|
|
500_000,
|
|
|
|
50.0,
|
|
|
|
)
|
2020-08-06 16:56:41 +02:00
|
|
|
.unwrap();
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
#[should_panic(expected = "InsufficientFunds")]
|
2020-10-26 14:16:25 -04:00
|
|
|
fn test_largest_first_coin_selection_insufficient_funds_high_fees() {
|
2020-08-06 16:56:41 +02:00
|
|
|
let utxos = get_test_utxos();
|
2020-10-14 15:21:22 +02:00
|
|
|
let database = MemoryDatabase::default();
|
2020-08-06 16:56:41 +02:00
|
|
|
|
2020-10-26 14:16:25 -04:00
|
|
|
LargestFirstCoinSelection::default()
|
2020-08-31 10:49:44 +02:00
|
|
|
.coin_select(
|
2020-10-14 15:21:22 +02:00
|
|
|
&database,
|
2020-10-14 14:03:12 +11:00
|
|
|
vec![],
|
2020-08-31 10:49:44 +02:00
|
|
|
utxos,
|
|
|
|
FeeRate::from_sat_per_vb(1000.0),
|
|
|
|
250_000,
|
|
|
|
50.0,
|
|
|
|
)
|
2020-08-06 16:56:41 +02:00
|
|
|
.unwrap();
|
|
|
|
}
|
2020-10-31 16:27:33 +01:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_bnb_coin_selection_success() {
|
|
|
|
// In this case bnb won't find a suitable match and single random draw will
|
|
|
|
// select three outputs
|
|
|
|
let utxos = generate_same_value_utxos(100_000, 20);
|
|
|
|
|
|
|
|
let database = MemoryDatabase::default();
|
|
|
|
|
|
|
|
let result = BranchAndBoundCoinSelection::default()
|
|
|
|
.coin_select(
|
|
|
|
&database,
|
|
|
|
vec![],
|
|
|
|
utxos,
|
|
|
|
FeeRate::from_sat_per_vb(1.0),
|
|
|
|
250_000,
|
|
|
|
50.0,
|
|
|
|
)
|
|
|
|
.unwrap();
|
|
|
|
|
|
|
|
assert_eq!(result.txin.len(), 3);
|
|
|
|
assert_eq!(result.selected_amount, 300_000);
|
|
|
|
assert_eq!(result.fee_amount, 254.0);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_bnb_coin_selection_required_are_enough() {
|
|
|
|
let utxos = get_test_utxos();
|
|
|
|
let database = MemoryDatabase::default();
|
|
|
|
|
|
|
|
let result = BranchAndBoundCoinSelection::default()
|
|
|
|
.coin_select(
|
|
|
|
&database,
|
|
|
|
utxos.clone(),
|
|
|
|
utxos,
|
|
|
|
FeeRate::from_sat_per_vb(1.0),
|
|
|
|
20_000,
|
|
|
|
50.0,
|
|
|
|
)
|
|
|
|
.unwrap();
|
|
|
|
|
|
|
|
assert_eq!(result.txin.len(), 2);
|
|
|
|
assert_eq!(result.selected_amount, 300_000);
|
|
|
|
assert_eq!(result.fee_amount, 186.0);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
#[should_panic(expected = "InsufficientFunds")]
|
|
|
|
fn test_bnb_coin_selection_insufficient_funds() {
|
|
|
|
let utxos = get_test_utxos();
|
|
|
|
let database = MemoryDatabase::default();
|
|
|
|
|
|
|
|
BranchAndBoundCoinSelection::default()
|
|
|
|
.coin_select(
|
|
|
|
&database,
|
|
|
|
vec![],
|
|
|
|
utxos,
|
|
|
|
FeeRate::from_sat_per_vb(1.0),
|
|
|
|
500_000,
|
|
|
|
50.0,
|
|
|
|
)
|
|
|
|
.unwrap();
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
#[should_panic(expected = "InsufficientFunds")]
|
|
|
|
fn test_bnb_coin_selection_insufficient_funds_high_fees() {
|
|
|
|
let utxos = get_test_utxos();
|
|
|
|
let database = MemoryDatabase::default();
|
|
|
|
|
|
|
|
BranchAndBoundCoinSelection::default()
|
|
|
|
.coin_select(
|
|
|
|
&database,
|
|
|
|
vec![],
|
|
|
|
utxos,
|
|
|
|
FeeRate::from_sat_per_vb(1000.0),
|
|
|
|
250_000,
|
|
|
|
50.0,
|
|
|
|
)
|
|
|
|
.unwrap();
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_bnb_coin_selection_check_fee_rate() {
|
|
|
|
let utxos = get_test_utxos();
|
|
|
|
let database = MemoryDatabase::default();
|
|
|
|
|
|
|
|
let result = BranchAndBoundCoinSelection::new(0)
|
|
|
|
.coin_select(
|
|
|
|
&database,
|
|
|
|
vec![],
|
|
|
|
utxos.clone(),
|
|
|
|
FeeRate::from_sat_per_vb(1.0),
|
|
|
|
99932, // first utxo's effective value
|
|
|
|
0.0,
|
|
|
|
)
|
|
|
|
.unwrap();
|
|
|
|
|
|
|
|
assert_eq!(result.txin.len(), 1);
|
|
|
|
assert_eq!(result.selected_amount, 100_000);
|
|
|
|
let result_size =
|
|
|
|
serialize(result.txin.first().unwrap()).len() as f32 + P2WPKH_WITNESS_SIZE as f32 / 4.0;
|
|
|
|
let epsilon = 0.5;
|
|
|
|
assert!((1.0 - (result.fee_amount / result_size)).abs() < epsilon);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_bnb_coin_selection_exact_match() {
|
|
|
|
let seed = [0; 32];
|
|
|
|
let mut rng: StdRng = SeedableRng::from_seed(seed);
|
|
|
|
let database = MemoryDatabase::default();
|
|
|
|
|
|
|
|
for _i in 0..200 {
|
|
|
|
let mut optional_utxos = generate_random_utxos(&mut rng, 16);
|
|
|
|
let target_amount = sum_random_utxos(&mut rng, &mut optional_utxos);
|
|
|
|
let result = BranchAndBoundCoinSelection::new(0)
|
|
|
|
.coin_select(
|
|
|
|
&database,
|
|
|
|
vec![],
|
|
|
|
optional_utxos,
|
|
|
|
FeeRate::from_sat_per_vb(0.0),
|
|
|
|
target_amount,
|
|
|
|
0.0,
|
|
|
|
)
|
|
|
|
.unwrap();
|
|
|
|
assert_eq!(result.selected_amount, target_amount);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-10-31 16:28:12 +01:00
|
|
|
#[test]
|
|
|
|
#[should_panic(expected = "BnBNoExactMatch")]
|
|
|
|
fn test_bnb_function_no_exact_match() {
|
|
|
|
let fee_rate = FeeRate::from_sat_per_vb(10.0);
|
|
|
|
let utxos: Vec<OutputGroup> = get_test_utxos()
|
|
|
|
.into_iter()
|
|
|
|
.map(|u| OutputGroup::new(u.0, u.1, fee_rate))
|
|
|
|
.collect();
|
|
|
|
|
|
|
|
let curr_available_value = utxos
|
|
|
|
.iter()
|
|
|
|
.fold(0, |acc, x| acc + x.effective_value as u64);
|
|
|
|
|
|
|
|
let size_of_change = 31;
|
|
|
|
let cost_of_change = size_of_change as f32 * fee_rate.as_sat_vb();
|
|
|
|
BranchAndBoundCoinSelection::new(size_of_change)
|
|
|
|
.bnb(
|
|
|
|
vec![],
|
|
|
|
utxos,
|
|
|
|
0,
|
|
|
|
curr_available_value,
|
|
|
|
20_000,
|
|
|
|
50.0,
|
|
|
|
cost_of_change,
|
|
|
|
)
|
|
|
|
.unwrap();
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
#[should_panic(expected = "BnBTotalTriesExceeded")]
|
|
|
|
fn test_bnb_function_tries_exceeded() {
|
|
|
|
let fee_rate = FeeRate::from_sat_per_vb(10.0);
|
|
|
|
let utxos: Vec<OutputGroup> = generate_same_value_utxos(100_000, 100_000)
|
|
|
|
.into_iter()
|
|
|
|
.map(|u| OutputGroup::new(u.0, u.1, fee_rate))
|
|
|
|
.collect();
|
|
|
|
|
|
|
|
let curr_available_value = utxos
|
|
|
|
.iter()
|
|
|
|
.fold(0, |acc, x| acc + x.effective_value as u64);
|
|
|
|
|
|
|
|
let size_of_change = 31;
|
|
|
|
let cost_of_change = size_of_change as f32 * fee_rate.as_sat_vb();
|
|
|
|
|
|
|
|
BranchAndBoundCoinSelection::new(size_of_change)
|
|
|
|
.bnb(
|
|
|
|
vec![],
|
|
|
|
utxos,
|
|
|
|
0,
|
|
|
|
curr_available_value,
|
|
|
|
20_000,
|
|
|
|
50.0,
|
|
|
|
cost_of_change,
|
|
|
|
)
|
|
|
|
.unwrap();
|
|
|
|
}
|
|
|
|
|
|
|
|
// The match won't be exact but still in the range
|
|
|
|
#[test]
|
|
|
|
fn test_bnb_function_almost_exact_match_with_fees() {
|
|
|
|
let fee_rate = FeeRate::from_sat_per_vb(1.0);
|
|
|
|
let size_of_change = 31;
|
|
|
|
let cost_of_change = size_of_change as f32 * fee_rate.as_sat_vb();
|
|
|
|
let fee_amount = 50.0;
|
|
|
|
|
|
|
|
let utxos: Vec<_> = generate_same_value_utxos(50_000, 10)
|
|
|
|
.into_iter()
|
|
|
|
.map(|u| OutputGroup::new(u.0, u.1, fee_rate))
|
|
|
|
.collect();
|
|
|
|
|
|
|
|
let curr_value = 0;
|
|
|
|
|
|
|
|
let curr_available_value = utxos
|
|
|
|
.iter()
|
|
|
|
.fold(0, |acc, x| acc + x.effective_value as u64);
|
|
|
|
|
|
|
|
// 2*(value of 1 utxo) - 2*(1 utxo fees with 1.0sat/vbyte fee rate) -
|
|
|
|
// cost_of_change + 5.
|
|
|
|
let target_amount = 2 * 50_000 - 2 * 67 - cost_of_change.ceil() as u64 + 5;
|
|
|
|
|
|
|
|
let result = BranchAndBoundCoinSelection::new(size_of_change)
|
|
|
|
.bnb(
|
|
|
|
vec![],
|
|
|
|
utxos,
|
|
|
|
curr_value,
|
|
|
|
curr_available_value,
|
|
|
|
target_amount,
|
|
|
|
fee_amount,
|
|
|
|
cost_of_change,
|
|
|
|
)
|
|
|
|
.unwrap();
|
|
|
|
assert_eq!(result.fee_amount, 186.0);
|
|
|
|
assert_eq!(result.selected_amount, 100_000);
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO: bnb() function should be optimized, and this test should be done with more utxos
|
|
|
|
#[test]
|
|
|
|
fn test_bnb_function_exact_match_more_utxos() {
|
|
|
|
let seed = [0; 32];
|
|
|
|
let mut rng: StdRng = SeedableRng::from_seed(seed);
|
|
|
|
let fee_rate = FeeRate::from_sat_per_vb(0.0);
|
|
|
|
|
|
|
|
for _ in 0..200 {
|
|
|
|
let optional_utxos: Vec<_> = generate_random_utxos(&mut rng, 40)
|
|
|
|
.into_iter()
|
|
|
|
.map(|u| OutputGroup::new(u.0, u.1, fee_rate))
|
|
|
|
.collect();
|
|
|
|
|
|
|
|
let curr_value = 0;
|
|
|
|
|
|
|
|
let curr_available_value = optional_utxos
|
|
|
|
.iter()
|
|
|
|
.fold(0, |acc, x| acc + x.effective_value as u64);
|
|
|
|
|
|
|
|
let target_amount = optional_utxos[3].effective_value as u64
|
|
|
|
+ optional_utxos[23].effective_value as u64;
|
|
|
|
|
|
|
|
let result = BranchAndBoundCoinSelection::new(0)
|
|
|
|
.bnb(
|
|
|
|
vec![],
|
|
|
|
optional_utxos,
|
|
|
|
curr_value,
|
|
|
|
curr_available_value,
|
|
|
|
target_amount,
|
|
|
|
0.0,
|
|
|
|
0.0,
|
|
|
|
)
|
|
|
|
.unwrap();
|
|
|
|
assert_eq!(result.selected_amount, target_amount);
|
|
|
|
}
|
|
|
|
}
|
2020-08-06 16:56:41 +02:00
|
|
|
}
|