Skip to content

Commit 36c430c

Browse files
authored
solana: harden missing token account checks (#188)
Co-authored-by: A5 Pickle <a5-pickle@users.noreply.github.com>
1 parent 35b854f commit 36c430c

File tree

6 files changed

+56
-20
lines changed

6 files changed

+56
-20
lines changed

solana/programs/matching-engine/src/composite/mod.rs

+6-3
Original file line numberDiff line numberDiff line change
@@ -384,9 +384,12 @@ pub struct ExecuteOrder<'info> {
384384
)]
385385
pub active_auction: ActiveAuction<'info>,
386386

387-
/// CHECK: Must be a token account, whose mint is [common::USDC_MINT].
388-
#[account(mut)]
389-
pub executor_token: UncheckedAccount<'info>,
387+
/// Must be a token account, whose mint is [common::USDC_MINT].
388+
#[account(
389+
mut,
390+
token::mint = common::USDC_MINT,
391+
)]
392+
pub executor_token: Box<Account<'info, token::TokenAccount>>,
390393

391394
/// CHECK: Mutable. Must equal [initial_offer](Auction::initial_offer).
392395
#[account(

solana/programs/matching-engine/src/processor/auction/execute_fast_order/mod.rs

+19-16
Original file line numberDiff line numberDiff line change
@@ -111,20 +111,21 @@ fn prepare_order_execution<'info>(
111111

112112
// If the initial offer token account doesn't exist anymore, we have nowhere to send the
113113
// init auction fee. The executor will get these funds instead.
114-
if !initial_offer_token.data_is_empty() {
115-
// Deserialize to token account to find owner. We know this is a legitimate token
116-
// account, so it is safe to borrow and unwrap here.
117-
{
118-
let mut acc_data: &[_] = &initial_offer_token.data.borrow();
119-
let token_data = token::TokenAccount::try_deserialize(&mut acc_data).unwrap();
120-
require_keys_eq!(
121-
token_data.owner,
122-
initial_participant.key(),
123-
ErrorCode::ConstraintTokenOwner
124-
);
114+
//
115+
// Deserialize to token account to find owner. We check that this is a legitimate token
116+
// account.
117+
if let Some(token_data) =
118+
utils::checked_deserialize_token_account(initial_offer_token, &custody_token.mint)
119+
{
120+
// Before setting the beneficiary to the initial participant, we need to make sure that
121+
// he is the owner of this token account.
122+
require_keys_eq!(
123+
token_data.owner,
124+
initial_participant.key(),
125+
ErrorCode::ConstraintTokenOwner
126+
);
125127

126-
beneficiary.replace(initial_participant.to_account_info());
127-
}
128+
beneficiary.replace(initial_participant.to_account_info());
128129

129130
if best_offer_token.key() != initial_offer_token.key() {
130131
// Pay the auction initiator their fee.
@@ -155,8 +156,8 @@ fn prepare_order_execution<'info>(
155156
// Return the security deposit and the fee to the highest bidder.
156157
//
157158
if best_offer_token.key() == executor_token.key() {
158-
// If the best offer token is equal to the executor token, just send whatever remains in the
159-
// custody token account.
159+
// If the best offer token is equal to the executor token, just send whatever remains in
160+
// the custody token account.
160161
//
161162
// NOTE: This will revert if the best offer token does not exist. But this will present
162163
// an opportunity for another executor to execute this order and take what the best
@@ -177,7 +178,9 @@ fn prepare_order_execution<'info>(
177178
// Otherwise, send the deposit and fee to the best offer token. If the best offer token
178179
// doesn't exist at this point (which would be unusual), we will reserve these funds
179180
// for the executor token.
180-
if !best_offer_token.data_is_empty() {
181+
if utils::checked_deserialize_token_account(best_offer_token, &custody_token.mint)
182+
.is_some()
183+
{
181184
token::transfer(
182185
CpiContext::new_with_signer(
183186
token_program.to_account_info(),

solana/programs/matching-engine/src/processor/auction/offer/improve.rs

+3-1
Original file line numberDiff line numberDiff line change
@@ -79,7 +79,9 @@ pub fn improve_offer(ctx: Context<ImproveOffer>, offer_price: u64) -> Result<()>
7979
// If the best offer token happens to be closed, we will just keep the funds in the
8080
// auction custody account. The executor token account will collect these funds when the
8181
// order is executed.
82-
if !best_offer_token.data_is_empty() {
82+
if utils::checked_deserialize_token_account(best_offer_token, &custody_token.mint)
83+
.is_some()
84+
{
8385
token::transfer(
8486
CpiContext::new_with_signer(
8587
token_program.to_account_info(),

solana/programs/matching-engine/src/utils/mod.rs

+16
Original file line numberDiff line numberDiff line change
@@ -4,6 +4,7 @@ pub mod auction;
44

55
use crate::{error::MatchingEngineError, state::RouterEndpoint};
66
use anchor_lang::prelude::*;
7+
use anchor_spl::token;
78
use common::wormhole_cctp_solana::wormhole::{VaaAccount, SOLANA_CHAIN};
89

910
pub trait VaaDigest {
@@ -40,3 +41,18 @@ pub fn require_local_endpoint(endpoint: &RouterEndpoint) -> Result<bool> {
4041

4142
Ok(true)
4243
}
44+
45+
pub fn checked_deserialize_token_account(
46+
acc_info: &AccountInfo,
47+
expected_mint: &Pubkey,
48+
) -> Option<token::TokenAccount> {
49+
if acc_info.owner != &token::ID {
50+
None
51+
} else {
52+
let data = acc_info.try_borrow_data().ok()?;
53+
54+
token::TokenAccount::try_deserialize(&mut &data[..])
55+
.ok()
56+
.filter(|token_data| &token_data.mint == expected_mint && !token_data.is_frozen())
57+
}
58+
}

solana/ts/src/idl/json/matching_engine.json

+6
Original file line numberDiff line numberDiff line change
@@ -685,6 +685,9 @@
685685
},
686686
{
687687
"name": "executor_token",
688+
"docs": [
689+
"Must be a token account, whose mint is [common::USDC_MINT]."
690+
],
688691
"writable": true
689692
},
690693
{
@@ -877,6 +880,9 @@
877880
},
878881
{
879882
"name": "executor_token",
883+
"docs": [
884+
"Must be a token account, whose mint is [common::USDC_MINT]."
885+
],
880886
"writable": true
881887
},
882888
{

solana/ts/src/idl/ts/matching_engine.ts

+6
Original file line numberDiff line numberDiff line change
@@ -691,6 +691,9 @@ export type MatchingEngine = {
691691
},
692692
{
693693
"name": "executorToken",
694+
"docs": [
695+
"Must be a token account, whose mint is [common::USDC_MINT]."
696+
],
694697
"writable": true
695698
},
696699
{
@@ -883,6 +886,9 @@ export type MatchingEngine = {
883886
},
884887
{
885888
"name": "executorToken",
889+
"docs": [
890+
"Must be a token account, whose mint is [common::USDC_MINT]."
891+
],
886892
"writable": true
887893
},
888894
{

0 commit comments

Comments
 (0)