2021-03-17 21:36:29 -04:00
|
|
|
use crate::bitcoin;
|
2021-03-16 04:11:14 -04:00
|
|
|
use crate::bitcoin::wallet::Watchable;
|
2021-02-15 00:09:42 -05:00
|
|
|
use crate::bitcoin::{
|
2021-02-14 20:19:43 -05:00
|
|
|
build_shared_output_descriptor, Address, Amount, BlockHeight, PublicKey, Transaction, TxLock,
|
2021-02-15 00:09:42 -05:00
|
|
|
};
|
2022-08-27 06:26:55 -04:00
|
|
|
use ::bitcoin::util::sighash::SighashCache;
|
2022-11-22 08:39:42 -05:00
|
|
|
use ::bitcoin::{
|
|
|
|
EcdsaSighashType, OutPoint, PackedLockTime, Script, Sequence, Sighash, TxIn, TxOut, Txid,
|
|
|
|
};
|
2021-02-15 00:09:42 -05:00
|
|
|
use anyhow::Result;
|
2022-11-22 08:39:42 -05:00
|
|
|
use bdk::miniscript::Descriptor;
|
2021-02-15 00:09:42 -05:00
|
|
|
use ecdsa_fun::Signature;
|
2021-02-14 20:19:43 -05:00
|
|
|
use serde::{Deserialize, Serialize};
|
2021-03-11 02:16:00 -05:00
|
|
|
use std::cmp::Ordering;
|
2021-03-03 19:28:58 -05:00
|
|
|
use std::collections::HashMap;
|
2021-07-06 02:42:05 -04:00
|
|
|
use std::fmt;
|
2021-03-03 19:28:58 -05:00
|
|
|
use std::ops::Add;
|
2021-02-14 20:19:43 -05:00
|
|
|
|
|
|
|
/// Represent a timelock, expressed in relative block height as defined in
|
|
|
|
/// [BIP68](https://github.com/bitcoin/bips/blob/master/bip-0068.mediawiki).
|
|
|
|
/// E.g. The timelock expires 10 blocks after the reference transaction is
|
|
|
|
/// mined.
|
|
|
|
#[derive(Debug, Copy, Clone, Serialize, Deserialize, Eq, PartialEq)]
|
|
|
|
#[serde(transparent)]
|
|
|
|
pub struct CancelTimelock(u32);
|
|
|
|
|
RPC server for API Interface (#1276)
* saving: implementing internal api shared by cli and rpc server
* writing async rpc methods and using arc for shared struct references
* cleaning up, renamed Init to Context
* saving: cleaning up and initial work for tests
* Respond with bitcoin withdraw txid
* Print RPC server address
* Cleanup, formatting, add `get_seller`, `get_swap_start_date` RPC endpoints
* fixing tests in cli module
* uncommenting and fixing more tests
* split api module and propagate errors with rpc server
* moving methods to api and validating addresses for rpc
* add broadcast channel to handle shutdowns gracefully and prepare for RPC server test
* added files
* Update rpc.rs
* adding new unfinished RPC tests
* updating rpc-server tests
* fixing warnings
* fixing formatting and cargo clippy warnings
* fix missing import in test
* fix: add data_dir to config to make config command work
* set server listen address manually and return file locations in JSON on Config
* Add called api method and swap_id to tracing for context, reduced boilerplate
* Pass server_address properly to RpcServer
* Update Cargo.lock
* dprint fmt
* Add cancel_refund RPC endpoint
* Combine Cmd and Params
* Disallow concurrent swaps
* Use RwLock instead of Mutex to allow for parallel reads and add get_current_swap endpoint
* Return wallet descriptor to RPC API caller
* Append all cli logs to single log file
After careful consideration, I've concluded that it's not practical/possible to ensure that the previous behaviour (one log file per swap) is preserved due to limitations of the tracing-subscriber crate and a big in the built in JSON formatter
* Add get_swap_expired_timelock timelock, other small refactoring
- Add get_swap_expired_timelock endpoint to return expired timelock if one exists. Fails if bitcoin lock tx has not yet published or if swap is already finished.
- Rename current_epoch to expired_timelock to enforce consistent method names
- Add blocks left until current expired timelock expires (next timelock expires) to ExpiredTimelock struct
- Change .expect() to .unwrap() in rpc server method register because those will only fail if we register the same method twice which will never happen
* initiating swaps in a separate task and handling shutdown signals with broadcast queues
* Replace get_swap_start_date, get_seller, get_expired_timelock with one get_swap_info rpc method
* WIP: Struct for concurrent swaps manager
* Ensure correct tracing spans
* Add note regarding Request, Method structs
* Update request.rs
* Add tracing span attribute log_reference_id to logs caused by rpc call
* Sync bitcoin wallet before initial max_giveable call
* use Span::current() to pass down to tracing span to spawned tasks
* Remove unused shutdown channel
* Add `get_monero_recovery_info` RPC endpoint
- Add `get_monero_recovery_info` RPC endpoint
- format PrivateViewKey using Display
* Rename `Method::RawHistory` to `Method::GetRawStates`
* Wait for swap to be suspended after sending signal
* Remove notes
* Add tracing span attribute log_reference_id to logs caused by rpc call
* Sync bitcoin wallet before initial max_giveable call
* use Span::current() to pass down to tracing span to spawned tasks
* Remove unused shutdown channel
* Add `get_monero_recovery_info` RPC endpoint
- Add `get_monero_recovery_info` RPC endpoint
- format PrivateViewKey using Display
* Rename `Method::RawHistory` to `Method::GetRawStates`
* Wait for swap to be suspended after sending signal
* Return additonal info on GetSwapInfo
* Update wallet.rs
* fix compile issues for tests and use serial_test crate
* fix rpc tests, only check for RPC errors and not returned values
* Rename `get_raw_history` tp `get_raw_states`
* Fix typo in rpc server stopped tracing log
* Remove unnecessary success property on suspend_current_swap response
* fixing test_cli_arguments and other tests
* WIP: RPC server integration tests
* WIP: Integration tests for RPC server
* Update rpc tests
* fix compile and warnings in tests/rpc.rs
* test: fix assert
* clippy --fix
* remove otp file
* cargo clippy fixes
* move resume swap initialization code out of spawned task
* Use `in_current_span` to pass down tracing span to spawned tasks
* moving buy_xmr initialization code out of spawned tasks
* cargo fmt
* Moving swap initialization code inside tokio select block to handle swap lock release logic
* Remove unnecessary swap suspension listener from determine_btc_to_swap call in BuyXmr
* Spawn event loop before requesting quote
* Release swap lock after receiving shutdown signal
* Remove inner tokio::select in BuyXmr and Resume
* Improve debug text for swap resume
* Return error to API caller if bid quote request fails
* Print error if one occurs during process invoked by API call
* Return bid quote to API caller
* Use type safe query! macro for database retrieval of states
* Return tx_lock_fee to API caller on GetSwapInfo call
Update request.rs
* Allow API caller to retrieve last synced bitcoin balane and avoid costly sync
* Return restore height on MoneroRecovery command to API Caller
* Include entire error cause-chain in API response
* Add span to bitcoin wallet logs
* Log event loop connection properties as tracing fields
* Wait for background tasks to complete before exiting CLI
* clippy
* specify sqlx patch version explicitly
* remove mem::forget and replace with _guard
* ci: add rpc test job
* test: wrap rpc test in #[cfg(test)]
* add missing tokio::test attribute
* fix and merge rpc tests, parse uuuid and multiaddr from serde_json value
* default Tor socks port to 9050, Cargo fmt
* Update swap/sqlite_dev_setup.sh: add version
Co-authored-by: Byron Hambly <byron@hambly.dev>
* ci: free up space on ubuntu test job
* Update swap/src/bitcoin/wallet.rs
Co-authored-by: Byron Hambly <byron@hambly.dev>
* Update swap/src/bitcoin/wallet.rs
Co-authored-by: Byron Hambly <byron@hambly.dev>
* fmt
---------
Co-authored-by: binarybaron <86064887+binarybaron@users.noreply.github.com>
Co-authored-by: Byron Hambly <byron@hambly.dev>
2024-05-22 09:12:58 -04:00
|
|
|
impl From<CancelTimelock> for u32 {
|
|
|
|
fn from(cancel_timelock: CancelTimelock) -> Self {
|
|
|
|
cancel_timelock.0
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-02-14 20:19:43 -05:00
|
|
|
impl CancelTimelock {
|
|
|
|
pub const fn new(number_of_blocks: u32) -> Self {
|
|
|
|
Self(number_of_blocks)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Add<CancelTimelock> for BlockHeight {
|
|
|
|
type Output = BlockHeight;
|
|
|
|
|
|
|
|
fn add(self, rhs: CancelTimelock) -> Self::Output {
|
|
|
|
self + rhs.0
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-03-11 02:16:00 -05:00
|
|
|
impl PartialOrd<CancelTimelock> for u32 {
|
|
|
|
fn partial_cmp(&self, other: &CancelTimelock) -> Option<Ordering> {
|
|
|
|
self.partial_cmp(&other.0)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl PartialEq<CancelTimelock> for u32 {
|
|
|
|
fn eq(&self, other: &CancelTimelock) -> bool {
|
|
|
|
self.eq(&other.0)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-07-06 02:42:05 -04:00
|
|
|
impl fmt::Display for CancelTimelock {
|
|
|
|
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
|
|
|
|
write!(f, "{} blocks", self.0)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-02-14 20:19:43 -05:00
|
|
|
/// Represent a timelock, expressed in relative block height as defined in
|
|
|
|
/// [BIP68](https://github.com/bitcoin/bips/blob/master/bip-0068.mediawiki).
|
|
|
|
/// E.g. The timelock expires 10 blocks after the reference transaction is
|
|
|
|
/// mined.
|
|
|
|
#[derive(Debug, Copy, Clone, Serialize, Deserialize, Eq, PartialEq)]
|
|
|
|
#[serde(transparent)]
|
|
|
|
pub struct PunishTimelock(u32);
|
|
|
|
|
RPC server for API Interface (#1276)
* saving: implementing internal api shared by cli and rpc server
* writing async rpc methods and using arc for shared struct references
* cleaning up, renamed Init to Context
* saving: cleaning up and initial work for tests
* Respond with bitcoin withdraw txid
* Print RPC server address
* Cleanup, formatting, add `get_seller`, `get_swap_start_date` RPC endpoints
* fixing tests in cli module
* uncommenting and fixing more tests
* split api module and propagate errors with rpc server
* moving methods to api and validating addresses for rpc
* add broadcast channel to handle shutdowns gracefully and prepare for RPC server test
* added files
* Update rpc.rs
* adding new unfinished RPC tests
* updating rpc-server tests
* fixing warnings
* fixing formatting and cargo clippy warnings
* fix missing import in test
* fix: add data_dir to config to make config command work
* set server listen address manually and return file locations in JSON on Config
* Add called api method and swap_id to tracing for context, reduced boilerplate
* Pass server_address properly to RpcServer
* Update Cargo.lock
* dprint fmt
* Add cancel_refund RPC endpoint
* Combine Cmd and Params
* Disallow concurrent swaps
* Use RwLock instead of Mutex to allow for parallel reads and add get_current_swap endpoint
* Return wallet descriptor to RPC API caller
* Append all cli logs to single log file
After careful consideration, I've concluded that it's not practical/possible to ensure that the previous behaviour (one log file per swap) is preserved due to limitations of the tracing-subscriber crate and a big in the built in JSON formatter
* Add get_swap_expired_timelock timelock, other small refactoring
- Add get_swap_expired_timelock endpoint to return expired timelock if one exists. Fails if bitcoin lock tx has not yet published or if swap is already finished.
- Rename current_epoch to expired_timelock to enforce consistent method names
- Add blocks left until current expired timelock expires (next timelock expires) to ExpiredTimelock struct
- Change .expect() to .unwrap() in rpc server method register because those will only fail if we register the same method twice which will never happen
* initiating swaps in a separate task and handling shutdown signals with broadcast queues
* Replace get_swap_start_date, get_seller, get_expired_timelock with one get_swap_info rpc method
* WIP: Struct for concurrent swaps manager
* Ensure correct tracing spans
* Add note regarding Request, Method structs
* Update request.rs
* Add tracing span attribute log_reference_id to logs caused by rpc call
* Sync bitcoin wallet before initial max_giveable call
* use Span::current() to pass down to tracing span to spawned tasks
* Remove unused shutdown channel
* Add `get_monero_recovery_info` RPC endpoint
- Add `get_monero_recovery_info` RPC endpoint
- format PrivateViewKey using Display
* Rename `Method::RawHistory` to `Method::GetRawStates`
* Wait for swap to be suspended after sending signal
* Remove notes
* Add tracing span attribute log_reference_id to logs caused by rpc call
* Sync bitcoin wallet before initial max_giveable call
* use Span::current() to pass down to tracing span to spawned tasks
* Remove unused shutdown channel
* Add `get_monero_recovery_info` RPC endpoint
- Add `get_monero_recovery_info` RPC endpoint
- format PrivateViewKey using Display
* Rename `Method::RawHistory` to `Method::GetRawStates`
* Wait for swap to be suspended after sending signal
* Return additonal info on GetSwapInfo
* Update wallet.rs
* fix compile issues for tests and use serial_test crate
* fix rpc tests, only check for RPC errors and not returned values
* Rename `get_raw_history` tp `get_raw_states`
* Fix typo in rpc server stopped tracing log
* Remove unnecessary success property on suspend_current_swap response
* fixing test_cli_arguments and other tests
* WIP: RPC server integration tests
* WIP: Integration tests for RPC server
* Update rpc tests
* fix compile and warnings in tests/rpc.rs
* test: fix assert
* clippy --fix
* remove otp file
* cargo clippy fixes
* move resume swap initialization code out of spawned task
* Use `in_current_span` to pass down tracing span to spawned tasks
* moving buy_xmr initialization code out of spawned tasks
* cargo fmt
* Moving swap initialization code inside tokio select block to handle swap lock release logic
* Remove unnecessary swap suspension listener from determine_btc_to_swap call in BuyXmr
* Spawn event loop before requesting quote
* Release swap lock after receiving shutdown signal
* Remove inner tokio::select in BuyXmr and Resume
* Improve debug text for swap resume
* Return error to API caller if bid quote request fails
* Print error if one occurs during process invoked by API call
* Return bid quote to API caller
* Use type safe query! macro for database retrieval of states
* Return tx_lock_fee to API caller on GetSwapInfo call
Update request.rs
* Allow API caller to retrieve last synced bitcoin balane and avoid costly sync
* Return restore height on MoneroRecovery command to API Caller
* Include entire error cause-chain in API response
* Add span to bitcoin wallet logs
* Log event loop connection properties as tracing fields
* Wait for background tasks to complete before exiting CLI
* clippy
* specify sqlx patch version explicitly
* remove mem::forget and replace with _guard
* ci: add rpc test job
* test: wrap rpc test in #[cfg(test)]
* add missing tokio::test attribute
* fix and merge rpc tests, parse uuuid and multiaddr from serde_json value
* default Tor socks port to 9050, Cargo fmt
* Update swap/sqlite_dev_setup.sh: add version
Co-authored-by: Byron Hambly <byron@hambly.dev>
* ci: free up space on ubuntu test job
* Update swap/src/bitcoin/wallet.rs
Co-authored-by: Byron Hambly <byron@hambly.dev>
* Update swap/src/bitcoin/wallet.rs
Co-authored-by: Byron Hambly <byron@hambly.dev>
* fmt
---------
Co-authored-by: binarybaron <86064887+binarybaron@users.noreply.github.com>
Co-authored-by: Byron Hambly <byron@hambly.dev>
2024-05-22 09:12:58 -04:00
|
|
|
impl From<PunishTimelock> for u32 {
|
|
|
|
fn from(punish_timelock: PunishTimelock) -> Self {
|
|
|
|
punish_timelock.0
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-02-14 20:19:43 -05:00
|
|
|
impl PunishTimelock {
|
|
|
|
pub const fn new(number_of_blocks: u32) -> Self {
|
|
|
|
Self(number_of_blocks)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Add<PunishTimelock> for BlockHeight {
|
|
|
|
type Output = BlockHeight;
|
|
|
|
|
|
|
|
fn add(self, rhs: PunishTimelock) -> Self::Output {
|
|
|
|
self + rhs.0
|
|
|
|
}
|
|
|
|
}
|
2021-02-15 00:09:42 -05:00
|
|
|
|
2021-03-11 02:16:00 -05:00
|
|
|
impl PartialOrd<PunishTimelock> for u32 {
|
|
|
|
fn partial_cmp(&self, other: &PunishTimelock) -> Option<Ordering> {
|
|
|
|
self.partial_cmp(&other.0)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl PartialEq<PunishTimelock> for u32 {
|
|
|
|
fn eq(&self, other: &PunishTimelock) -> bool {
|
|
|
|
self.eq(&other.0)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-03-16 04:11:14 -04:00
|
|
|
#[derive(Debug)]
|
2021-02-15 00:09:42 -05:00
|
|
|
pub struct TxCancel {
|
|
|
|
inner: Transaction,
|
2022-08-27 06:26:55 -04:00
|
|
|
digest: Sighash,
|
2021-02-15 00:09:42 -05:00
|
|
|
pub(in crate::bitcoin) output_descriptor: Descriptor<::bitcoin::PublicKey>,
|
2021-02-24 21:52:05 -05:00
|
|
|
lock_output_descriptor: Descriptor<::bitcoin::PublicKey>,
|
2021-02-15 00:09:42 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
impl TxCancel {
|
|
|
|
pub fn new(
|
|
|
|
tx_lock: &TxLock,
|
|
|
|
cancel_timelock: CancelTimelock,
|
|
|
|
A: PublicKey,
|
|
|
|
B: PublicKey,
|
2021-04-28 20:59:40 -04:00
|
|
|
spending_fee: Amount,
|
2021-02-15 00:09:42 -05:00
|
|
|
) -> Self {
|
|
|
|
let cancel_output_descriptor = build_shared_output_descriptor(A.0, B.0);
|
|
|
|
|
|
|
|
let tx_in = TxIn {
|
|
|
|
previous_output: tx_lock.as_outpoint(),
|
|
|
|
script_sig: Default::default(),
|
2022-11-22 08:39:42 -05:00
|
|
|
sequence: Sequence(cancel_timelock.0),
|
2022-08-27 06:26:55 -04:00
|
|
|
witness: Default::default(),
|
2021-02-15 00:09:42 -05:00
|
|
|
};
|
|
|
|
|
|
|
|
let tx_out = TxOut {
|
2022-11-22 08:39:42 -05:00
|
|
|
value: tx_lock.lock_amount().to_sat() - spending_fee.to_sat(),
|
2021-02-17 21:33:50 -05:00
|
|
|
script_pubkey: cancel_output_descriptor.script_pubkey(),
|
2021-02-15 00:09:42 -05:00
|
|
|
};
|
|
|
|
|
|
|
|
let transaction = Transaction {
|
|
|
|
version: 2,
|
2022-11-22 08:39:42 -05:00
|
|
|
lock_time: PackedLockTime(0),
|
2021-02-15 00:09:42 -05:00
|
|
|
input: vec![tx_in],
|
|
|
|
output: vec![tx_out],
|
|
|
|
};
|
|
|
|
|
2022-08-27 06:26:55 -04:00
|
|
|
let digest = SighashCache::new(&transaction)
|
|
|
|
.segwit_signature_hash(
|
|
|
|
0, // Only one input: lock_input (lock transaction)
|
|
|
|
&tx_lock.output_descriptor.script_code().expect("scriptcode"),
|
2022-11-22 08:39:42 -05:00
|
|
|
tx_lock.lock_amount().to_sat(),
|
2022-08-27 06:26:55 -04:00
|
|
|
EcdsaSighashType::All,
|
|
|
|
)
|
|
|
|
.expect("sighash");
|
2021-02-15 00:09:42 -05:00
|
|
|
|
|
|
|
Self {
|
|
|
|
inner: transaction,
|
|
|
|
digest,
|
|
|
|
output_descriptor: cancel_output_descriptor,
|
2021-02-24 21:52:05 -05:00
|
|
|
lock_output_descriptor: tx_lock.output_descriptor.clone(),
|
2021-02-15 00:09:42 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn txid(&self) -> Txid {
|
|
|
|
self.inner.txid()
|
|
|
|
}
|
|
|
|
|
2022-08-27 06:26:55 -04:00
|
|
|
pub fn digest(&self) -> Sighash {
|
2021-02-15 00:09:42 -05:00
|
|
|
self.digest
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn amount(&self) -> Amount {
|
|
|
|
Amount::from_sat(self.inner.output[0].value)
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn as_outpoint(&self) -> OutPoint {
|
|
|
|
OutPoint::new(self.inner.txid(), 0)
|
|
|
|
}
|
|
|
|
|
2021-03-17 21:36:29 -04:00
|
|
|
pub fn complete_as_alice(
|
|
|
|
self,
|
|
|
|
a: bitcoin::SecretKey,
|
|
|
|
B: bitcoin::PublicKey,
|
|
|
|
tx_cancel_sig_B: bitcoin::Signature,
|
|
|
|
) -> Result<Transaction> {
|
|
|
|
let sig_a = a.sign(self.digest());
|
|
|
|
let sig_b = tx_cancel_sig_B;
|
|
|
|
|
|
|
|
let tx_cancel = self
|
|
|
|
.add_signatures((a.public(), sig_a), (B, sig_b))
|
|
|
|
.expect("sig_{a,b} to be valid signatures for tx_cancel");
|
|
|
|
|
|
|
|
Ok(tx_cancel)
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn complete_as_bob(
|
|
|
|
self,
|
|
|
|
A: bitcoin::PublicKey,
|
|
|
|
b: bitcoin::SecretKey,
|
|
|
|
tx_cancel_sig_A: bitcoin::Signature,
|
|
|
|
) -> Result<Transaction> {
|
|
|
|
let sig_a = tx_cancel_sig_A;
|
|
|
|
let sig_b = b.sign(self.digest());
|
|
|
|
|
|
|
|
let tx_cancel = self
|
|
|
|
.add_signatures((A, sig_a), (b.public(), sig_b))
|
|
|
|
.expect("sig_{a,b} to be valid signatures for tx_cancel");
|
|
|
|
|
|
|
|
Ok(tx_cancel)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn add_signatures(
|
2021-02-15 00:09:42 -05:00
|
|
|
self,
|
|
|
|
(A, sig_a): (PublicKey, Signature),
|
|
|
|
(B, sig_b): (PublicKey, Signature),
|
|
|
|
) -> Result<Transaction> {
|
|
|
|
let satisfier = {
|
|
|
|
let mut satisfier = HashMap::with_capacity(2);
|
|
|
|
|
|
|
|
let A = ::bitcoin::PublicKey {
|
|
|
|
compressed: true,
|
2022-08-27 06:26:55 -04:00
|
|
|
inner: A.0.into(),
|
2021-02-15 00:09:42 -05:00
|
|
|
};
|
|
|
|
let B = ::bitcoin::PublicKey {
|
|
|
|
compressed: true,
|
2022-08-27 06:26:55 -04:00
|
|
|
inner: B.0.into(),
|
2021-02-15 00:09:42 -05:00
|
|
|
};
|
|
|
|
|
|
|
|
// The order in which these are inserted doesn't matter
|
2023-03-15 08:12:28 -04:00
|
|
|
satisfier.insert(
|
|
|
|
A,
|
|
|
|
::bitcoin::EcdsaSig {
|
|
|
|
sig: sig_a.into(),
|
|
|
|
hash_ty: EcdsaSighashType::All,
|
|
|
|
},
|
|
|
|
);
|
|
|
|
satisfier.insert(
|
|
|
|
B,
|
|
|
|
::bitcoin::EcdsaSig {
|
|
|
|
sig: sig_b.into(),
|
|
|
|
hash_ty: EcdsaSighashType::All,
|
|
|
|
},
|
|
|
|
);
|
2021-02-15 00:09:42 -05:00
|
|
|
|
|
|
|
satisfier
|
|
|
|
};
|
|
|
|
|
|
|
|
let mut tx_cancel = self.inner;
|
2021-02-24 21:52:05 -05:00
|
|
|
self.lock_output_descriptor
|
2021-02-17 21:33:50 -05:00
|
|
|
.satisfy(&mut tx_cancel.input[0], satisfier)?;
|
2021-02-15 00:09:42 -05:00
|
|
|
|
|
|
|
Ok(tx_cancel)
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn build_spend_transaction(
|
|
|
|
&self,
|
|
|
|
spend_address: &Address,
|
2021-02-14 20:19:43 -05:00
|
|
|
sequence: Option<PunishTimelock>,
|
2021-04-28 20:40:04 -04:00
|
|
|
spending_fee: Amount,
|
2021-02-15 00:09:42 -05:00
|
|
|
) -> Transaction {
|
|
|
|
let previous_output = self.as_outpoint();
|
|
|
|
|
2022-11-22 08:39:42 -05:00
|
|
|
let sequence = Sequence(sequence.map(|seq| seq.0).unwrap_or(0xFFFF_FFFF));
|
2021-02-15 00:09:42 -05:00
|
|
|
let tx_in = TxIn {
|
|
|
|
previous_output,
|
|
|
|
script_sig: Default::default(),
|
2022-11-22 08:39:42 -05:00
|
|
|
sequence,
|
2022-08-27 06:26:55 -04:00
|
|
|
witness: Default::default(),
|
2021-02-15 00:09:42 -05:00
|
|
|
};
|
|
|
|
|
|
|
|
let tx_out = TxOut {
|
2022-11-22 08:39:42 -05:00
|
|
|
value: self.amount().to_sat() - spending_fee.to_sat(),
|
2021-02-15 00:09:42 -05:00
|
|
|
script_pubkey: spend_address.script_pubkey(),
|
|
|
|
};
|
|
|
|
|
|
|
|
Transaction {
|
|
|
|
version: 2,
|
2022-11-22 08:39:42 -05:00
|
|
|
lock_time: PackedLockTime(0),
|
2021-02-15 00:09:42 -05:00
|
|
|
input: vec![tx_in],
|
|
|
|
output: vec![tx_out],
|
|
|
|
}
|
|
|
|
}
|
2021-05-03 03:35:41 -04:00
|
|
|
|
|
|
|
pub fn weight() -> usize {
|
|
|
|
596
|
|
|
|
}
|
2021-02-15 00:09:42 -05:00
|
|
|
}
|
2021-03-16 04:11:14 -04:00
|
|
|
|
|
|
|
impl Watchable for TxCancel {
|
|
|
|
fn id(&self) -> Txid {
|
|
|
|
self.txid()
|
|
|
|
}
|
|
|
|
|
|
|
|
fn script(&self) -> Script {
|
|
|
|
self.output_descriptor.script_pubkey()
|
|
|
|
}
|
|
|
|
}
|