use crate::cfg_client;
use crate::prelude::*;
#[derive(Accounts)]
#[instruction(params:FunctionServiceVerifyParams)]
pub struct FunctionServiceVerify<'info> {
#[account(mut)]
pub service: AccountInfo<'info>, pub function_enclave_signer: AccountInfo<'info>, #[account(mut)]
pub escrow_wallet: AccountInfo<'info>, #[account(mut)]
pub escrow_token_wallet: AccountInfo<'info>, #[account(mut)]
pub function: AccountInfo<'info>, #[account(mut)]
pub function_escrow_token_wallet: Option<AccountInfo<'info>>, pub verifier_quote: AccountInfo<'info>, #[account(signer)]
pub verifier_enclave_signer: AccountInfo<'info>, pub verifier_permission: AccountInfo<'info>, pub attestation_queue: AccountInfo<'info>, #[account(mut)]
pub receiver: AccountInfo<'info>, #[account(address = anchor_spl::token::ID)]
pub token_program: AccountInfo<'info>,
}
#[derive(Clone, AnchorSerialize, AnchorDeserialize, Debug)]
pub struct FunctionServiceVerifyParams {
pub observed_time: i64,
pub mr_enclave: [u8; 32],
}
impl InstructionData for FunctionServiceVerifyParams {}
impl Discriminator for FunctionServiceVerifyParams {
const DISCRIMINATOR: [u8; 8] = [58, 127, 172, 41, 218, 178, 104, 226];
}
impl Discriminator for FunctionServiceVerify<'_> {
const DISCRIMINATOR: [u8; 8] = [58, 127, 172, 41, 218, 178, 104, 226];
}
cfg_client! {
pub struct FunctionServiceVerifyAccounts {
pub service: Pubkey,
pub function_enclave_signer: Pubkey,
pub escrow_wallet: Pubkey,
pub function: Pubkey,
pub function_escrow_token_wallet: Option<Pubkey>,
pub verifier: Pubkey,
pub verifier_enclave_signer: Pubkey,
pub reward_receiver: Pubkey,
pub attestation_queue: Pubkey,
pub queue_authority: Pubkey,
}
impl ToAccountMetas for FunctionServiceVerifyAccounts {
fn to_account_metas(&self, _: Option<bool>) -> Vec<AccountMeta> {
vec![
AccountMeta::new(self.service, false),
AccountMeta::new_readonly(self.function_enclave_signer, false),
AccountMeta::new(self.escrow_wallet, false),
AccountMeta::new(find_associated_token_address(&self.escrow_wallet, &NativeMint::id()), false),
AccountMeta::new(self.function, false),
AccountMeta::new(self.function_escrow_token_wallet.unwrap_or(*SWITCHBOARD_ATTESTATION_PROGRAM_ID), false),
AccountMeta::new_readonly(self.verifier, false),
AccountMeta::new_readonly(self.verifier_enclave_signer, true),
AccountMeta::new_readonly(
AttestationPermissionAccountData::get_pda(
&self.queue_authority,
&self.attestation_queue,
&self.verifier
),
false
),
AccountMeta::new_readonly(self.attestation_queue, false),
AccountMeta::new(self.reward_receiver, false),
AccountMeta::new_readonly(anchor_spl::token::ID, false),
]
}
}
}
impl<'info> FunctionServiceVerify<'info> {
pub fn get_instruction(
&self,
program_id: Pubkey,
params: &FunctionServiceVerifyParams,
) -> anchor_lang::Result<Instruction> {
let accounts = self.to_account_metas(None);
let mut data: Vec<u8> = FunctionServiceVerify::discriminator().try_to_vec()?;
let mut param_vec: Vec<u8> = params.try_to_vec()?;
data.append(&mut param_vec);
let instruction = Instruction::new_with_bytes(program_id, &data, accounts);
Ok(instruction)
}
pub fn invoke(
&self,
program: AccountInfo<'info>,
params: &FunctionServiceVerifyParams,
) -> ProgramResult {
let instruction = self.get_instruction(*program.key, params)?;
let account_infos = self.to_account_infos();
invoke(&instruction, &account_infos[..])
}
pub fn invoke_signed(
&self,
program: AccountInfo<'info>,
params: &FunctionServiceVerifyParams,
signer_seeds: &[&[&[u8]]],
) -> ProgramResult {
let instruction = self.get_instruction(*program.key, params)?;
let account_infos = self.to_account_infos();
invoke_signed(&instruction, &account_infos[..], signer_seeds)
}
fn to_account_infos(&self) -> Vec<AccountInfo<'info>> {
let mut account_infos = Vec::new();
account_infos.extend(self.service.to_account_infos());
account_infos.extend(self.function_enclave_signer.to_account_infos());
account_infos.extend(self.escrow_wallet.to_account_infos());
account_infos.extend(self.escrow_token_wallet.to_account_infos());
account_infos.extend(self.function.to_account_infos());
account_infos.extend(self.function_escrow_token_wallet.to_account_infos());
account_infos.extend(self.verifier_quote.to_account_infos());
account_infos.extend(self.verifier_enclave_signer.to_account_infos());
account_infos.extend(self.verifier_permission.to_account_infos());
account_infos.extend(self.attestation_queue.to_account_infos());
account_infos.extend(self.receiver.to_account_infos());
account_infos.extend(self.token_program.to_account_infos());
account_infos
}
#[allow(unused_variables)]
fn to_account_metas(&self, is_signer: Option<bool>) -> Vec<AccountMeta> {
let mut account_metas = Vec::new();
account_metas.extend(self.service.to_account_metas(None));
account_metas.extend(self.function_enclave_signer.to_account_metas(None));
account_metas.extend(self.escrow_wallet.to_account_metas(None));
account_metas.extend(self.escrow_token_wallet.to_account_metas(None));
account_metas.extend(self.function.to_account_metas(None));
if let Some(function_escrow_token_wallet) = &self.function_escrow_token_wallet {
account_metas.extend(function_escrow_token_wallet.to_account_metas(None));
} else {
account_metas.push(AccountMeta::new_readonly(
*SWITCHBOARD_ATTESTATION_PROGRAM_ID,
false,
));
}
account_metas.extend(self.verifier_quote.to_account_metas(None));
account_metas.extend(self.verifier_enclave_signer.to_account_metas(None));
account_metas.extend(self.verifier_permission.to_account_metas(None));
account_metas.extend(self.attestation_queue.to_account_metas(None));
account_metas.extend(self.receiver.to_account_metas(None));
account_metas.extend(self.token_program.to_account_metas(None));
account_metas
}
cfg_client! {
pub fn build_ix(
accounts: &FunctionServiceVerifyAccounts,
params: &FunctionServiceVerifyParams,
) -> Result<Instruction, SbError> {
Ok(
crate::utils::build_ix(
&*SWITCHBOARD_ATTESTATION_PROGRAM_ID,
accounts,
params,
)
)
}
}
}