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-02-05 11:59:02 +01:00
|
|
|
use std::convert::AsRef;
|
|
|
|
|
|
|
|
use bitcoin::blockdata::transaction::{OutPoint, Transaction, TxOut};
|
2021-02-08 15:40:56 +11:00
|
|
|
use bitcoin::{hash_types::Txid, util::psbt};
|
2020-02-05 11:59:02 +01:00
|
|
|
|
|
|
|
use serde::{Deserialize, Serialize};
|
|
|
|
|
2020-12-16 11:33:34 -08:00
|
|
|
/// Types of keychains
|
2020-08-10 10:49:34 +02:00
|
|
|
#[derive(Serialize, Deserialize, Debug, Clone, Copy, PartialEq, Eq, Hash)]
|
2020-12-14 17:14:24 +01:00
|
|
|
pub enum KeychainKind {
|
2020-12-04 16:13:15 +01:00
|
|
|
/// External
|
2020-02-05 11:59:02 +01:00
|
|
|
External = 0,
|
2020-12-04 16:13:15 +01:00
|
|
|
/// Internal, usually used for change outputs
|
2020-02-05 11:59:02 +01:00
|
|
|
Internal = 1,
|
|
|
|
}
|
|
|
|
|
2020-12-14 17:14:24 +01:00
|
|
|
impl KeychainKind {
|
2020-12-16 11:33:34 -08:00
|
|
|
/// Return [`KeychainKind`] as a byte
|
2020-02-05 11:59:02 +01:00
|
|
|
pub fn as_byte(&self) -> u8 {
|
|
|
|
match self {
|
2020-12-14 17:14:24 +01:00
|
|
|
KeychainKind::External => b'e',
|
|
|
|
KeychainKind::Internal => b'i',
|
2020-02-05 11:59:02 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-12-14 17:14:24 +01:00
|
|
|
impl AsRef<[u8]> for KeychainKind {
|
2020-02-05 11:59:02 +01:00
|
|
|
fn as_ref(&self) -> &[u8] {
|
|
|
|
match self {
|
2020-12-14 17:14:24 +01:00
|
|
|
KeychainKind::External => b"e",
|
|
|
|
KeychainKind::Internal => b"i",
|
2020-02-05 11:59:02 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-09-04 11:44:49 +02:00
|
|
|
/// Fee rate
|
2020-08-31 10:49:44 +02:00
|
|
|
#[derive(Debug, Copy, Clone, PartialEq, PartialOrd)]
|
|
|
|
// Internally stored as satoshi/vbyte
|
|
|
|
pub struct FeeRate(f32);
|
|
|
|
|
|
|
|
impl FeeRate {
|
2020-09-04 11:44:49 +02:00
|
|
|
/// Create a new instance of [`FeeRate`] given a float fee rate in btc/kvbytes
|
2020-08-31 10:49:44 +02:00
|
|
|
pub fn from_btc_per_kvb(btc_per_kvb: f32) -> Self {
|
|
|
|
FeeRate(btc_per_kvb * 1e5)
|
|
|
|
}
|
|
|
|
|
2020-09-04 11:44:49 +02:00
|
|
|
/// Create a new instance of [`FeeRate`] given a float fee rate in satoshi/vbyte
|
2020-08-31 10:49:44 +02:00
|
|
|
pub fn from_sat_per_vb(sat_per_vb: f32) -> Self {
|
|
|
|
FeeRate(sat_per_vb)
|
|
|
|
}
|
|
|
|
|
2020-09-04 11:44:49 +02:00
|
|
|
/// Create a new [`FeeRate`] with the default min relay fee value
|
2020-08-31 10:49:44 +02:00
|
|
|
pub fn default_min_relay_fee() -> Self {
|
|
|
|
FeeRate(1.0)
|
|
|
|
}
|
|
|
|
|
2020-09-04 11:44:49 +02:00
|
|
|
/// Return the value as satoshi/vbyte
|
2020-08-31 10:49:44 +02:00
|
|
|
pub fn as_sat_vb(&self) -> f32 {
|
|
|
|
self.0
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl std::default::Default for FeeRate {
|
|
|
|
fn default() -> Self {
|
|
|
|
FeeRate::default_min_relay_fee()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-02-08 15:40:56 +11:00
|
|
|
/// An unspent output owned by a [`Wallet`].
|
|
|
|
///
|
|
|
|
/// [`Wallet`]: crate::Wallet
|
2020-02-06 12:17:28 +01:00
|
|
|
#[derive(Serialize, Deserialize, Debug, Clone, PartialEq, Eq)]
|
2021-02-04 12:09:53 +11:00
|
|
|
pub struct LocalUtxo {
|
2020-12-16 11:33:34 -08:00
|
|
|
/// Reference to a transaction output
|
2020-02-05 11:59:02 +01:00
|
|
|
pub outpoint: OutPoint,
|
2020-12-16 11:33:34 -08:00
|
|
|
/// Transaction output
|
2020-02-05 11:59:02 +01:00
|
|
|
pub txout: TxOut,
|
2020-12-16 11:33:34 -08:00
|
|
|
/// Type of keychain
|
2020-12-14 17:14:24 +01:00
|
|
|
pub keychain: KeychainKind,
|
2020-02-05 11:59:02 +01:00
|
|
|
}
|
|
|
|
|
2021-02-08 15:40:56 +11:00
|
|
|
/// A [`Utxo`] with its `satisfaction_weight`.
|
|
|
|
#[derive(Debug, Clone, PartialEq)]
|
|
|
|
pub struct WeightedUtxo {
|
|
|
|
/// The weight of the witness data or `scriptSig`.
|
|
|
|
/// This is used to properly maintain the feerate when doing coin selection.
|
|
|
|
pub satisfaction_weight: usize,
|
|
|
|
/// The UTXO
|
|
|
|
pub utxo: Utxo,
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Debug, Clone, PartialEq)]
|
|
|
|
/// An unspent transaction output (UTXO).
|
|
|
|
pub enum Utxo {
|
|
|
|
/// A UTXO owned by the local wallet.
|
|
|
|
Local(LocalUtxo),
|
|
|
|
/// A UTXO owned by another wallet.
|
|
|
|
Foreign {
|
|
|
|
/// The location of the output.
|
|
|
|
outpoint: OutPoint,
|
|
|
|
/// The information about the input we require to add it to a PSBT.
|
|
|
|
// Box it to stop the type being too big.
|
|
|
|
psbt_input: Box<psbt::Input>,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Utxo {
|
|
|
|
/// Get the location of the UTXO
|
|
|
|
pub fn outpoint(&self) -> OutPoint {
|
|
|
|
match &self {
|
|
|
|
Utxo::Local(local) => local.outpoint,
|
|
|
|
Utxo::Foreign { outpoint, .. } => *outpoint,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Get the `TxOut` of the UTXO
|
|
|
|
pub fn txout(&self) -> &TxOut {
|
|
|
|
match &self {
|
|
|
|
Utxo::Local(local) => &local.txout,
|
|
|
|
Utxo::Foreign {
|
|
|
|
outpoint,
|
|
|
|
psbt_input,
|
|
|
|
} => {
|
|
|
|
if let Some(prev_tx) = &psbt_input.non_witness_utxo {
|
|
|
|
return &prev_tx.output[outpoint.vout as usize];
|
|
|
|
}
|
|
|
|
|
|
|
|
if let Some(txout) = &psbt_input.witness_utxo {
|
|
|
|
return &txout;
|
|
|
|
}
|
|
|
|
|
|
|
|
unreachable!("Foreign UTXOs will always have one of these set")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-09-04 11:44:49 +02:00
|
|
|
/// A wallet transaction
|
2020-02-07 23:22:28 +01:00
|
|
|
#[derive(Serialize, Deserialize, Debug, Clone, PartialEq, Eq, Default)]
|
2020-02-05 11:59:02 +01:00
|
|
|
pub struct TransactionDetails {
|
2020-12-16 11:33:34 -08:00
|
|
|
/// Optional transaction
|
2020-02-05 11:59:02 +01:00
|
|
|
pub transaction: Option<Transaction>,
|
2020-12-16 11:33:34 -08:00
|
|
|
/// Transaction id
|
2020-02-05 11:59:02 +01:00
|
|
|
pub txid: Txid,
|
2020-12-16 11:33:34 -08:00
|
|
|
/// Timestamp
|
2020-02-05 11:59:02 +01:00
|
|
|
pub timestamp: u64,
|
2020-12-16 11:33:34 -08:00
|
|
|
/// Received value (sats)
|
2020-02-05 11:59:02 +01:00
|
|
|
pub received: u64,
|
2020-12-16 11:33:34 -08:00
|
|
|
/// Sent value (sats)
|
2020-02-05 11:59:02 +01:00
|
|
|
pub sent: u64,
|
2020-12-16 11:33:34 -08:00
|
|
|
/// Fee value (sats)
|
2020-08-10 10:49:34 +02:00
|
|
|
pub fees: u64,
|
2020-12-16 11:33:34 -08:00
|
|
|
/// Confirmed in block height, `None` means unconfirmed
|
2020-02-05 11:59:02 +01:00
|
|
|
pub height: Option<u32>,
|
|
|
|
}
|