#[cfg(feature = "pinocchio")]
type Ref<'a, T> = pinocchio::account_info::Ref<'a, T>;
#[cfg(not(feature = "pinocchio"))]
use std::cell::Ref;
use solana_program::sysvar::clock::Clock;
use crate::anchor_traits::*;
#[allow(unused_imports)]
use crate::impl_account_deserialize;
use crate::AccountInfo;
use crate::{cfg_client, get_sb_program_id, OnDemandError, Quote};
cfg_client! {
use crate::address_lookup_table;
use spl_associated_token_account::solana_program::address_lookup_table::AddressLookupTableAccount;
use spl_associated_token_account::solana_program::address_lookup_table::instruction::derive_lookup_table_address;
use crate::find_lut_signer;
}
#[cfg(feature = "anchor")]
use anchor_lang::solana_program;
#[cfg(not(feature = "anchor"))]
use crate::solana_program;
use solana_program::pubkey::Pubkey;
pub const ORACLE_FEED_STATS_SEED: &[u8; 15] = b"OracleFeedStats";
#[repr(u8)]
#[derive(Copy, Clone, Default, Debug, Eq, PartialEq)]
pub enum VerificationStatus {
#[default]
None = 0,
VerificationPending = 1 << 0,
VerificationFailure = 1 << 1,
VerificationSuccess = 1 << 2,
VerificationOverride = 1 << 3,
}
impl From<VerificationStatus> for u8 {
fn from(value: VerificationStatus) -> Self {
match value {
VerificationStatus::VerificationPending => 1 << 0,
VerificationStatus::VerificationFailure => 1 << 1,
VerificationStatus::VerificationSuccess => 1 << 2,
VerificationStatus::VerificationOverride => 1 << 3,
_ => 0,
}
}
}
impl From<u8> for VerificationStatus {
fn from(value: u8) -> Self {
match value {
1 => VerificationStatus::VerificationPending,
2 => VerificationStatus::VerificationFailure,
4 => VerificationStatus::VerificationSuccess,
8 => VerificationStatus::VerificationOverride,
_ => VerificationStatus::default(),
}
}
}
#[repr(C)]
#[derive(bytemuck::Zeroable, bytemuck::Pod, Debug, Copy, Clone)]
pub struct OracleAccountData {
pub enclave: Quote,
pub authority: Pubkey,
pub queue: Pubkey,
pub created_at: i64,
pub last_heartbeat: i64,
pub secp_authority: [u8; 64],
pub gateway_uri: [u8; 64],
pub permissions: u64,
pub is_on_queue: u8,
_padding1: [u8; 7],
pub lut_slot: u64,
pub last_reward_epoch: u64,
pub operator: Pubkey,
_ebuf3: [u8; 16],
_ebuf2: [u8; 64],
_ebuf1: [u8; 1024],
}
cfg_client! {
impl_account_deserialize!(OracleAccountData);
}
impl Discriminator for OracleAccountData {
const DISCRIMINATOR: &'static [u8] = &[128, 30, 16, 241, 170, 73, 55, 54];
}
impl Owner for OracleAccountData {
fn owner() -> Pubkey {
if crate::utils::is_devnet() {
get_sb_program_id("devnet")
} else {
get_sb_program_id("mainnet")
}
}
}
impl OracleAccountData {
pub fn size() -> usize {
8 + std::mem::size_of::<OracleAccountData>()
}
pub fn new<'info>(
quote_account_info: &'info AccountInfo,
) -> Result<Ref<'info, OracleAccountData>, OnDemandError> {
let data = quote_account_info
.try_borrow_data()
.map_err(|_| OnDemandError::AccountBorrowError)?;
if data.len() < OracleAccountData::DISCRIMINATOR.len() {
return Err(OnDemandError::InvalidDiscriminator);
}
let mut disc_bytes = [0u8; 8];
disc_bytes.copy_from_slice(&data[..8]);
if disc_bytes != *OracleAccountData::DISCRIMINATOR {
return Err(OnDemandError::InvalidDiscriminator);
}
let expected_size = std::mem::size_of::<OracleAccountData>() + 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::<OracleAccountData>() {
return Err(OnDemandError::InvalidData);
}
match bytemuck::try_from_bytes::<OracleAccountData>(slice_to_parse) {
Ok(_) => {
Ok(Ref::map(data, |data| {
bytemuck::from_bytes::<OracleAccountData>(
&data[8..std::mem::size_of::<OracleAccountData>() + 8],
)
}))
}
Err(_) => Err(OnDemandError::AccountDeserializeError),
}
}
pub fn new_from_bytes(data: &[u8]) -> Result<&OracleAccountData, OnDemandError> {
if data.len() < OracleAccountData::DISCRIMINATOR.len() {
return Err(OnDemandError::InvalidDiscriminator);
}
let mut disc_bytes = [0u8; 8];
disc_bytes.copy_from_slice(&data[..8]);
if disc_bytes != *OracleAccountData::DISCRIMINATOR {
return Err(OnDemandError::InvalidDiscriminator);
}
let expected_size = std::mem::size_of::<OracleAccountData>() + 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::<OracleAccountData>() {
return Err(OnDemandError::InvalidData);
}
match bytemuck::try_from_bytes::<OracleAccountData>(slice_to_parse) {
Ok(oracle_data) => Ok(oracle_data),
Err(_) => Err(OnDemandError::AccountDeserializeError),
}
}
pub fn signer(&self) -> &Pubkey {
&self.enclave.enclave_signer
}
pub fn is_verified(&self, clock: &Clock) -> bool {
match self.enclave.verification_status.into() {
VerificationStatus::VerificationOverride => true,
VerificationStatus::VerificationSuccess => {
self.enclave.valid_until > clock.unix_timestamp
}
_ => false,
}
}
pub fn verify(&self, clock: &Clock) -> std::result::Result<(), OnDemandError> {
if !self.is_verified(clock) {
return Err(OnDemandError::InvalidQuote);
}
Ok(())
}
pub fn gateway_uri(&self) -> Option<String> {
let uri = self.gateway_uri;
let uri = String::from_utf8_lossy(&uri);
let uri = uri
.split_at(uri.find('\0').unwrap_or(uri.len()))
.0
.to_string();
if uri.is_empty() {
return None;
}
Some(uri)
}
pub fn ed25519_signer(&self) -> Option<Pubkey> {
let key = self.enclave.enclave_signer;
if key == Pubkey::default() {
return None;
}
Some(key)
}
pub fn secp_authority(&self) -> Option<[u8; 64]> {
let key = self.secp_authority;
if key == [0u8; 64] {
return None;
}
Some(key)
}
pub fn secp256k1_signer(&self) -> Option<[u8; 64]> {
let key = self.enclave.secp256k1_signer;
if key == [0u8; 64] {
return None;
}
Some(key)
}
pub fn libsecp256k1_signer(&self) -> Option<libsecp256k1::PublicKey> {
let bytes = self.secp256k1_signer()?;
let tag_full_pubkey: Vec<u8> = vec![4u8];
let bytes = [tag_full_pubkey, bytes.into()].concat().try_into().ok()?;
libsecp256k1::PublicKey::parse(&bytes).ok()
}
pub fn stats_key(key: &Pubkey) -> Pubkey {
let pid = OracleAccountData::owner();
let oracle_stats_seed = b"OracleStats";
let (key, _) =
Pubkey::find_program_address(&[oracle_stats_seed.as_slice(), &key.to_bytes()], &pid);
key
}
pub fn feed_stats_key(feed: &Pubkey, oracle: &Pubkey) -> (Pubkey, u8) {
let pid = OracleAccountData::owner();
Pubkey::find_program_address(
&Self::feed_stats_seed(&feed.to_bytes(), &oracle.to_bytes(), &[]),
&pid,
)
}
pub fn feed_stats_seed<'a>(feed: &'a [u8], oracle: &'a [u8], bump: &'a [u8]) -> [&'a [u8]; 4] {
[ORACLE_FEED_STATS_SEED.as_slice(), feed, oracle, bump]
}
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_many(
client: &crate::RpcClient,
oracles: &[Pubkey],
) -> std::result::Result<Vec<OracleAccountData>, crate::OnDemandError> {
let converted_oracles: Vec<anchor_client::solana_sdk::pubkey::Pubkey> = oracles.iter().map(|pk| pk.to_bytes().into()).collect();
Ok(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())
}
pub async fn fetch_lut(
&self,
oracle_pubkey: &Pubkey,
client: &crate::RpcClient,
) -> std::result::Result<AddressLookupTableAccount, crate::OnDemandError> {
let lut_slot = self.lut_slot;
let lut_signer = find_lut_signer(oracle_pubkey);
let lut = derive_lookup_table_address(&lut_signer.to_bytes().into(), lut_slot).0;
address_lookup_table::fetch(client, &lut.to_bytes().into()).await
}
}
}