use anchor_lang::prelude::*;
use anchor_lang::{Discriminator, Owner, ZeroCopy};
use bytemuck::{Pod, Zeroable};
use std::cell::Ref;
use crate::SWITCHBOARD_ATTESTATION_PROGRAM_ID;
#[zero_copy(unsafe)]
#[repr(packed)]
#[derive(Debug)]
pub struct AttestationQueueAccountData {
pub authority: Pubkey,
pub mr_enclaves: [[u8; 32]; 32],
pub mr_enclaves_len: u32,
pub data: [Pubkey; 128],
pub data_len: u32,
pub allow_authority_override_after: i64,
pub require_authority_heartbeat_permission: bool,
pub require_usage_permissions: bool,
pub max_quote_verification_age: i64,
pub reward: u32, pub last_heartbeat: i64,
pub node_timeout: i64,
pub curr_idx: u32,
pub gc_idx: u32,
pub _ebuf: [u8; 1024],
}
unsafe impl Pod for AttestationQueueAccountData {}
unsafe impl Zeroable for AttestationQueueAccountData {}
impl Discriminator for AttestationQueueAccountData {
const DISCRIMINATOR: [u8; 8] = [192, 53, 130, 67, 234, 207, 39, 171];
}
impl Owner for AttestationQueueAccountData {
fn owner() -> Pubkey {
SWITCHBOARD_ATTESTATION_PROGRAM_ID
}
}
impl ZeroCopy for AttestationQueueAccountData {}
impl AttestationQueueAccountData {
pub fn new<'info>(
attestation_queue_account_info: &'info AccountInfo<'info>,
) -> anchor_lang::Result<Ref<'info, AttestationQueueAccountData>> {
let data = attestation_queue_account_info.try_borrow_data()?;
if data.len() < AttestationQueueAccountData::discriminator().len() {
return Err(ErrorCode::AccountDiscriminatorNotFound.into());
}
let mut disc_bytes = [0u8; 8];
disc_bytes.copy_from_slice(&data[..8]);
if disc_bytes != AttestationQueueAccountData::discriminator() {
return Err(ErrorCode::AccountDiscriminatorMismatch.into());
}
Ok(Ref::map(data, |data| {
bytemuck::from_bytes(&data[8..std::mem::size_of::<AttestationQueueAccountData>() + 8])
}))
}
pub fn new_from_bytes(data: &[u8]) -> anchor_lang::Result<&AttestationQueueAccountData> {
if data.len() < AttestationQueueAccountData::discriminator().len() {
return Err(ErrorCode::AccountDiscriminatorNotFound.into());
}
let mut disc_bytes = [0u8; 8];
disc_bytes.copy_from_slice(&data[..8]);
if disc_bytes != AttestationQueueAccountData::discriminator() {
return Err(ErrorCode::AccountDiscriminatorMismatch.into());
}
Ok(bytemuck::from_bytes(
&data[8..std::mem::size_of::<AttestationQueueAccountData>() + 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())
}
#[cfg(feature = "client")]
#[cfg_attr(docsrs, doc(cfg(feature = "client")))]
pub async fn fetch(
client: &solana_client::rpc_client::RpcClient,
pubkey: Pubkey,
) -> std::result::Result<Self, switchboard_common::Error> {
crate::client::load_account(client, pubkey).await
}
}