-
Notifications
You must be signed in to change notification settings - Fork 17
/
Copy pathimprove.rs
157 lines (140 loc) · 5.5 KB
/
improve.rs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
use crate::{composite::*, error::MatchingEngineError, state::Auction, utils};
use anchor_lang::prelude::*;
use anchor_spl::token;
use common::TRANSFER_AUTHORITY_SEED_PREFIX;
#[derive(Accounts)]
#[instruction(offer_price: u64)]
#[event_cpi]
pub struct ImproveOffer<'info> {
/// The auction participant needs to set approval to this PDA.
///
/// CHECK: Seeds must be \["transfer-authority", auction.key(), offer_price.to_be_bytes()\].
#[account(
seeds = [
TRANSFER_AUTHORITY_SEED_PREFIX,
active_auction.key().as_ref(),
&offer_price.to_be_bytes()
],
bump
)]
transfer_authority: UncheckedAccount<'info>,
#[account(
constraint = {
// This is safe because we know that this is an active auction.
let info = active_auction.info.as_ref().unwrap();
require!(
info.within_auction_duration(&active_auction.config),
MatchingEngineError::AuctionPeriodExpired
);
require!(
offer_price
< utils::auction::compute_min_allowed_offer(&active_auction.config, info),
MatchingEngineError::CarpingNotAllowed
);
true
}
)]
active_auction: ActiveAuction<'info>,
#[account(
constraint = {
offer_token.key() != active_auction.custody_token.key()
} @ MatchingEngineError::InvalidOfferToken,
)]
offer_token: Account<'info, token::TokenAccount>,
token_program: Program<'info, token::Token>,
}
pub fn improve_offer(ctx: Context<ImproveOffer>, offer_price: u64) -> Result<()> {
let offer_token = &ctx.accounts.offer_token;
{
let ActiveAuction {
auction,
custody_token,
best_offer_token,
config: _,
} = &ctx.accounts.active_auction;
let token_program = &ctx.accounts.token_program;
// Transfer funds from the `offer_token` token account to the `best_offer_token` token account,
// but only if the pubkeys are different.
if offer_token.key() != best_offer_token.key() {
// These operations will seem silly, but we do this as a safety measure to ensure that
// nothing terrible happened with the auction's custody account.
let total_deposit = ctx
.accounts
.active_auction
.info
.as_ref()
.unwrap()
.total_deposit();
// If the best offer token happens to be closed, we will just keep the funds in the
// auction custody account. The executor token account will collect these funds when the
// order is executed.
if utils::checked_deserialize_token_account(best_offer_token, &common::USDC_MINT)
.is_some()
{
token::transfer(
CpiContext::new_with_signer(
token_program.to_account_info(),
anchor_spl::token::Transfer {
from: custody_token.to_account_info(),
to: best_offer_token.to_account_info(),
authority: auction.to_account_info(),
},
&[&[
Auction::SEED_PREFIX,
auction.vaa_hash.as_ref(),
&[auction.bump],
]],
),
total_deposit,
)?;
}
token::transfer(
CpiContext::new_with_signer(
token_program.to_account_info(),
anchor_spl::token::Transfer {
from: offer_token.to_account_info(),
to: custody_token.to_account_info(),
authority: ctx.accounts.transfer_authority.to_account_info(),
},
&[&[
TRANSFER_AUTHORITY_SEED_PREFIX,
auction.key().as_ref(),
&offer_price.to_be_bytes(),
&[ctx.bumps.transfer_authority],
]],
),
total_deposit,
)?;
}
}
// Update info before we emit event.
{
let info = ctx.accounts.active_auction.info.as_mut().unwrap();
info.best_offer_token = offer_token.key();
info.offer_price = offer_price;
}
// Emit the auction updated event.
{
let auction = &ctx.accounts.active_auction;
let config = &auction.config;
let info = auction.info.as_ref().unwrap();
// Emit event for auction participants to listen to.
emit_cpi!(crate::utils::log_emit(crate::events::AuctionUpdated {
config_id: info.config_id,
fast_vaa_hash: auction.vaa_hash,
vaa: Default::default(),
source_chain: info.source_chain,
target_protocol: auction.target_protocol,
redeemer_message_len: info.redeemer_message_len,
end_slot: info.auction_end_slot(config),
best_offer_token: offer_token.key(),
token_balance_before: offer_token.amount,
amount_in: info.amount_in,
total_deposit: info.total_deposit(),
max_offer_price_allowed: utils::auction::compute_min_allowed_offer(config, info)
.checked_sub(1),
}));
}
// Done.
Ok(())
}