Skip to content

Commit b00b375

Browse files
authored
solana: fix ATA constraint for executor (#168)
Co-authored-by: A5 Pickle <a5-pickle@users.noreply.github.com>
1 parent 8cfe602 commit b00b375

File tree

7 files changed

+78
-14
lines changed

7 files changed

+78
-14
lines changed
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,4 @@
1+
# Matching Engine Program
2+
3+
A program to facilitate the transfer of USDC between networks that allow Wormhole and CCTP bridging.
4+
With the help of solvers, allowing USDC to be transferred faster than finality.

solana/programs/matching-engine/src/processor/auction/settle/complete.rs

+21-11
Original file line numberDiff line numberDiff line change
@@ -3,7 +3,7 @@ use crate::{
33
state::{Auction, AuctionStatus, PreparedOrderResponse},
44
};
55
use anchor_lang::prelude::*;
6-
use anchor_spl::token;
6+
use anchor_spl::{associated_token::get_associated_token_address, token};
77

88
#[derive(Accounts)]
99
pub struct SettleAuctionComplete<'info> {
@@ -16,8 +16,8 @@ pub struct SettleAuctionComplete<'info> {
1616

1717
#[account(
1818
mut,
19-
associated_token::mint = best_offer_token.mint,
20-
associated_token::authority = executor,
19+
token::mint = best_offer_token.mint,
20+
token::authority = executor,
2121
)]
2222
executor_token: Account<'info, token::TokenAccount>,
2323

@@ -89,17 +89,18 @@ fn handle_settle_auction_complete(
8989
total_penalty: execute_penalty.map(|v| v.saturating_add(base_fee)),
9090
};
9191

92-
let executor_token = &ctx.accounts.executor_token;
93-
let best_offer_token = &ctx.accounts.best_offer_token;
94-
let prepared_custody_token = &ctx.accounts.prepared_custody_token;
95-
let token_program = &ctx.accounts.token_program;
96-
9792
let prepared_order_response_signer_seeds = &[
9893
PreparedOrderResponse::SEED_PREFIX,
9994
prepared_order_response.seeds.fast_vaa_hash.as_ref(),
10095
&[prepared_order_response.seeds.bump],
10196
];
10297

98+
let executor = &ctx.accounts.executor;
99+
let executor_token = &ctx.accounts.executor_token;
100+
let best_offer_token = &ctx.accounts.best_offer_token;
101+
let token_program = &ctx.accounts.token_program;
102+
let prepared_custody_token = &ctx.accounts.prepared_custody_token;
103+
103104
// We may deduct from this account if the winning participant was penalized.
104105
let mut repayment = ctx.accounts.prepared_custody_token.amount;
105106

@@ -116,7 +117,7 @@ fn handle_settle_auction_complete(
116117
executor_token.key(),
117118
best_offer_token.key(),
118119
MatchingEngineError::ExecutorTokenMismatch
119-
)
120+
);
120121
}
121122
_ => {
122123
// If there is a penalty, we want to return the lamports back to the person who paid to
@@ -125,7 +126,7 @@ fn handle_settle_auction_complete(
125126
// The executor's intention here would be to collect the base fee to cover the cost to
126127
// post the finalized VAA.
127128
require_keys_eq!(
128-
executor_token.owner,
129+
executor.key(),
129130
prepared_order_response.prepared_by,
130131
MatchingEngineError::ExecutorNotPreparedBy
131132
);
@@ -134,6 +135,15 @@ fn handle_settle_auction_complete(
134135
// Because the auction participant was penalized for executing the order late, he
135136
// will be deducted the base fee. This base fee will be sent to the executor token
136137
// account if it is not the same as the best offer token account.
138+
139+
// We require that the executor token account be an ATA.
140+
require_keys_eq!(
141+
executor_token.key(),
142+
get_associated_token_address(&executor_token.owner, &executor_token.mint),
143+
ErrorCode::AccountNotAssociatedTokenAccount
144+
);
145+
146+
// Transfer base fee to the executor.
137147
token::transfer(
138148
CpiContext::new_with_signer(
139149
token_program.to_account_info(),
@@ -178,7 +188,7 @@ fn handle_settle_auction_complete(
178188
token_program.to_account_info(),
179189
token::CloseAccount {
180190
account: prepared_custody_token.to_account_info(),
181-
destination: ctx.accounts.executor.to_account_info(),
191+
destination: executor.to_account_info(),
182192
authority: prepared_order_response.to_account_info(),
183193
},
184194
&[prepared_order_response_signer_seeds],

solana/programs/matching-engine/src/state/auction_history.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -88,7 +88,7 @@ impl AccountDeserialize for AuctionHistoryInternal {
8888
}
8989

9090
fn try_deserialize_unchecked(buf: &mut &[u8]) -> Result<Self> {
91-
*buf = &mut &buf[8..];
91+
*buf = &buf[8..];
9292
Ok(Self {
9393
header: AnchorDeserialize::deserialize(buf)?,
9494
num_entries: AnchorDeserialize::deserialize(buf)?,
+4
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,4 @@
1+
# Token Router Program
2+
3+
A program which is used to send and receive USDC from other networks via CCTP accompanied by a
4+
Wormhole message.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
# Upgrade Manager Program
2+
3+
This program is used to perform upgrades for the Matching Engine and Token Router programs.
4+
5+
Only the owner of these programs can perform these upgrades.

solana/ts/src/matchingEngine/index.ts

+4-2
Original file line numberDiff line numberDiff line change
@@ -1594,10 +1594,12 @@ export class MatchingEngineProgram {
15941594
preparedOrderResponse: PublicKey;
15951595
auction?: PublicKey;
15961596
bestOfferToken?: PublicKey;
1597+
executorToken?: PublicKey;
15971598
}) {
15981599
const { executor, preparedOrderResponse } = accounts;
15991600

1600-
let { auction, bestOfferToken } = accounts;
1601+
let { auction, bestOfferToken, executorToken } = accounts;
1602+
executorToken ??= splToken.getAssociatedTokenAddressSync(this.mint, executor);
16011603

16021604
if (auction === undefined) {
16031605
const { seeds } = await this.fetchPreparedOrderResponse({
@@ -1620,7 +1622,7 @@ export class MatchingEngineProgram {
16201622
.settleAuctionComplete()
16211623
.accounts({
16221624
executor,
1623-
executorToken: splToken.getAssociatedTokenAddressSync(this.mint, executor),
1625+
executorToken,
16241626
preparedOrderResponse,
16251627
preparedCustodyToken: this.preparedCustodyTokenAddress(preparedOrderResponse),
16261628
auction,

solana/ts/tests/01__matchingEngine.ts

+39
Original file line numberDiff line numberDiff line change
@@ -3670,6 +3670,44 @@ describe("Matching Engine", function () {
36703670
);
36713671
});
36723672

3673+
it("Cannot Settle Completed with Penalty (Executor is not ATA)", async function () {
3674+
const executorTokenSigner = Keypair.generate();
3675+
const executorToken = executorTokenSigner.publicKey;
3676+
3677+
await expectIxOk(
3678+
connection,
3679+
[
3680+
SystemProgram.createAccount({
3681+
fromPubkey: payer.publicKey,
3682+
newAccountPubkey: executorToken,
3683+
lamports: await connection.getMinimumBalanceForRentExemption(
3684+
splToken.ACCOUNT_SIZE,
3685+
),
3686+
space: splToken.ACCOUNT_SIZE,
3687+
programId: splToken.TOKEN_PROGRAM_ID,
3688+
}),
3689+
splToken.createInitializeAccount3Instruction(
3690+
executorToken,
3691+
engine.mint,
3692+
playerTwo.publicKey,
3693+
),
3694+
],
3695+
[payer, executorTokenSigner],
3696+
);
3697+
3698+
await settleAuctionCompleteForTest(
3699+
{
3700+
executor: playerTwo.publicKey,
3701+
executorToken,
3702+
},
3703+
{
3704+
prepareSigners: [playerTwo],
3705+
executeWithinGracePeriod: false,
3706+
errorMsg: "Error Code: AccountNotAssociatedTokenAccount",
3707+
},
3708+
);
3709+
});
3710+
36733711
it("Settle Completed with Penalty (Executor != Best Offer)", async function () {
36743712
await settleAuctionCompleteForTest(
36753713
{
@@ -4744,6 +4782,7 @@ describe("Matching Engine", function () {
47444782
async function settleAuctionCompleteForTest(
47454783
accounts: {
47464784
executor?: PublicKey;
4785+
executorToken?: PublicKey;
47474786
preparedOrderResponse?: PublicKey;
47484787
auction?: PublicKey;
47494788
bestOfferToken?: PublicKey;

0 commit comments

Comments
 (0)