Browse Source

Merge #227

227: Handle collab settlement response from the maker r=klochowicz a=klochowicz

- store TakerId for proposals to be able to respond to specific taker about
  settlement proposal when the user takes action
- send the collab settlement response (accept or reject) from the maker
- handle rejection (remove from the map), CFD state in the UI returns to "Open"

Co-authored-by: Mariusz Klochowicz <mariusz@klochowicz.com>
feature/integration-tests
bors[bot] 3 years ago
committed by GitHub
parent
commit
7232269205
No known key found for this signature in database GPG Key ID: 4AEE18F83AFDEB23
  1. 160
      daemon/src/maker_cfd.rs
  2. 10
      daemon/src/maker_inc_connections.rs
  3. 37
      daemon/src/taker_cfd.rs
  4. 4
      daemon/src/wire.rs

160
daemon/src/maker_cfd.rs

@ -85,7 +85,8 @@ pub struct Actor {
setup_state: SetupState, setup_state: SetupState,
latest_announcements: Option<BTreeMap<OracleEventId, oracle::Announcement>>, latest_announcements: Option<BTreeMap<OracleEventId, oracle::Announcement>>,
oracle_actor: Address<oracle::Actor<Actor, monitor::Actor<Actor>>>, oracle_actor: Address<oracle::Actor<Actor, monitor::Actor<Actor>>>,
current_pending_proposals: UpdateCfdProposals, // Maker needs to also store TakerId to be able to send a reply back
current_pending_proposals: HashMap<OrderId, (UpdateCfdProposal, TakerId)>,
} }
enum SetupState { enum SetupState {
@ -126,10 +127,33 @@ impl Actor {
} }
} }
/// Send pending proposals for the purposes of UI updates.
/// Filters out the TakerIds, as they are an implementation detail inside of
/// the actor
fn send_pending_proposals(&self) -> Result<()> { fn send_pending_proposals(&self) -> Result<()> {
Ok(self Ok(self.update_cfd_feed_sender.send(
.update_cfd_feed_sender self.current_pending_proposals
.send(self.current_pending_proposals.clone())?) .iter()
.map(|(order_id, (update_cfd, _))| (*order_id, (update_cfd.clone())))
.collect(),
)?)
}
fn get_taker_id_of_proposal(&self, order_id: &OrderId) -> Result<TakerId> {
let (_, taker_id) = self
.current_pending_proposals
.get(order_id)
.context("Could not find proposal for given order id")?;
Ok(*taker_id)
}
/// Removes a proposal and updates the update cfd proposals' feed
fn remove_pending_proposal(&mut self, order_id: &OrderId) -> Result<()> {
if self.current_pending_proposals.remove(order_id).is_none() {
anyhow::bail!("Could not find proposal with order id: {}", &order_id)
}
self.send_pending_proposals()?;
Ok(())
} }
async fn handle_new_order( async fn handle_new_order(
@ -193,35 +217,50 @@ impl Actor {
Ok(()) Ok(())
} }
async fn handle_propose_settlement(&mut self, proposal: SettlementProposal) -> Result<()> { async fn handle_propose_settlement(
&mut self,
taker_id: TakerId,
proposal: SettlementProposal,
) -> Result<()> {
tracing::info!( tracing::info!(
"Received settlement proposal from the taker: {:?}", "Received settlement proposal from the taker: {:?}",
proposal proposal
); );
self.current_pending_proposals.insert( self.current_pending_proposals.insert(
proposal.order_id, proposal.order_id,
UpdateCfdProposal::Settlement { (
proposal, UpdateCfdProposal::Settlement {
direction: SettlementKind::Incoming, proposal,
}, direction: SettlementKind::Incoming,
},
taker_id,
),
); );
self.send_pending_proposals()?; self.send_pending_proposals()?;
Ok(()) Ok(())
} }
async fn handle_propose_roll_over(&mut self, proposal: RollOverProposal) -> Result<()> { async fn handle_propose_roll_over(
&mut self,
taker_id: TakerId,
proposal: RollOverProposal,
) -> Result<()> {
tracing::info!( tracing::info!(
"Received proposal from the taker: {:?} to roll over order {}", "Received proposal from the taker {}: {:?} to roll over order {}",
taker_id,
proposal, proposal,
proposal.order_id proposal.order_id
); );
self.current_pending_proposals.insert( self.current_pending_proposals.insert(
proposal.order_id, proposal.order_id,
UpdateCfdProposal::RollOverProposal { (
proposal, UpdateCfdProposal::RollOverProposal {
direction: SettlementKind::Incoming, proposal,
}, direction: SettlementKind::Incoming,
},
taker_id,
),
); );
self.send_pending_proposals()?; self.send_pending_proposals()?;
@ -516,47 +555,57 @@ impl Actor {
async fn handle_accept_settlement(&mut self, order_id: OrderId) -> Result<()> { async fn handle_accept_settlement(&mut self, order_id: OrderId) -> Result<()> {
tracing::debug!(%order_id, "Maker accepts a settlement proposal" ); tracing::debug!(%order_id, "Maker accepts a settlement proposal" );
// TODO: Initiate the settlement
self.current_pending_proposals let taker_id = self.get_taker_id_of_proposal(&order_id)?;
.remove(&order_id)
.context("Could not find proposal for given order id")?; // TODO: Initiate the settlement - should we start calculating the
self.send_pending_proposals()?; // signature here?
self.takers
.do_send_async(maker_inc_connections::TakerMessage {
taker_id,
command: TakerCommand::NotifySettlementAccepted { id: order_id },
})
.await?;
self.remove_pending_proposal(&order_id)
.context("accepted settlement")?;
Ok(()) Ok(())
} }
async fn handle_reject_settlement(&mut self, order_id: OrderId) -> Result<()> { async fn handle_reject_settlement(&mut self, order_id: OrderId) -> Result<()> {
tracing::debug!(%order_id, "Maker rejects a settlement proposal" ); tracing::debug!(%order_id, "Maker rejects a settlement proposal" );
// TODO: Handle rejection offer:
// - notify the taker that the settlement was rejected
self.current_pending_proposals let taker_id = self.get_taker_id_of_proposal(&order_id)?;
.remove(&order_id)
.context("Could not find proposal for given order id")?; self.takers
self.send_pending_proposals()?; .do_send_async(maker_inc_connections::TakerMessage {
taker_id,
command: TakerCommand::NotifySettlementRejected { id: order_id },
})
.await?;
self.remove_pending_proposal(&order_id)
.context("rejected settlement")?;
Ok(()) Ok(())
} }
async fn handle_accept_roll_over(&mut self, order_id: OrderId) -> Result<()> { async fn handle_accept_roll_over(&mut self, order_id: OrderId) -> Result<()> {
tracing::debug!(%order_id, "Maker accepts a roll over proposal" ); tracing::debug!(%order_id, "Maker accepts a rollover proposal" );
// TODO: Initiate the roll over logic // TODO: Initiate the roll over logic
self.current_pending_proposals self.remove_pending_proposal(&order_id)
.remove(&order_id) .context("accepted rollover")?;
.context("Could not find roll over proposal for given order id")?;
self.send_pending_proposals()?;
Ok(()) Ok(())
} }
async fn handle_reject_roll_over(&mut self, order_id: OrderId) -> Result<()> { async fn handle_reject_roll_over(&mut self, order_id: OrderId) -> Result<()> {
tracing::debug!(%order_id, "Maker rejects a roll over proposal" ); tracing::debug!(%order_id, "Maker rejects a rollover proposal" );
// TODO: Handle rejection and notify the taker that the roll over was rejected // TODO: Handle rejection and notify the taker that the rollover was rejected
self.current_pending_proposals self.remove_pending_proposal(&order_id)
.remove(&order_id) .context("rejected rollover")?;
.context("Could not find roll over proposal for given order id")?;
self.send_pending_proposals()?;
Ok(()) Ok(())
} }
@ -726,16 +775,13 @@ impl Handler<monitor::Event> for Actor {
#[async_trait] #[async_trait]
impl Handler<TakerStreamMessage> for Actor { impl Handler<TakerStreamMessage> for Actor {
async fn handle(&mut self, msg: TakerStreamMessage, _ctx: &mut Context<Self>) -> KeepRunning { async fn handle(&mut self, msg: TakerStreamMessage, _ctx: &mut Context<Self>) -> KeepRunning {
let TakerStreamMessage { let TakerStreamMessage { taker_id, item } = msg;
taker_id: taker,
item,
} = msg;
let msg = match item { let msg = match item {
Ok(msg) => msg, Ok(msg) => msg,
Err(e) => { Err(e) => {
tracing::warn!( tracing::warn!(
"Error while receiving message from taker {}: {:#}", "Error while receiving message from taker {}: {:#}",
taker, taker_id,
e e
); );
return KeepRunning::Yes; return KeepRunning::Yes;
@ -744,7 +790,7 @@ impl Handler<TakerStreamMessage> for Actor {
match msg { match msg {
wire::TakerToMaker::TakeOrder { order_id, quantity } => { wire::TakerToMaker::TakeOrder { order_id, quantity } => {
log_error!(self.handle_take_order(taker, order_id, quantity)) log_error!(self.handle_take_order(taker_id, order_id, quantity))
} }
wire::TakerToMaker::ProposeSettlement { wire::TakerToMaker::ProposeSettlement {
order_id, order_id,
@ -752,24 +798,30 @@ impl Handler<TakerStreamMessage> for Actor {
taker, taker,
maker, maker,
} => { } => {
log_error!(self.handle_propose_settlement(SettlementProposal { log_error!(self.handle_propose_settlement(
order_id, taker_id,
timestamp, SettlementProposal {
taker, order_id,
maker timestamp,
})) taker,
maker
}
))
} }
wire::TakerToMaker::Protocol(msg) => { wire::TakerToMaker::Protocol(msg) => {
log_error!(self.handle_inc_protocol_msg(taker, msg)) log_error!(self.handle_inc_protocol_msg(taker_id, msg))
} }
TakerToMaker::ProposeRollOver { TakerToMaker::ProposeRollOver {
order_id, order_id,
timestamp, timestamp,
} => { } => {
log_error!(self.handle_propose_roll_over(RollOverProposal { log_error!(self.handle_propose_roll_over(
order_id, taker_id,
timestamp, RollOverProposal {
})) order_id,
timestamp,
}
))
} }
} }

10
daemon/src/maker_inc_connections.rs

@ -22,6 +22,8 @@ pub enum TakerCommand {
NotifyInvalidOrderId { id: OrderId }, NotifyInvalidOrderId { id: OrderId },
NotifyOrderAccepted { id: OrderId }, NotifyOrderAccepted { id: OrderId },
NotifyOrderRejected { id: OrderId }, NotifyOrderRejected { id: OrderId },
NotifySettlementAccepted { id: OrderId },
NotifySettlementRejected { id: OrderId },
Protocol(wire::SetupMsg), Protocol(wire::SetupMsg),
} }
@ -94,6 +96,14 @@ impl Actor {
self.send_to_taker(msg.taker_id, wire::MakerToTaker::RejectOrder(id)) self.send_to_taker(msg.taker_id, wire::MakerToTaker::RejectOrder(id))
.await?; .await?;
} }
TakerCommand::NotifySettlementAccepted { id } => {
self.send_to_taker(msg.taker_id, wire::MakerToTaker::ConfirmSettlement(id))
.await?;
}
TakerCommand::NotifySettlementRejected { id } => {
self.send_to_taker(msg.taker_id, wire::MakerToTaker::RejectSettlement(id))
.await?;
}
TakerCommand::Protocol(setup_msg) => { TakerCommand::Protocol(setup_msg) => {
self.send_to_taker(msg.taker_id, wire::MakerToTaker::Protocol(setup_msg)) self.send_to_taker(msg.taker_id, wire::MakerToTaker::Protocol(setup_msg))
.await?; .await?;

37
daemon/src/taker_cfd.rs

@ -107,6 +107,15 @@ impl Actor {
.send(self.current_pending_proposals.clone())?) .send(self.current_pending_proposals.clone())?)
} }
/// Removes a proposal and updates the update cfd proposals' feed
fn remove_pending_proposal(&mut self, order_id: &OrderId) -> Result<()> {
if self.current_pending_proposals.remove(order_id).is_none() {
anyhow::bail!("Could not find proposal with order id: {}", &order_id)
}
self.send_pending_update_proposals()?;
Ok(())
}
async fn handle_take_offer(&mut self, order_id: OrderId, quantity: Usd) -> Result<()> { async fn handle_take_offer(&mut self, order_id: OrderId, quantity: Usd) -> Result<()> {
let mut conn = self.db.acquire().await?; let mut conn = self.db.acquire().await?;
@ -311,6 +320,28 @@ impl Actor {
Ok(()) Ok(())
} }
async fn handle_settlement_accepted(
&mut self,
order_id: OrderId,
_ctx: &mut Context<Self>,
) -> Result<()> {
tracing::info!(%order_id, "Settlement proposal got accepted");
// TODO: Initiate collaborative settlement
self.remove_pending_proposal(&order_id)?;
Ok(())
}
async fn handle_settlement_rejected(&mut self, order_id: OrderId) -> Result<()> {
tracing::info!(%order_id, "Settlement proposal got rejected");
self.remove_pending_proposal(&order_id)?;
Ok(())
}
async fn handle_inc_protocol_msg(&mut self, msg: SetupMsg) -> Result<()> { async fn handle_inc_protocol_msg(&mut self, msg: SetupMsg) -> Result<()> {
match &mut self.setup_state { match &mut self.setup_state {
SetupState::Active { sender } => { SetupState::Active { sender } => {
@ -530,6 +561,12 @@ impl Handler<MakerStreamMessage> for Actor {
wire::MakerToTaker::RejectOrder(order_id) => { wire::MakerToTaker::RejectOrder(order_id) => {
log_error!(self.handle_order_rejected(order_id)) log_error!(self.handle_order_rejected(order_id))
} }
wire::MakerToTaker::ConfirmSettlement(order_id) => {
log_error!(self.handle_settlement_accepted(order_id, ctx))
}
wire::MakerToTaker::RejectSettlement(order_id) => {
log_error!(self.handle_settlement_rejected(order_id))
}
wire::MakerToTaker::InvalidOrderId(_) => todo!(), wire::MakerToTaker::InvalidOrderId(_) => todo!(),
wire::MakerToTaker::Protocol(setup_msg) => { wire::MakerToTaker::Protocol(setup_msg) => {
log_error!(self.handle_inc_protocol_msg(setup_msg)) log_error!(self.handle_inc_protocol_msg(setup_msg))

4
daemon/src/wire.rs

@ -58,6 +58,8 @@ pub enum MakerToTaker {
CurrentOrder(Option<Order>), CurrentOrder(Option<Order>),
ConfirmOrder(OrderId), // TODO: Include payout curve in "accept" message from maker ConfirmOrder(OrderId), // TODO: Include payout curve in "accept" message from maker
RejectOrder(OrderId), RejectOrder(OrderId),
ConfirmSettlement(OrderId),
RejectSettlement(OrderId),
InvalidOrderId(OrderId), InvalidOrderId(OrderId),
Protocol(SetupMsg), Protocol(SetupMsg),
} }
@ -68,6 +70,8 @@ impl fmt::Display for MakerToTaker {
MakerToTaker::CurrentOrder(_) => write!(f, "CurrentOrder"), MakerToTaker::CurrentOrder(_) => write!(f, "CurrentOrder"),
MakerToTaker::ConfirmOrder(_) => write!(f, "ConfirmOrder"), MakerToTaker::ConfirmOrder(_) => write!(f, "ConfirmOrder"),
MakerToTaker::RejectOrder(_) => write!(f, "RejectOrder"), MakerToTaker::RejectOrder(_) => write!(f, "RejectOrder"),
MakerToTaker::ConfirmSettlement(_) => write!(f, "ConfirmSettlement"),
MakerToTaker::RejectSettlement(_) => write!(f, "RejectSettlement"),
MakerToTaker::InvalidOrderId(_) => write!(f, "InvalidOrderId"), MakerToTaker::InvalidOrderId(_) => write!(f, "InvalidOrderId"),
MakerToTaker::Protocol(_) => write!(f, "Protocol"), MakerToTaker::Protocol(_) => write!(f, "Protocol"),
} }

Loading…
Cancel
Save