lib.rs 6.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239
  1. #![allow(unused_imports)]
  2. #![allow(unused_variables)]
  3. #![allow(unused_mut)]
  4. pub mod dot;
  5. use anchor_lang::prelude::*;
  6. use anchor_spl::{
  7. associated_token::{self, AssociatedToken},
  8. token::{self, Mint, Token, TokenAccount},
  9. };
  10. use dot::program::*;
  11. use std::{cell::RefCell, rc::Rc};
  12. declare_id!("2RjL4mpTANyGxz7fLWEbQtmdEDti7c4CqsLR96mgvcaV");
  13. pub mod seahorse_util {
  14. use super::*;
  15. #[cfg(feature = "pyth-sdk-solana")]
  16. pub use pyth_sdk_solana::{load_price_feed_from_account_info, PriceFeed};
  17. use std::{collections::HashMap, fmt::Debug, ops::Deref};
  18. pub struct Mutable<T>(Rc<RefCell<T>>);
  19. impl<T> Mutable<T> {
  20. pub fn new(obj: T) -> Self {
  21. Self(Rc::new(RefCell::new(obj)))
  22. }
  23. }
  24. impl<T> Clone for Mutable<T> {
  25. fn clone(&self) -> Self {
  26. Self(self.0.clone())
  27. }
  28. }
  29. impl<T> Deref for Mutable<T> {
  30. type Target = Rc<RefCell<T>>;
  31. fn deref(&self) -> &Self::Target {
  32. &self.0
  33. }
  34. }
  35. impl<T: Debug> Debug for Mutable<T> {
  36. fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
  37. write!(f, "{:?}", self.0)
  38. }
  39. }
  40. impl<T: Default> Default for Mutable<T> {
  41. fn default() -> Self {
  42. Self::new(T::default())
  43. }
  44. }
  45. impl<T: Clone> Mutable<Vec<T>> {
  46. pub fn wrapped_index(&self, mut index: i128) -> usize {
  47. if index >= 0 {
  48. return index.try_into().unwrap();
  49. }
  50. index += self.borrow().len() as i128;
  51. return index.try_into().unwrap();
  52. }
  53. }
  54. impl<T: Clone, const N: usize> Mutable<[T; N]> {
  55. pub fn wrapped_index(&self, mut index: i128) -> usize {
  56. if index >= 0 {
  57. return index.try_into().unwrap();
  58. }
  59. index += self.borrow().len() as i128;
  60. return index.try_into().unwrap();
  61. }
  62. }
  63. #[derive(Clone)]
  64. pub struct Empty<T: Clone> {
  65. pub account: T,
  66. pub bump: Option<u8>,
  67. }
  68. #[derive(Clone, Debug)]
  69. pub struct ProgramsMap<'info>(pub HashMap<&'static str, AccountInfo<'info>>);
  70. impl<'info> ProgramsMap<'info> {
  71. pub fn get(&self, name: &'static str) -> AccountInfo<'info> {
  72. self.0.get(name).unwrap().clone()
  73. }
  74. }
  75. #[derive(Clone, Debug)]
  76. pub struct WithPrograms<'info, 'entrypoint, A> {
  77. pub account: &'entrypoint A,
  78. pub programs: &'entrypoint ProgramsMap<'info>,
  79. }
  80. impl<'info, 'entrypoint, A> Deref for WithPrograms<'info, 'entrypoint, A> {
  81. type Target = A;
  82. fn deref(&self) -> &Self::Target {
  83. &self.account
  84. }
  85. }
  86. pub type SeahorseAccount<'info, 'entrypoint, A> =
  87. WithPrograms<'info, 'entrypoint, Box<Account<'info, A>>>;
  88. pub type SeahorseSigner<'info, 'entrypoint> = WithPrograms<'info, 'entrypoint, Signer<'info>>;
  89. #[derive(Clone, Debug)]
  90. pub struct CpiAccount<'info> {
  91. #[doc = "CHECK: CpiAccounts temporarily store AccountInfos."]
  92. pub account_info: AccountInfo<'info>,
  93. pub is_writable: bool,
  94. pub is_signer: bool,
  95. pub seeds: Option<Vec<Vec<u8>>>,
  96. }
  97. #[macro_export]
  98. macro_rules! seahorse_const {
  99. ($ name : ident , $ value : expr) => {
  100. macro_rules! $name {
  101. () => {
  102. $value
  103. };
  104. }
  105. pub(crate) use $name;
  106. };
  107. }
  108. #[macro_export]
  109. macro_rules! assign {
  110. ($ lval : expr , $ rval : expr) => {{
  111. let temp = $rval;
  112. $lval = temp;
  113. }};
  114. }
  115. #[macro_export]
  116. macro_rules! index_assign {
  117. ($ lval : expr , $ idx : expr , $ rval : expr) => {
  118. let temp_rval = $rval;
  119. let temp_idx = $idx;
  120. $lval[temp_idx] = temp_rval;
  121. };
  122. }
  123. pub(crate) use assign;
  124. pub(crate) use index_assign;
  125. pub(crate) use seahorse_const;
  126. }
  127. #[program]
  128. mod seahorse {
  129. use super::*;
  130. use seahorse_util::*;
  131. use std::collections::HashMap;
  132. #[derive(Accounts)]
  133. pub struct InitMockAccount<'info> {
  134. #[account(mut)]
  135. pub signer: Signer<'info>,
  136. # [account (init , space = std :: mem :: size_of :: < dot :: program :: MockAccount > () + 8 , payer = signer , seeds = ["mock_account" . as_bytes () . as_ref ()] , bump)]
  137. pub mock_account: Box<Account<'info, dot::program::MockAccount>>,
  138. pub rent: Sysvar<'info, Rent>,
  139. pub system_program: Program<'info, System>,
  140. }
  141. pub fn init_mock_account(ctx: Context<InitMockAccount>) -> Result<()> {
  142. let mut programs = HashMap::new();
  143. programs.insert(
  144. "system_program",
  145. ctx.accounts.system_program.to_account_info(),
  146. );
  147. let programs_map = ProgramsMap(programs);
  148. let signer = SeahorseSigner {
  149. account: &ctx.accounts.signer,
  150. programs: &programs_map,
  151. };
  152. let mock_account = Empty {
  153. account: dot::program::MockAccount::load(&mut ctx.accounts.mock_account, &programs_map),
  154. bump: ctx.bumps.get("mock_account").map(|bump| *bump),
  155. };
  156. init_mock_account_handler(signer.clone(), mock_account.clone());
  157. dot::program::MockAccount::store(mock_account.account);
  158. return Ok(());
  159. }
  160. #[derive(Accounts)]
  161. # [instruction (amount : u64)]
  162. pub struct TransferSolWithCpi<'info> {
  163. #[account(mut)]
  164. pub sender: Signer<'info>,
  165. #[account(mut)]
  166. pub recipient: Box<Account<'info, dot::program::MockAccount>>,
  167. pub system_program: Program<'info, System>,
  168. }
  169. pub fn transfer_sol_with_cpi(ctx: Context<TransferSolWithCpi>, amount: u64) -> Result<()> {
  170. let mut programs = HashMap::new();
  171. programs.insert(
  172. "system_program",
  173. ctx.accounts.system_program.to_account_info(),
  174. );
  175. let programs_map = ProgramsMap(programs);
  176. let sender = SeahorseSigner {
  177. account: &ctx.accounts.sender,
  178. programs: &programs_map,
  179. };
  180. let recipient = dot::program::MockAccount::load(&mut ctx.accounts.recipient, &programs_map);
  181. transfer_sol_with_cpi_handler(sender.clone(), recipient.clone(), amount);
  182. dot::program::MockAccount::store(recipient);
  183. return Ok(());
  184. }
  185. }