Skip to content

Commit d445c13

Browse files
committed
solana: settle complete executor must always be prepared_by
1 parent 40b5170 commit d445c13

File tree

2 files changed

+18
-48
lines changed

2 files changed

+18
-48
lines changed

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

+17-34
Original file line numberDiff line numberDiff line change
@@ -11,19 +11,19 @@ use anchor_spl::{
1111

1212
#[derive(Accounts)]
1313
pub struct SettleAuctionComplete<'info> {
14-
/// CHECK: To prevent squatters from preparing order responses on behalf of the auction winner,
15-
/// we will always reward the owner of the executor token account with the lamports from the
16-
/// prepared order response and its custody token account when we close these accounts. This
17-
/// means we disregard the `prepared_by` field in the prepared order response.
18-
#[account(mut)]
14+
/// CHECK: Must equal prepared_order_response.prepared_by, who paid the rent to post the
15+
/// finalized VAA.
16+
#[account(
17+
mut,
18+
address = prepared_order_response.prepared_by,
19+
)]
1920
executor: UncheckedAccount<'info>,
2021

2122
#[account(
2223
mut,
2324
token::mint = common::USDC_MINT,
24-
token::authority = executor,
2525
)]
26-
executor_token: Account<'info, TokenAccount>,
26+
executor_token: Box<Account<'info, TokenAccount>>,
2727

2828
/// Destination token account, which the redeemer may not own. But because the redeemer is a
2929
/// signer and is the one encoded in the Deposit Fill message, he may have the tokens be sent
@@ -46,7 +46,7 @@ pub struct SettleAuctionComplete<'info> {
4646
],
4747
bump = prepared_order_response.seeds.bump,
4848
)]
49-
prepared_order_response: Account<'info, PreparedOrderResponse>,
49+
prepared_order_response: Box<Account<'info, PreparedOrderResponse>>,
5050

5151
/// CHECK: Seeds must be \["prepared-custody"\, prepared_order_response.key()].
5252
#[account(
@@ -57,7 +57,7 @@ pub struct SettleAuctionComplete<'info> {
5757
],
5858
bump,
5959
)]
60-
prepared_custody_token: Account<'info, TokenAccount>,
60+
prepared_custody_token: Box<Account<'info, TokenAccount>>,
6161

6262
#[account(
6363
mut,
@@ -67,7 +67,7 @@ pub struct SettleAuctionComplete<'info> {
6767
],
6868
bump = auction.bump,
6969
)]
70-
auction: Account<'info, Auction>,
70+
auction: Box<Account<'info, Auction>>,
7171

7272
token_program: Program<'info, token::Token>,
7373
}
@@ -115,19 +115,6 @@ fn handle_settle_auction_complete(
115115

116116
let (executor_result, best_offer_result) = match execute_penalty {
117117
None => {
118-
// If there is no penalty, we require that the executor token and best offer token be
119-
// equal. The winning offer should not be penalized for calling this instruction when he
120-
// has executed the order within the grace period.
121-
//
122-
// By requiring that these pubkeys are equal, we enforce that the owner of the best
123-
// offer token gets rewarded the lamports from the prepared order response and its
124-
// custody account.
125-
require_keys_eq!(
126-
executor_token.key(),
127-
best_offer_token.key(),
128-
MatchingEngineError::ExecutorTokenMismatch
129-
);
130-
131118
// If the token account happens to not exist anymore, we will revert.
132119
match TokenAccount::try_deserialize(&mut &best_offer_token.data.borrow()[..]) {
133120
Ok(best_offer) => (
@@ -142,17 +129,6 @@ fn handle_settle_auction_complete(
142129
}
143130
}
144131
_ => {
145-
// If there is a penalty, we want to return the lamports back to the person who paid to
146-
// create the prepared order response and custody token accounts.
147-
//
148-
// The executor's intention here would be to collect the base fee to cover the cost to
149-
// post the finalized VAA.
150-
require_keys_eq!(
151-
executor.key(),
152-
prepared_order_response.prepared_by,
153-
MatchingEngineError::ExecutorNotPreparedBy
154-
);
155-
156132
// If the token account happens to not exist anymore, we will give everything to the
157133
// executor.
158134
match TokenAccount::try_deserialize(&mut &best_offer_token.data.borrow()[..]) {
@@ -182,6 +158,13 @@ fn handle_settle_auction_complete(
182158
ErrorCode::AccountNotAssociatedTokenAccount
183159
);
184160

161+
// And enforce that the owner of this ATA is the executor.
162+
require_keys_eq!(
163+
executor.key(),
164+
executor_token.owner,
165+
ErrorCode::ConstraintTokenOwner,
166+
);
167+
185168
(
186169
TokenAccountResult {
187170
balance_before: executor_token.amount,

solana/ts/tests/01__matchingEngine.ts

+1-14
Original file line numberDiff line numberDiff line change
@@ -3606,19 +3606,6 @@ describe("Matching Engine", function () {
36063606
);
36073607
});
36083608

3609-
it("Cannot Settle Completed Auction with No Penalty (Executor != Best Offer)", async function () {
3610-
await settleAuctionCompleteForTest(
3611-
{
3612-
executor: payer.publicKey,
3613-
},
3614-
{
3615-
prepareSigners: [payer],
3616-
executeWithinGracePeriod: true,
3617-
errorMsg: "Error Code: ExecutorTokenMismatch",
3618-
},
3619-
);
3620-
});
3621-
36223609
it("Settle Completed without Penalty", async function () {
36233610
await settleAuctionCompleteForTest(
36243611
{
@@ -3652,7 +3639,7 @@ describe("Matching Engine", function () {
36523639
{
36533640
executeWithinGracePeriod: false,
36543641
executorIsPreparer: false,
3655-
errorMsg: "Error Code: ExecutorNotPreparedBy",
3642+
errorMsg: "executor. Error Code: ConstraintAddress",
36563643
},
36573644
);
36583645
});

0 commit comments

Comments
 (0)