#[cfg(feature = "pinocchio")]
type Ref<'a, T> = pinocchio::account_info::Ref<'a, T>;
#[cfg(not(feature = "pinocchio"))]
use std::cell::Ref;
use bytemuck::{Pod, Zeroable};
#[allow(unused_imports)]
use crate::impl_account_deserialize;
use crate::AccountInfo;
#[allow(unused_imports)]
use crate::OracleAccountData;
use crate::{cfg_client, get_sb_program_id, OnDemandError};
cfg_client! {
use crate::solana_compat::AddressLookupTableAccount;
}
use crate::Pubkey;
#[repr(C)]
#[derive(Debug, Copy, Clone)]
pub struct QueueAccountData {
pub authority: Pubkey,
pub mr_enclaves: [[u8; 32]; 32],
pub oracle_keys: [Pubkey; 78],
reserved1: [u8; 40],
pub secp_oracle_signing_keys: [[u8; 20]; 30],
pub ed25519_oracle_signing_keys: [Pubkey; 30],
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,
pub signer_bump: u8,
pub mint: Pubkey,
pub lut_slot: u64,
pub allow_subsidies: u8,
_ebuf6: [u8; 15],
pub ncn: Pubkey,
_resrved: u64, pub vaults: [VaultInfo; 4],
pub last_reward_epoch: u64,
_ebuf4: [u8; 32],
_ebuf2: [u8; 256],
_ebuf1: [u8; 504], }
unsafe impl Pod for QueueAccountData {}
unsafe impl Zeroable for QueueAccountData {}
#[cfg(feature = "idl-build")]
impl anchor_lang::IdlBuild for QueueAccountData {}
#[repr(C)]
#[derive(PartialEq, Debug, Copy, Clone)]
pub struct VaultInfo {
pub vault_key: Pubkey,
pub last_reward_epoch: u64,
}
unsafe impl Pod for VaultInfo {}
unsafe impl Zeroable for VaultInfo {}
#[cfg(feature = "idl-build")]
impl anchor_lang::IdlBuild for VaultInfo {}
cfg_client! {
impl_account_deserialize!(QueueAccountData);
}
pub const QUEUE_ACCOUNT_DISCRIMINATOR: [u8; 8] = [217, 194, 55, 127, 184, 83, 138, 1];
impl crate::anchor_traits::Discriminator for QueueAccountData {
const DISCRIMINATOR: &'static [u8] = &QUEUE_ACCOUNT_DISCRIMINATOR;
}
impl crate::anchor_traits::Owner for QueueAccountData {
fn owner() -> Pubkey {
let pid = if crate::utils::is_devnet() {
get_sb_program_id("devnet")
} else {
get_sb_program_id("mainnet")
};
pid.to_bytes().into()
}
}
impl crate::anchor_traits::ZeroCopy for QueueAccountData {}
#[cfg(feature = "anchor")]
impl anchor_lang::Discriminator for QueueAccountData {
const DISCRIMINATOR: &'static [u8] = &QUEUE_ACCOUNT_DISCRIMINATOR;
}
#[cfg(feature = "anchor")]
impl anchor_lang::Owner for QueueAccountData {
fn owner() -> Pubkey {
let pid = if crate::utils::is_devnet() {
get_sb_program_id("devnet")
} else {
get_sb_program_id("mainnet")
};
pid.to_bytes().into()
}
}
#[cfg(feature = "anchor")]
impl anchor_lang::ZeroCopy for QueueAccountData {}
impl QueueAccountData {
pub fn size() -> usize {
8 + std::mem::size_of::<QueueAccountData>()
}
pub fn new<'info>(
attestation_queue_account_info: &'info AccountInfo,
) -> Result<Ref<'info, QueueAccountData>, OnDemandError> {
let data = attestation_queue_account_info
.try_borrow_data()
.map_err(|_| OnDemandError::AccountBorrowError)?;
if data.len() < QUEUE_ACCOUNT_DISCRIMINATOR.len() {
return Err(OnDemandError::InvalidDiscriminator);
}
let mut disc_bytes = [0u8; 8];
disc_bytes.copy_from_slice(&data[..8]);
if disc_bytes != QUEUE_ACCOUNT_DISCRIMINATOR {
return Err(OnDemandError::InvalidDiscriminator);
}
let expected_size = std::mem::size_of::<QueueAccountData>() + 8;
if data.len() < expected_size {
return Err(OnDemandError::InvalidData);
}
let slice_to_parse = &data[8..expected_size];
if slice_to_parse.len() != std::mem::size_of::<QueueAccountData>() {
return Err(OnDemandError::InvalidData);
}
match bytemuck::try_from_bytes::<QueueAccountData>(slice_to_parse) {
Ok(_) => {
Ok(Ref::map(data, |data| {
bytemuck::from_bytes::<QueueAccountData>(
&data[8..std::mem::size_of::<QueueAccountData>() + 8],
)
}))
}
Err(_) => Err(OnDemandError::AccountDeserializeError),
}
}
pub fn new_from_bytes(data: &[u8]) -> Result<&QueueAccountData, OnDemandError> {
if data.len() < QUEUE_ACCOUNT_DISCRIMINATOR.len() {
return Err(OnDemandError::InvalidDiscriminator);
}
let mut disc_bytes = [0u8; 8];
disc_bytes.copy_from_slice(&data[..8]);
if disc_bytes != QUEUE_ACCOUNT_DISCRIMINATOR {
return Err(OnDemandError::InvalidDiscriminator);
}
let expected_size = std::mem::size_of::<QueueAccountData>() + 8;
if data.len() < expected_size {
return Err(OnDemandError::InvalidData);
}
let slice_to_parse = &data[8..expected_size];
if slice_to_parse.len() != std::mem::size_of::<QueueAccountData>() {
return Err(OnDemandError::InvalidData);
}
match bytemuck::try_from_bytes::<QueueAccountData>(slice_to_parse) {
Ok(queue_data) => Ok(queue_data),
Err(_) => Err(OnDemandError::AccountDeserializeError),
}
}
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: &crate::RpcClient,
pubkey: Pubkey,
) -> std::result::Result<Self, crate::OnDemandError> {
let pubkey = pubkey.to_bytes().into();
crate::client::fetch_zerocopy_account(client, pubkey).await
}
pub async fn fetch_oracles(
&self,
client: &crate::RpcClient,
) -> std::result::Result<Vec<(Pubkey, OracleAccountData)>, crate::OnDemandError> {
let oracles = &self.oracle_keys[..self.oracle_keys_len as usize];
let converted_oracles: Vec<crate::solana_compat::solana_sdk::pubkey::Pubkey> = oracles.iter().map(|pk| pk.to_bytes().into()).collect();
let datas: Vec<_> = client
.get_multiple_accounts(&converted_oracles)
.await
.map_err(|_e| crate::OnDemandError::NetworkError)?
.into_iter()
.flatten()
.map(|x| x.data.clone())
.collect::<Vec<_>>()
.iter()
.map(|x| OracleAccountData::new_from_bytes(x))
.filter_map(|x| x.ok())
.copied()
.collect();
Ok(oracles.iter().cloned().zip(datas).collect())
}
pub async fn fetch_lut(
&self,
pubkey: &Pubkey,
client: &crate::RpcClient,
) -> std::result::Result<AddressLookupTableAccount, crate::OnDemandError> {
use crate::derive_lookup_table_address;
let lut_slot = self.lut_slot;
let lut_signer: Pubkey = crate::find_lut_signer(pubkey);
let lut = derive_lookup_table_address(&lut_signer.to_bytes().into(), lut_slot).0;
crate::address_lookup_table::fetch(client, &lut.to_bytes().into()).await
}
}
}