@ -2,12 +2,15 @@ use crate::cfd_actors::{self, append_cfd_state, insert_cfd_and_send_to_feed};
use crate ::db ::{ insert_order , load_cfd_by_order_id , load_order_by_id } ;
use crate ::db ::{ insert_order , load_cfd_by_order_id , load_order_by_id } ;
use crate ::model ::cfd ::{
use crate ::model ::cfd ::{
Cfd , CfdState , CfdStateCommon , CollaborativeSettlement , Dlc , Order , OrderId , Origin , Role ,
Cfd , CfdState , CfdStateCommon , CollaborativeSettlement , Dlc , Order , OrderId , Origin , Role ,
RollOverProposal , SettlementKind , SettlementProposal , UpdateCfdProposal , UpdateCfdProposals ,
RollOverProposal , SettlementKind , SettlementProposal , UpdateCfdProposal ,
} ;
} ;
use crate ::model ::{ Price , TakerId , Timestamp , Usd } ;
use crate ::model ::{ Price , TakerId , Timestamp , Usd } ;
use crate ::monitor ::MonitorParams ;
use crate ::monitor ::MonitorParams ;
use crate ::tokio_ext ::FutureExt ;
use crate ::tokio_ext ::FutureExt ;
use crate ::{ log_error , maker_inc_connections , monitor , oracle , setup_contract , wallet , wire } ;
use crate ::{
log_error , maker_inc_connections , monitor , oracle , projection , setup_contract , wallet , wire ,
UpdateCfdProposals ,
} ;
use anyhow ::{ Context as _ , Result } ;
use anyhow ::{ Context as _ , Result } ;
use async_trait ::async_trait ;
use async_trait ::async_trait ;
use bdk ::bitcoin ::secp256k1 ::schnorrsig ;
use bdk ::bitcoin ::secp256k1 ::schnorrsig ;
@ -19,7 +22,6 @@ use sqlx::pool::PoolConnection;
use sqlx ::Sqlite ;
use sqlx ::Sqlite ;
use std ::collections ::HashMap ;
use std ::collections ::HashMap ;
use time ::Duration ;
use time ::Duration ;
use tokio ::sync ::watch ;
use xtra ::prelude ::* ;
use xtra ::prelude ::* ;
pub enum CfdAction {
pub enum CfdAction {
@ -62,9 +64,7 @@ pub struct Actor<O, M, T, W> {
wallet : Address < W > ,
wallet : Address < W > ,
settlement_time_interval_hours : Duration ,
settlement_time_interval_hours : Duration ,
oracle_pk : schnorrsig ::PublicKey ,
oracle_pk : schnorrsig ::PublicKey ,
cfd_feed_actor_inbox : watch ::Sender < Vec < Cfd > > ,
projection_actor : Address < projection ::Actor > ,
order_feed_sender : watch ::Sender < Option < Order > > ,
update_cfd_feed_sender : watch ::Sender < UpdateCfdProposals > ,
takers : Address < T > ,
takers : Address < T > ,
current_order_id : Option < OrderId > ,
current_order_id : Option < OrderId > ,
monitor_actor : Address < M > ,
monitor_actor : Address < M > ,
@ -102,9 +102,7 @@ impl<O, M, T, W> Actor<O, M, T, W> {
wallet : Address < W > ,
wallet : Address < W > ,
settlement_time_interval_hours : Duration ,
settlement_time_interval_hours : Duration ,
oracle_pk : schnorrsig ::PublicKey ,
oracle_pk : schnorrsig ::PublicKey ,
cfd_feed_actor_inbox : watch ::Sender < Vec < Cfd > > ,
projection_actor : Address < projection ::Actor > ,
order_feed_sender : watch ::Sender < Option < Order > > ,
update_cfd_feed_sender : watch ::Sender < UpdateCfdProposals > ,
takers : Address < T > ,
takers : Address < T > ,
monitor_actor : Address < M > ,
monitor_actor : Address < M > ,
oracle_actor : Address < O > ,
oracle_actor : Address < O > ,
@ -115,9 +113,7 @@ impl<O, M, T, W> Actor<O, M, T, W> {
wallet ,
wallet ,
settlement_time_interval_hours ,
settlement_time_interval_hours ,
oracle_pk ,
oracle_pk ,
cfd_feed_actor_inbox ,
projection_actor ,
order_feed_sender ,
update_cfd_feed_sender ,
takers ,
takers ,
current_order_id : None ,
current_order_id : None ,
monitor_actor ,
monitor_actor ,
@ -165,7 +161,7 @@ impl<O, M, T, W> Actor<O, M, T, W> {
taker_id ,
taker_id ,
) ,
) ,
) ;
) ;
self . send_pending_proposals ( ) ? ;
self . send_pending_proposals ( ) . await ? ;
Ok ( ( ) )
Ok ( ( ) )
}
}
@ -189,7 +185,7 @@ impl<O, M, T, W> Actor<O, M, T, W> {
taker_id ,
taker_id ,
) ,
) ,
) ;
) ;
self . send_pending_proposals ( ) ? ;
self . send_pending_proposals ( ) . await ? ;
Ok ( ( ) )
Ok ( ( ) )
}
}
@ -235,21 +231,25 @@ impl<O, M, T, W> Actor<O, M, T, W> {
/// Send pending proposals for the purposes of UI updates.
/// Send pending proposals for the purposes of UI updates.
/// Filters out the TakerIds, as they are an implementation detail inside of
/// Filters out the TakerIds, as they are an implementation detail inside of
/// the actor
/// the actor
fn send_pending_proposals ( & self ) -> Result < ( ) > {
async fn send_pending_proposals ( & self ) -> Result < ( ) > {
Ok ( self . update_cfd_feed_sender . send (
let pending_proposal = self
self . current_pending_proposals
. current_pending_proposals
. iter ( )
. iter ( )
. map ( | ( order_id , ( update_cfd , _ ) ) | ( * order_id , ( update_cfd . clone ( ) ) ) )
. map ( | ( order_id , ( update_cfd , _ ) ) | ( * order_id , ( update_cfd . clone ( ) ) ) )
. collect ( ) ,
. collect ::< UpdateCfdProposals > ( ) ;
) ? )
let _ = self
. projection_actor
. send ( projection ::Update ( pending_proposal ) )
. await ? ;
Ok ( ( ) )
}
}
/// Removes a proposal and updates the update cfd proposals' feed
/// Removes a proposal and updates the update cfd proposals' feed
fn remove_pending_proposal ( & mut self , order_id : & OrderId ) -> Result < ( ) > {
async fn remove_pending_proposal ( & mut self , order_id : & OrderId ) -> Result < ( ) > {
if self . current_pending_proposals . remove ( order_id ) . is_none ( ) {
if self . current_pending_proposals . remove ( order_id ) . is_none ( ) {
anyhow ::bail ! ( "Could not find proposal with order id: {}" , & order_id )
anyhow ::bail ! ( "Could not find proposal with order id: {}" , & order_id )
}
}
self . send_pending_proposals ( ) ? ;
self . send_pending_proposals ( ) . await ? ;
Ok ( ( ) )
Ok ( ( ) )
}
}
@ -283,26 +283,16 @@ where
{
{
async fn handle_commit ( & mut self , order_id : OrderId ) -> Result < ( ) > {
async fn handle_commit ( & mut self , order_id : OrderId ) -> Result < ( ) > {
let mut conn = self . db . acquire ( ) . await ? ;
let mut conn = self . db . acquire ( ) . await ? ;
cfd_actors ::handle_commit (
cfd_actors ::handle_commit ( order_id , & mut conn , & self . wallet , & self . projection_actor )
order_id ,
. await ? ;
& mut conn ,
& self . wallet ,
& self . cfd_feed_actor_inbox ,
)
. await ? ;
Ok ( ( ) )
Ok ( ( ) )
}
}
async fn handle_monitoring_event ( & mut self , event : monitor ::Event ) -> Result < ( ) > {
async fn handle_monitoring_event ( & mut self , event : monitor ::Event ) -> Result < ( ) > {
let mut conn = self . db . acquire ( ) . await ? ;
let mut conn = self . db . acquire ( ) . await ? ;
cfd_actors ::handle_monitoring_event (
cfd_actors ::handle_monitoring_event ( event , & mut conn , & self . wallet , & self . projection_actor )
event ,
. await ? ;
& mut conn ,
& self . wallet ,
& self . cfd_feed_actor_inbox ,
)
. await ? ;
Ok ( ( ) )
Ok ( ( ) )
}
}
@ -312,7 +302,7 @@ where
attestation ,
attestation ,
& mut conn ,
& mut conn ,
& self . wallet ,
& self . wallet ,
& self . cfd_feed_actor_inbox ,
& self . projection_actor ,
)
)
. await ? ;
. await ? ;
Ok ( ( ) )
Ok ( ( ) )
@ -361,11 +351,13 @@ where
self . current_agreed_proposals
self . current_agreed_proposals
. insert ( order_id , self . get_settlement_proposal ( order_id ) ? ) ;
. insert ( order_id , self . get_settlement_proposal ( order_id ) ? ) ;
self . remove_pending_proposal ( & order_id )
self . remove_pending_proposal ( & order_id )
. await
. context ( "accepted settlement" ) ? ;
. context ( "accepted settlement" ) ? ;
}
}
Err ( e ) = > {
Err ( e ) = > {
tracing ::warn ! ( "Failed to notify taker of accepted settlement: {}" , e ) ;
tracing ::warn ! ( "Failed to notify taker of accepted settlement: {}" , e ) ;
self . remove_pending_proposal ( & order_id )
self . remove_pending_proposal ( & order_id )
. await
. context ( "accepted settlement" ) ? ;
. context ( "accepted settlement" ) ? ;
}
}
}
}
@ -381,6 +373,7 @@ where
// clean-up state ahead of sending to ensure consistency in case we fail to deliver the
// clean-up state ahead of sending to ensure consistency in case we fail to deliver the
// message
// message
self . remove_pending_proposal ( & order_id )
self . remove_pending_proposal ( & order_id )
. await
. context ( "rejected settlement" ) ? ;
. context ( "rejected settlement" ) ? ;
self . takers
self . takers
@ -413,6 +406,7 @@ where
// clean-up state ahead of sending to ensure consistency in case we fail to deliver the
// clean-up state ahead of sending to ensure consistency in case we fail to deliver the
// message
// message
self . remove_pending_proposal ( & order_id )
self . remove_pending_proposal ( & order_id )
. await
. context ( "rejected roll_over" ) ? ;
. context ( "rejected roll_over" ) ? ;
self . takers
self . takers
@ -472,7 +466,7 @@ where
self . takers
self . takers
. send ( maker_inc_connections ::BroadcastOrder ( None ) )
. send ( maker_inc_connections ::BroadcastOrder ( None ) )
. await ? ;
. await ? ;
self . order_feed_sender . send ( None ) ? ;
self . projection_actor . send ( projection ::Update ( None ) ) . await ? ;
// 3. Insert CFD in DB
// 3. Insert CFD in DB
let cfd = Cfd ::new (
let cfd = Cfd ::new (
@ -485,7 +479,7 @@ where
taker_id ,
taker_id ,
} ,
} ,
) ;
) ;
insert_cfd_and_send_to_feed ( & cfd , & mut conn , & self . cfd_feed_actor_inbox ) . await ? ;
insert_cfd_and_send_to_feed ( & cfd , & mut conn , & self . projection_actor ) . await ? ;
// 4. check if order has acceptable amounts and if not reject the cfd
// 4. check if order has acceptable amounts and if not reject the cfd
// Since rejection is tied to the cfd state at the moment we can only do this after creating
// Since rejection is tied to the cfd state at the moment we can only do this after creating
@ -533,7 +527,7 @@ where
) -> Result < ( ) > {
) -> Result < ( ) > {
cfd . state = CfdState ::rejected ( ) ;
cfd . state = CfdState ::rejected ( ) ;
append_cfd_state ( & cfd , & mut conn , & self . cfd_feed_actor_inbox ) . await ? ;
append_cfd_state ( & cfd , & mut conn , & self . projection_actor ) . await ? ;
self . takers
self . takers
. send ( maker_inc_connections ::TakerMessage {
. send ( maker_inc_connections ::TakerMessage {
@ -598,7 +592,7 @@ where
// 4. Insert that we are in contract setup and refresh our own feed
// 4. Insert that we are in contract setup and refresh our own feed
cfd . state = CfdState ::contract_setup ( ) ;
cfd . state = CfdState ::contract_setup ( ) ;
append_cfd_state ( & cfd , & mut conn , & self . cfd_feed_actor_inbox ) . await ? ;
append_cfd_state ( & cfd , & mut conn , & self . projection_actor ) . await ? ;
// 5. Spawn away the contract setup
// 5. Spawn away the contract setup
let ( sender , receiver ) = mpsc ::unbounded ( ) ;
let ( sender , receiver ) = mpsc ::unbounded ( ) ;
@ -665,7 +659,7 @@ where
info : e . to_string ( ) ,
info : e . to_string ( ) ,
} ;
} ;
append_cfd_state ( & cfd , & mut conn , & self . cfd_feed_actor_inbox ) . await ? ;
append_cfd_state ( & cfd , & mut conn , & self . projection_actor ) . await ? ;
return Err ( e ) ;
return Err ( e ) ;
}
}
@ -677,7 +671,7 @@ where
attestation : None ,
attestation : None ,
} ;
} ;
append_cfd_state ( & cfd , & mut conn , & self . cfd_feed_actor_inbox ) . await ? ;
append_cfd_state ( & cfd , & mut conn , & self . projection_actor ) . await ? ;
let txid = self
let txid = self
. wallet
. wallet
@ -802,6 +796,7 @@ where
} ;
} ;
self . remove_pending_proposal ( & order_id )
self . remove_pending_proposal ( & order_id )
. await
. context ( "accepted roll_over" ) ? ;
. context ( "accepted roll_over" ) ? ;
Ok ( ( ) )
Ok ( ( ) )
}
}
@ -827,7 +822,7 @@ where
attestation : None ,
attestation : None ,
collaborative_close : None ,
collaborative_close : None ,
} ;
} ;
append_cfd_state ( & cfd , & mut conn , & self . cfd_feed_actor_inbox ) . await ? ;
append_cfd_state ( & cfd , & mut conn , & self . projection_actor ) . await ? ;
self . monitor_actor
self . monitor_actor
. send ( monitor ::StartMonitoring {
. send ( monitor ::StartMonitoring {
@ -887,7 +882,7 @@ where
own_script_pubkey . clone ( ) ,
own_script_pubkey . clone ( ) ,
proposal . price ,
proposal . price ,
) ? ) ? ;
) ? ) ? ;
append_cfd_state ( & cfd , & mut conn , & self . cfd_feed_actor_inbox ) . await ? ;
append_cfd_state ( & cfd , & mut conn , & self . projection_actor ) . await ? ;
let spend_tx = dlc . finalize_spend_transaction ( ( tx , sig_maker ) , sig_taker ) ? ;
let spend_tx = dlc . finalize_spend_transaction ( ( tx , sig_maker ) , sig_taker ) ? ;
@ -976,7 +971,9 @@ where
self . current_order_id . replace ( order . id ) ;
self . current_order_id . replace ( order . id ) ;
// 3. Notify UI via feed
// 3. Notify UI via feed
self . order_feed_sender . send ( Some ( order . clone ( ) ) ) ? ;
self . projection_actor
. send ( projection ::Update ( Some ( order . clone ( ) ) ) )
. await ? ;
// 4. Inform connected takers
// 4. Inform connected takers
self . takers
self . takers