1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
use async_trait::async_trait;
use bitcoin::{sha256, Hash};
use runtime::{AccountId, Error as RuntimeError, InterBtcParachain, UtilFuncs, VaultRegistryPallet};
use std::fmt;
#[async_trait]
pub trait RandomDelay: fmt::Debug {
async fn delay(&self, seed_data: &[u8; 32]) -> Result<(), RuntimeError>;
}
#[derive(Clone)]
pub struct OrderedVaultsDelay {
btc_parachain: InterBtcParachain,
vaults: Vec<AccountId>,
}
impl OrderedVaultsDelay {
pub async fn new(btc_parachain: InterBtcParachain) -> Result<Self, RuntimeError> {
let vaults = btc_parachain
.get_all_vaults()
.await?
.into_iter()
.map(|vault| vault.id.account_id)
.collect();
Ok(Self { btc_parachain, vaults })
}
}
impl fmt::Debug for OrderedVaultsDelay {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
f.debug_struct("OrderedDelayableVaults")
.field("btc_parachain", &self.btc_parachain.get_account_id())
.field("vaults", &self.vaults)
.finish()
}
}
#[async_trait]
impl RandomDelay for OrderedVaultsDelay {
async fn delay(&self, seed_data: &[u8; 32]) -> Result<(), RuntimeError> {
fn hash_vault(data: &[u8; 32], account_id: &AccountId) -> sha256::Hash {
let account_id = account_id.0.clone();
let account_id: [u8; 32] = (*account_id).clone().into();
let xor = data.zip(account_id).map(|(a, b)| a ^ b);
sha256::Hash::hash(&xor)
}
let self_hash = hash_vault(seed_data, self.btc_parachain.get_account_id());
let random_ordering = self
.vaults
.iter()
.filter(|account_id| hash_vault(seed_data, account_id) < self_hash)
.count();
let delay: u32 = (random_ordering + 1).ilog2();
self.btc_parachain.delay_for_blocks(delay).await
}
}
#[derive(Clone, Debug)]
pub struct ZeroDelay;
#[async_trait]
impl RandomDelay for ZeroDelay {
async fn delay(&self, _seed_data: &[u8; 32]) -> Result<(), RuntimeError> {
Ok(())
}
}