Browse Source

Oracle attestation should not affect all cfds

Only the attestation relevant for the cfds, based on the event id, should trigger monitoring for the CET.
We now properly filter the monitoring parameters by event id upon attestation.

Without the filter we ran into the error `No CET for oracle event found`, which is expected when trying to find CETs for an event that does not match.
Since an error in the loop can also have unwanted side effects we wrap it with a `try_continue`.
fix/sql-oddness
Daniel Karzel 3 years ago
parent
commit
32351efe67
No known key found for this signature in database GPG Key ID: 30C3FC2E438ADB6E
  1. 12
      daemon/src/maker_cfd.rs
  2. 35
      daemon/src/monitor.rs
  3. 12
      daemon/src/taker_cfd.rs

12
daemon/src/maker_cfd.rs

@ -704,7 +704,11 @@ where
self.monitor_actor self.monitor_actor
.do_send_async(monitor::StartMonitoring { .do_send_async(monitor::StartMonitoring {
id: order_id, id: order_id,
params: MonitorParams::from_dlc_and_timelocks(dlc, cfd.refund_timelock_in_blocks()), params: MonitorParams::new(
dlc,
cfd.refund_timelock_in_blocks(),
cfd.order.oracle_event_id,
),
}) })
.await?; .await?;
@ -837,7 +841,11 @@ where
self.monitor_actor self.monitor_actor
.do_send_async(monitor::StartMonitoring { .do_send_async(monitor::StartMonitoring {
id: order_id, id: order_id,
params: MonitorParams::from_dlc_and_timelocks(dlc, cfd.refund_timelock_in_blocks()), params: MonitorParams::new(
dlc,
cfd.refund_timelock_in_blocks(),
cfd.order.oracle_event_id,
),
}) })
.await?; .await?;

35
daemon/src/monitor.rs

@ -1,7 +1,7 @@
use crate::model::cfd::{CetStatus, Cfd, CfdState, Dlc, OrderId}; use crate::model::cfd::{CetStatus, Cfd, CfdState, Dlc, OrderId};
use crate::model::BitMexPriceEventId; use crate::model::BitMexPriceEventId;
use crate::oracle::Attestation; use crate::oracle::Attestation;
use crate::{log_error, model, oracle}; use crate::{log_error, model, oracle, try_continue};
use anyhow::{Context, Result}; use anyhow::{Context, Result};
use async_trait::async_trait; use async_trait::async_trait;
use bdk::bitcoin::{PublicKey, Script, Txid}; use bdk::bitcoin::{PublicKey, Script, Txid};
@ -36,6 +36,7 @@ pub struct MonitorParams {
cets: HashMap<BitMexPriceEventId, Vec<Cet>>, cets: HashMap<BitMexPriceEventId, Vec<Cet>>,
refund: (Txid, Script, u32), refund: (Txid, Script, u32),
revoked_commits: Vec<(Txid, Script)>, revoked_commits: Vec<(Txid, Script)>,
event_id: BitMexPriceEventId,
} }
pub struct Sync; pub struct Sync;
@ -77,12 +78,12 @@ impl Actor<bdk::electrum_client::Client> {
match cfd.state.clone() { match cfd.state.clone() {
// In PendingOpen we know the complete dlc setup and assume that the lock transaction will be published // In PendingOpen we know the complete dlc setup and assume that the lock transaction will be published
CfdState::PendingOpen { dlc, .. } => { CfdState::PendingOpen { dlc, .. } => {
let params = MonitorParams::from_dlc_and_timelocks(dlc.clone(), cfd.refund_timelock_in_blocks()); 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.cfds.insert(cfd.order.id, params.clone());
actor.monitor_all(&params, cfd.order.id); actor.monitor_all(&params, cfd.order.id);
} }
CfdState::Open { dlc, .. } | CfdState::PendingCommit { dlc, .. } => { CfdState::Open { dlc, .. } | CfdState::PendingCommit { dlc, .. } => {
let params = MonitorParams::from_dlc_and_timelocks(dlc.clone(), cfd.refund_timelock_in_blocks()); 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.cfds.insert(cfd.order.id, params.clone());
actor.monitor_commit_finality(&params, cfd.order.id); actor.monitor_commit_finality(&params, cfd.order.id);
@ -98,7 +99,7 @@ impl Actor<bdk::electrum_client::Client> {
} }
} }
CfdState::OpenCommitted { dlc, cet_status, .. } => { CfdState::OpenCommitted { dlc, cet_status, .. } => {
let params = MonitorParams::from_dlc_and_timelocks(dlc.clone(), cfd.refund_timelock_in_blocks()); 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.cfds.insert(cfd.order.id, params.clone());
match cet_status { match cet_status {
@ -125,7 +126,7 @@ impl Actor<bdk::electrum_client::Client> {
} }
} }
CfdState::PendingCet { dlc, attestation, .. } => { CfdState::PendingCet { dlc, attestation, .. } => {
let params = MonitorParams::from_dlc_and_timelocks(dlc.clone(), cfd.refund_timelock_in_blocks()); 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.cfds.insert(cfd.order.id, params.clone());
actor.monitor_cet_finality(map_cets(dlc.cets), attestation.into(), cfd.order.id)?; actor.monitor_cet_finality(map_cets(dlc.cets), attestation.into(), cfd.order.id)?;
@ -133,7 +134,7 @@ impl Actor<bdk::electrum_client::Client> {
actor.monitor_refund_finality(&params,cfd.order.id); actor.monitor_refund_finality(&params,cfd.order.id);
} }
CfdState::PendingRefund { dlc, .. } => { CfdState::PendingRefund { dlc, .. } => {
let params = MonitorParams::from_dlc_and_timelocks(dlc.clone(), cfd.refund_timelock_in_blocks()); 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.cfds.insert(cfd.order.id, params.clone());
actor.monitor_commit_refund_timelock(&params, cfd.order.id); actor.monitor_commit_refund_timelock(&params, cfd.order.id);
@ -225,9 +226,13 @@ where
attestation: Attestation, attestation: Attestation,
order_id: OrderId, order_id: OrderId,
) -> Result<()> { ) -> Result<()> {
let cets = cets let attestation_id = attestation.id;
.get(&attestation.id) let cets = cets.get(&attestation_id).with_context(|| {
.context("No CET for oracle event found")?; format!(
"No CET for oracle event {} and cfd with order id {}",
attestation_id, order_id
)
})?;
let (txid, script_pubkey) = cets let (txid, script_pubkey) = cets
.iter() .iter()
@ -314,8 +319,13 @@ where
} }
async fn handle_oracle_attestation(&mut self, attestation: oracle::Attestation) -> Result<()> { async fn handle_oracle_attestation(&mut self, attestation: oracle::Attestation) -> Result<()> {
for (order_id, MonitorParams { cets, .. }) in self.cfds.clone().into_iter() { for (order_id, MonitorParams { cets, .. }) in self
self.monitor_cet_finality(cets, attestation.clone(), order_id)?; .cfds
.clone()
.into_iter()
.filter(|(_, params)| params.event_id == attestation.id)
{
try_continue!(self.monitor_cet_finality(cets, attestation.clone(), order_id))
} }
Ok(()) Ok(())
@ -551,7 +561,7 @@ impl Event {
} }
impl MonitorParams { impl MonitorParams {
pub fn from_dlc_and_timelocks(dlc: Dlc, refund_timelock_in_blocks: u32) -> Self { pub fn new(dlc: Dlc, refund_timelock_in_blocks: u32, event_id: BitMexPriceEventId) -> Self {
let script_pubkey = dlc.maker_address.script_pubkey(); let script_pubkey = dlc.maker_address.script_pubkey();
MonitorParams { MonitorParams {
lock: (dlc.lock.0.txid(), dlc.lock.1), lock: (dlc.lock.0.txid(), dlc.lock.1),
@ -567,6 +577,7 @@ impl MonitorParams {
.iter() .iter()
.map(|rev_commit| (rev_commit.txid, rev_commit.script_pubkey.clone())) .map(|rev_commit| (rev_commit.txid, rev_commit.script_pubkey.clone()))
.collect(), .collect(),
event_id,
} }
} }
} }

12
daemon/src/taker_cfd.rs

@ -421,7 +421,11 @@ where
self.monitor_actor self.monitor_actor
.do_send_async(monitor::StartMonitoring { .do_send_async(monitor::StartMonitoring {
id: order_id, id: order_id,
params: MonitorParams::from_dlc_and_timelocks(dlc, cfd.refund_timelock_in_blocks()), params: MonitorParams::new(
dlc,
cfd.refund_timelock_in_blocks(),
cfd.order.oracle_event_id,
),
}) })
.await?; .await?;
@ -590,7 +594,11 @@ where
self.monitor_actor self.monitor_actor
.do_send_async(monitor::StartMonitoring { .do_send_async(monitor::StartMonitoring {
id: order_id, id: order_id,
params: MonitorParams::from_dlc_and_timelocks(dlc, cfd.refund_timelock_in_blocks()), params: MonitorParams::new(
dlc,
cfd.refund_timelock_in_blocks(),
cfd.order.oracle_event_id,
),
}) })
.await?; .await?;

Loading…
Cancel
Save