| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466 |
- use borsh::BorshDeserialize;
- use byteorder::{
- BigEndian,
- WriteBytesExt,
- };
- use libsecp256k1::{
- PublicKey,
- SecretKey,
- };
- use sha3::Digest;
- use solana_program::{
- instruction::Instruction,
- pubkey::Pubkey,
- system_instruction,
- };
- use solana_program_test::{
- BanksClient,
- BanksClientError,
- ProgramTest,
- };
- use solana_sdk::{
- commitment_config::CommitmentLevel,
- secp256k1_instruction::new_secp256k1_instruction,
- signature::{
- Keypair,
- Signer,
- },
- signers::Signers,
- transaction::Transaction,
- };
- use std::{
- env,
- io::{
- Cursor,
- Write,
- },
- time::SystemTime,
- };
- use bridge::{
- accounts::FeeCollector,
- instruction,
- instructions,
- types::ConsistencyLevel,
- PostVAAData,
- VerifySignaturesData,
- };
- use solitaire::processors::seeded::Seeded;
- pub use helpers::*;
- /// Simple API wrapper for quickly preparing and sending transactions.
- pub async fn execute<T: Signers>(
- client: &mut BanksClient,
- payer: &Keypair,
- signers: &T,
- instructions: &[Instruction],
- commitment_level: CommitmentLevel,
- ) -> Result<(), BanksClientError> {
- let mut transaction = Transaction::new_with_payer(instructions, Some(&payer.pubkey()));
- let recent_blockhash = client.get_latest_blockhash().await?;
- transaction.sign(signers, recent_blockhash);
- client
- .process_transaction_with_commitment(transaction, commitment_level)
- .await
- }
- mod helpers {
- use super::*;
- use solana_program_test::processor;
- /// Initialize the test environment, spins up a solana-test-validator in the background so that
- /// each test has a fresh environment to work within.
- pub async fn setup() -> (BanksClient, Keypair, Pubkey) {
- let program = env::var("BRIDGE_PROGRAM")
- .unwrap_or_else(|_| "Bridge1p5gheXUvJ6jGWGeCsgPKgnE3YgdGKRVCMY9o".to_string())
- .parse::<Pubkey>()
- .unwrap();
- let builder = ProgramTest::new("bridge", program, processor!(instruction::solitaire));
- let (client, payer, _) = builder.start().await;
- (client, payer, program)
- }
- /// Fetch account data, the loop is there to re-attempt until data is available.
- pub async fn get_account_data<T: BorshDeserialize>(
- client: &mut BanksClient,
- account: Pubkey,
- ) -> T {
- let account = client.get_account(account).await.unwrap().unwrap();
- T::try_from_slice(&account.data).unwrap()
- }
- /// Fetch account balance
- pub async fn get_account_balance(client: &mut BanksClient, account: Pubkey) -> u64 {
- client.get_account(account).await.unwrap().unwrap().lamports
- }
- /// Generate `count` secp256k1 private keys, along with their ethereum-styled public key
- /// encoding: 0x0123456789ABCDEF01234
- pub fn generate_keys(count: u8) -> (Vec<[u8; 20]>, Vec<SecretKey>) {
- let mut rng = rand::thread_rng();
- // Generate Guardian Keys
- let secret_keys: Vec<SecretKey> = std::iter::repeat_with(|| SecretKey::random(&mut rng))
- .take(count as usize)
- .collect();
- (
- secret_keys
- .iter()
- .map(|key| {
- let public_key = PublicKey::from_secret_key(key);
- let mut h = sha3::Keccak256::default();
- h.write_all(&public_key.serialize()[1..]).unwrap();
- let key: [u8; 32] = h.finalize().into();
- let mut address = [0u8; 20];
- address.copy_from_slice(&key[12..]);
- address
- })
- .collect(),
- secret_keys,
- )
- }
- /// Utility function for generating VAA's from message data.
- pub fn generate_vaa(
- emitter: &Keypair,
- data: Vec<u8>,
- nonce: u32,
- sequence: u64,
- guardian_set_index: u32,
- emitter_chain: u16,
- ) -> (PostVAAData, [u8; 32], [u8; 32]) {
- let vaa = PostVAAData {
- version: 0,
- guardian_set_index,
- // Body part
- emitter_chain,
- emitter_address: emitter.pubkey().to_bytes(),
- sequence,
- payload: data,
- timestamp: SystemTime::now()
- .duration_since(SystemTime::UNIX_EPOCH)
- .unwrap()
- .as_secs() as u32,
- nonce,
- consistency_level: ConsistencyLevel::Confirmed as u8,
- };
- // Hash data, the thing we wish to actually sign.
- let body = {
- let mut v = Cursor::new(Vec::new());
- v.write_u32::<BigEndian>(vaa.timestamp).unwrap();
- v.write_u32::<BigEndian>(vaa.nonce).unwrap();
- v.write_u16::<BigEndian>(vaa.emitter_chain).unwrap();
- v.write_all(&vaa.emitter_address).unwrap();
- v.write_u64::<BigEndian>(vaa.sequence).unwrap();
- v.write_u8(vaa.consistency_level).unwrap();
- v.write_all(&vaa.payload).unwrap();
- v.into_inner()
- };
- // Hash this body, which is expected to be the same as the hash currently stored in the
- // signature account, binding that set of signatures to this VAA.
- let body: [u8; 32] = {
- let mut h = sha3::Keccak256::default();
- h.write_all(body.as_slice()).unwrap();
- h.finalize().into()
- };
- let body_hash: [u8; 32] = {
- let mut h = sha3::Keccak256::default();
- h.write_all(&body).unwrap();
- h.finalize().into()
- };
- (vaa, body, body_hash)
- }
- pub async fn initialize(
- client: &mut BanksClient,
- program: Pubkey,
- payer: &Keypair,
- initial_guardians: &[[u8; 20]],
- fee: u64,
- ) -> Result<(), BanksClientError> {
- execute(
- client,
- payer,
- &[payer],
- &[instructions::initialize(
- program,
- payer.pubkey(),
- fee,
- 2_000_000_000,
- initial_guardians,
- )
- .unwrap()],
- CommitmentLevel::Processed,
- )
- .await
- }
- #[allow(clippy::too_many_arguments)]
- pub async fn post_message(
- client: &mut BanksClient,
- program: &Pubkey,
- payer: &Keypair,
- emitter: &Keypair,
- // when None, then a new keypair is generated
- message: Option<&Keypair>,
- nonce: u32,
- data: Vec<u8>,
- fee: u64,
- ) -> Result<Pubkey, BanksClientError> {
- // Transfer money into the fee collector as it needs a balance/must exist.
- let fee_collector = FeeCollector::<'_>::key(None, program);
- let new_message_pair = &Keypair::new();
- let message: &Keypair = match message {
- Some(keypair) => keypair,
- None => new_message_pair,
- };
- // Capture the resulting message, later functions will need this.
- let instruction = instructions::post_message(
- *program,
- payer.pubkey(),
- emitter.pubkey(),
- message.pubkey(),
- nonce,
- data,
- ConsistencyLevel::Confirmed,
- )
- .unwrap();
- execute(
- client,
- payer,
- &[payer, emitter, message],
- &[
- system_instruction::transfer(&payer.pubkey(), &fee_collector, fee),
- instruction,
- ],
- CommitmentLevel::Processed,
- )
- .await?;
- Ok(message.pubkey())
- }
- #[allow(clippy::too_many_arguments)]
- pub async fn post_message_unreliable(
- client: &mut BanksClient,
- program: &Pubkey,
- payer: &Keypair,
- emitter: &Keypair,
- message: &Keypair,
- nonce: u32,
- data: Vec<u8>,
- fee: u64,
- ) -> Result<(), BanksClientError> {
- // Transfer money into the fee collector as it needs a balance/must exist.
- let fee_collector = FeeCollector::<'_>::key(None, program);
- // Capture the resulting message, later functions will need this.
- let instruction = instructions::post_message_unreliable(
- *program,
- payer.pubkey(),
- emitter.pubkey(),
- message.pubkey(),
- nonce,
- data,
- ConsistencyLevel::Confirmed,
- )
- .unwrap();
- execute(
- client,
- payer,
- &[payer, emitter, message],
- &[
- system_instruction::transfer(&payer.pubkey(), &fee_collector, fee),
- instruction,
- ],
- CommitmentLevel::Processed,
- )
- .await
- }
- pub async fn verify_signatures(
- client: &mut BanksClient,
- program: &Pubkey,
- payer: &Keypair,
- body: [u8; 32],
- secret_keys: &[SecretKey],
- guardian_set_version: u32,
- ) -> Result<Pubkey, BanksClientError> {
- let signature_set = Keypair::new();
- let tx_signers = [payer, &signature_set];
- // Push Secp256k1 instructions for each signature we want to verify.
- for (i, key) in secret_keys.iter().enumerate() {
- // Set this signers signature position as present at 0.
- let mut signers = [-1; 19];
- signers[i] = 0;
- execute(
- client,
- payer,
- &tx_signers,
- &[
- new_secp256k1_instruction(key, &body),
- instructions::verify_signatures(
- *program,
- payer.pubkey(),
- guardian_set_version,
- signature_set.pubkey(),
- VerifySignaturesData { signers },
- )
- .unwrap(),
- ],
- CommitmentLevel::Processed,
- )
- .await?;
- }
- Ok(signature_set.pubkey())
- }
- pub async fn post_vaa(
- client: &mut BanksClient,
- program: &Pubkey,
- payer: &Keypair,
- signature_set: Pubkey,
- vaa: PostVAAData,
- ) -> Result<(), BanksClientError> {
- execute(
- client,
- payer,
- &[payer],
- &[instructions::post_vaa(
- *program,
- payer.pubkey(),
- signature_set,
- vaa,
- )],
- CommitmentLevel::Processed,
- )
- .await
- }
- #[allow(clippy::too_many_arguments)]
- pub async fn upgrade_guardian_set(
- client: &mut BanksClient,
- program: &Pubkey,
- payer: &Keypair,
- payload_message: Pubkey,
- emitter: Pubkey,
- old_index: u32,
- new_index: u32,
- sequence: u64,
- ) -> Result<(), BanksClientError> {
- execute(
- client,
- payer,
- &[payer],
- &[instructions::upgrade_guardian_set(
- *program,
- payer.pubkey(),
- payload_message,
- emitter,
- old_index,
- new_index,
- sequence,
- )],
- CommitmentLevel::Processed,
- )
- .await
- }
- #[allow(clippy::too_many_arguments)]
- pub async fn upgrade_contract(
- client: &mut BanksClient,
- program: &Pubkey,
- payer: &Keypair,
- payload_message: Pubkey,
- emitter: Pubkey,
- new_contract: Pubkey,
- spill: Pubkey,
- sequence: u64,
- ) -> Result<(), BanksClientError> {
- execute(
- client,
- payer,
- &[payer],
- &[instructions::upgrade_contract(
- *program,
- payer.pubkey(),
- payload_message,
- emitter,
- new_contract,
- spill,
- sequence,
- )],
- CommitmentLevel::Processed,
- )
- .await
- }
- pub async fn set_fees(
- client: &mut BanksClient,
- program: &Pubkey,
- payer: &Keypair,
- message: Pubkey,
- emitter: Pubkey,
- sequence: u64,
- ) -> Result<(), BanksClientError> {
- execute(
- client,
- payer,
- &[payer],
- &[instructions::set_fees(
- *program,
- payer.pubkey(),
- message,
- emitter,
- sequence,
- )],
- CommitmentLevel::Processed,
- )
- .await
- }
- pub async fn transfer_fees(
- client: &mut BanksClient,
- program: &Pubkey,
- payer: &Keypair,
- message: Pubkey,
- emitter: Pubkey,
- recipient: Pubkey,
- sequence: u64,
- ) -> Result<(), BanksClientError> {
- execute(
- client,
- payer,
- &[payer],
- &[instructions::transfer_fees(
- *program,
- payer.pubkey(),
- message,
- emitter,
- sequence,
- recipient,
- )],
- CommitmentLevel::Processed,
- )
- .await
- }
- }
|