You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

1010 lines
35 KiB

use crate::model::cfd::{CetStatus, Cfd, CfdState, Dlc, OrderId};
use crate::model::BitMexPriceEventId;
use crate::oracle::Attestation;
use crate::{log_error, model, oracle, try_continue};
use anyhow::{Context, Result};
use async_trait::async_trait;
use bdk::bitcoin::{PublicKey, Script, Txid};
use bdk::descriptor::Descriptor;
use bdk::electrum_client::{ElectrumApi, GetHistoryRes, HeaderNotification};
use bdk::miniscript::DescriptorTrait;
use std::collections::hash_map::Entry;
use std::collections::{BTreeMap, HashMap};
use std::convert::{TryFrom, TryInto};
use std::fmt;
use std::marker::Send;
use std::ops::{Add, RangeInclusive};
use xtra::prelude::StrongMessageChannel;
use xtra_productivity::xtra_productivity;
const FINALITY_CONFIRMATIONS: u32 = 1;
pub struct StartMonitoring {
pub id: OrderId,
pub params: MonitorParams,
}
pub struct CollaborativeSettlement {
pub order_id: OrderId,
pub tx: (Txid, Script),
}
#[derive(Clone)]
pub struct MonitorParams {
lock: (Txid, Descriptor<PublicKey>),
commit: (Txid, Descriptor<PublicKey>),
cets: HashMap<BitMexPriceEventId, Vec<Cet>>,
refund: (Txid, Script, u32),
revoked_commits: Vec<(Txid, Script)>,
event_id: BitMexPriceEventId,
}
pub struct Sync;
pub struct Actor<C = bdk::electrum_client::Client> {
cfds: HashMap<OrderId, MonitorParams>,
event_channel: Box<dyn StrongMessageChannel<Event>>,
client: C,
latest_block_height: BlockHeight,
current_status: BTreeMap<(Txid, Script), ScriptStatus>,
awaiting_status: HashMap<(Txid, Script), Vec<(ScriptStatus, Event)>>,
}
impl Actor<bdk::electrum_client::Client> {
pub async fn new(
electrum_rpc_url: String,
event_channel: Box<dyn StrongMessageChannel<Event>>,
cfds: Vec<Cfd>,
) -> Result<Self> {
let client = bdk::electrum_client::Client::new(&electrum_rpc_url)
.context("Failed to initialize Electrum RPC client")?;
// Initially fetch the latest block for storing the height.
// We do not act on this subscription after this call.
let latest_block = client
.block_headers_subscribe()
.context("Failed to subscribe to header notifications")?;
let mut actor = Self {
cfds: HashMap::new(),
event_channel,
client,
latest_block_height: BlockHeight::try_from(latest_block)?,
current_status: BTreeMap::default(),
awaiting_status: HashMap::default(),
};
for cfd in cfds {
match cfd.state.clone() {
// In PendingOpen we know the complete dlc setup and assume that the lock transaction will be published
CfdState::PendingOpen { dlc, .. } => {
let params = MonitorParams::new(dlc.clone(), cfd.refund_timelock_in_blocks(), cfd.order.oracle_event_id);
actor.cfds.insert(cfd.order.id, params.clone());
actor.monitor_all(&params, cfd.order.id);
}
CfdState::Open { dlc, .. } | CfdState::PendingCommit { dlc, .. } => {
let params = MonitorParams::new(dlc.clone(), cfd.refund_timelock_in_blocks(), cfd.order.oracle_event_id);
actor.cfds.insert(cfd.order.id, params.clone());
actor.monitor_commit_finality(&params, cfd.order.id);
actor.monitor_commit_cet_timelock(&params, cfd.order.id);
actor.monitor_commit_refund_timelock(&params, cfd.order.id);
actor.monitor_refund_finality(&params,cfd.order.id);
if let Some(model::cfd::CollaborativeSettlement { tx, ..}
) = cfd.state.get_collaborative_close() {
let close_params = (tx.txid(),
tx.output.first().context("transaction has zero outputs")?.script_pubkey.clone());
actor.monitor_close_finality(close_params,cfd.order.id);
}
}
CfdState::OpenCommitted { dlc, cet_status, .. } => {
let params = MonitorParams::new(dlc.clone(), cfd.refund_timelock_in_blocks(), cfd.order.oracle_event_id);
actor.cfds.insert(cfd.order.id, params.clone());
match cet_status {
CetStatus::Unprepared => {
actor.monitor_commit_cet_timelock(&params, cfd.order.id);
actor.monitor_commit_refund_timelock(&params, cfd.order.id);
actor.monitor_refund_finality(&params,cfd.order.id);
}
CetStatus::OracleSigned(attestation) => {
actor.monitor_cet_finality(map_cets(dlc.cets), attestation.into(), cfd.order.id)?;
actor.monitor_commit_cet_timelock(&params, cfd.order.id);
actor.monitor_commit_refund_timelock(&params, cfd.order.id);
actor.monitor_refund_finality(&params,cfd.order.id);
}
CetStatus::TimelockExpired => {
actor.monitor_commit_refund_timelock(&params, cfd.order.id);
actor.monitor_refund_finality(&params,cfd.order.id);
}
CetStatus::Ready(attestation) => {
actor.monitor_cet_finality(map_cets(dlc.cets), attestation.into(), cfd.order.id)?;
actor.monitor_commit_refund_timelock(&params, cfd.order.id);
actor.monitor_refund_finality(&params,cfd.order.id);
}
}
}
CfdState::PendingCet { dlc, attestation, .. } => {
let params = MonitorParams::new(dlc.clone(), cfd.refund_timelock_in_blocks(), cfd.order.oracle_event_id);
actor.cfds.insert(cfd.order.id, params.clone());
actor.monitor_cet_finality(map_cets(dlc.cets), attestation.into(), cfd.order.id)?;
actor.monitor_commit_refund_timelock(&params, cfd.order.id);
actor.monitor_refund_finality(&params,cfd.order.id);
}
CfdState::PendingRefund { dlc, .. } => {
let params = MonitorParams::new(dlc.clone(), cfd.refund_timelock_in_blocks(), cfd.order.oracle_event_id);
actor.cfds.insert(cfd.order.id, params.clone());
actor.monitor_commit_refund_timelock(&params, cfd.order.id);
actor.monitor_refund_finality(&params,cfd.order.id);
}
// too early to monitor
CfdState::OutgoingOrderRequest { .. }
| CfdState::IncomingOrderRequest { .. }
| CfdState::Accepted { .. }
| CfdState::ContractSetup { .. }
// final states
| CfdState::Closed { .. }
| CfdState::Rejected { .. }
| CfdState::Refunded { .. }
| CfdState::SetupFailed { .. } => ()
}
}
Ok(actor)
}
}
impl<C> Actor<C>
where
C: bdk::electrum_client::ElectrumApi,
{
fn monitor_all(&mut self, params: &MonitorParams, order_id: OrderId) {
self.monitor_lock_finality(params, order_id);
self.monitor_commit_finality(params, order_id);
self.monitor_commit_cet_timelock(params, order_id);
self.monitor_commit_refund_timelock(params, order_id);
self.monitor_refund_finality(params, order_id);
self.monitor_revoked_commit_transactions(params, order_id);
}
fn monitor_lock_finality(&mut self, params: &MonitorParams, order_id: OrderId) {
self.awaiting_status
.entry((params.lock.0, params.lock.1.script_pubkey()))
.or_default()
.push((ScriptStatus::finality(), Event::LockFinality(order_id)));
}
fn monitor_commit_finality(&mut self, params: &MonitorParams, order_id: OrderId) {
self.awaiting_status
.entry((params.commit.0, params.commit.1.script_pubkey()))
.or_default()
.push((ScriptStatus::finality(), Event::CommitFinality(order_id)));
}
fn monitor_close_finality(&mut self, close_params: (Txid, Script), order_id: OrderId) {
self.awaiting_status
.entry(close_params)
.or_default()
.push((ScriptStatus::finality(), Event::CloseFinality(order_id)));
}
fn monitor_commit_cet_timelock(&mut self, params: &MonitorParams, order_id: OrderId) {
self.awaiting_status
.entry((params.commit.0, params.commit.1.script_pubkey()))
.or_default()
.push((
ScriptStatus::with_confirmations(Cfd::CET_TIMELOCK),
Event::CetTimelockExpired(order_id),
));
}
fn monitor_commit_refund_timelock(&mut self, params: &MonitorParams, order_id: OrderId) {
self.awaiting_status
.entry((params.commit.0, params.commit.1.script_pubkey()))
.or_default()
.push((
ScriptStatus::with_confirmations(params.refund.2),
Event::RefundTimelockExpired(order_id),
));
}
fn monitor_refund_finality(&mut self, params: &MonitorParams, order_id: OrderId) {
self.awaiting_status
.entry((params.refund.0, params.refund.1.clone()))
.or_default()
.push((ScriptStatus::finality(), Event::RefundFinality(order_id)));
}
fn monitor_cet_finality(
&mut self,
cets: HashMap<BitMexPriceEventId, Vec<Cet>>,
attestation: Attestation,
order_id: OrderId,
) -> Result<()> {
let attestation_id = attestation.id;
let cets = cets.get(&attestation_id).with_context(|| {
format!(
"No CET for oracle event {} and cfd with order id {}",
attestation_id, order_id
)
})?;
let (txid, script_pubkey) = cets
.iter()
.find_map(
|Cet {
txid,
script,
range,
..
}| { range.contains(&attestation.price).then(|| (txid, script)) },
)
.context("No price range match for oracle attestation")?;
self.awaiting_status
.entry((*txid, script_pubkey.clone()))
.or_default()
.push((ScriptStatus::finality(), Event::CetFinality(order_id)));
Ok(())
}
fn monitor_revoked_commit_transactions(&mut self, params: &MonitorParams, order_id: OrderId) {
for revoked_commit_tx in params.revoked_commits.iter() {
self.awaiting_status
.entry((revoked_commit_tx.0, revoked_commit_tx.1.clone()))
.or_default()
.push((
ScriptStatus::InMempool,
Event::RevokedTransactionFound(order_id),
));
}
}
async fn sync(&mut self) -> Result<()> {
// Fetch the latest block for storing the height.
// We do not act on this subscription after this call, as we cannot rely on
// subscription push notifications because eventually the Electrum server will
// close the connection and subscriptions are not automatically renewed
// upon renewing the connection.
let latest_block_height = self
.client
.block_headers_subscribe()
.context("Failed to subscribe to header notifications")?
.try_into()?;
tracing::trace!(
"Updating status of {} transactions",
self.awaiting_status.len()
);
let txid_to_script = self
.awaiting_status
.keys()
.cloned()
.collect::<HashMap<_, _>>();
let histories = self
.client
.batch_script_get_history(self.awaiting_status.keys().map(|(_, script)| script))
.context("Failed to get script histories")?;
let mut histories_grouped_by_txid = HashMap::new();
for history in histories {
for response in history {
let txid = response.tx_hash;
let script = match txid_to_script.get(&txid) {
None => {
tracing::trace!(
"Could not find script in own state for txid {}, ignoring",
txid
);
continue;
}
Some(script) => script,
};
histories_grouped_by_txid.insert((txid, script.clone()), response);
}
}
self.update_state(latest_block_height, histories_grouped_by_txid)
.await?;
Ok(())
}
async fn handle_oracle_attestation(&mut self, attestation: oracle::Attestation) -> Result<()> {
for (order_id, MonitorParams { cets, .. }) in self
.cfds
.clone()
.into_iter()
.filter(|(_, params)| params.event_id == attestation.id)
{
try_continue!(self.monitor_cet_finality(cets, attestation.clone(), order_id))
}
Ok(())
}
async fn update_state(
&mut self,
latest_block_height: BlockHeight,
histories: HashMap<(Txid, Script), GetHistoryRes>,
) -> Result<()> {
if latest_block_height > self.latest_block_height {
tracing::debug!(
block_height = u32::from(latest_block_height),
"Got notification for new block"
);
self.latest_block_height = latest_block_height;
}
// 1. Decide new status based on script history
let new_status = self
.awaiting_status
.iter()
.map(|(key, _old_status)| {
let new_script_status = match histories.get(key) {
None => ScriptStatus::Unseen,
Some(history_entry) => {
if history_entry.height <= 0 {
ScriptStatus::InMempool
} else {
ScriptStatus::Confirmed(Confirmed::from_inclusion_and_latest_block(
u32::try_from(history_entry.height)
.expect("we checked that height is > 0"),
u32::from(self.latest_block_height),
))
}
}
};
(key.clone(), new_script_status)
})
.collect::<BTreeMap<_, _>>();
// 2. log any changes since our last sync
for ((txid, script), status) in new_status.iter() {
let old = self.current_status.get(&(*txid, script.clone()));
print_status_change(*txid, old, status);
}
// 3. update local state
self.current_status = new_status;
// 4. check for finished monitoring tasks
for ((txid, script), status) in self.current_status.iter() {
match self.awaiting_status.entry((*txid, script.clone())) {
Entry::Vacant(_) => {
unreachable!("we are only fetching the status of scripts we are waiting for")
}
Entry::Occupied(mut occupied) => {
let targets = occupied.insert(Vec::new());
// Split vec into two lists, all the ones for which we reached the target and
// the ones which we need to still monitor
let (reached_monitoring_target, remaining) = targets
.into_iter()
.partition::<Vec<_>, _>(|(target_status, event)| {
tracing::trace!(
"{:?} requires {} and we have {}",
event,
target_status,
status
);
status >= target_status
});
tracing::trace!("{} subscriptions reached their monitoring target, {} remaining for this script", reached_monitoring_target.len(), remaining.len());
// TODO: When reaching finality of a final tx (CET, refund_tx,
// collaborate_close_tx) we have to remove the remaining "competing"
// transactions. This is not critical, but when fetching
// `GetHistoryRes` by script we can have entries that we don't care about
// anymore.
if remaining.is_empty() {
occupied.remove();
} else {
occupied.insert(remaining);
}
for (target_status, event) in reached_monitoring_target {
tracing::info!(%txid, target = %target_status, current = %status, "Bitcoin transaction reached monitoring target");
self.event_channel.send(event).await?;
}
}
}
}
Ok(())
}
}
fn print_status_change(txid: Txid, old: Option<&ScriptStatus>, new: &ScriptStatus) {
match (old, new) {
(None, new_status) if new_status > &ScriptStatus::Unseen => {
tracing::debug!(%txid, status = %new_status, "Found relevant Bitcoin transaction");
}
(Some(old_status), new_status) if old_status != new_status => {
tracing::debug!(%txid, %new_status, %old_status, "Bitcoin transaction status changed");
}
_ => {}
}
}
#[derive(Debug, Copy, Clone, PartialEq, PartialOrd)]
struct Confirmed {
/// The depth of this transaction within the blockchain.
///
/// Will be zero if the transaction is included in the latest block.
depth: u32,
}
impl Confirmed {
fn with_confirmations(blocks: u32) -> Self {
Self { depth: blocks - 1 }
}
/// Compute the depth of a transaction based on its inclusion height and the
/// latest known block.
///
/// Our information about the latest block might be outdated. To avoid an
/// overflow, we make sure the depth is 0 in case the inclusion height
/// exceeds our latest known block,
fn from_inclusion_and_latest_block(inclusion_height: u32, latest_block: u32) -> Self {
let depth = latest_block.saturating_sub(inclusion_height);
Self { depth }
}
fn confirmations(&self) -> u32 {
self.depth + 1
}
}
#[derive(Debug, Copy, Clone, PartialEq, PartialOrd)]
enum ScriptStatus {
Unseen,
InMempool,
Confirmed(Confirmed),
}
impl ScriptStatus {
fn with_confirmations(confirmations: u32) -> Self {
Self::Confirmed(Confirmed::with_confirmations(confirmations))
}
fn finality() -> Self {
Self::with_confirmations(FINALITY_CONFIRMATIONS)
}
}
impl fmt::Display for ScriptStatus {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
match self {
ScriptStatus::Unseen => write!(f, "unseen"),
ScriptStatus::InMempool => write!(f, "in mempool"),
ScriptStatus::Confirmed(inner) => {
write!(f, "confirmed with {} blocks", inner.confirmations())
}
}
}
}
/// Represent a block height, or block number, expressed in absolute block
/// count. E.g. The transaction was included in block #655123, 655123 block
/// after the genesis block.
#[derive(Debug, Copy, Clone, PartialEq, Eq, Ord, PartialOrd)]
struct BlockHeight(u32);
impl From<BlockHeight> for u32 {
fn from(height: BlockHeight) -> Self {
height.0
}
}
impl TryFrom<HeaderNotification> for BlockHeight {
type Error = anyhow::Error;
fn try_from(value: HeaderNotification) -> Result<Self, Self::Error> {
Ok(Self(
value
.height
.try_into()
.context("Failed to fit usize into u32")?,
))
}
}
impl Add<u32> for BlockHeight {
type Output = BlockHeight;
fn add(self, rhs: u32) -> Self::Output {
BlockHeight(self.0 + rhs)
}
}
#[derive(Debug, Clone, PartialEq)]
pub enum Event {
LockFinality(OrderId),
CommitFinality(OrderId),
CloseFinality(OrderId),
CetTimelockExpired(OrderId),
CetFinality(OrderId),
RefundTimelockExpired(OrderId),
RefundFinality(OrderId),
RevokedTransactionFound(OrderId),
}
impl Event {
pub fn order_id(&self) -> OrderId {
let order_id = match self {
Event::LockFinality(order_id) => order_id,
Event::CommitFinality(order_id) => order_id,
Event::CloseFinality(order_id) => order_id,
Event::CetTimelockExpired(order_id) => order_id,
Event::RefundTimelockExpired(order_id) => order_id,
Event::RefundFinality(order_id) => order_id,
Event::CetFinality(order_id) => order_id,
Event::RevokedTransactionFound(order_id) => order_id,
};
*order_id
}
}
impl MonitorParams {
pub fn new(dlc: Dlc, refund_timelock_in_blocks: u32, event_id: BitMexPriceEventId) -> Self {
let script_pubkey = dlc.maker_address.script_pubkey();
MonitorParams {
lock: (dlc.lock.0.txid(), dlc.lock.1),
commit: (dlc.commit.0.txid(), dlc.commit.2),
cets: map_cets(dlc.cets),
refund: (
dlc.refund.0.txid(),
script_pubkey,
refund_timelock_in_blocks,
),
revoked_commits: dlc
.revoked_commit
.iter()
.map(|rev_commit| (rev_commit.txid, rev_commit.script_pubkey.clone()))
.collect(),
event_id,
}
}
}
#[derive(Clone)]
struct Cet {
txid: Txid,
script: Script,
range: RangeInclusive<u64>,
n_bits: usize,
}
impl From<model::cfd::Cet> for Cet {
fn from(cet: model::cfd::Cet) -> Self {
Cet {
txid: cet.tx.txid(),
script: cet.tx.output[0].script_pubkey.clone(),
range: cet.range.clone(),
n_bits: cet.n_bits,
}
}
}
fn map_cets(
cets: HashMap<BitMexPriceEventId, Vec<model::cfd::Cet>>,
) -> HashMap<BitMexPriceEventId, Vec<Cet>> {
cets.into_iter()
.map(|(event_id, cets)| {
(
event_id,
cets.into_iter().map(Cet::from).collect::<Vec<_>>(),
)
})
.collect()
}
impl xtra::Message for Event {
type Result = ();
}
impl xtra::Message for Sync {
type Result = ();
}
impl<C> xtra::Actor for Actor<C> where C: Send + 'static {}
#[xtra_productivity]
impl<C> Actor<C>
where
C: bdk::electrum_client::ElectrumApi + Send + 'static,
{
async fn handle_start_monitoring(
&mut self,
msg: StartMonitoring,
_ctx: &mut xtra::Context<Self>,
) {
let StartMonitoring { id, params } = msg;
self.monitor_all(&params, id);
self.cfds.insert(id, params);
}
fn handle_collaborative_settlement(
&mut self,
collaborative_settlement: CollaborativeSettlement,
) {
self.monitor_close_finality(
collaborative_settlement.tx,
collaborative_settlement.order_id,
);
}
}
#[async_trait]
impl<C> xtra::Handler<Sync> for Actor<C>
where
C: bdk::electrum_client::ElectrumApi + Send + 'static,
{
async fn handle(&mut self, _: Sync, _ctx: &mut xtra::Context<Self>) {
log_error!(self.sync());
}
}
#[async_trait]
impl xtra::Handler<oracle::Attestation> for Actor {
async fn handle(&mut self, msg: oracle::Attestation, _ctx: &mut xtra::Context<Self>) {
log_error!(self.handle_oracle_attestation(msg));
}
}
#[cfg(test)]
mod tests {
use super::*;
use bdk::bitcoin::blockdata::block;
use bdk::electrum_client::{
Batch, Error, GetBalanceRes, GetHeadersRes, GetMerkleRes, ListUnspentRes,
RawHeaderNotification, ServerFeaturesRes,
};
use std::iter::FromIterator;
use tracing_subscriber::prelude::*;
#[tokio::test]
async fn can_handle_multiple_subscriptions_on_the_same_transaction() {
let _guard = tracing_subscriber::fmt()
.with_env_filter("trace")
.with_test_writer()
.set_default();
let (recorder_address, mut recorder_context) =
xtra::Context::<MessageRecordingActor>::new(None);
let mut recorder = MessageRecordingActor::default();
let commit_finality = Event::CommitFinality(OrderId::default());
let refund_expired = Event::RefundTimelockExpired(OrderId::default());
let mut monitor = Actor::for_test(
Box::new(recorder_address),
[(
(txid1(), script1()),
vec![
(ScriptStatus::finality(), commit_finality.clone()),
(
ScriptStatus::with_confirmations(Cfd::CET_TIMELOCK),
refund_expired.clone(),
),
],
)],
);
monitor.client.include_tx(txid1(), 5);
monitor.client.advance_to_height(10);
recorder_context
.handle_while(&mut recorder, monitor.sync())
.await
.unwrap();
assert_eq!(recorder.events[0], commit_finality);
monitor.client.advance_to_height(20);
recorder_context
.handle_while(&mut recorder, monitor.sync())
.await
.unwrap();
assert_eq!(recorder.events[1], refund_expired);
}
#[tokio::test]
async fn update_for_a_script_only_results_in_event_for_corresponding_transaction() {
let _guard = tracing_subscriber::fmt()
.with_env_filter("trace")
.with_test_writer()
.set_default();
let (recorder_address, mut recorder_context) =
xtra::Context::<MessageRecordingActor>::new(None);
let mut recorder = MessageRecordingActor::default();
let cet_finality = Event::CetFinality(OrderId::default());
let refund_finality = Event::RefundFinality(OrderId::default());
let mut monitor = Actor::for_test(
Box::new(recorder_address),
[
(
(txid1(), script1()),
vec![(ScriptStatus::finality(), cet_finality.clone())],
),
(
(txid2(), script1()),
vec![(ScriptStatus::finality(), refund_finality.clone())],
),
],
);
monitor.client.include_tx(txid1(), 5);
recorder_context
.handle_while(&mut recorder, monitor.sync())
.await
.unwrap();
assert!(recorder.events.contains(&cet_finality));
assert!(!recorder.events.contains(&refund_finality));
}
#[tokio::test]
async fn stop_monitoring_after_target_reached() {
let _guard = tracing_subscriber::fmt()
.with_env_filter("trace")
.with_test_writer()
.set_default();
let (recorder_address, mut recorder_context) =
xtra::Context::<MessageRecordingActor>::new(None);
let mut recorder = MessageRecordingActor::default();
let cet_finality = Event::CetFinality(OrderId::default());
let mut monitor = Actor::for_test(
Box::new(recorder_address),
[(
(txid1(), script1()),
vec![(ScriptStatus::finality(), cet_finality.clone())],
)],
);
monitor.client.include_tx(txid1(), 5);
recorder_context
.handle_while(&mut recorder, monitor.sync())
.await
.unwrap();
assert!(recorder.events.contains(&cet_finality));
assert!(monitor.awaiting_status.is_empty());
}
impl Actor<stub::Client> {
#[allow(clippy::type_complexity)]
fn for_test<const N: usize>(
event_channel: Box<dyn StrongMessageChannel<Event>>,
subscriptions: [((Txid, Script), Vec<(ScriptStatus, Event)>); N],
) -> Self {
Actor {
cfds: HashMap::default(),
event_channel,
client: stub::Client::default(),
latest_block_height: BlockHeight(0),
current_status: BTreeMap::default(),
awaiting_status: HashMap::from_iter(subscriptions),
}
}
}
fn txid1() -> Txid {
"1278ef8104c2f63c03d4d52bace29bed28bd5e664e67543735ddc95a39bfdc0f"
.parse()
.unwrap()
}
fn txid2() -> Txid {
"07ade6a49e34ad4cc3ca3f79d78df462685f8f1fbc8a9b05af51ec503ea5b960"
.parse()
.unwrap()
}
fn script1() -> Script {
"6a4c50001d97ca0002d3829148f63cc8ee21241e3f1c5eaee58781dd45a7d814710fac571b92aadff583e85d5a295f61856f469b401efe615657bf040c32f1000065bce011a420ca9ea3657fff154d95d1a95c".parse().unwrap()
}
#[derive(Default)]
struct MessageRecordingActor {
events: Vec<Event>,
}
impl xtra::Actor for MessageRecordingActor {}
#[async_trait]
impl xtra::Handler<Event> for MessageRecordingActor {
async fn handle(&mut self, message: Event, _ctx: &mut xtra::Context<Self>) {
self.events.push(message);
}
}
mod stub {
use super::*;
use bdk::electrum_client::ScriptStatus;
#[derive(Default)]
pub struct Client {
transactions: HashMap<Txid, i32>,
block_height: usize,
}
impl Client {
pub fn include_tx(&mut self, tx: Txid, height: i32) {
self.transactions.insert(tx, height);
}
pub fn advance_to_height(&mut self, height: usize) {
self.block_height = height;
}
}
impl ElectrumApi for Client {
fn block_headers_subscribe(&self) -> Result<HeaderNotification, Error> {
Ok(HeaderNotification {
height: self.block_height,
header: block::BlockHeader {
version: 0,
prev_blockhash: Default::default(),
merkle_root: Default::default(),
time: 0,
bits: 0,
nonce: 0,
},
})
}
fn batch_script_get_history<'s, I>(
&self,
_: I,
) -> Result<Vec<Vec<GetHistoryRes>>, Error>
where
I: IntoIterator<Item = &'s Script> + Clone,
{
Ok(self
.transactions
.iter()
.map(|(tx, included_at)| {
vec![GetHistoryRes {
height: *included_at,
tx_hash: *tx,
fee: None,
}]
})
.collect())
}
fn batch_call(&self, _batch: &Batch) -> Result<Vec<serde_json::Value>, Error> {
unreachable!("This is a test.")
}
fn block_headers_subscribe_raw(&self) -> Result<RawHeaderNotification, Error> {
unreachable!("This is a test.")
}
fn block_headers_pop_raw(&self) -> Result<Option<RawHeaderNotification>, Error> {
unreachable!("This is a test.")
}
fn block_header_raw(&self, _height: usize) -> Result<Vec<u8>, Error> {
unreachable!("This is a test.")
}
fn block_headers(&self, _: usize, _: usize) -> Result<GetHeadersRes, Error> {
unreachable!("This is a test.")
}
fn estimate_fee(&self, _number: usize) -> Result<f64, Error> {
unreachable!("This is a test.")
}
fn relay_fee(&self) -> Result<f64, Error> {
unreachable!("This is a test.")
}
fn script_subscribe(&self, _script: &Script) -> Result<Option<ScriptStatus>, Error> {
unreachable!("This is a test.")
}
fn script_unsubscribe(&self, _script: &Script) -> Result<bool, Error> {
unreachable!("This is a test.")
}
fn script_pop(&self, _: &Script) -> Result<Option<ScriptStatus>, Error> {
unreachable!("This is a test.")
}
fn script_get_balance(&self, _script: &Script) -> Result<GetBalanceRes, Error> {
unreachable!("This is a test.")
}
fn batch_script_get_balance<'s, I>(&self, _: I) -> Result<Vec<GetBalanceRes>, Error>
where
I: IntoIterator<Item = &'s Script> + Clone,
{
unreachable!("This is a test.")
}
fn script_get_history(&self, _script: &Script) -> Result<Vec<GetHistoryRes>, Error> {
unreachable!("This is a test.")
}
fn script_list_unspent(&self, _script: &Script) -> Result<Vec<ListUnspentRes>, Error> {
unreachable!("This is a test.")
}
fn batch_script_list_unspent<'s, I>(
&self,
_: I,
) -> Result<Vec<Vec<ListUnspentRes>>, Error>
where
I: IntoIterator<Item = &'s Script> + Clone,
{
unreachable!("This is a test.")
}
fn transaction_get_raw(&self, _txid: &Txid) -> Result<Vec<u8>, Error> {
unreachable!("This is a test.")
}
fn batch_transaction_get_raw<'t, I>(&self, _txids: I) -> Result<Vec<Vec<u8>>, Error>
where
I: IntoIterator<Item = &'t Txid> + Clone,
{
unreachable!("This is a test.")
}
fn batch_block_header_raw<I>(&self, _heights: I) -> Result<Vec<Vec<u8>>, Error>
where
I: IntoIterator<Item = u32> + Clone,
{
unreachable!("This is a test.")
}
fn batch_estimate_fee<I>(&self, _numbers: I) -> Result<Vec<f64>, Error>
where
I: IntoIterator<Item = usize> + Clone,
{
unreachable!("This is a test.")
}
fn transaction_broadcast_raw(&self, _raw_tx: &[u8]) -> Result<Txid, Error> {
unreachable!("This is a test.")
}
fn transaction_get_merkle(&self, _: &Txid, _: usize) -> Result<GetMerkleRes, Error> {
unreachable!("This is a test.")
}
fn server_features(&self) -> Result<ServerFeaturesRes, Error> {
unreachable!("This is a test.")
}
fn ping(&self) -> Result<(), Error> {
unreachable!("This is a test.")
}
}
}
}