batch.rs 6.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223
  1. mod setup;
  2. use {
  3. crate::setup::TOKEN_PROGRAM_ID,
  4. solana_program_test::{tokio, ProgramTest},
  5. solana_sdk::{
  6. instruction::{AccountMeta, Instruction},
  7. program_error::ProgramError,
  8. program_pack::Pack,
  9. pubkey::Pubkey,
  10. signature::{Keypair, Signer},
  11. system_instruction,
  12. transaction::Transaction,
  13. },
  14. };
  15. fn batch_instruction(instructions: Vec<Instruction>) -> Result<Instruction, ProgramError> {
  16. // Create a `Vec` of ordered `AccountMeta`s
  17. let mut accounts: Vec<AccountMeta> = vec![];
  18. // Start with the batch discriminator
  19. let mut data: Vec<u8> = vec![0xff];
  20. for instruction in instructions {
  21. // Error out on non-token IX.
  22. if instruction.program_id.ne(&spl_token::ID) {
  23. return Err(ProgramError::IncorrectProgramId);
  24. }
  25. data.push(instruction.accounts.len() as u8);
  26. data.push(instruction.data.len() as u8);
  27. data.extend_from_slice(&instruction.data);
  28. accounts.extend_from_slice(&instruction.accounts);
  29. }
  30. Ok(Instruction {
  31. program_id: spl_token::ID,
  32. data,
  33. accounts,
  34. })
  35. }
  36. #[test_case::test_case(TOKEN_PROGRAM_ID ; "p-token")]
  37. #[tokio::test]
  38. async fn batch(token_program: Pubkey) {
  39. let context = ProgramTest::new("pinocchio_token_program", TOKEN_PROGRAM_ID, None)
  40. .start_with_context()
  41. .await;
  42. let rent = context.banks_client.get_rent().await.unwrap();
  43. let mint_len = spl_token::state::Mint::LEN;
  44. let mint_rent = rent.minimum_balance(mint_len);
  45. let account_len = spl_token::state::Account::LEN;
  46. let account_rent = rent.minimum_balance(account_len);
  47. // Create a mint
  48. let mint_a = Keypair::new();
  49. let mint_authority = Keypair::new();
  50. let create_mint_a = system_instruction::create_account(
  51. &context.payer.pubkey(),
  52. &mint_a.pubkey(),
  53. mint_rent,
  54. mint_len as u64,
  55. &token_program,
  56. );
  57. let initialize_mint_ix = spl_token::instruction::initialize_mint(
  58. &token_program,
  59. &mint_a.pubkey(),
  60. &mint_authority.pubkey(),
  61. None,
  62. 6,
  63. )
  64. .unwrap();
  65. // Create a mint 2 with a freeze authority
  66. let mint_b = Keypair::new();
  67. let freeze_authority = Pubkey::new_unique();
  68. let create_mint_b = system_instruction::create_account(
  69. &context.payer.pubkey(),
  70. &mint_b.pubkey(),
  71. mint_rent,
  72. mint_len as u64,
  73. &token_program,
  74. );
  75. let initialize_mint_with_freeze_authority_ix = spl_token::instruction::initialize_mint2(
  76. &token_program,
  77. &mint_b.pubkey(),
  78. &mint_authority.pubkey(),
  79. Some(&freeze_authority),
  80. 6,
  81. )
  82. .unwrap();
  83. // Create 2 token accounts for mint A and 1 for mint B
  84. let owner_a = Keypair::new();
  85. let owner_b = Keypair::new();
  86. let owner_a_ta_a = Keypair::new();
  87. let owner_b_ta_a = Keypair::new();
  88. let create_owner_a_ta_a = system_instruction::create_account(
  89. &context.payer.pubkey(),
  90. &owner_a_ta_a.pubkey(),
  91. account_rent,
  92. account_len as u64,
  93. &token_program,
  94. );
  95. let create_owner_b_ta_a = system_instruction::create_account(
  96. &context.payer.pubkey(),
  97. &owner_b_ta_a.pubkey(),
  98. account_rent,
  99. account_len as u64,
  100. &token_program,
  101. );
  102. let intialize_owner_a_ta_a = spl_token::instruction::initialize_account3(
  103. &token_program,
  104. &owner_a_ta_a.pubkey(),
  105. &mint_a.pubkey(),
  106. &owner_a.pubkey(),
  107. )
  108. .unwrap();
  109. let intialize_owner_b_ta_a = spl_token::instruction::initialize_account3(
  110. &token_program,
  111. &owner_b_ta_a.pubkey(),
  112. &mint_a.pubkey(),
  113. &owner_b.pubkey(),
  114. )
  115. .unwrap();
  116. // Mint Token A to Owner A
  117. let mint_token_a_to_owner_a = spl_token::instruction::mint_to(
  118. &token_program,
  119. &mint_a.pubkey(),
  120. &owner_a_ta_a.pubkey(),
  121. &mint_authority.pubkey(),
  122. &[],
  123. 1_000_000,
  124. )
  125. .unwrap();
  126. // Transfer Token A from Owner A to Owner B
  127. let transfer_token_a_to_owner_b = spl_token::instruction::transfer(
  128. &token_program,
  129. &owner_a_ta_a.pubkey(),
  130. &owner_b_ta_a.pubkey(),
  131. &owner_a.pubkey(),
  132. &[],
  133. 1_000_000,
  134. )
  135. .unwrap();
  136. // Close Token A
  137. let close_owner_a_ta_a = spl_token::instruction::close_account(
  138. &token_program,
  139. &owner_a_ta_a.pubkey(),
  140. &owner_a.pubkey(),
  141. &owner_a.pubkey(),
  142. &[],
  143. )
  144. .unwrap();
  145. let batch_ix = batch_instruction(vec![
  146. initialize_mint_ix,
  147. initialize_mint_with_freeze_authority_ix,
  148. intialize_owner_a_ta_a,
  149. intialize_owner_b_ta_a,
  150. mint_token_a_to_owner_a,
  151. transfer_token_a_to_owner_b,
  152. close_owner_a_ta_a,
  153. ])
  154. .unwrap();
  155. let tx = Transaction::new_signed_with_payer(
  156. &[
  157. create_mint_a,
  158. create_mint_b,
  159. create_owner_a_ta_a,
  160. create_owner_b_ta_a,
  161. batch_ix,
  162. ],
  163. Some(&context.payer.pubkey()),
  164. &vec![
  165. &context.payer,
  166. &mint_a,
  167. &mint_b,
  168. &owner_a_ta_a,
  169. &owner_b_ta_a,
  170. &mint_authority,
  171. &owner_a,
  172. ],
  173. context.last_blockhash,
  174. );
  175. context.banks_client.process_transaction(tx).await.unwrap();
  176. let mint_a_account = context
  177. .banks_client
  178. .get_account(mint_a.pubkey())
  179. .await
  180. .unwrap();
  181. assert!(mint_a_account.is_some());
  182. let mint_a_account = spl_token::state::Mint::unpack(&mint_a_account.unwrap().data).unwrap();
  183. assert_eq!(mint_a_account.supply, 1000000);
  184. let mint_b_account = context
  185. .banks_client
  186. .get_account(mint_b.pubkey())
  187. .await
  188. .unwrap();
  189. assert!(mint_b_account.is_some());
  190. let mint_b_account = spl_token::state::Mint::unpack(&mint_b_account.unwrap().data).unwrap();
  191. assert_eq!(mint_b_account.supply, 0);
  192. let owner_b_ta_a_account = context
  193. .banks_client
  194. .get_account(owner_b_ta_a.pubkey())
  195. .await
  196. .unwrap();
  197. assert!(owner_b_ta_a_account.is_some());
  198. let owner_b_ta_a_account =
  199. spl_token::state::Account::unpack(&owner_b_ta_a_account.unwrap().data).unwrap();
  200. assert_eq!(owner_b_ta_a_account.amount, 1000000);
  201. }