Skip to content

Commit d0a90eb

Browse files
authored
[solana] Add timestamp check in CastVote instruction (#221)
* Add timestamp check in CastVote instruction * Use strict condition in cast_vote * Fix tests * Apply yarn format * Fix tests. Apply yarn format
1 parent d0d1849 commit d0a90eb

File tree

4 files changed

+92
-22
lines changed

4 files changed

+92
-22
lines changed

solana/programs/staking/src/error.rs

+2
Original file line numberDiff line numberDiff line change
@@ -49,6 +49,8 @@ pub enum ErrorCode {
4949
ExceedsMaxAllowableVoteWeightWindowLength,
5050
#[msg("Invalid next voter checkpoints")]
5151
InvalidNextVoterCheckpoints,
52+
#[msg("Proposal inactive")]
53+
ProposalInactive,
5254
#[msg("Other")]
5355
Other,
5456
}

solana/programs/staking/src/lib.rs

+2
Original file line numberDiff line numberDiff line change
@@ -502,7 +502,9 @@ pub mod staking {
502502
let proposal = &mut ctx.accounts.proposal;
503503
let config = &ctx.accounts.config;
504504

505+
let current_timestamp: u64 = utils::clock::get_current_time().try_into()?;
505506
let vote_start = proposal.vote_start;
507+
require!(current_timestamp > vote_start, ErrorCode::ProposalInactive);
506508

507509
let (_, window_length) = find_window_length_le(
508510
&ctx.accounts.vote_weight_window_lengths.to_account_info(),

solana/tests/api_test.ts

+70-5
Original file line numberDiff line numberDiff line change
@@ -959,17 +959,66 @@ describe("api", async () => {
959959
});
960960

961961
describe("castVote", () => {
962+
it("should fail to castVote if proposal inactive", async () => {
963+
await user6StakeConnection.delegate(
964+
user6,
965+
WHTokenBalance.fromString("50"),
966+
);
967+
968+
let proposalIdInput = await addTestProposal(
969+
user6StakeConnection,
970+
Math.floor(Date.now() / 1000) + 20,
971+
);
972+
973+
let stakeAccountMetadataAddress =
974+
await user6StakeConnection.getStakeMetadataAddress(
975+
user6StakeConnection.userPublicKey(),
976+
);
977+
let previousStakeAccountCheckpointsAddress =
978+
await user6StakeConnection.getStakeAccountCheckpointsAddressByMetadata(
979+
stakeAccountMetadataAddress,
980+
false,
981+
);
982+
983+
const { proposalAccount } =
984+
await user6StakeConnection.fetchProposalAccount(proposalIdInput);
985+
986+
try {
987+
await user6StakeConnection.program.methods
988+
.castVote(
989+
Array.from(proposalIdInput),
990+
new BN(10),
991+
new BN(20),
992+
new BN(12),
993+
0,
994+
)
995+
.accountsPartial({
996+
proposal: proposalAccount,
997+
voterCheckpoints: previousStakeAccountCheckpointsAddress,
998+
voterCheckpointsNext: null,
999+
})
1000+
.rpc();
1001+
1002+
assert.fail("Expected an error but none was thrown");
1003+
} catch (e) {
1004+
assert(
1005+
(e as AnchorError).error?.errorCode?.code === "ProposalInactive",
1006+
);
1007+
}
1008+
});
1009+
9621010
it("should fail to castVote if votes were added in the voteWeightWindow", async () => {
9631011
await user6StakeConnection.delegate(
9641012
user6,
965-
WHTokenBalance.fromString("150"),
1013+
WHTokenBalance.fromString("100"),
9661014
);
9671015

9681016
// voteWeightWindow is 10s
9691017
let proposalIdInput = await addTestProposal(
9701018
user6StakeConnection,
9711019
Math.floor(Date.now() / 1000) + 3,
9721020
);
1021+
await sleep(4000);
9731022

9741023
let stakeAccountMetadataAddress =
9751024
await user6StakeConnection.getStakeMetadataAddress(
@@ -1014,11 +1063,16 @@ describe("api", async () => {
10141063
WHTokenBalance.fromString("150"),
10151064
);
10161065

1066+
let voteStart = Math.floor(Date.now() / 1000) + 12;
10171067
let proposalIdInput = await addTestProposal(
10181068
user3StakeConnection,
1019-
Math.floor(Date.now() / 1000) + 12,
1069+
voteStart,
10201070
);
10211071

1072+
while (voteStart >= Math.floor(Date.now() / 1000)) {
1073+
await sleep(1000);
1074+
}
1075+
await sleep(1000);
10221076
await user3StakeConnection.castVote(
10231077
proposalIdInput,
10241078
new BN(10),
@@ -1053,6 +1107,7 @@ describe("api", async () => {
10531107
it("should cast vote with the correct weight", async () => {
10541108
let stakeAccountCheckpointsAddress;
10551109
let proposalIdInput;
1110+
let voteStart;
10561111

10571112
// Create 6 checkpoints, 1 second apart
10581113
for (let i = 0; i < 6; i++) {
@@ -1064,15 +1119,20 @@ describe("api", async () => {
10641119
// Create a proposal with a start time 10 seconds in the future in iteration 5
10651120
// We do this because the vote weight window is 10 seconds
10661121
if (i == 4) {
1122+
voteStart = Math.floor(Date.now() / 1000) + 10;
10671123
proposalIdInput = await addTestProposal(
10681124
user7StakeConnection,
1069-
Math.floor(Date.now() / 1000) + 10,
1125+
voteStart,
10701126
);
10711127
}
10721128

10731129
await sleep(1000);
10741130
}
10751131

1132+
while (voteStart >= Math.floor(Date.now() / 1000)) {
1133+
await sleep(1000);
1134+
}
1135+
await sleep(1000);
10761136
await user7StakeConnection.castVote(
10771137
proposalIdInput,
10781138
new BN(10),
@@ -1088,7 +1148,6 @@ describe("api", async () => {
10881148
assert.equal(againstVotes.toString(), "10");
10891149
assert.equal(forVotes.toString(), "20");
10901150
assert.equal(abstainVotes.toString(), "12");
1091-
10921151
});
10931152

10941153
it("should fail to castVote if next voter checkpoints are invalid", async () => {
@@ -1103,6 +1162,7 @@ describe("api", async () => {
11031162
user4StakeConnection,
11041163
voteStart,
11051164
);
1165+
11061166
const { proposalAccount } =
11071167
await user4StakeConnection.fetchProposalAccount(proposalIdInput);
11081168

@@ -1114,6 +1174,10 @@ describe("api", async () => {
11141174
WHTokenBalance.fromString("5"),
11151175
);
11161176
}
1177+
await sleep(5000);
1178+
while (voteStart >= Math.floor(Date.now() / 1000)) {
1179+
await sleep(1000);
1180+
}
11171181

11181182
let currentStakeAccountCheckpointsAddress =
11191183
await user4StakeConnection.getStakeAccountCheckpointsAddress(
@@ -1178,8 +1242,9 @@ describe("api", async () => {
11781242

11791243
let proposalIdInput = await addTestProposal(
11801244
user4StakeConnection,
1181-
Math.floor(Date.now() / 1000) + 12,
1245+
Math.floor(Date.now() / 1000) + 11,
11821246
);
1247+
await sleep(12000);
11831248

11841249
const { proposalAccount } =
11851250
await user4StakeConnection.fetchProposalAccount(proposalIdInput);

solana/tests/vesting.ts

+18-17
Original file line numberDiff line numberDiff line change
@@ -550,11 +550,11 @@ describe("vesting", () => {
550550

551551
await stakeConnection.program.methods
552552
.initializeVestingConfig(seed2)
553-
.accounts({
554-
...accounts,
555-
config: config2,
556-
vault: vault2,
557-
})
553+
.accounts({
554+
...accounts,
555+
config: config2,
556+
vault: vault2,
557+
})
558558
.signers([whMintAuthority])
559559
.rpc()
560560
.then(confirm);
@@ -675,12 +675,12 @@ describe("vesting", () => {
675675
it("Create another matured vests", async () => {
676676
await stakeConnection.program.methods
677677
.createVesting(NOW, new BN(100e6))
678-
.accounts({
679-
...accounts,
680-
config: config2,
681-
vestingBalance: vestingBalance2,
682-
vest: vestNow2,
683-
})
678+
.accounts({
679+
...accounts,
680+
config: config2,
681+
vestingBalance: vestingBalance2,
682+
vest: vestNow2,
683+
})
684684
.signers([whMintAuthority])
685685
.rpc({
686686
skipPreflight: true,
@@ -1015,12 +1015,13 @@ describe("vesting", () => {
10151015
);
10161016

10171017
await sleep(2000);
1018-
let stakeAccountCheckpointsAddress = await vesterStakeConnection.delegateWithVest(
1019-
vesterStakeConnection.userPublicKey(),
1020-
WHTokenBalance.fromString("0"),
1021-
true,
1022-
config2,
1023-
);
1018+
let stakeAccountCheckpointsAddress =
1019+
await vesterStakeConnection.delegateWithVest(
1020+
vesterStakeConnection.userPublicKey(),
1021+
WHTokenBalance.fromString("0"),
1022+
true,
1023+
config2,
1024+
);
10241025

10251026
let stakeAccountCheckpointsData =
10261027
await vesterStakeConnection.program.account.checkpointData.fetch(

0 commit comments

Comments
 (0)