use crate::anchor_traits::*;
use crate::cfg_client;
#[allow(unused_imports)]
use crate::impl_account_deserialize;
use crate::OnDemandError;
#[allow(unused_imports)]
use crate::OracleAccountData;
use crate::get_sb_program_id;
use bytemuck::{Pod, Zeroable};
use solana_program::account_info::AccountInfo;
use solana_program::pubkey::Pubkey;
use std::cell::Ref;
#[repr(C)]
#[derive(Debug, Copy, Clone)]
pub struct QueueAccountData {
pub authority: Pubkey,
pub mr_enclaves: [[u8; 32]; 32],
pub oracle_keys: [Pubkey; 128],
pub max_quote_verification_age: i64,
pub last_heartbeat: i64,
pub node_timeout: i64,
pub oracle_min_stake: u64,
pub allow_authority_override_after: i64,
pub mr_enclaves_len: u32,
pub oracle_keys_len: u32,
pub reward: u32, pub curr_idx: u32,
pub gc_idx: u32,
pub require_authority_heartbeat_permission: u8,
pub require_authority_verify_permission: u8,
pub require_usage_permissions: u8,
padding1: [u8; 1],
pub mint: Pubkey,
pub _ebuf: [u8; 1024],
}
unsafe impl Pod for QueueAccountData {}
unsafe impl Zeroable for QueueAccountData {}
cfg_client! {
impl_account_deserialize!(QueueAccountData);
}
impl Discriminator for QueueAccountData {
const DISCRIMINATOR: [u8; 8] = [217, 194, 55, 127, 184, 83, 138, 1];
}
impl Owner for QueueAccountData {
fn owner() -> Pubkey {
let cluster = std::env::var("CLUSTER").unwrap_or("mainnet".to_string());
get_sb_program_id(&cluster)
}
}
impl QueueAccountData {
pub fn size() -> usize {
8 + std::mem::size_of::<QueueAccountData>()
}
pub fn new<'info>(
attestation_queue_account_info: &'info AccountInfo<'info>,
) -> Result<Ref<'info, QueueAccountData>, OnDemandError> {
let data = attestation_queue_account_info
.try_borrow_data()
.map_err(|_| OnDemandError::AccountBorrowError)?;
if data.len() < QueueAccountData::discriminator().len() {
return Err(OnDemandError::InvalidDiscriminator);
}
let mut disc_bytes = [0u8; 8];
disc_bytes.copy_from_slice(&data[..8]);
if disc_bytes != QueueAccountData::discriminator() {
return Err(OnDemandError::InvalidDiscriminator);
}
Ok(Ref::map(data, |data| {
bytemuck::from_bytes(&data[8..std::mem::size_of::<QueueAccountData>() + 8])
}))
}
pub fn new_from_bytes(data: &[u8]) -> Result<&QueueAccountData, OnDemandError> {
if data.len() < QueueAccountData::discriminator().len() {
return Err(OnDemandError::InvalidDiscriminator);
}
let mut disc_bytes = [0u8; 8];
disc_bytes.copy_from_slice(&data[..8]);
if disc_bytes != QueueAccountData::discriminator() {
return Err(OnDemandError::InvalidDiscriminator);
}
Ok(bytemuck::from_bytes(
&data[8..std::mem::size_of::<QueueAccountData>() + 8],
))
}
pub fn has_mr_enclave(&self, mr_enclave: &[u8]) -> bool {
self.mr_enclaves[..self.mr_enclaves_len as usize]
.iter()
.any(|x| x.to_vec() == mr_enclave.to_vec())
}
pub fn permitted_enclaves(&self) -> Vec<[u8; 32]> {
self.mr_enclaves[..self.mr_enclaves_len as usize].to_vec()
}
pub fn garbage_collection_node(&self) -> Option<Pubkey> {
let gc_node = self.oracle_keys[self.gc_idx as usize];
if gc_node != Pubkey::default() {
Some(gc_node)
} else {
None
}
}
pub fn idx_of_oracle(&self, oracle: &Pubkey) -> Option<usize> {
self.oracle_keys[..self.oracle_keys_len as usize]
.iter()
.position(|x| x == oracle)
}
pub fn oracle_keys(&self) -> Vec<Pubkey> {
self.oracle_keys[..self.oracle_keys_len as usize].to_vec()
}
cfg_client! {
pub async fn fetch_async(
client: &solana_client::nonblocking::rpc_client::RpcClient,
pubkey: Pubkey,
) -> std::result::Result<Self, crate::OnDemandError> {
crate::client::fetch_zerocopy_account_async(client, pubkey).await
}
pub async fn fetch_oracles(
&self,
client: &solana_client::nonblocking::rpc_client::RpcClient,
) -> std::result::Result<Vec<(Pubkey, OracleAccountData)>, crate::OnDemandError> {
let oracles = &self.oracle_keys[..self.oracle_keys_len as usize];
let datas: Vec<_> = client
.get_multiple_accounts(&oracles)
.await
.map_err(|_e| crate::OnDemandError::NetworkError)?
.into_iter()
.filter_map(|x| x)
.map(|x| x.data.clone())
.collect::<Vec<_>>()
.iter()
.map(|x| OracleAccountData::new_from_bytes(x))
.filter_map(|x| x.ok())
.map(|x| x.clone())
.collect();
Ok(oracles.iter().cloned().zip(datas).collect())
}
}
}