Skip to content

Commit a064aa9

Browse files
authored
[solana] Fix vesting logic (#216)
* Remove the ability to vest multiple different mint types * Add vestingConfig and vestingBalance consistency checking in Delegate instruction * Fix overwriting vest.amount when transferring vest * Refactoring transfer_vesting: remove update_balance function * Add test for TransferVesting instruction if another vester with an existing vest of the same kind * Fix new_vesting_balance.vester in TransferVesting instruction * Refactoring transfer_vesting: update set_inner * Rename whTokenMint to votingTokenMint * Fix vesting tests * Apple yarn format * Fix Underflow/Overflow error name
1 parent 6b21165 commit a064aa9

16 files changed

+450
-307
lines changed

solana/app/StakeConnection.ts

+11-8
Original file line numberDiff line numberDiff line change
@@ -335,7 +335,10 @@ export class StakeConnection {
335335
).amount;
336336

337337
const totalSupply = (
338-
await getMint(this.program.provider.connection, this.config.whTokenMint)
338+
await getMint(
339+
this.program.provider.connection,
340+
this.config.votingTokenMint,
341+
)
339342
).supply;
340343

341344
return new StakeAccount(
@@ -366,7 +369,7 @@ export class StakeConnection {
366369
await this.program.methods
367370
.createStakeAccount()
368371
.accounts({
369-
mint: this.config.whTokenMint,
372+
mint: this.config.votingTokenMint,
370373
})
371374
.instruction(),
372375
);
@@ -389,7 +392,7 @@ export class StakeConnection {
389392
await this.program.methods
390393
.createStakeAccount()
391394
.accounts({
392-
mint: this.config.whTokenMint,
395+
mint: this.config.votingTokenMint,
393396
})
394397
.instruction(),
395398
);
@@ -402,7 +405,7 @@ export class StakeConnection {
402405
amount: BN,
403406
): Promise<TransactionInstruction> {
404407
const from_account = await getAssociatedTokenAddress(
405-
this.config.whTokenMint,
408+
this.config.votingTokenMint,
406409
this.provider.wallet.publicKey,
407410
true,
408411
);
@@ -515,7 +518,7 @@ export class StakeConnection {
515518
delegateeStakeAccountCheckpointsAddress,
516519
vestingConfig: vestingConfigAccount,
517520
vestingBalance: vestingBalanceAccount,
518-
mint: this.config.whTokenMint,
521+
mint: this.config.votingTokenMint,
519522
})
520523
.instruction(),
521524
);
@@ -602,7 +605,7 @@ export class StakeConnection {
602605
delegateeStakeAccountCheckpointsAddress,
603606
vestingConfig: null,
604607
vestingBalance: null,
605-
mint: this.config.whTokenMint,
608+
mint: this.config.votingTokenMint,
606609
})
607610
.instruction(),
608611
);
@@ -739,7 +742,7 @@ export class StakeConnection {
739742
}
740743

741744
const toAccount = await getAssociatedTokenAddress(
742-
this.config.whTokenMint,
745+
this.config.votingTokenMint,
743746
this.provider.wallet.publicKey,
744747
true,
745748
);
@@ -751,7 +754,7 @@ export class StakeConnection {
751754
this.provider.wallet.publicKey,
752755
toAccount,
753756
this.provider.wallet.publicKey,
754-
this.config.whTokenMint,
757+
this.config.votingTokenMint,
755758
),
756759
);
757760
}

solana/app/deploy/1_init_staking.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -19,7 +19,7 @@ async function main() {
1919
const globalConfig = {
2020
bump: 255,
2121
governanceAuthority: DEPLOYER_AUTHORITY_KEYPAIR.publicKey,
22-
whTokenMint: WORMHOLE_TOKEN,
22+
votingTokenMint: WORMHOLE_TOKEN,
2323
vestingAdmin: DEPLOYER_AUTHORITY_KEYPAIR.publicKey,
2424
maxCheckpointsAccountLimit: CHECKPOINTS_ACCOUNT_LIMIT,
2525
};

solana/programs/staking/src/context.rs

+2-2
Original file line numberDiff line numberDiff line change
@@ -131,7 +131,7 @@ pub struct Delegate<'info> {
131131
#[account(seeds = [CONFIG_SEED.as_bytes()], bump = config.bump)]
132132
pub config: Box<Account<'info, global_config::GlobalConfig>>,
133133
// Wormhole token mint:
134-
#[account(address = config.wh_token_mint)]
134+
#[account(address = config.voting_token_mint)]
135135
pub mint: Account<'info, Mint>,
136136
pub system_program: Program<'info, System>,
137137
}
@@ -446,7 +446,7 @@ pub struct CreateStakeAccount<'info> {
446446
#[account(seeds = [CONFIG_SEED.as_bytes()], bump = config.bump)]
447447
pub config: Box<Account<'info, global_config::GlobalConfig>>,
448448
// Wormhole token mint:
449-
#[account(address = config.wh_token_mint)]
449+
#[account(address = config.voting_token_mint)]
450450
pub mint: Account<'info, Mint>,
451451
#[account(
452452
init,

solana/programs/staking/src/contexts/claim_vesting.rs

-10
Original file line numberDiff line numberDiff line change
@@ -95,16 +95,6 @@ impl<'info> ClaimVesting<'info> {
9595
);
9696
drop(loaded_checkpoints);
9797

98-
require!(
99-
self.config.mint == self.global_config.wh_token_mint,
100-
// This error can never happen here, because for the condition above
101-
// (self.vesting_balance.stake_account_metadata != Pubkey::default())
102-
// to be met, the delegate instruction must be executed.
103-
// However, delegate cannot be executed when self.config.mint !=
104-
// self.global_config.wh_token_mint.
105-
VestingError::InvalidVestingMint
106-
);
107-
10898
// Additional checks to ensure the owner matches
10999
require!(
110100
stake_account_metadata.owner == self.vesting_balance.vester,

solana/programs/staking/src/contexts/initialize.rs

+1
Original file line numberDiff line numberDiff line change
@@ -11,6 +11,7 @@ use anchor_spl::token_interface::{Mint, TokenAccount, TokenInterface};
1111
pub struct Initialize<'info> {
1212
#[account(mut)]
1313
admin: Signer<'info>,
14+
#[account(address = global_config.voting_token_mint)]
1415
mint: InterfaceAccount<'info, Mint>,
1516
// Initialize a vault for us to store our money in escrow for vesting
1617
#[account(

solana/programs/staking/src/contexts/transfer_vesting.rs

+35-32
Original file line numberDiff line numberDiff line change
@@ -87,16 +87,11 @@ pub struct TransferVesting<'info> {
8787
}
8888

8989
impl<'info> crate::contexts::TransferVesting<'info> {
90-
pub fn transfer_vesting(&mut self, bump: u8) -> Result<()> {
91-
fn update_balance(balance: &mut u64, amount: u64, is_subtract: bool) -> Result<()> {
92-
if is_subtract {
93-
*balance = balance.checked_sub(amount).ok_or(VestingError::Underflow)?;
94-
} else {
95-
*balance = balance.checked_add(amount).ok_or(VestingError::Underflow)?;
96-
}
97-
Ok(())
98-
}
99-
90+
pub fn transfer_vesting(
91+
&mut self,
92+
new_vest_bump: u8,
93+
new_vesting_balance_bump: u8,
94+
) -> Result<()> {
10095
if self.new_vester_ta.owner.key() == self.vester_ta.owner.key() {
10196
return err!(VestingError::TransferVestToMyself);
10297
}
@@ -149,11 +144,10 @@ impl<'info> crate::contexts::TransferVesting<'info> {
149144
VestingError::InvalidStakeAccountMetadataPDA
150145
);
151146

152-
update_balance(
153-
&mut stake_account_metadata.recorded_vesting_balance,
154-
self.vest.amount,
155-
true,
156-
)?;
147+
stake_account_metadata.recorded_vesting_balance = stake_account_metadata
148+
.recorded_vesting_balance
149+
.checked_sub(self.vest.amount)
150+
.ok_or(VestingError::Underflow)?;
157151

158152
// Update checkpoints
159153
let current_delegate_checkpoints_account_info =
@@ -232,11 +226,10 @@ impl<'info> crate::contexts::TransferVesting<'info> {
232226
VestingError::InvalidStakeAccountMetadataPDA
233227
);
234228

235-
update_balance(
236-
&mut new_stake_account_metadata.recorded_vesting_balance,
237-
self.vest.amount,
238-
false,
239-
)?;
229+
new_stake_account_metadata.recorded_vesting_balance = new_stake_account_metadata
230+
.recorded_vesting_balance
231+
.checked_add(self.vest.amount)
232+
.ok_or(VestingError::Overflow)?;
240233

241234
let current_delegate_checkpoints_account_info =
242235
new_stake_account_checkpoints.to_account_info();
@@ -267,21 +260,31 @@ impl<'info> crate::contexts::TransferVesting<'info> {
267260
self.new_vest.set_inner(Vesting {
268261
vester_ta: self.new_vester_ta.key(),
269262
config: self.vest.config,
270-
amount: self.vest.amount,
263+
amount: self
264+
.new_vest
265+
.amount
266+
.checked_add(self.vest.amount)
267+
.ok_or(VestingError::Overflow)?,
271268
maturation: self.vest.maturation,
272-
bump,
269+
bump: new_vest_bump,
270+
});
271+
272+
self.new_vesting_balance.set_inner(VestingBalance {
273+
vester: self.new_vester_ta.owner.key(),
274+
stake_account_metadata: self.new_vesting_balance.stake_account_metadata,
275+
total_vesting_balance: self
276+
.new_vesting_balance
277+
.total_vesting_balance
278+
.checked_add(self.vest.amount)
279+
.ok_or(VestingError::Overflow)?,
280+
bump: new_vesting_balance_bump,
273281
});
274282

275-
update_balance(
276-
&mut self.vesting_balance.total_vesting_balance,
277-
self.vest.amount,
278-
true,
279-
)?;
280-
update_balance(
281-
&mut self.new_vesting_balance.total_vesting_balance,
282-
self.vest.amount,
283-
false,
284-
)?;
283+
self.vesting_balance.total_vesting_balance = self
284+
.vesting_balance
285+
.total_vesting_balance
286+
.checked_sub(self.vest.amount)
287+
.ok_or(VestingError::Underflow)?;
285288

286289
Ok(())
287290
}

solana/programs/staking/src/error.rs

+4-2
Original file line numberDiff line numberDiff line change
@@ -7,6 +7,8 @@ pub enum ErrorCode {
77
TooManyCheckpoints,
88
#[msg("An arithmetic operation unexpectedly overflowed")]
99
GenericOverflow,
10+
#[msg("An arithmetic operation unexpectedly underflowed")]
11+
GenericUnderflow,
1012
#[msg("Error deserializing checkpoint")]
1113
CheckpointSerDe,
1214
#[msg("Checkpoint out of bounds")]
@@ -69,8 +71,6 @@ pub enum VestingError {
6971
InvalidStakeAccountCheckpoints,
7072
#[msg("Error parsing stake_account_metadata and stake_account_checkpoints")]
7173
ErrorOfStakeAccountParsing,
72-
#[msg("Invalid vesting mint")]
73-
InvalidVestingMint,
7474
#[msg("Invalid stake account owner")]
7575
InvalidStakeAccountOwner,
7676
#[msg("Invalid vesting admin")]
@@ -81,6 +81,8 @@ pub enum VestingError {
8181
InvalidStakeAccountMetadataPDA,
8282
#[msg("Invalid stake account checkpoints PDA")]
8383
InvalidStakeAccountCheckpointsPDA,
84+
#[msg("Invalid vesting balance PDA")]
85+
InvalidVestingBalancePDA,
8486
#[msg("Transfer vest to myself")]
8587
TransferVestToMyself,
8688
}

solana/programs/staking/src/lib.rs

+13-7
Original file line numberDiff line numberDiff line change
@@ -78,7 +78,7 @@ pub mod staking {
7878
let config_account = &mut ctx.accounts.config_account;
7979
config_account.bump = ctx.bumps.config_account;
8080
config_account.governance_authority = global_config.governance_authority;
81-
config_account.wh_token_mint = global_config.wh_token_mint;
81+
config_account.voting_token_mint = global_config.voting_token_mint;
8282
config_account.vesting_admin = global_config.vesting_admin;
8383
config_account.max_checkpoints_account_limit = global_config.max_checkpoints_account_limit;
8484

@@ -188,14 +188,19 @@ pub mod staking {
188188
if let Some(vesting_config) = &mut ctx.accounts.vesting_config {
189189
if vesting_config.finalized {
190190
if let Some(vesting_balance) = &mut ctx.accounts.vesting_balance {
191-
require!(
192-
vesting_balance.vester == stake_account_metadata.owner,
193-
VestingError::InvalidStakeAccountOwner
191+
let (expected_vesting_balance_pda, _) = Pubkey::find_program_address(
192+
&[
193+
VESTING_BALANCE_SEED.as_bytes(),
194+
vesting_config.key().as_ref(),
195+
stake_account_metadata.owner.as_ref(),
196+
],
197+
&crate::ID,
194198
);
195199
require!(
196-
vesting_config.mint == config.wh_token_mint,
197-
VestingError::InvalidVestingMint
200+
expected_vesting_balance_pda == vesting_balance.key(),
201+
VestingError::InvalidVestingBalancePDA
198202
);
203+
199204
vesting_balance.stake_account_metadata = stake_account_metadata.key();
200205
stake_account_metadata.recorded_vesting_balance =
201206
vesting_balance.total_vesting_balance;
@@ -664,7 +669,8 @@ pub mod staking {
664669

665670
// Transfer Vesting from and send to new Vester
666671
pub fn transfer_vesting(ctx: Context<TransferVesting>) -> Result<()> {
667-
ctx.accounts.transfer_vesting(ctx.bumps.new_vest)
672+
ctx.accounts
673+
.transfer_vesting(ctx.bumps.new_vest, ctx.bumps.new_vesting_balance)
668674
}
669675

670676
// Cancel and close a Vesting account for a non-finalized Config

solana/programs/staking/src/state/checkpoints.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -280,7 +280,7 @@ fn calc_new_checkpoint(
280280
.ok_or_else(|| error!(ErrorCode::GenericOverflow))?,
281281
Operation::Subtract => current_value
282282
.checked_sub(amount_delta)
283-
.ok_or_else(|| error!(ErrorCode::GenericOverflow))?,
283+
.ok_or_else(|| error!(ErrorCode::GenericUnderflow))?,
284284
};
285285

286286
let new_checkpoint = Checkpoint {

solana/programs/staking/src/state/global_config.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -9,7 +9,7 @@ pub struct GlobalConfig {
99
// Maximum number of checkpoints that can be stored in a single account
1010
pub max_checkpoints_account_limit: u32,
1111
pub governance_authority: Pubkey,
12-
pub wh_token_mint: Pubkey,
12+
pub voting_token_mint: Pubkey,
1313
pub vesting_admin: Pubkey,
1414
}
1515

solana/programs/staking/src/state/vesting.rs

+1
Original file line numberDiff line numberDiff line change
@@ -2,6 +2,7 @@ use anchor_lang::prelude::*;
22
use std::mem::size_of;
33

44
#[account]
5+
#[derive(Default)]
56
pub struct Vesting {
67
pub vester_ta: Pubkey,
78
pub config: Pubkey,

solana/programs/staking/src/state/vesting_balance.rs

+1
Original file line numberDiff line numberDiff line change
@@ -4,6 +4,7 @@ use std::mem::size_of;
44
/// Used to store the total vesting balance of a single vester
55
/// It is also used to delegate vesting
66
#[account]
7+
#[derive(Default)]
78
pub struct VestingBalance {
89
pub vester: Pubkey,
910
pub total_vesting_balance: u64,

solana/tests/api_test.ts

+5-5
Original file line numberDiff line numberDiff line change
@@ -632,7 +632,7 @@ describe("api", async () => {
632632
user3StakeAccountCheckpointsAddress,
633633
vestingConfig: null,
634634
vestingBalance: null,
635-
mint: stakeConnection.config.whTokenMint,
635+
mint: stakeConnection.config.votingTokenMint,
636636
})
637637
.rpc();
638638

@@ -677,7 +677,7 @@ describe("api", async () => {
677677
user3StakeAccountCheckpointsAddress,
678678
vestingConfig: null,
679679
vestingBalance: null,
680-
mint: stakeConnection.config.whTokenMint,
680+
mint: stakeConnection.config.votingTokenMint,
681681
})
682682
.rpc();
683683

@@ -726,7 +726,7 @@ describe("api", async () => {
726726
ownerStakeAccountCheckpointsAddress, // Invalid delegatee checkpoints account
727727
vestingConfig: null,
728728
vestingBalance: null,
729-
mint: stakeConnection.config.whTokenMint,
729+
mint: stakeConnection.config.votingTokenMint,
730730
})
731731
.rpc();
732732

@@ -801,7 +801,7 @@ describe("api", async () => {
801801
);
802802

803803
const toAccount = await getAssociatedTokenAddress(
804-
stakeConnection.config.whTokenMint,
804+
stakeConnection.config.votingTokenMint,
805805
owner,
806806
true,
807807
);
@@ -842,7 +842,7 @@ describe("api", async () => {
842842
assert.equal(currentDelegate.toBase58(), user2.toBase58());
843843

844844
const toAccount = await getAssociatedTokenAddress(
845-
stakeConnection.config.whTokenMint,
845+
stakeConnection.config.votingTokenMint,
846846
owner,
847847
true,
848848
);

0 commit comments

Comments
 (0)