1use {
4 crate::{exchange_instruction::*, exchange_state::*, faucet},
5 log::*,
6 num_derive::{FromPrimitive, ToPrimitive},
7 serde_derive::Serialize,
8 solana_metrics::inc_new_counter_info,
9 solana_sdk::{
10 account::{ReadableAccount, WritableAccount},
11 decode_error::DecodeError,
12 instruction::InstructionError,
13 keyed_account::KeyedAccount,
14 process_instruction::InvokeContext,
15 program_utils::limited_deserialize,
16 pubkey::Pubkey,
17 },
18 std::cmp,
19 thiserror::Error,
20};
21
22#[derive(Error, Debug, Serialize, Clone, PartialEq, FromPrimitive, ToPrimitive)]
23pub enum ExchangeError {
24 #[error("Signer does not own account")]
25 SignerDoesNotOwnAccount,
26 #[error("Signer does not own order")]
27 SignerDoesNotOwnOrder,
28 #[error("The From account balance is too low")]
29 FromAccountBalanceTooLow,
30 #[error("Attmept operation on mismatched tokens")]
31 TokenMismatch,
32 #[error("From trade balance is too low")]
33 FromTradeBalanceTooLow,
34 #[error("Serialization failed")]
35 SerializeFailed,
36}
37impl<T> DecodeError<T> for ExchangeError {
38 fn type_of() -> &'static str {
39 "ExchangeError"
40 }
41}
42
43pub struct ExchangeProcessor {}
44
45impl ExchangeProcessor {
46 #[allow(clippy::needless_pass_by_value)]
47 fn map_to_invalid_arg(err: std::boxed::Box<bincode::ErrorKind>) -> InstructionError {
48 warn!("Deserialize failed, not a valid state: {:?}", err);
49 InstructionError::InvalidArgument
50 }
51
52 fn is_account_unallocated(data: &[u8]) -> Result<(), InstructionError> {
53 let state: ExchangeState = bincode::deserialize(data).map_err(Self::map_to_invalid_arg)?;
54 if let ExchangeState::Unallocated = state {
55 Ok(())
56 } else {
57 error!("New account is already in use");
58 Err(InstructionError::InvalidAccountData)
59 }
60 }
61
62 fn deserialize_account(data: &[u8]) -> Result<TokenAccountInfo, InstructionError> {
63 let state: ExchangeState = bincode::deserialize(data).map_err(Self::map_to_invalid_arg)?;
64 if let ExchangeState::Account(account) = state {
65 Ok(account)
66 } else {
67 error!("Not a valid account");
68 Err(InstructionError::InvalidAccountData)
69 }
70 }
71
72 fn deserialize_order(data: &[u8]) -> Result<OrderInfo, InstructionError> {
73 let state: ExchangeState = bincode::deserialize(data).map_err(Self::map_to_invalid_arg)?;
74 if let ExchangeState::Trade(info) = state {
75 Ok(info)
76 } else {
77 error!("Not a valid trade");
78 Err(InstructionError::InvalidAccountData)
79 }
80 }
81
82 fn serialize(state: &ExchangeState, data: &mut [u8]) -> Result<(), InstructionError> {
83 let writer = std::io::BufWriter::new(data);
84 match bincode::serialize_into(writer, state) {
85 Ok(_) => Ok(()),
86 Err(e) => {
87 error!("Serialize failed: {:?}", e);
88 Err(ExchangeError::SerializeFailed.into())
89 }
90 }
91 }
92
93 fn trade_to_token_account(trade: &OrderInfo) -> TokenAccountInfo {
94 let token = match trade.side {
97 OrderSide::Ask => trade.pair.Quote,
98 OrderSide::Bid => trade.pair.Base,
99 };
100
101 let mut account = TokenAccountInfo::default().owner(&trade.owner);
102 account.tokens[token] = trade.tokens_settled;
103 account
104 }
105
106 fn calculate_swap(
107 scaler: u64,
108 to_trade: &mut OrderInfo,
109 from_trade: &mut OrderInfo,
110 profit_account: &mut TokenAccountInfo,
111 ) -> Result<(), InstructionError> {
112 if to_trade.tokens == 0 || from_trade.tokens == 0 {
113 error!("Inactive Trade, balance is zero");
114 return Err(InstructionError::InvalidArgument);
115 }
116 if to_trade.price == 0 || from_trade.price == 0 {
117 error!("Inactive Trade, price is zero");
118 return Err(InstructionError::InvalidArgument);
119 }
120
121 trace!("tt {} ft {}", to_trade.tokens, from_trade.tokens);
124 trace!("tp {} fp {}", to_trade.price, from_trade.price);
125
126 let max_to_secondary = to_trade.tokens * to_trade.price / scaler;
127 let max_to_primary = from_trade.tokens * scaler / from_trade.price;
128
129 trace!("mtp {} mts {}", max_to_primary, max_to_secondary);
130
131 let max_primary = cmp::min(max_to_primary, to_trade.tokens);
132 let max_secondary = cmp::min(max_to_secondary, from_trade.tokens);
133
134 trace!("mp {} ms {}", max_primary, max_secondary);
135
136 let primary_tokens = if max_secondary < max_primary {
137 max_secondary * scaler / from_trade.price
138 } else {
139 max_primary
140 };
141 let secondary_tokens = if max_secondary < max_primary {
142 max_secondary
143 } else {
144 max_primary * to_trade.price / scaler
145 };
146
147 if primary_tokens == 0 || secondary_tokens == 0 {
148 error!("Trade quantities to low to be fulfilled");
149 return Err(InstructionError::InvalidArgument);
150 }
151
152 trace!("pt {} st {}", primary_tokens, secondary_tokens);
153
154 let primary_cost = cmp::max(primary_tokens, secondary_tokens * scaler / to_trade.price);
155 let secondary_cost = cmp::max(secondary_tokens, primary_tokens * from_trade.price / scaler);
156
157 trace!("pc {} sc {}", primary_cost, secondary_cost);
158
159 let primary_profit = primary_cost - primary_tokens;
160 let secondary_profit = secondary_cost - secondary_tokens;
161
162 trace!("pp {} sp {}", primary_profit, secondary_profit);
163
164 let primary_token = to_trade.pair.Base;
165 let secondary_token = from_trade.pair.Quote;
166
167 if to_trade.tokens < primary_cost {
170 error!("Not enough tokens in to account");
171 return Err(InstructionError::InvalidArgument);
172 }
173 if from_trade.tokens < secondary_cost {
174 error!("Not enough tokens in from account");
175 return Err(InstructionError::InvalidArgument);
176 }
177 to_trade.tokens -= primary_cost;
178 to_trade.tokens_settled += secondary_tokens;
179 from_trade.tokens -= secondary_cost;
180 from_trade.tokens_settled += primary_tokens;
181
182 profit_account.tokens[primary_token] += primary_profit;
183 profit_account.tokens[secondary_token] += secondary_profit;
184
185 Ok(())
186 }
187
188 fn do_account_request(keyed_accounts: &[KeyedAccount]) -> Result<(), InstructionError> {
189 const OWNER_INDEX: usize = 0;
190 const NEW_ACCOUNT_INDEX: usize = 1;
191
192 if keyed_accounts.len() < 2 {
193 error!("Not enough accounts");
194 return Err(InstructionError::InvalidArgument);
195 }
196 Self::is_account_unallocated(keyed_accounts[NEW_ACCOUNT_INDEX].try_account_ref()?.data())?;
197 Self::serialize(
198 &ExchangeState::Account(
199 TokenAccountInfo::default()
200 .owner(keyed_accounts[OWNER_INDEX].unsigned_key())
201 .tokens(100_000, 100_000, 100_000, 100_000),
202 ),
203 &mut keyed_accounts[NEW_ACCOUNT_INDEX]
204 .try_account_ref_mut()?
205 .data_as_mut_slice(),
206 )
207 }
208
209 fn do_transfer_request(
210 keyed_accounts: &[KeyedAccount],
211 token: Token,
212 tokens: u64,
213 ) -> Result<(), InstructionError> {
214 const OWNER_INDEX: usize = 0;
215 const TO_ACCOUNT_INDEX: usize = 1;
216 const FROM_ACCOUNT_INDEX: usize = 2;
217
218 if keyed_accounts.len() < 3 {
219 error!("Not enough accounts");
220 return Err(InstructionError::InvalidArgument);
221 }
222
223 let mut to_account =
224 Self::deserialize_account(keyed_accounts[TO_ACCOUNT_INDEX].try_account_ref()?.data())?;
225
226 if &faucet::id() == keyed_accounts[FROM_ACCOUNT_INDEX].unsigned_key() {
227 to_account.tokens[token] += tokens;
228 } else {
229 let state: ExchangeState =
230 bincode::deserialize(keyed_accounts[FROM_ACCOUNT_INDEX].try_account_ref()?.data())
231 .map_err(Self::map_to_invalid_arg)?;
232 match state {
233 ExchangeState::Account(mut from_account) => {
234 if &from_account.owner != keyed_accounts[OWNER_INDEX].unsigned_key() {
235 error!("Signer does not own from account");
236 return Err(ExchangeError::SignerDoesNotOwnAccount.into());
237 }
238
239 if from_account.tokens[token] < tokens {
240 error!("From account balance too low");
241 return Err(ExchangeError::FromAccountBalanceTooLow.into());
242 }
243
244 from_account.tokens[token] -= tokens;
245 to_account.tokens[token] += tokens;
246
247 Self::serialize(
248 &ExchangeState::Account(from_account),
249 &mut keyed_accounts[FROM_ACCOUNT_INDEX]
250 .try_account_ref_mut()?
251 .data_as_mut_slice(),
252 )?;
253 }
254 ExchangeState::Trade(mut from_trade) => {
255 if &from_trade.owner != keyed_accounts[OWNER_INDEX].unsigned_key() {
256 error!("Signer does not own from account");
257 return Err(ExchangeError::SignerDoesNotOwnAccount.into());
258 }
259
260 let from_token = match from_trade.side {
261 OrderSide::Ask => from_trade.pair.Quote,
262 OrderSide::Bid => from_trade.pair.Base,
263 };
264 if token != from_token {
265 error!("Trade to transfer from does not hold correct token");
266 return Err(ExchangeError::TokenMismatch.into());
267 }
268
269 if from_trade.tokens_settled < tokens {
270 error!("From trade balance too low");
271 return Err(ExchangeError::FromTradeBalanceTooLow.into());
272 }
273
274 from_trade.tokens_settled -= tokens;
275 to_account.tokens[token] += tokens;
276
277 Self::serialize(
278 &ExchangeState::Trade(from_trade),
279 &mut keyed_accounts[FROM_ACCOUNT_INDEX]
280 .try_account_ref_mut()?
281 .data_as_mut_slice(),
282 )?;
283 }
284 _ => {
285 error!("Not a valid from account for transfer");
286 return Err(InstructionError::InvalidArgument);
287 }
288 }
289 }
290
291 Self::serialize(
292 &ExchangeState::Account(to_account),
293 &mut keyed_accounts[TO_ACCOUNT_INDEX]
294 .try_account_ref_mut()?
295 .data_as_mut_slice(),
296 )
297 }
298
299 fn do_order_request(
300 keyed_accounts: &[KeyedAccount],
301 info: &OrderRequestInfo,
302 ) -> Result<(), InstructionError> {
303 const OWNER_INDEX: usize = 0;
304 const ORDER_INDEX: usize = 1;
305 const ACCOUNT_INDEX: usize = 2;
306
307 if keyed_accounts.len() < 3 {
308 error!("Not enough accounts");
309 return Err(InstructionError::InvalidArgument);
310 }
311
312 Self::is_account_unallocated(keyed_accounts[ORDER_INDEX].try_account_ref()?.data())?;
313
314 let mut account =
315 Self::deserialize_account(keyed_accounts[ACCOUNT_INDEX].try_account_ref_mut()?.data())?;
316
317 if &account.owner != keyed_accounts[OWNER_INDEX].unsigned_key() {
318 error!("Signer does not own account");
319 return Err(ExchangeError::SignerDoesNotOwnAccount.into());
320 }
321 let from_token = match info.side {
322 OrderSide::Ask => info.pair.Base,
323 OrderSide::Bid => info.pair.Quote,
324 };
325 if account.tokens[from_token] < info.tokens {
326 error!("From token balance is too low");
327 return Err(ExchangeError::FromAccountBalanceTooLow.into());
328 }
329
330 if let Err(e) = check_trade(info.side, info.tokens, info.price) {
331 bincode::serialize(&e).unwrap();
332 }
333
334 account.tokens[from_token] -= info.tokens;
336
337 inc_new_counter_info!("exchange_processor-trades", 1);
338
339 Self::serialize(
340 &ExchangeState::Trade(OrderInfo {
341 owner: *keyed_accounts[OWNER_INDEX].unsigned_key(),
342 side: info.side,
343 pair: info.pair,
344 tokens: info.tokens,
345 price: info.price,
346 tokens_settled: 0,
347 }),
348 &mut keyed_accounts[ORDER_INDEX]
349 .try_account_ref_mut()?
350 .data_as_mut_slice(),
351 )?;
352 Self::serialize(
353 &ExchangeState::Account(account),
354 &mut keyed_accounts[ACCOUNT_INDEX]
355 .try_account_ref_mut()?
356 .data_as_mut_slice(),
357 )
358 }
359
360 fn do_order_cancellation(keyed_accounts: &[KeyedAccount]) -> Result<(), InstructionError> {
361 const OWNER_INDEX: usize = 0;
362 const ORDER_INDEX: usize = 1;
363
364 if keyed_accounts.len() < 2 {
365 error!("Not enough accounts");
366 return Err(InstructionError::InvalidArgument);
367 }
368
369 let order = Self::deserialize_order(keyed_accounts[ORDER_INDEX].try_account_ref()?.data())?;
370
371 if &order.owner != keyed_accounts[OWNER_INDEX].unsigned_key() {
372 error!("Signer does not own order");
373 return Err(ExchangeError::SignerDoesNotOwnOrder.into());
374 }
375
376 let token = match order.side {
377 OrderSide::Ask => order.pair.Base,
378 OrderSide::Bid => order.pair.Quote,
379 };
380
381 let mut account = TokenAccountInfo::default().owner(&order.owner);
382 account.tokens[token] = order.tokens;
383 account.tokens[token] += order.tokens_settled;
384
385 Self::serialize(
387 &ExchangeState::Account(account),
388 &mut keyed_accounts[ORDER_INDEX]
389 .try_account_ref_mut()?
390 .data_as_mut_slice(),
391 )
392 }
393
394 fn do_swap_request(keyed_accounts: &[KeyedAccount]) -> Result<(), InstructionError> {
395 const TO_ORDER_INDEX: usize = 1;
396 const FROM_ORDER_INDEX: usize = 2;
397 const PROFIT_ACCOUNT_INDEX: usize = 3;
398
399 if keyed_accounts.len() < 4 {
400 error!("Not enough accounts");
401 return Err(InstructionError::InvalidArgument);
402 }
403
404 let mut to_order =
405 Self::deserialize_order(keyed_accounts[TO_ORDER_INDEX].try_account_ref()?.data())?;
406 let mut from_order =
407 Self::deserialize_order(keyed_accounts[FROM_ORDER_INDEX].try_account_ref()?.data())?;
408 let mut profit_account = Self::deserialize_account(
409 keyed_accounts[PROFIT_ACCOUNT_INDEX]
410 .try_account_ref()?
411 .data(),
412 )?;
413
414 if to_order.side != OrderSide::Ask {
415 error!("To trade is not a To");
416 return Err(InstructionError::InvalidArgument);
417 }
418 if from_order.side != OrderSide::Bid {
419 error!("From trade is not a From");
420 return Err(InstructionError::InvalidArgument);
421 }
422 if to_order.pair != from_order.pair {
423 error!("Mismatched token pairs");
424 return Err(InstructionError::InvalidArgument);
425 }
426 if to_order.side == from_order.side {
427 error!("Matching trade sides");
428 return Err(InstructionError::InvalidArgument);
429 }
430
431 if let Err(e) =
432 Self::calculate_swap(SCALER, &mut to_order, &mut from_order, &mut profit_account)
433 {
434 error!(
435 "Swap calculation failed from {} for {} to {} for {}",
436 from_order.tokens, from_order.price, to_order.tokens, to_order.price,
437 );
438 return Err(e);
439 }
440
441 inc_new_counter_info!("exchange_processor-swaps", 1);
442
443 if to_order.tokens == 0 {
444 Self::serialize(
446 &ExchangeState::Account(Self::trade_to_token_account(&from_order)),
447 &mut keyed_accounts[TO_ORDER_INDEX]
448 .try_account_ref_mut()?
449 .data_as_mut_slice(),
450 )?;
451 } else {
452 Self::serialize(
453 &ExchangeState::Trade(to_order),
454 &mut keyed_accounts[TO_ORDER_INDEX]
455 .try_account_ref_mut()?
456 .data_as_mut_slice(),
457 )?;
458 }
459
460 if from_order.tokens == 0 {
461 Self::serialize(
463 &ExchangeState::Account(Self::trade_to_token_account(&from_order)),
464 &mut keyed_accounts[FROM_ORDER_INDEX]
465 .try_account_ref_mut()?
466 .data_as_mut_slice(),
467 )?;
468 } else {
469 Self::serialize(
470 &ExchangeState::Trade(from_order),
471 &mut keyed_accounts[FROM_ORDER_INDEX]
472 .try_account_ref_mut()?
473 .data_as_mut_slice(),
474 )?;
475 }
476
477 Self::serialize(
478 &ExchangeState::Account(profit_account),
479 &mut keyed_accounts[PROFIT_ACCOUNT_INDEX]
480 .try_account_ref_mut()?
481 .data_as_mut_slice(),
482 )
483 }
484}
485
486pub fn process_instruction(
487 _program_id: &Pubkey,
488 data: &[u8],
489 invoke_context: &mut dyn InvokeContext,
490) -> Result<(), InstructionError> {
491 let keyed_accounts = invoke_context.get_keyed_accounts()?;
492
493 solana_logger::setup();
494 match limited_deserialize::<ExchangeInstruction>(data)? {
495 ExchangeInstruction::AccountRequest => {
496 ExchangeProcessor::do_account_request(keyed_accounts)
497 }
498 ExchangeInstruction::TransferRequest(token, tokens) => {
499 ExchangeProcessor::do_transfer_request(keyed_accounts, token, tokens)
500 }
501 ExchangeInstruction::OrderRequest(info) => {
502 ExchangeProcessor::do_order_request(keyed_accounts, &info)
503 }
504 ExchangeInstruction::OrderCancellation => {
505 ExchangeProcessor::do_order_cancellation(keyed_accounts)
506 }
507 ExchangeInstruction::SwapRequest => ExchangeProcessor::do_swap_request(keyed_accounts),
508 }
509}
510
511#[cfg(test)]
512mod test {
513 use {
514 super::*,
515 crate::{exchange_instruction, id},
516 solana_runtime::{bank::Bank, bank_client::BankClient},
517 solana_sdk::{
518 client::SyncClient,
519 genesis_config::create_genesis_config,
520 message::Message,
521 signature::{Keypair, Signer},
522 system_instruction,
523 },
524 std::mem,
525 };
526
527 #[allow(clippy::too_many_arguments)]
528 fn try_calc(
529 scaler: u64,
530 primary_tokens: u64,
531 primary_price: u64,
532 secondary_tokens: u64,
533 secondary_price: u64,
534 primary_tokens_expect: u64,
535 secondary_tokens_expect: u64,
536 primary_tokens_settled_expect: u64,
537 secondary_tokens_settled_expect: u64,
538 profit_account_tokens: Tokens,
539 ) -> Result<(), InstructionError> {
540 trace!(
541 "Swap {} for {} to {} for {}",
542 primary_tokens,
543 primary_price,
544 secondary_tokens,
545 secondary_price,
546 );
547 let mut to_trade = OrderInfo::default();
548 let mut from_trade = OrderInfo::default().side(OrderSide::Bid);
549 let mut profit_account = TokenAccountInfo::default();
550
551 to_trade.tokens = primary_tokens;
552 to_trade.price = primary_price;
553 from_trade.tokens = secondary_tokens;
554 from_trade.price = secondary_price;
555 ExchangeProcessor::calculate_swap(
556 scaler,
557 &mut to_trade,
558 &mut from_trade,
559 &mut profit_account,
560 )?;
561
562 trace!(
563 "{:?} {:?} {:?} {:?}\n{:?}\n{:?}\n{:?}\n{:?}",
564 to_trade.tokens,
565 primary_tokens_expect,
566 from_trade.tokens,
567 secondary_tokens_expect,
568 primary_tokens_settled_expect,
569 secondary_tokens_settled_expect,
570 profit_account.tokens,
571 profit_account_tokens
572 );
573
574 assert_eq!(to_trade.tokens, primary_tokens_expect);
575 assert_eq!(from_trade.tokens, secondary_tokens_expect);
576 assert_eq!(to_trade.tokens_settled, primary_tokens_settled_expect);
577 assert_eq!(from_trade.tokens_settled, secondary_tokens_settled_expect);
578 assert_eq!(profit_account.tokens, profit_account_tokens);
579 Ok(())
580 }
581
582 #[test]
583 #[rustfmt::skip]
584 fn test_calculate_swap() {
585 solana_logger::setup();
586
587 try_calc(1, 50, 2, 50, 1, 0, 0, 50, 50, Tokens::new( 0, 0, 0, 0)).unwrap_err();
588 try_calc(1, 50, 1, 0, 1, 0, 0, 50, 50, Tokens::new( 0, 0, 0, 0)).unwrap_err();
589 try_calc(1, 0, 1, 50, 1, 0, 0, 50, 50, Tokens::new( 0, 0, 0, 0)).unwrap_err();
590 try_calc(1, 50, 1, 50, 0, 0, 0, 50, 50, Tokens::new( 0, 0, 0, 0)).unwrap_err();
591 try_calc(1, 50, 0, 50, 1, 0, 0, 50, 50, Tokens::new( 0, 0, 0, 0)).unwrap_err();
592 try_calc(1, 1, 2, 2, 3, 1, 2, 0, 0, Tokens::new( 0, 0, 0, 0)).unwrap_err();
593
594 try_calc(1, 50, 1, 50, 1, 0, 0, 50, 50, Tokens::new( 0, 0, 0, 0)).unwrap();
595 try_calc(1, 1, 2, 3, 3, 0, 0, 2, 1, Tokens::new( 0, 1, 0, 0)).unwrap();
596 try_calc(1, 2, 2, 3, 3, 1, 0, 2, 1, Tokens::new( 0, 1, 0, 0)).unwrap();
597 try_calc(1, 3, 2, 3, 3, 2, 0, 2, 1, Tokens::new( 0, 1, 0, 0)).unwrap();
598 try_calc(1, 3, 2, 6, 3, 1, 0, 4, 2, Tokens::new( 0, 2, 0, 0)).unwrap();
599 try_calc(1000, 1, 2000, 3, 3000, 0, 0, 2, 1, Tokens::new( 0, 1, 0, 0)).unwrap();
600 try_calc(1, 3, 2, 7, 3, 1, 1, 4, 2, Tokens::new( 0, 2, 0, 0)).unwrap();
601 try_calc(1000, 3000, 333, 1000, 500, 0, 1,999, 1998, Tokens::new(1002, 0, 0, 0)).unwrap();
602 try_calc(1000, 50, 100, 50, 101, 0,45, 5, 49, Tokens::new( 1, 0, 0, 0)).unwrap();
603 }
604
605 fn create_bank(lamports: u64) -> (Bank, Keypair) {
606 let (genesis_config, mint_keypair) = create_genesis_config(lamports);
607 let mut bank = Bank::new(&genesis_config);
608 bank.add_builtin("exchange_program", id(), process_instruction);
609 (bank, mint_keypair)
610 }
611
612 fn create_client(bank: Bank, mint_keypair: Keypair) -> (BankClient, Keypair) {
613 let owner = Keypair::new();
614 let bank_client = BankClient::new(bank);
615 bank_client
616 .transfer_and_confirm(42, &mint_keypair, &owner.pubkey())
617 .unwrap();
618
619 (bank_client, owner)
620 }
621
622 fn create_account(client: &BankClient, owner: &Keypair) -> Pubkey {
623 let new = Keypair::new();
624
625 let instruction = system_instruction::create_account(
626 &owner.pubkey(),
627 &new.pubkey(),
628 1,
629 mem::size_of::<ExchangeState>() as u64,
630 &id(),
631 );
632
633 client
634 .send_and_confirm_message(
635 &[owner, &new],
636 Message::new(&[instruction], Some(&owner.pubkey())),
637 )
638 .unwrap_or_else(|_| panic!("{}:{}", line!(), file!()));
639 new.pubkey()
640 }
641
642 fn create_token_account(client: &BankClient, owner: &Keypair) -> Pubkey {
643 let new = create_account(client, owner);
644 let instruction = exchange_instruction::account_request(&owner.pubkey(), &new);
645 client
646 .send_and_confirm_instruction(owner, instruction)
647 .unwrap_or_else(|_| panic!("{}:{}", line!(), file!()));
648 new
649 }
650
651 fn transfer(client: &BankClient, owner: &Keypair, to: &Pubkey, token: Token, tokens: u64) {
652 let instruction = exchange_instruction::transfer_request(
653 &owner.pubkey(),
654 to,
655 &faucet::id(),
656 token,
657 tokens,
658 );
659 client
660 .send_and_confirm_instruction(owner, instruction)
661 .unwrap_or_else(|_| panic!("{}:{}", line!(), file!()));
662 }
663
664 fn trade(
665 client: &BankClient,
666 owner: &Keypair,
667 side: OrderSide,
668 pair: AssetPair,
669 from_token: Token,
670 src_tokens: u64,
671 trade_tokens: u64,
672 price: u64,
673 ) -> (Pubkey, Pubkey) {
674 let trade = create_account(client, owner);
675 let src = create_token_account(client, owner);
676 transfer(client, owner, &src, from_token, src_tokens);
677
678 let instruction = exchange_instruction::trade_request(
679 &owner.pubkey(),
680 &trade,
681 side,
682 pair,
683 trade_tokens,
684 price,
685 &src,
686 );
687 client
688 .send_and_confirm_instruction(owner, instruction)
689 .unwrap_or_else(|_| panic!("{}:{}", line!(), file!()));
690 (trade, src)
691 }
692
693 #[test]
694 fn test_exchange_new_account() {
695 solana_logger::setup();
696 let (bank, mint_keypair) = create_bank(10_000);
697 let (client, owner) = create_client(bank, mint_keypair);
698
699 let new = create_token_account(&client, &owner);
700 let new_account_data = client.get_account_data(&new).unwrap().unwrap();
701
702 assert_eq!(
705 TokenAccountInfo::default()
706 .owner(&owner.pubkey())
707 .tokens(100_000, 100_000, 100_000, 100_000),
708 ExchangeProcessor::deserialize_account(&new_account_data).unwrap()
709 );
710 }
711
712 #[test]
713 fn test_exchange_new_account_not_unallocated() {
714 solana_logger::setup();
715 let (bank, mint_keypair) = create_bank(10_000);
716 let (client, owner) = create_client(bank, mint_keypair);
717
718 let new = create_token_account(&client, &owner);
719 let instruction = exchange_instruction::account_request(&owner.pubkey(), &new);
720 client
721 .send_and_confirm_instruction(&owner, instruction)
722 .expect_err(&format!("{}:{}", line!(), file!()));
723 }
724
725 #[test]
726 fn test_exchange_new_transfer_request() {
727 solana_logger::setup();
728 let (bank, mint_keypair) = create_bank(10_000);
729 let (client, owner) = create_client(bank, mint_keypair);
730
731 let new = create_token_account(&client, &owner);
732
733 let instruction = exchange_instruction::transfer_request(
734 &owner.pubkey(),
735 &new,
736 &faucet::id(),
737 Token::A,
738 42,
739 );
740 client
741 .send_and_confirm_instruction(&owner, instruction)
742 .unwrap_or_else(|_| panic!("{}:{}", line!(), file!()));
743
744 let new_account_data = client.get_account_data(&new).unwrap().unwrap();
745
746 assert_eq!(
749 TokenAccountInfo::default()
750 .owner(&owner.pubkey())
751 .tokens(100_042, 100_000, 100_000, 100_000),
752 ExchangeProcessor::deserialize_account(&new_account_data).unwrap()
753 );
754 }
755
756 #[test]
757 fn test_exchange_new_trade_request() {
758 solana_logger::setup();
759 let (bank, mint_keypair) = create_bank(10_000);
760 let (client, owner) = create_client(bank, mint_keypair);
761
762 let (trade, src) = trade(
763 &client,
764 &owner,
765 OrderSide::Ask,
766 AssetPair::default(),
767 Token::A,
768 42,
769 2,
770 1000,
771 );
772
773 let trade_account_data = client.get_account_data(&trade).unwrap().unwrap();
774 let src_account_data = client.get_account_data(&src).unwrap().unwrap();
775
776 assert_eq!(
779 OrderInfo {
780 owner: owner.pubkey(),
781 side: OrderSide::Ask,
782 pair: AssetPair::default(),
783 tokens: 2,
784 price: 1000,
785 tokens_settled: 0
786 },
787 ExchangeProcessor::deserialize_order(&trade_account_data).unwrap()
788 );
789 assert_eq!(
790 TokenAccountInfo::default()
791 .owner(&owner.pubkey())
792 .tokens(100_040, 100_000, 100_000, 100_000),
793 ExchangeProcessor::deserialize_account(&src_account_data).unwrap()
794 );
795 }
796
797 #[test]
798 fn test_exchange_new_swap_request() {
799 solana_logger::setup();
800 let (bank, mint_keypair) = create_bank(10_000);
801 let (client, owner) = create_client(bank, mint_keypair);
802
803 let profit = create_token_account(&client, &owner);
804 let (to_trade, _) = trade(
805 &client,
806 &owner,
807 OrderSide::Ask,
808 AssetPair::default(),
809 Token::A,
810 2,
811 2,
812 2000,
813 );
814 let (from_trade, _) = trade(
815 &client,
816 &owner,
817 OrderSide::Bid,
818 AssetPair::default(),
819 Token::B,
820 3,
821 3,
822 3000,
823 );
824
825 let instruction =
826 exchange_instruction::swap_request(&owner.pubkey(), &to_trade, &from_trade, &profit);
827 client
828 .send_and_confirm_instruction(&owner, instruction)
829 .unwrap_or_else(|_| panic!("{}:{}", line!(), file!()));
830
831 let to_trade_account_data = client.get_account_data(&to_trade).unwrap().unwrap();
832 let from_trade_account_data = client.get_account_data(&from_trade).unwrap().unwrap();
833 let profit_account_data = client.get_account_data(&profit).unwrap().unwrap();
834
835 assert_eq!(
838 OrderInfo {
839 owner: owner.pubkey(),
840 side: OrderSide::Ask,
841 pair: AssetPair::default(),
842 tokens: 1,
843 price: 2000,
844 tokens_settled: 2,
845 },
846 ExchangeProcessor::deserialize_order(&to_trade_account_data).unwrap()
847 );
848
849 assert_eq!(
850 TokenAccountInfo::default()
851 .owner(&owner.pubkey())
852 .tokens(1, 0, 0, 0),
853 ExchangeProcessor::deserialize_account(&from_trade_account_data).unwrap()
854 );
855
856 assert_eq!(
857 TokenAccountInfo::default()
858 .owner(&owner.pubkey())
859 .tokens(100_000, 100_001, 100_000, 100_000),
860 ExchangeProcessor::deserialize_account(&profit_account_data).unwrap()
861 );
862 }
863
864 #[test]
865 fn test_exchange_trade_to_token_account() {
866 solana_logger::setup();
867 let (bank, mint_keypair) = create_bank(10_000);
868 let (client, owner) = create_client(bank, mint_keypair);
869
870 let profit = create_token_account(&client, &owner);
871 let (to_trade, _) = trade(
872 &client,
873 &owner,
874 OrderSide::Ask,
875 AssetPair::default(),
876 Token::A,
877 3,
878 3,
879 2000,
880 );
881 let (from_trade, _) = trade(
882 &client,
883 &owner,
884 OrderSide::Bid,
885 AssetPair::default(),
886 Token::B,
887 3,
888 3,
889 3000,
890 );
891
892 let instruction =
893 exchange_instruction::swap_request(&owner.pubkey(), &to_trade, &from_trade, &profit);
894 client
895 .send_and_confirm_instruction(&owner, instruction)
896 .unwrap_or_else(|_| panic!("{}:{}", line!(), file!()));
897
898 let new = create_token_account(&client, &owner);
899
900 let instruction =
901 exchange_instruction::transfer_request(&owner.pubkey(), &new, &to_trade, Token::B, 1);
902 client
903 .send_and_confirm_instruction(&owner, instruction)
904 .unwrap_or_else(|_| panic!("{}:{}", line!(), file!()));
905
906 let instruction =
907 exchange_instruction::transfer_request(&owner.pubkey(), &new, &from_trade, Token::A, 1);
908 client
909 .send_and_confirm_instruction(&owner, instruction)
910 .unwrap_or_else(|_| panic!("{}:{}", line!(), file!()));
911
912 let new_account_data = client.get_account_data(&new).unwrap().unwrap();
913
914 assert_eq!(
917 TokenAccountInfo::default()
918 .owner(&owner.pubkey())
919 .tokens(100_001, 100_001, 100_000, 100_000),
920 ExchangeProcessor::deserialize_account(&new_account_data).unwrap()
921 );
922 }
923}