Skip to content

Commit d427ea2

Browse files
authored
solana: add fill time (#146)
Co-authored-by: A5 Pickle <a5-pickle@users.noreply.github.com>
1 parent ff04006 commit d427ea2

File tree

16 files changed

+278
-55
lines changed

16 files changed

+278
-55
lines changed

.github/workflows/solana.yml

+3-3
Original file line numberDiff line numberDiff line change
@@ -44,7 +44,7 @@ jobs:
4444
run: make lint
4545
working-directory: ./solana
4646
make-anchor-build-idl:
47-
name: make anchor-build-idl
47+
name: make idl
4848
runs-on: ubuntu-latest
4949
steps:
5050
- uses: actions/checkout@v4
@@ -56,8 +56,8 @@ jobs:
5656
- name: Set default Rust toolchain
5757
run: rustup default stable
5858
working-directory: ./solana
59-
- name: make anchor-build-idl
60-
run: make anchor-build-idl
59+
- name: make idl
60+
run: make idl
6161
working-directory: ./solana
6262
make-anchor-test:
6363
name: make anchor-test

solana/Makefile

+6-6
Original file line numberDiff line numberDiff line change
@@ -25,7 +25,9 @@ clean:
2525
anchor clean
2626
rm -rf node_modules artifacts-mainnet artifacts-testnet artifacts-localnet ts/tests/artifacts
2727

28-
node_modules: ../package-lock.json
28+
node_modules: ../node_modules
29+
30+
../node_modules:
2931
cd .. && npm ci
3032

3133
.PHONY: cargo-test
@@ -53,16 +55,14 @@ anchor-test-setup: node_modules ts/tests/artifacts $(CLONED_MAINNET_PROGRAMS)
5355
cp target/deploy/upgrade_manager.so ts/tests/artifacts/testnet_upgrade_manager.so
5456
anchor build -- --features integration-test
5557

56-
.PHONY: anchor-build-idl
57-
anchor-build-idl:
58+
.PHONY: idl
59+
idl:
5860
anchor build -- --features localnet
59-
# git diff --exit-code
60-
61-
idl: anchor-build-idl
6261
mkdir -p ts/src/idl/json
6362
mkdir -p ts/src/idl/ts
6463
cp -r target/idl/* ts/src/idl/json/
6564
cp -r target/types/* ts/src/idl/ts/
65+
git diff --exit-code
6666

6767
$(BUILD_$(NETWORK)): cargo-test
6868

solana/programs/matching-engine/src/processor/fast_fill/close_redeemed.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -9,7 +9,7 @@ pub struct CloseRedeemedFastFill<'info> {
99
/// CHECK: Must equal the `prepared_by` field of the `fast_fill` account.
1010
#[account(
1111
mut,
12-
address = fast_fill.prepared_by,
12+
address = fast_fill.info.prepared_by,
1313
)]
1414
prepared_by: UncheckedAccount<'info>,
1515

solana/programs/matching-engine/src/processor/fast_fill/complete.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -82,7 +82,7 @@ pub fn complete_fast_fill(ctx: Context<CompleteFastFill>) -> Result<()> {
8282

8383
// Emit event that the fast fill is redeemed. Listeners can close this account.
8484
emit!(crate::events::FastFillRedeemed {
85-
prepared_by: ctx.accounts.fast_fill.prepared_by,
85+
prepared_by: ctx.accounts.fast_fill.info.prepared_by,
8686
fast_fill: ctx.accounts.fast_fill.key(),
8787
});
8888

solana/programs/matching-engine/src/state/fast_fill/mod.rs

+23-3
Original file line numberDiff line numberDiff line change
@@ -9,24 +9,44 @@ use common::messages::Fill;
99

1010
#[derive(Debug, AnchorSerialize, AnchorDeserialize, Clone, Copy, InitSpace)]
1111
pub struct FastFillInfo {
12+
/// Who paid the lamports to create the [FastFill] account.
13+
pub prepared_by: Pubkey,
14+
15+
/// Fill amount.
1216
pub amount: u64,
17+
18+
/// Authority allowed to redeem [FastFill].
1319
pub redeemer: Pubkey,
20+
21+
/// Timestamp at the time a fill was issued. When the fast fill is created, it is set using the
22+
/// current [Clock] unix timestamp.
23+
pub timestamp: i64,
1424
}
1525

1626
#[derive(Debug, AnchorSerialize, AnchorDeserialize, Clone, Copy, InitSpace)]
1727
pub struct FastFillSeeds {
28+
/// Wormhole chain ID reflecting where the order was created.
1829
pub source_chain: u16,
30+
31+
/// Universal address of the order sender.
1932
pub order_sender: [u8; 32],
33+
34+
/// Sequence generated by the [FastFillSequencer](crate::state::FastFillSequencer) when it
35+
/// reserved a sequence number for this fill.
2036
pub sequence: u64,
37+
38+
/// Bump seed for the [FastFill] account.
2139
pub bump: u8,
2240
}
2341

2442
#[account]
2543
#[derive(Debug)]
2644
pub struct FastFill {
2745
pub seeds: FastFillSeeds,
28-
pub prepared_by: Pubkey,
46+
47+
/// Whether the [FastFill] has been redeemed via the local Token Router.
2948
pub redeemed: bool,
49+
3050
pub info: FastFillInfo,
3151
pub redeemer_message: Vec<u8>,
3252
}
@@ -37,7 +57,6 @@ impl FastFill {
3757
pub(crate) fn checked_compute_size(redeemer_message_len: usize) -> Option<usize> {
3858
const FIXED: usize = 8 // DISCRIMINATOR
3959
+ FastFillSeeds::INIT_SPACE
40-
+ 32 // prepared_by
4160
+ 1 // redeemed
4261
+ FastFillInfo::INIT_SPACE
4362
+ 4 // redeemer_message len
@@ -62,11 +81,12 @@ impl FastFill {
6281
sequence,
6382
bump,
6483
},
65-
prepared_by,
6684
redeemed: Default::default(),
6785
info: FastFillInfo {
86+
prepared_by,
6887
amount,
6988
redeemer: Pubkey::from(redeemer),
89+
timestamp: Clock::get().unwrap().unix_timestamp,
7090
},
7191
redeemer_message: redeemer_message.into(),
7292
}

solana/programs/token-router/src/processor/redeem_fill/cctp.rs

+1
Original file line numberDiff line numberDiff line change
@@ -262,6 +262,7 @@ fn handle_redeem_fill_cctp(ctx: Context<RedeemCctpFill>, args: CctpMessageArgs)
262262
fill_type: FillType::WormholeCctpDeposit,
263263
source_chain: fill.source_chain(),
264264
order_sender: fill.order_sender(),
265+
timestamp: vaa.timestamp().into(),
265266
},
266267
redeemer_message: fill.message_to_vec(),
267268
});

solana/programs/token-router/src/processor/redeem_fill/fast.rs

+1
Original file line numberDiff line numberDiff line change
@@ -134,6 +134,7 @@ fn handle_redeem_fast_fill(ctx: Context<RedeemFastFill>) -> Result<()> {
134134
fill_type: FillType::FastFill,
135135
source_chain: fast_fill.seeds.source_chain,
136136
order_sender: fast_fill.seeds.order_sender,
137+
timestamp: fast_fill.info.timestamp,
137138
},
138139
redeemer_message,
139140
});

solana/programs/token-router/src/state/prepared_fill.rs

+17-6
Original file line numberDiff line numberDiff line change
@@ -15,15 +15,31 @@ pub struct PreparedFillSeeds {
1515

1616
#[derive(Debug, AnchorSerialize, AnchorDeserialize, Clone, InitSpace)]
1717
pub struct PreparedFillInfo {
18+
/// Bump seed for the custody token account associated with [PreparedFill].
1819
pub prepared_custody_token_bump: u8,
1920

21+
/// Who paid the lamports to create the [PreparedFill] account.
2022
pub prepared_by: Pubkey,
2123

24+
/// NOTE: If [FillType::Unset], the [PreparedFill] account is invalid.
2225
pub fill_type: FillType,
2326

27+
/// Wormhole chain ID reflecting where the order was created.
2428
pub source_chain: u16,
29+
30+
/// Universal address of the order sender.
2531
pub order_sender: [u8; 32],
32+
33+
/// Authority allowed to redeem [PreparedFill].
2634
pub redeemer: Pubkey,
35+
36+
/// Timestamp at the time a fill was issued. This time will either be a VAA time for a direct
37+
/// fill from another Token Router or timestamp from [matching_engine::state::FastFill] as a
38+
/// result of a market order.
39+
///
40+
/// NOTE: This timestamp is not used by the Token Router. It only provides more information for
41+
/// an integrator so he can perform special handling based on when the fill happened.
42+
pub timestamp: i64,
2743
}
2844

2945
#[account]
@@ -40,12 +56,7 @@ impl PreparedFill {
4056
pub fn checked_compute_size(payload_len: usize) -> Option<usize> {
4157
const FIXED: usize = 8 // DISCRIMINATOR
4258
+ PreparedFillSeeds::INIT_SPACE
43-
+ 1 // prepared_custody_token_bump
44-
+ 32 // prepared_by
45-
+ FillType::INIT_SPACE
46-
+ 2 // source_chain
47-
+ 32 // order_sender
48-
+ 32 // redeemer
59+
+ PreparedFillInfo::INIT_SPACE
4960
+ 4 // payload len
5061
;
5162

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

+39-6
Original file line numberDiff line numberDiff line change
@@ -3360,7 +3360,7 @@
33603360
"docs": [
33613361
"Length of the redeemer message, which may impact the expense to execute the auction."
33623362
],
3363-
"type": "u32"
3363+
"type": "u16"
33643364
},
33653365
{
33663366
"name": "destination_asset_info",
@@ -3556,7 +3556,7 @@
35563556
},
35573557
{
35583558
"name": "redeemer_message_len",
3559-
"type": "u32"
3559+
"type": "u16"
35603560
},
35613561
{
35623562
"name": "end_slot",
@@ -3732,12 +3732,11 @@
37323732
}
37333733
}
37343734
},
3735-
{
3736-
"name": "prepared_by",
3737-
"type": "pubkey"
3738-
},
37393735
{
37403736
"name": "redeemed",
3737+
"docs": [
3738+
"Whether the [FastFill] has been redeemed via the local Token Router."
3739+
],
37413740
"type": "bool"
37423741
},
37433742
{
@@ -3760,13 +3759,34 @@
37603759
"type": {
37613760
"kind": "struct",
37623761
"fields": [
3762+
{
3763+
"name": "prepared_by",
3764+
"docs": [
3765+
"Who paid the lamports to create the [FastFill] account."
3766+
],
3767+
"type": "pubkey"
3768+
},
37633769
{
37643770
"name": "amount",
3771+
"docs": [
3772+
"Fill amount."
3773+
],
37653774
"type": "u64"
37663775
},
37673776
{
37683777
"name": "redeemer",
3778+
"docs": [
3779+
"Authority allowed to redeem [FastFill]."
3780+
],
37693781
"type": "pubkey"
3782+
},
3783+
{
3784+
"name": "timestamp",
3785+
"docs": [
3786+
"Timestamp at the time a fill was issued. When the fast fill is created, it is set using the",
3787+
"current [Clock] unix timestamp."
3788+
],
3789+
"type": "i64"
37703790
}
37713791
]
37723792
}
@@ -3794,10 +3814,16 @@
37943814
"fields": [
37953815
{
37963816
"name": "source_chain",
3817+
"docs": [
3818+
"Wormhole chain ID reflecting where the order was created."
3819+
],
37973820
"type": "u16"
37983821
},
37993822
{
38003823
"name": "order_sender",
3824+
"docs": [
3825+
"Universal address of the order sender."
3826+
],
38013827
"type": {
38023828
"array": [
38033829
"u8",
@@ -3807,10 +3833,17 @@
38073833
},
38083834
{
38093835
"name": "sequence",
3836+
"docs": [
3837+
"Sequence generated by the [FastFillSequencer](crate::state::FastFillSequencer) when it",
3838+
"reserved a sequence number for this fill."
3839+
],
38103840
"type": "u64"
38113841
},
38123842
{
38133843
"name": "bump",
3844+
"docs": [
3845+
"Bump seed for the [FastFill] account."
3846+
],
38143847
"type": "u8"
38153848
}
38163849
]

0 commit comments

Comments
 (0)