use anchor_lang::prelude::*;
use crate::errors::*;
use crate::state::*;
use crate::utils::*;
#[derive(AnchorSerialize, AnchorDeserialize)]
pub struct VaultTransactionCreateArgs {
pub vault_index: u8,
pub ephemeral_signers: u8,
pub transaction_message: Vec<u8>,
pub memo: Option<String>,
}
#[derive(Accounts)]
#[instruction(args: VaultTransactionCreateArgs)]
pub struct VaultTransactionCreate<'info> {
#[account(
mut,
seeds = [SEED_PREFIX, SEED_MULTISIG, multisig.create_key.as_ref()],
bump = multisig.bump,
)]
pub multisig: Account<'info, Multisig>,
#[account(
init,
payer = rent_payer,
space = VaultTransaction::size(args.ephemeral_signers, &args.transaction_message)?,
seeds = [
SEED_PREFIX,
multisig.key().as_ref(),
SEED_TRANSACTION,
&multisig.transaction_index.checked_add(1).unwrap().to_le_bytes(),
],
bump
)]
pub transaction: Account<'info, VaultTransaction>,
pub creator: Signer<'info>,
#[account(mut)]
pub rent_payer: Signer<'info>,
pub system_program: Program<'info, System>,
}
impl VaultTransactionCreate<'_> {
fn validate(&self) -> Result<()> {
let Self {
multisig, creator, ..
} = self;
require!(
multisig.is_member(creator.key()).is_some(),
MultisigError::NotAMember
);
require!(
multisig.member_has_permission(creator.key(), Permission::Initiate),
MultisigError::Unauthorized
);
Ok(())
}
#[access_control(ctx.accounts.validate())]
pub fn vault_transaction_create(
ctx: Context<Self>,
args: VaultTransactionCreateArgs,
) -> Result<()> {
let multisig = &mut ctx.accounts.multisig;
let transaction = &mut ctx.accounts.transaction;
let creator = &mut ctx.accounts.creator;
let transaction_message =
TransactionMessage::deserialize(&mut args.transaction_message.as_slice())?;
let multisig_key = multisig.key();
let transaction_key = transaction.key();
let vault_seeds = &[
SEED_PREFIX,
multisig_key.as_ref(),
SEED_VAULT,
&args.vault_index.to_le_bytes(),
];
let (_, vault_bump) = Pubkey::find_program_address(vault_seeds, ctx.program_id);
let ephemeral_signer_bumps: Vec<u8> = (0..args.ephemeral_signers)
.into_iter()
.map(|ephemeral_signer_index| {
let ephemeral_signer_seeds = &[
SEED_PREFIX,
transaction_key.as_ref(),
SEED_EPHEMERAL_SIGNER,
&ephemeral_signer_index.to_le_bytes(),
];
let (_, bump) =
Pubkey::find_program_address(ephemeral_signer_seeds, ctx.program_id);
bump
})
.collect();
let transaction_index = multisig.transaction_index.checked_add(1).unwrap();
transaction.multisig = multisig_key;
transaction.creator = creator.key();
transaction.index = transaction_index;
transaction.bump = *ctx.bumps.get("transaction").unwrap();
transaction.vault_index = args.vault_index;
transaction.vault_bump = vault_bump;
transaction.ephemeral_signer_bumps = ephemeral_signer_bumps;
transaction.message = transaction_message.try_into()?;
multisig.transaction_index = transaction_index;
multisig.invariant()?;
msg!("transaction index: {}", transaction_index);
Ok(())
}
}
#[derive(AnchorSerialize, AnchorDeserialize, Clone)]
pub struct TransactionMessage {
pub num_signers: u8,
pub num_writable_signers: u8,
pub num_writable_non_signers: u8,
pub account_keys: SmallVec<u8, Pubkey>,
pub instructions: SmallVec<u8, CompiledInstruction>,
pub address_table_lookups: SmallVec<u8, MessageAddressTableLookup>,
}
#[derive(AnchorSerialize, AnchorDeserialize, Clone)]
pub struct CompiledInstruction {
pub program_id_index: u8,
pub account_indexes: SmallVec<u8, u8>,
pub data: SmallVec<u16, u8>,
}
#[derive(AnchorSerialize, AnchorDeserialize, Clone)]
pub struct MessageAddressTableLookup {
pub account_key: Pubkey,
pub writable_indexes: SmallVec<u8, u8>,
pub readonly_indexes: SmallVec<u8, u8>,
}