Skip to content

Commit 799cd04

Browse files
authored
solana: fix lamport beneficiaries (#109)
Co-authored-by: A5 Pickle <a5-pickle@users.noreply.github.com>
1 parent 7504b53 commit 799cd04

File tree

12 files changed

+100
-26
lines changed

12 files changed

+100
-26
lines changed

solana/.gitignore

+1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,6 @@
11
.anchor
22
.env
3+
.validator_pid
34
**/*.rs.bk
45
/artifacts-*
56
/cfg/**/*.json

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

+5
Original file line numberDiff line numberDiff line change
@@ -380,6 +380,11 @@ pub struct ExecuteOrder<'info> {
380380
address = active_auction.info.as_ref().unwrap().initial_offer_token,
381381
)]
382382
pub initial_offer_token: UncheckedAccount<'info>,
383+
384+
/// CHECK: Must be the owner of initial offer token account. If the initial offer token account
385+
/// does not exist anymore, we will attempt to perform this check.
386+
#[account(mut)]
387+
pub initial_participant: UncheckedAccount<'info>,
383388
}
384389

385390
#[derive(Accounts)]

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

+2-1
Original file line numberDiff line numberDiff line change
@@ -79,6 +79,7 @@ pub fn handle_execute_fast_order_cctp(
7979
let super::PreparedOrderExecution {
8080
user_amount: amount,
8181
fill,
82+
beneficiary,
8283
} = super::prepare_order_execution(super::PrepareFastExecution {
8384
execute_order: &mut ctx.accounts.execute_order,
8485
custodian: &ctx.accounts.custodian,
@@ -187,7 +188,7 @@ pub fn handle_execute_fast_order_cctp(
187188
token_program.to_account_info(),
188189
token::CloseAccount {
189190
account: auction_custody_token.to_account_info(),
190-
destination: payer.to_account_info(),
191+
destination: beneficiary.unwrap_or(payer.to_account_info()),
191192
authority: custodian.to_account_info(),
192193
},
193194
&[Custodian::SIGNER_SEEDS],

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

+2-1
Original file line numberDiff line numberDiff line change
@@ -60,6 +60,7 @@ pub fn execute_fast_order_local(ctx: Context<ExecuteFastOrderLocal>) -> Result<(
6060
let super::PreparedOrderExecution {
6161
user_amount: amount,
6262
fill,
63+
beneficiary,
6364
} = super::prepare_order_execution(super::PrepareFastExecution {
6465
execute_order: &mut ctx.accounts.execute_order,
6566
custodian,
@@ -108,7 +109,7 @@ pub fn execute_fast_order_local(ctx: Context<ExecuteFastOrderLocal>) -> Result<(
108109
token_program.to_account_info(),
109110
token::CloseAccount {
110111
account: auction_custody_token.to_account_info(),
111-
destination: payer.to_account_info(),
112+
destination: beneficiary.unwrap_or(payer.to_account_info()),
112113
authority: custodian.to_account_info(),
113114
},
114115
&[Custodian::SIGNER_SEEDS],

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

+32-16
Original file line numberDiff line numberDiff line change
@@ -23,38 +23,36 @@ struct PrepareFastExecution<'ctx, 'info> {
2323
token_program: &'ctx Program<'info, token::Token>,
2424
}
2525

26-
struct PreparedOrderExecution {
26+
struct PreparedOrderExecution<'info> {
2727
pub user_amount: u64,
2828
pub fill: Fill,
29+
pub beneficiary: Option<AccountInfo<'info>>,
2930
}
3031

31-
fn prepare_order_execution(accounts: PrepareFastExecution) -> Result<PreparedOrderExecution> {
32+
fn prepare_order_execution<'info>(
33+
accounts: PrepareFastExecution<'_, 'info>,
34+
) -> Result<PreparedOrderExecution<'info>> {
3235
let PrepareFastExecution {
3336
execute_order,
3437
custodian,
3538
token_program,
3639
} = accounts;
3740

38-
let ExecuteOrder {
39-
fast_vaa,
40-
active_auction,
41-
executor_token,
42-
initial_offer_token,
43-
} = execute_order;
44-
45-
let ActiveAuction {
46-
auction,
47-
custody_token,
48-
config,
49-
best_offer_token,
50-
} = active_auction;
41+
let auction = &mut execute_order.active_auction.auction;
42+
let fast_vaa = &execute_order.fast_vaa;
43+
let custody_token = &execute_order.active_auction.custody_token;
44+
let config = &execute_order.active_auction.config;
45+
let executor_token = &execute_order.executor_token;
46+
let best_offer_token = &execute_order.active_auction.best_offer_token;
47+
let initial_offer_token = &execute_order.initial_offer_token;
48+
let initial_participant = &execute_order.initial_participant;
5149

5250
let vaa = fast_vaa.load_unchecked();
5351
let order = LiquidityLayerMessage::try_from(vaa.payload())
5452
.unwrap()
5553
.to_fast_market_order_unchecked();
5654

57-
let (user_amount, new_status) = {
55+
let (user_amount, new_status, beneficiary) = {
5856
let auction_info = auction.info.as_ref().unwrap();
5957

6058
let current_slot = Clock::get().unwrap().slot;
@@ -98,9 +96,25 @@ fn prepare_order_execution(accounts: PrepareFastExecution) -> Result<PreparedOrd
9896
deposit_and_fee = deposit_and_fee.saturating_sub(penalty);
9997
}
10098

99+
let mut beneficiary = None;
100+
101101
// If the initial offer token account doesn't exist anymore, we have nowhere to send the
102102
// init auction fee. The executor will get these funds instead.
103103
if !initial_offer_token.data_is_empty() {
104+
// Deserialize to token account to find owner. We know this is a legitimate token
105+
// account, so it is safe to borrow and unwrap here.
106+
{
107+
let mut acc_data: &[_] = &initial_offer_token.data.borrow();
108+
let token_data = token::TokenAccount::try_deserialize(&mut acc_data).unwrap();
109+
require_keys_eq!(
110+
token_data.owner,
111+
initial_participant.key(),
112+
ErrorCode::ConstraintTokenOwner
113+
);
114+
115+
beneficiary.replace(initial_participant.to_account_info());
116+
}
117+
104118
if best_offer_token.key() != initial_offer_token.key() {
105119
// Pay the auction initiator their fee.
106120
token::transfer(
@@ -218,6 +232,7 @@ fn prepare_order_execution(accounts: PrepareFastExecution) -> Result<PreparedOrd
218232
slot: current_slot,
219233
execute_penalty: if penalized { Some(penalty) } else { None },
220234
},
235+
beneficiary,
221236
)
222237
};
223238

@@ -235,5 +250,6 @@ fn prepare_order_execution(accounts: PrepareFastExecution) -> Result<PreparedOrd
235250
.try_into()
236251
.map_err(|_| MatchingEngineError::RedeemerMessageTooLarge)?,
237252
},
253+
beneficiary,
238254
})
239255
}

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

+2-1
Original file line numberDiff line numberDiff line change
@@ -84,6 +84,7 @@ fn handle_settle_auction_none_cctp(
8484
ctx: Context<SettleAuctionNoneCctp>,
8585
destination_cctp_domain: u32,
8686
) -> Result<()> {
87+
let prepared_by = &ctx.accounts.prepared.by;
8788
let prepared_custody_token = &ctx.accounts.prepared.custody_token;
8889
let custodian = &ctx.accounts.custodian;
8990
let token_program = &ctx.accounts.token_program;
@@ -206,7 +207,7 @@ fn handle_settle_auction_none_cctp(
206207
token_program.to_account_info(),
207208
token::CloseAccount {
208209
account: prepared_custody_token.to_account_info(),
209-
destination: payer.to_account_info(),
210+
destination: prepared_by.to_account_info(),
210211
authority: custodian.to_account_info(),
211212
},
212213
&[Custodian::SIGNER_SEEDS],

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

+2-1
Original file line numberDiff line numberDiff line change
@@ -70,6 +70,7 @@ pub struct SettleAuctionNoneLocal<'info> {
7070
}
7171

7272
pub fn settle_auction_none_local(ctx: Context<SettleAuctionNoneLocal>) -> Result<()> {
73+
let prepared_by = &ctx.accounts.prepared.by;
7374
let prepared_custody_token = &ctx.accounts.prepared.custody_token;
7475
let custodian = &ctx.accounts.custodian;
7576
let token_program = &ctx.accounts.token_program;
@@ -124,7 +125,7 @@ pub fn settle_auction_none_local(ctx: Context<SettleAuctionNoneLocal>) -> Result
124125
token_program.to_account_info(),
125126
token::CloseAccount {
126127
account: prepared_custody_token.to_account_info(),
127-
destination: payer.to_account_info(),
128+
destination: prepared_by.to_account_info(),
128129
authority: custodian.to_account_info(),
129130
},
130131
&[Custodian::SIGNER_SEEDS],

solana/programs/token-router/src/processor/market_order/place_cctp.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -290,7 +290,7 @@ fn handle_place_market_order_cctp(
290290
token_program.to_account_info(),
291291
token::CloseAccount {
292292
account: prepared_custody_token.to_account_info(),
293-
destination: payer.to_account_info(),
293+
destination: ctx.accounts.prepared_by.to_account_info(),
294294
authority: custodian.to_account_info(),
295295
},
296296
&[Custodian::SIGNER_SEEDS],

solana/target/idl/matching_engine.json

+14
Original file line numberDiff line numberDiff line change
@@ -643,6 +643,13 @@
643643
{
644644
"name": "initial_offer_token",
645645
"writable": true
646+
},
647+
{
648+
"name": "initial_participant",
649+
"docs": [
650+
"does not exist anymore, we will attempt to perform this check."
651+
],
652+
"writable": true
646653
}
647654
]
648655
},
@@ -831,6 +838,13 @@
831838
{
832839
"name": "initial_offer_token",
833840
"writable": true
841+
},
842+
{
843+
"name": "initial_participant",
844+
"docs": [
845+
"does not exist anymore, we will attempt to perform this check."
846+
],
847+
"writable": true
834848
}
835849
]
836850
},

solana/target/types/matching_engine.ts

+14
Original file line numberDiff line numberDiff line change
@@ -649,6 +649,13 @@ export type MatchingEngine = {
649649
{
650650
"name": "initialOfferToken",
651651
"writable": true
652+
},
653+
{
654+
"name": "initialParticipant",
655+
"docs": [
656+
"does not exist anymore, we will attempt to perform this check."
657+
],
658+
"writable": true
652659
}
653660
]
654661
},
@@ -837,6 +844,13 @@ export type MatchingEngine = {
837844
{
838845
"name": "initialOfferToken",
839846
"writable": true
847+
},
848+
{
849+
"name": "initialParticipant",
850+
"docs": [
851+
"does not exist anymore, we will attempt to perform this check."
852+
],
853+
"writable": true
840854
}
841855
]
842856
},

solana/ts/src/matchingEngine/index.ts

+24-5
Original file line numberDiff line numberDiff line change
@@ -1704,26 +1704,29 @@ export class MatchingEngineProgram {
17041704
auctionConfig?: PublicKey;
17051705
bestOfferToken?: PublicKey;
17061706
initialOfferToken?: PublicKey;
1707+
initialParticipant?: PublicKey;
17071708
},
17081709
opts: {
17091710
targetChain?: wormholeSdk.ChainId;
17101711
} = {},
17111712
) {
1713+
const connection = this.program.provider.connection;
1714+
17121715
const { payer, fastVaa, auctionConfig, bestOfferToken } = accounts;
17131716

1714-
let { auction, executorToken, initialOfferToken } = accounts;
1717+
let { auction, executorToken, initialOfferToken, initialParticipant } = accounts;
17151718
let { targetChain } = opts;
17161719

17171720
executorToken ??= splToken.getAssociatedTokenAddressSync(this.mint, payer);
17181721

17191722
let fastVaaAccount: VaaAccount | undefined;
17201723
if (auction === undefined) {
1721-
fastVaaAccount = await VaaAccount.fetch(this.program.provider.connection, fastVaa);
1724+
fastVaaAccount = await VaaAccount.fetch(connection, fastVaa);
17221725
auction = this.auctionAddress(fastVaaAccount.digest());
17231726
}
17241727

17251728
if (targetChain === undefined) {
1726-
fastVaaAccount ??= await VaaAccount.fetch(this.program.provider.connection, fastVaa);
1729+
fastVaaAccount ??= await VaaAccount.fetch(connection, fastVaa);
17271730

17281731
const { fastMarketOrder } = LiquidityLayerMessage.decode(fastVaaAccount.payload());
17291732
if (fastMarketOrder === undefined) {
@@ -1742,6 +1745,11 @@ export class MatchingEngineProgram {
17421745
initialOfferToken = info.initialOfferToken;
17431746
}
17441747

1748+
if (initialParticipant === undefined) {
1749+
const token = await splToken.getAccount(connection, initialOfferToken);
1750+
initialParticipant = token.owner;
1751+
}
1752+
17451753
const {
17461754
custodian,
17471755
routerEndpoint: toRouterEndpoint,
@@ -1781,6 +1789,7 @@ export class MatchingEngineProgram {
17811789
),
17821790
executorToken,
17831791
initialOfferToken,
1792+
initialParticipant,
17841793
},
17851794
toRouterEndpoint: this.routerEndpointComposite(toRouterEndpoint),
17861795
custodian: this.checkedCustodianComposite(custodian),
@@ -1818,21 +1827,25 @@ export class MatchingEngineProgram {
18181827
auctionConfig?: PublicKey;
18191828
bestOfferToken?: PublicKey;
18201829
initialOfferToken?: PublicKey;
1830+
initialParticipant?: PublicKey;
18211831
toRouterEndpoint?: PublicKey;
18221832
},
18231833
opts: {
18241834
sourceChain?: wormholeSdk.ChainId;
18251835
} = {},
18261836
) {
1837+
const connection = this.program.provider.connection;
1838+
18271839
const { payer, fastVaa, auctionConfig, bestOfferToken } = accounts;
18281840

1829-
let { auction, executorToken, toRouterEndpoint, initialOfferToken } = accounts;
1841+
let { auction, executorToken, toRouterEndpoint, initialOfferToken, initialParticipant } =
1842+
accounts;
18301843
let { sourceChain } = opts;
18311844
executorToken ??= splToken.getAssociatedTokenAddressSync(this.mint, payer);
18321845
toRouterEndpoint ??= this.routerEndpointAddress(wormholeSdk.CHAIN_ID_SOLANA);
18331846

18341847
if (auction === undefined) {
1835-
const vaaAccount = await VaaAccount.fetch(this.program.provider.connection, fastVaa);
1848+
const vaaAccount = await VaaAccount.fetch(connection, fastVaa);
18361849
auction = this.auctionAddress(vaaAccount.digest());
18371850
}
18381851

@@ -1852,6 +1865,11 @@ export class MatchingEngineProgram {
18521865
initialOfferToken ??= auctionInfo.initialOfferToken;
18531866
}
18541867

1868+
if (initialParticipant === undefined) {
1869+
const token = await splToken.getAccount(connection, initialOfferToken);
1870+
initialParticipant = token.owner;
1871+
}
1872+
18551873
const {
18561874
custodian,
18571875
coreMessage,
@@ -1879,6 +1897,7 @@ export class MatchingEngineProgram {
18791897
),
18801898
executorToken,
18811899
initialOfferToken,
1900+
initialParticipant,
18821901
},
18831902
toRouterEndpoint: this.routerEndpointComposite(toRouterEndpoint),
18841903
wormhole: {

solana/ts/tests/01__matchingEngine.ts

+1
Original file line numberDiff line numberDiff line change
@@ -2544,6 +2544,7 @@ describe("Matching Engine", function () {
25442544
const ix = await engine.executeFastOrderCctpIx({
25452545
payer: liquidator.publicKey,
25462546
fastVaa,
2547+
initialParticipant: payer.publicKey,
25472548
});
25482549

25492550
const computeIx = ComputeBudgetProgram.setComputeUnitLimit({

0 commit comments

Comments
 (0)