2023-09-12 18:37:30 +02:00
|
|
|
// SPDX-License-Identifier: UNLICENSED
|
2024-11-08 00:56:09 -03:00
|
|
|
pragma solidity ^0.8.26;
|
2023-09-12 18:37:30 +02:00
|
|
|
|
2023-10-10 15:44:16 +02:00
|
|
|
import { ERC20 } from "@openzeppelin/contracts/token/ERC20/ERC20.sol";
|
|
|
|
|
2024-02-24 16:35:34 -03:00
|
|
|
import { Test, console } from "forge-std/Test.sol";
|
2023-09-12 18:37:30 +02:00
|
|
|
import { Deploy } from "../script/Deploy.s.sol";
|
2024-02-28 13:46:50 +01:00
|
|
|
import { DeployMigrationStakeManager } from "../script/DeployMigrationStakeManager.s.sol";
|
2023-09-12 18:37:30 +02:00
|
|
|
import { DeploymentConfig } from "../script/DeploymentConfig.s.sol";
|
2024-11-07 10:50:13 -03:00
|
|
|
import { StakeManager, IStakeManager, ExpiredStakeStorage } from "../contracts/StakeManager.sol";
|
|
|
|
import { ITrustedCodehashAccess } from "../contracts/access/ITrustedCodehashAccess.sol";
|
|
|
|
import { MultiplierPointMath } from "../contracts/MultiplierPointMath.sol";
|
2023-10-10 15:44:16 +02:00
|
|
|
import { StakeVault } from "../contracts/StakeVault.sol";
|
2023-11-07 09:49:22 +01:00
|
|
|
import { VaultFactory } from "../contracts/VaultFactory.sol";
|
2023-09-12 18:37:30 +02:00
|
|
|
|
|
|
|
contract StakeManagerTest is Test {
|
|
|
|
DeploymentConfig internal deploymentConfig;
|
|
|
|
StakeManager internal stakeManager;
|
2023-11-07 09:49:22 +01:00
|
|
|
VaultFactory internal vaultFactory;
|
2023-09-12 18:37:30 +02:00
|
|
|
|
2023-10-10 13:53:03 +02:00
|
|
|
address internal stakeToken;
|
|
|
|
address internal deployer;
|
2023-10-10 15:44:16 +02:00
|
|
|
address internal testUser = makeAddr("testUser");
|
2024-01-12 13:07:31 +01:00
|
|
|
address internal testUser2 = makeAddr("testUser2");
|
2023-10-10 13:53:03 +02:00
|
|
|
|
2023-09-12 18:37:30 +02:00
|
|
|
function setUp() public virtual {
|
|
|
|
Deploy deployment = new Deploy();
|
2023-11-07 09:49:22 +01:00
|
|
|
(vaultFactory, stakeManager, deploymentConfig) = deployment.run();
|
2023-10-10 13:53:03 +02:00
|
|
|
(deployer, stakeToken) = deploymentConfig.activeNetworkConfig();
|
2023-09-12 18:37:30 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
function testDeployment() public {
|
|
|
|
assertEq(stakeManager.owner(), deployer);
|
2023-10-10 13:53:03 +02:00
|
|
|
assertEq(stakeManager.currentEpoch(), 0);
|
|
|
|
assertEq(stakeManager.pendingReward(), 0);
|
2024-11-07 10:50:13 -03:00
|
|
|
assertEq(stakeManager.totalMP(), 0);
|
|
|
|
assertEq(stakeManager.totalStaked(), 0);
|
2024-11-08 00:56:09 -03:00
|
|
|
assertEq(address(stakeManager.REWARD_TOKEN()), stakeToken);
|
2024-06-19 11:26:03 +02:00
|
|
|
assertEq(address(stakeManager.previousManager()), address(0));
|
2023-09-12 18:37:30 +02:00
|
|
|
assertEq(stakeManager.totalSupply(), 0);
|
|
|
|
}
|
2023-10-10 15:44:16 +02:00
|
|
|
|
|
|
|
function _createTestVault(address owner) internal returns (StakeVault vault) {
|
|
|
|
vm.prank(owner);
|
2023-11-07 09:49:22 +01:00
|
|
|
vault = vaultFactory.createVault();
|
2023-10-10 15:44:16 +02:00
|
|
|
|
2024-09-28 15:21:43 -03:00
|
|
|
if (!stakeManager.isTrustedCodehash(address(vault).codehash)) {
|
2024-01-12 13:07:31 +01:00
|
|
|
vm.prank(deployer);
|
2024-09-28 15:21:43 -03:00
|
|
|
stakeManager.setTrustedCodehash(address(vault).codehash, true);
|
2024-01-12 13:07:31 +01:00
|
|
|
}
|
2023-10-10 15:44:16 +02:00
|
|
|
}
|
2024-02-28 00:03:35 -03:00
|
|
|
|
|
|
|
function _createStakingAccount(address owner, uint256 amount) internal returns (StakeVault userVault) {
|
|
|
|
return _createStakingAccount(owner, amount, 0, amount);
|
|
|
|
}
|
|
|
|
|
|
|
|
function _createStakingAccount(
|
|
|
|
address owner,
|
|
|
|
uint256 amount,
|
|
|
|
uint256 lockTime
|
|
|
|
)
|
|
|
|
internal
|
|
|
|
returns (StakeVault userVault)
|
|
|
|
{
|
|
|
|
return _createStakingAccount(owner, amount, lockTime, amount);
|
|
|
|
}
|
|
|
|
|
|
|
|
function _createStakingAccount(
|
|
|
|
address owner,
|
|
|
|
uint256 amount,
|
|
|
|
uint256 lockTime,
|
|
|
|
uint256 mintAmount
|
|
|
|
)
|
|
|
|
internal
|
|
|
|
returns (StakeVault userVault)
|
|
|
|
{
|
|
|
|
deal(stakeToken, owner, mintAmount);
|
|
|
|
userVault = _createTestVault(owner);
|
|
|
|
vm.startPrank(owner);
|
2024-03-01 19:19:45 -03:00
|
|
|
ERC20(stakeToken).approve(address(userVault), mintAmount);
|
2024-02-28 00:03:35 -03:00
|
|
|
userVault.stake(amount, lockTime);
|
|
|
|
vm.stopPrank();
|
|
|
|
}
|
2023-09-12 18:37:30 +02:00
|
|
|
}
|
2023-10-10 15:32:46 +02:00
|
|
|
|
|
|
|
contract StakeTest is StakeManagerTest {
|
|
|
|
function test_RevertWhen_SenderIsNotVault() public {
|
2024-11-07 10:50:13 -03:00
|
|
|
vm.expectRevert(ITrustedCodehashAccess.TrustedCodehashAccess__UnauthorizedCodehash.selector);
|
2023-10-10 15:32:46 +02:00
|
|
|
stakeManager.stake(100, 1);
|
|
|
|
}
|
2023-12-06 12:10:07 +01:00
|
|
|
|
2024-09-12 02:09:45 -03:00
|
|
|
function test_StakeWithLockBonusMP() public {
|
|
|
|
uint256 stakeAmount = 10_000;
|
|
|
|
uint256 lockTime = stakeManager.MIN_LOCKUP_PERIOD();
|
|
|
|
|
|
|
|
StakeVault userVault = _createStakingAccount(testUser, stakeAmount, 0, stakeAmount);
|
|
|
|
|
|
|
|
(, uint256 balance, uint256 bonusMP, uint256 totalMP,,,,) = stakeManager.accounts(address(userVault));
|
|
|
|
assertEq(balance, stakeAmount, "balance of user vault should be equal to stake amount after stake");
|
|
|
|
assertEq(bonusMP, stakeAmount, "bonusMP of user vault should be equal to stake amount after stake if no lock");
|
2024-11-07 10:50:13 -03:00
|
|
|
assertEq(
|
|
|
|
totalMP, stakeAmount, "totalMP of user vault should be equal to stakeAmount after stake if no epochs passed"
|
|
|
|
);
|
2024-09-12 02:09:45 -03:00
|
|
|
|
|
|
|
vm.prank(testUser);
|
|
|
|
userVault.lock(lockTime);
|
2024-11-07 10:50:13 -03:00
|
|
|
uint256 estimatedBonusMp = stakeAmount + stakeManager.calculateMP(stakeAmount, lockTime);
|
2024-09-12 02:09:45 -03:00
|
|
|
|
|
|
|
(, balance, bonusMP, totalMP,,,,) = stakeManager.accounts(address(userVault));
|
|
|
|
assertEq(balance, stakeAmount, "balance of user vault should be equal to stake amount after lock");
|
|
|
|
assertEq(bonusMP, estimatedBonusMp, "bonusMP of user vault should be equal to estimated bonusMP after lock");
|
|
|
|
assertEq(totalMP, bonusMP, "totalMP of user vault should be equal to bonusMP after lock if no epochs passed");
|
|
|
|
|
|
|
|
StakeVault userVault2 = _createStakingAccount(testUser, stakeAmount, lockTime, stakeAmount);
|
|
|
|
|
|
|
|
(, balance, bonusMP, totalMP,,,,) = stakeManager.accounts(address(userVault2));
|
|
|
|
assertEq(balance, stakeAmount, "balance of user vault should be equal to stake amount after stake locked");
|
|
|
|
assertEq(
|
|
|
|
bonusMP, estimatedBonusMp, "bonusMP of user vault should be equal to estimated bonusMP after stake locked"
|
|
|
|
);
|
|
|
|
assertEq(
|
|
|
|
totalMP, bonusMP, "totalMP of user vault should be equal to bonusMP after stake locked if no epochs passed"
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2023-12-06 12:10:07 +01:00
|
|
|
function test_RevertWhen_InvalidLockupPeriod() public {
|
|
|
|
// ensure user has funds
|
|
|
|
deal(stakeToken, testUser, 1000);
|
|
|
|
StakeVault userVault = _createTestVault(testUser);
|
|
|
|
|
|
|
|
vm.startPrank(testUser);
|
|
|
|
ERC20(stakeToken).approve(address(userVault), 100);
|
|
|
|
|
|
|
|
uint256 lockTime = stakeManager.MIN_LOCKUP_PERIOD() - 1;
|
2024-11-07 10:50:13 -03:00
|
|
|
vm.expectRevert(IStakeManager.StakeManager__InvalidLockTime.selector);
|
2023-12-06 12:10:07 +01:00
|
|
|
userVault.stake(100, lockTime);
|
|
|
|
|
|
|
|
lockTime = stakeManager.MAX_LOCKUP_PERIOD() + 1;
|
2024-11-07 10:50:13 -03:00
|
|
|
vm.expectRevert(IStakeManager.StakeManager__InvalidLockTime.selector);
|
2023-12-06 12:10:07 +01:00
|
|
|
userVault.stake(100, lockTime);
|
|
|
|
}
|
2024-01-12 13:07:31 +01:00
|
|
|
|
2024-09-04 09:54:45 +02:00
|
|
|
function test_RevertWhen_StakeIsTooLow() public {
|
|
|
|
StakeVault userVault = _createTestVault(testUser);
|
|
|
|
vm.startPrank(testUser);
|
2024-11-07 10:50:13 -03:00
|
|
|
vm.expectRevert(IStakeManager.StakeManager__StakeIsTooLow.selector);
|
2024-09-04 09:54:45 +02:00
|
|
|
userVault.stake(0, 0);
|
2024-03-04 23:00:06 -03:00
|
|
|
}
|
|
|
|
|
2024-09-04 09:54:45 +02:00
|
|
|
function test_RevertWhen_Restake() public {
|
|
|
|
// ensure user has funds
|
|
|
|
deal(stakeToken, testUser, 1000);
|
|
|
|
StakeVault userVault = _createTestVault(testUser);
|
|
|
|
vm.startPrank(testUser);
|
|
|
|
ERC20(stakeToken).approve(address(userVault), 1000);
|
2024-03-01 19:19:45 -03:00
|
|
|
|
2024-09-04 09:54:45 +02:00
|
|
|
userVault.stake(100, 0);
|
|
|
|
vm.expectRevert(StakeManager.StakeManager__AlreadyStaked.selector);
|
|
|
|
userVault.stake(100, 0);
|
2024-03-01 19:19:45 -03:00
|
|
|
}
|
|
|
|
|
2024-09-04 09:54:45 +02:00
|
|
|
function test_RevertWhen_RestakeWithLock() public {
|
2024-03-01 19:19:45 -03:00
|
|
|
uint256 lockToIncrease = stakeManager.MIN_LOCKUP_PERIOD();
|
2024-09-04 09:54:45 +02:00
|
|
|
// ensure user has funds
|
|
|
|
deal(stakeToken, testUser, 1000);
|
|
|
|
StakeVault userVault = _createTestVault(testUser);
|
|
|
|
vm.startPrank(testUser);
|
|
|
|
ERC20(stakeToken).approve(address(userVault), 1000);
|
2024-03-01 19:19:45 -03:00
|
|
|
|
2024-09-04 09:54:45 +02:00
|
|
|
userVault.stake(100, lockToIncrease);
|
|
|
|
vm.expectRevert(StakeManager.StakeManager__AlreadyStaked.selector);
|
|
|
|
userVault.stake(100, 0);
|
2024-03-01 19:19:45 -03:00
|
|
|
}
|
|
|
|
|
2024-09-04 09:54:45 +02:00
|
|
|
function test_StakeWithoutLockUpTimeMintsMultiplierPoints() public {
|
2024-09-28 15:21:43 -03:00
|
|
|
uint256 stakeAmount = 54;
|
|
|
|
StakeVault userVault = _createStakingAccount(testUser, stakeAmount, 0, stakeAmount);
|
2024-03-01 19:19:45 -03:00
|
|
|
|
2024-09-04 09:54:45 +02:00
|
|
|
(,, uint256 totalMP,,,,,) = stakeManager.accounts(address(userVault));
|
2024-11-07 10:50:13 -03:00
|
|
|
assertEq(stakeManager.totalMP(), stakeAmount, "total multiplier point supply");
|
2024-09-04 09:54:45 +02:00
|
|
|
assertEq(totalMP, stakeAmount, "user multiplier points");
|
2024-03-01 19:19:45 -03:00
|
|
|
|
|
|
|
vm.prank(testUser);
|
2024-09-04 09:54:45 +02:00
|
|
|
userVault.unstake(stakeAmount);
|
2024-03-01 19:19:45 -03:00
|
|
|
|
2024-09-04 09:54:45 +02:00
|
|
|
(,,, totalMP,,,,) = stakeManager.accounts(address(userVault));
|
2024-11-07 10:50:13 -03:00
|
|
|
assertEq(stakeManager.totalMP(), 0, "totalMP burned after unstaking");
|
2024-09-04 09:54:45 +02:00
|
|
|
assertEq(totalMP, 0, "userMP burned after unstaking");
|
2024-03-01 19:19:45 -03:00
|
|
|
}
|
2023-10-10 15:32:46 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
contract UnstakeTest is StakeManagerTest {
|
|
|
|
function test_RevertWhen_SenderIsNotVault() public {
|
2024-11-07 10:50:13 -03:00
|
|
|
vm.expectRevert(ITrustedCodehashAccess.TrustedCodehashAccess__UnauthorizedCodehash.selector);
|
2024-02-18 18:37:15 -03:00
|
|
|
stakeManager.unstake(1);
|
2023-10-10 15:32:46 +02:00
|
|
|
}
|
2023-10-10 15:44:16 +02:00
|
|
|
|
|
|
|
function test_RevertWhen_FundsLocked() public {
|
2023-12-06 12:10:07 +01:00
|
|
|
uint256 lockTime = stakeManager.MIN_LOCKUP_PERIOD();
|
2024-02-28 00:03:35 -03:00
|
|
|
uint256 stakeAmount = 100;
|
|
|
|
uint256 mintAmount = stakeAmount * 10;
|
|
|
|
StakeVault userVault = _createStakingAccount(testUser, stakeAmount, lockTime, mintAmount);
|
2023-10-10 15:44:16 +02:00
|
|
|
|
2024-02-28 00:03:35 -03:00
|
|
|
vm.prank(testUser);
|
2024-11-07 10:50:13 -03:00
|
|
|
vm.expectRevert(IStakeManager.StakeManager__FundsLocked.selector);
|
2024-02-18 18:37:15 -03:00
|
|
|
userVault.unstake(1);
|
2024-02-28 00:03:35 -03:00
|
|
|
|
|
|
|
vm.prank(testUser);
|
2024-11-07 10:50:13 -03:00
|
|
|
vm.expectRevert(IStakeManager.StakeManager__FundsLocked.selector);
|
2024-02-28 00:03:35 -03:00
|
|
|
userVault.unstake(stakeAmount);
|
2023-10-10 15:44:16 +02:00
|
|
|
}
|
2024-01-11 14:54:39 +01:00
|
|
|
|
2024-02-28 00:03:35 -03:00
|
|
|
function test_UnstakeShouldReturnFund_NoLockUp() public {
|
|
|
|
uint256 lockTime = 0;
|
|
|
|
uint256 stakeAmount = 100;
|
|
|
|
uint256 mintAmount = stakeAmount * 10;
|
|
|
|
StakeVault userVault = _createStakingAccount(testUser, stakeAmount, lockTime, mintAmount);
|
|
|
|
assertEq(ERC20(stakeToken).balanceOf(testUser), 900);
|
2024-01-11 14:54:39 +01:00
|
|
|
|
2024-02-28 00:03:35 -03:00
|
|
|
vm.prank(testUser);
|
|
|
|
userVault.unstake(100);
|
|
|
|
|
2024-11-07 10:50:13 -03:00
|
|
|
assertEq(stakeManager.totalStaked(), 0);
|
2024-02-28 00:03:35 -03:00
|
|
|
assertEq(ERC20(stakeToken).balanceOf(address(userVault)), 0);
|
|
|
|
assertEq(ERC20(stakeToken).balanceOf(testUser), 1000);
|
|
|
|
}
|
2024-01-11 14:54:39 +01:00
|
|
|
|
2024-02-28 00:03:35 -03:00
|
|
|
function test_UnstakeShouldReturnFund_WithLockUp() public {
|
|
|
|
uint256 lockTime = stakeManager.MIN_LOCKUP_PERIOD();
|
|
|
|
uint256 stakeAmount = 100;
|
|
|
|
uint256 mintAmount = stakeAmount * 10;
|
|
|
|
StakeVault userVault = _createStakingAccount(testUser, stakeAmount, lockTime, mintAmount);
|
2024-01-11 14:54:39 +01:00
|
|
|
assertEq(ERC20(stakeToken).balanceOf(testUser), 900);
|
|
|
|
|
2024-02-28 00:03:35 -03:00
|
|
|
vm.warp(block.timestamp + lockTime + 1);
|
|
|
|
|
|
|
|
vm.prank(testUser);
|
2024-01-11 14:54:39 +01:00
|
|
|
userVault.unstake(100);
|
|
|
|
|
2024-11-07 10:50:13 -03:00
|
|
|
assertEq(stakeManager.totalStaked(), 0);
|
2024-01-11 14:54:39 +01:00
|
|
|
assertEq(ERC20(stakeToken).balanceOf(address(userVault)), 0);
|
|
|
|
assertEq(ERC20(stakeToken).balanceOf(testUser), 1000);
|
|
|
|
}
|
2024-02-24 16:35:34 -03:00
|
|
|
|
|
|
|
function test_UnstakeShouldBurnMultiplierPoints() public {
|
|
|
|
uint256 percentToBurn = 90;
|
2024-02-28 00:03:35 -03:00
|
|
|
uint256 stakeAmount = 100;
|
|
|
|
StakeVault userVault = _createStakingAccount(testUser, stakeAmount);
|
2024-02-24 16:35:34 -03:00
|
|
|
|
|
|
|
vm.startPrank(testUser);
|
|
|
|
|
2024-11-07 10:50:13 -03:00
|
|
|
assertEq(stakeManager.totalMP(), stakeAmount);
|
2024-02-24 16:35:34 -03:00
|
|
|
for (uint256 i = 0; i < 53; i++) {
|
|
|
|
vm.warp(stakeManager.epochEnd());
|
|
|
|
stakeManager.executeAccount(address(userVault), i + 1);
|
|
|
|
}
|
2024-09-04 09:54:45 +02:00
|
|
|
(, uint256 balanceBefore, uint256 bonusMPBefore, uint256 totalMPBefore,,,,) =
|
2024-02-24 16:35:34 -03:00
|
|
|
stakeManager.accounts(address(userVault));
|
2024-11-07 10:50:13 -03:00
|
|
|
uint256 totalSupplyMPBefore = stakeManager.totalMP();
|
2024-02-24 16:35:34 -03:00
|
|
|
uint256 unstakeAmount = stakeAmount * percentToBurn / 100;
|
|
|
|
console.log("unstake", unstakeAmount);
|
|
|
|
|
|
|
|
assertEq(ERC20(stakeToken).balanceOf(testUser), 0);
|
|
|
|
userVault.unstake(unstakeAmount);
|
2024-09-04 09:54:45 +02:00
|
|
|
(, uint256 balanceAfter, uint256 bonusMPAfter, uint256 totalMPAfter,,,,) =
|
2024-02-24 16:35:34 -03:00
|
|
|
stakeManager.accounts(address(userVault));
|
|
|
|
|
2024-11-07 10:50:13 -03:00
|
|
|
uint256 totalSupplyMPAfter = stakeManager.totalMP();
|
2024-02-24 16:35:34 -03:00
|
|
|
console.log("totalSupplyMPBefore", totalSupplyMPBefore);
|
|
|
|
console.log("totalSupplyMPAfter", totalSupplyMPAfter);
|
|
|
|
console.log("balanceBefore", balanceBefore);
|
|
|
|
console.log("balanceAfter", balanceAfter);
|
2024-06-25 12:48:23 +02:00
|
|
|
console.log("bonusMPBefore", bonusMPBefore);
|
|
|
|
console.log("bonusMPAfter", bonusMPAfter);
|
|
|
|
console.log("totalMPBefore", totalMPBefore);
|
|
|
|
console.log("totalMPAfter", totalMPAfter);
|
2024-02-24 16:35:34 -03:00
|
|
|
|
|
|
|
assertEq(balanceAfter, balanceBefore - (balanceBefore * percentToBurn / 100));
|
2024-06-25 12:48:23 +02:00
|
|
|
assertEq(bonusMPAfter, bonusMPBefore - (bonusMPBefore * percentToBurn / 100));
|
|
|
|
assertEq(totalMPAfter, totalMPBefore - (totalMPBefore * percentToBurn / 100));
|
|
|
|
assertEq(totalSupplyMPAfter, totalSupplyMPBefore - (totalMPBefore * percentToBurn / 100));
|
2024-02-24 16:35:34 -03:00
|
|
|
assertEq(ERC20(stakeToken).balanceOf(testUser), unstakeAmount);
|
|
|
|
}
|
2024-03-04 12:11:23 -03:00
|
|
|
|
|
|
|
function test_RevertWhen_AmountMoreThanBalance() public {
|
2024-09-04 09:54:45 +02:00
|
|
|
uint256 stakeAmount = 1000;
|
2024-03-04 12:11:23 -03:00
|
|
|
StakeVault userVault = _createStakingAccount(testUser, stakeAmount);
|
|
|
|
vm.startPrank(testUser);
|
2024-11-07 10:50:13 -03:00
|
|
|
vm.expectRevert(IStakeManager.StakeManager__InsufficientFunds.selector);
|
2024-03-04 12:11:23 -03:00
|
|
|
userVault.unstake(stakeAmount + 1);
|
|
|
|
}
|
2023-10-10 15:32:46 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
contract LockTest is StakeManagerTest {
|
|
|
|
function test_RevertWhen_SenderIsNotVault() public {
|
2024-11-07 10:50:13 -03:00
|
|
|
vm.expectRevert(ITrustedCodehashAccess.TrustedCodehashAccess__UnauthorizedCodehash.selector);
|
2023-10-10 15:32:46 +02:00
|
|
|
stakeManager.lock(100);
|
|
|
|
}
|
2023-10-10 15:49:34 +02:00
|
|
|
|
2024-03-04 23:00:06 -03:00
|
|
|
function test_NewLockupPeriod() public {
|
|
|
|
StakeVault userVault = _createStakingAccount(testUser, 1000);
|
2023-12-06 12:10:07 +01:00
|
|
|
|
2024-03-04 23:00:06 -03:00
|
|
|
uint256 lockTime = stakeManager.MAX_LOCKUP_PERIOD();
|
2023-12-06 12:10:07 +01:00
|
|
|
vm.startPrank(testUser);
|
2024-03-04 23:00:06 -03:00
|
|
|
userVault.lock(lockTime);
|
|
|
|
|
2024-09-04 09:54:45 +02:00
|
|
|
(, uint256 balance, uint256 bonusMP, uint256 totalMP,,,,) = stakeManager.accounts(address(userVault));
|
2024-03-04 23:00:06 -03:00
|
|
|
|
|
|
|
console.log("balance", balance);
|
2024-06-25 12:48:23 +02:00
|
|
|
console.log("bonusMP", bonusMP);
|
|
|
|
console.log("totalMP", totalMP);
|
2024-03-04 23:00:06 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
function test_RevertWhen_InvalidNewLockupPeriod() public {
|
|
|
|
StakeVault userVault = _createStakingAccount(testUser, 1000);
|
2023-12-06 12:10:07 +01:00
|
|
|
|
|
|
|
uint256 lockTime = stakeManager.MAX_LOCKUP_PERIOD() + 1;
|
2024-03-04 23:00:06 -03:00
|
|
|
vm.startPrank(testUser);
|
2024-11-07 10:50:13 -03:00
|
|
|
vm.expectRevert(IStakeManager.StakeManager__InvalidLockTime.selector);
|
2024-03-04 23:00:06 -03:00
|
|
|
userVault.lock(lockTime);
|
|
|
|
}
|
|
|
|
|
|
|
|
function test_UpdateLockupPeriod() public {
|
|
|
|
uint256 minLockup = stakeManager.MIN_LOCKUP_PERIOD();
|
|
|
|
StakeVault userVault = _createStakingAccount(testUser, 1000, minLockup, 1000);
|
|
|
|
|
|
|
|
vm.warp(block.timestamp + stakeManager.MIN_LOCKUP_PERIOD() - 1);
|
|
|
|
stakeManager.executeAccount(address(userVault), 1);
|
2024-09-04 09:54:45 +02:00
|
|
|
(, uint256 balance, uint256 bonusMP, uint256 totalMP,, uint256 lockUntil,,) =
|
2024-03-04 23:00:06 -03:00
|
|
|
stakeManager.accounts(address(userVault));
|
|
|
|
|
|
|
|
vm.startPrank(testUser);
|
|
|
|
userVault.lock(minLockup - 1);
|
2024-09-04 09:54:45 +02:00
|
|
|
(, balance, bonusMP, totalMP,, lockUntil,,) = stakeManager.accounts(address(userVault));
|
2024-03-04 23:00:06 -03:00
|
|
|
|
|
|
|
assertEq(lockUntil, block.timestamp + minLockup);
|
|
|
|
|
|
|
|
vm.warp(block.timestamp + stakeManager.MIN_LOCKUP_PERIOD());
|
|
|
|
userVault.lock(minLockup);
|
|
|
|
}
|
|
|
|
|
|
|
|
function test_RevertWhen_InvalidUpdateLockupPeriod() public {
|
|
|
|
uint256 minLockup = stakeManager.MIN_LOCKUP_PERIOD();
|
|
|
|
StakeVault userVault = _createStakingAccount(testUser, 1000, minLockup, 1000);
|
|
|
|
|
|
|
|
vm.warp(block.timestamp + stakeManager.MIN_LOCKUP_PERIOD());
|
|
|
|
stakeManager.executeAccount(address(userVault), 1);
|
|
|
|
|
2024-09-04 09:54:45 +02:00
|
|
|
(,,,,, uint256 lockUntil,,) = stakeManager.accounts(address(userVault));
|
2024-03-04 23:00:06 -03:00
|
|
|
console.log(lockUntil);
|
|
|
|
vm.startPrank(testUser);
|
2024-11-07 10:50:13 -03:00
|
|
|
vm.expectRevert(IStakeManager.StakeManager__InvalidLockTime.selector);
|
2024-03-04 23:00:06 -03:00
|
|
|
userVault.lock(minLockup - 1);
|
|
|
|
}
|
|
|
|
|
2024-06-25 12:48:23 +02:00
|
|
|
function test_ShouldIncreaseBonusMP() public {
|
2024-03-04 23:00:06 -03:00
|
|
|
uint256 stakeAmount = 100;
|
|
|
|
uint256 lockTime = stakeManager.MAX_LOCKUP_PERIOD();
|
|
|
|
StakeVault userVault = _createStakingAccount(testUser, stakeAmount);
|
2024-09-04 09:54:45 +02:00
|
|
|
(, uint256 balance, uint256 bonusMP, uint256 totalMP,,,,) = stakeManager.accounts(address(userVault));
|
2024-11-07 10:50:13 -03:00
|
|
|
uint256 totalSupplyMPBefore = stakeManager.totalMP();
|
2024-03-04 23:00:06 -03:00
|
|
|
|
|
|
|
vm.startPrank(testUser);
|
|
|
|
userVault.lock(lockTime);
|
|
|
|
|
2024-09-04 09:54:45 +02:00
|
|
|
//solhint-disable-next-line max-line-length
|
|
|
|
(, uint256 newBalance, uint256 newBonusMP, uint256 newCurrentMP,,,,) = stakeManager.accounts(address(userVault));
|
2024-11-07 10:50:13 -03:00
|
|
|
uint256 totalSupplyMPAfter = stakeManager.totalMP();
|
|
|
|
assertGt(totalSupplyMPAfter, totalSupplyMPBefore, "totalMP");
|
2024-06-25 12:48:23 +02:00
|
|
|
assertGt(newBonusMP, bonusMP, "bonusMP");
|
|
|
|
assertGt(newCurrentMP, totalMP, "totalMP");
|
2024-03-04 23:00:06 -03:00
|
|
|
assertEq(newBalance, balance, "balance");
|
2023-12-06 12:10:07 +01:00
|
|
|
}
|
2023-10-10 15:32:46 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
contract LeaveTest is StakeManagerTest {
|
|
|
|
function test_RevertWhen_SenderIsNotVault() public {
|
2024-11-07 10:50:13 -03:00
|
|
|
vm.expectRevert(ITrustedCodehashAccess.TrustedCodehashAccess__UnauthorizedCodehash.selector);
|
|
|
|
stakeManager.leave();
|
2023-10-10 15:32:46 +02:00
|
|
|
}
|
2023-10-10 15:49:34 +02:00
|
|
|
|
|
|
|
function test_RevertWhen_NoPendingMigration() public {
|
2024-02-28 00:03:35 -03:00
|
|
|
uint256 lockTime = 0;
|
|
|
|
uint256 stakeAmount = 100;
|
|
|
|
uint256 mintAmount = stakeAmount * 10;
|
|
|
|
StakeVault userVault = _createStakingAccount(testUser, stakeAmount, lockTime, mintAmount);
|
2024-02-18 18:37:15 -03:00
|
|
|
vm.startPrank(testUser);
|
|
|
|
|
2023-10-10 15:49:34 +02:00
|
|
|
vm.expectRevert(StakeManager.StakeManager__NoPendingMigration.selector);
|
2024-01-24 16:07:23 +01:00
|
|
|
userVault.acceptMigration();
|
2024-02-18 18:37:15 -03:00
|
|
|
|
2024-01-24 16:07:23 +01:00
|
|
|
vm.expectRevert(StakeManager.StakeManager__NoPendingMigration.selector);
|
2023-10-10 15:49:34 +02:00
|
|
|
userVault.leave();
|
2024-02-18 18:37:15 -03:00
|
|
|
vm.stopPrank();
|
2023-10-10 15:49:34 +02:00
|
|
|
}
|
2023-10-10 15:32:46 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
contract MigrateTest is StakeManagerTest {
|
|
|
|
function test_RevertWhen_SenderIsNotVault() public {
|
2024-11-07 10:50:13 -03:00
|
|
|
vm.expectRevert(ITrustedCodehashAccess.TrustedCodehashAccess__UnauthorizedCodehash.selector);
|
|
|
|
stakeManager.acceptUpdate();
|
2023-10-10 15:32:46 +02:00
|
|
|
}
|
2023-10-10 15:49:34 +02:00
|
|
|
|
|
|
|
function test_RevertWhen_NoPendingMigration() public {
|
2024-02-28 00:03:35 -03:00
|
|
|
uint256 lockTime = 0;
|
|
|
|
uint256 stakeAmount = 100;
|
|
|
|
uint256 mintAmount = stakeAmount * 10;
|
|
|
|
StakeVault userVault = _createStakingAccount(testUser, stakeAmount, lockTime, mintAmount);
|
2024-02-18 18:37:15 -03:00
|
|
|
vm.startPrank(testUser);
|
|
|
|
|
2023-10-10 15:49:34 +02:00
|
|
|
vm.expectRevert(StakeManager.StakeManager__NoPendingMigration.selector);
|
2024-01-24 16:07:23 +01:00
|
|
|
userVault.acceptMigration();
|
2024-02-18 18:37:15 -03:00
|
|
|
vm.stopPrank();
|
2023-10-10 15:49:34 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-03-01 10:56:19 +01:00
|
|
|
contract MigrationInitializeTest is StakeManagerTest {
|
|
|
|
function setUp() public override {
|
|
|
|
StakeManagerTest.setUp();
|
|
|
|
}
|
|
|
|
|
|
|
|
function test_RevertWhen_MigrationPending() public {
|
|
|
|
// first, create 2nd and 3rd generation stake manager
|
|
|
|
vm.startPrank(deployer);
|
|
|
|
StakeManager secondStakeManager = new StakeManager(stakeToken, address(stakeManager));
|
|
|
|
StakeManager thirdStakeManager = new StakeManager(stakeToken, address(secondStakeManager));
|
2024-09-04 09:54:45 +02:00
|
|
|
vm.stopPrank();
|
2024-03-01 10:56:19 +01:00
|
|
|
|
|
|
|
// first, ensure `secondStakeManager` is in migration mode itself
|
2024-09-28 15:21:43 -03:00
|
|
|
ExpiredStakeStorage db = stakeManager.expiredStakeStorage();
|
2024-09-04 09:54:45 +02:00
|
|
|
vm.prank(address(stakeManager));
|
|
|
|
db.transferOwnership(address(secondStakeManager));
|
|
|
|
|
|
|
|
vm.prank(address(deployer));
|
2024-03-01 10:56:19 +01:00
|
|
|
secondStakeManager.startMigration(thirdStakeManager);
|
|
|
|
|
|
|
|
uint256 currentEpoch = stakeManager.currentEpoch();
|
2024-11-07 10:50:13 -03:00
|
|
|
uint256 totalMP = stakeManager.totalMP();
|
|
|
|
uint256 totalBalance = stakeManager.totalStaked();
|
2024-03-01 10:56:19 +01:00
|
|
|
|
|
|
|
// `stakeManager` calling `migrationInitialize` while the new stake manager is
|
|
|
|
// in migration itself, should revert
|
|
|
|
vm.prank(address(stakeManager));
|
|
|
|
vm.expectRevert(StakeManager.StakeManager__PendingMigration.selector);
|
2024-09-04 09:54:45 +02:00
|
|
|
secondStakeManager.migrationInitialize(currentEpoch, totalMP, totalBalance, 0, 0, 0, 0);
|
2024-03-01 10:56:19 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-10-10 15:49:34 +02:00
|
|
|
contract ExecuteAccountTest is StakeManagerTest {
|
2024-02-24 16:35:34 -03:00
|
|
|
StakeVault[] private userVaults;
|
|
|
|
|
2023-10-10 15:49:34 +02:00
|
|
|
function test_RevertWhen_InvalidLimitEpoch() public {
|
2023-12-06 12:10:07 +01:00
|
|
|
uint256 lockTime = stakeManager.MIN_LOCKUP_PERIOD();
|
2024-02-28 00:03:35 -03:00
|
|
|
uint256 stakeAmount = 100;
|
|
|
|
uint256 mintAmount = stakeAmount * 10;
|
|
|
|
StakeVault userVault = _createStakingAccount(testUser, stakeAmount, lockTime, mintAmount);
|
|
|
|
vm.startPrank(testUser);
|
2023-10-10 15:49:34 +02:00
|
|
|
|
|
|
|
uint256 currentEpoch = stakeManager.currentEpoch();
|
|
|
|
|
2024-09-25 12:53:59 -03:00
|
|
|
vm.expectRevert(StakeManager.StakeManager__InvalidLimitEpoch.selector);
|
|
|
|
stakeManager.executeEpoch(currentEpoch + 1);
|
|
|
|
|
|
|
|
vm.expectRevert(StakeManager.StakeManager__InvalidLimitEpoch.selector);
|
|
|
|
stakeManager.executeAccount(address(userVault), currentEpoch + 1);
|
|
|
|
|
|
|
|
vm.warp(stakeManager.epochEnd() - 1);
|
|
|
|
|
|
|
|
vm.expectRevert(StakeManager.StakeManager__InvalidLimitEpoch.selector);
|
|
|
|
stakeManager.executeEpoch(currentEpoch + 1);
|
|
|
|
|
|
|
|
vm.expectRevert(StakeManager.StakeManager__InvalidLimitEpoch.selector);
|
|
|
|
stakeManager.executeAccount(address(userVault), currentEpoch + 1);
|
|
|
|
|
|
|
|
vm.warp(stakeManager.epochEnd());
|
|
|
|
|
|
|
|
stakeManager.executeAccount(address(userVault), currentEpoch + 1);
|
|
|
|
stakeManager.executeEpoch(currentEpoch + 1);
|
|
|
|
|
|
|
|
currentEpoch++;
|
|
|
|
|
|
|
|
vm.expectRevert(StakeManager.StakeManager__InvalidLimitEpoch.selector);
|
|
|
|
stakeManager.executeEpoch(currentEpoch + 1);
|
|
|
|
|
|
|
|
vm.expectRevert(StakeManager.StakeManager__InvalidLimitEpoch.selector);
|
|
|
|
stakeManager.executeAccount(address(userVault), currentEpoch + 1);
|
|
|
|
|
|
|
|
vm.warp(stakeManager.epochEnd() + stakeManager.EPOCH_SIZE() - 1);
|
|
|
|
|
|
|
|
vm.expectRevert(StakeManager.StakeManager__InvalidLimitEpoch.selector);
|
|
|
|
stakeManager.executeEpoch(currentEpoch + 2);
|
|
|
|
|
|
|
|
vm.expectRevert(StakeManager.StakeManager__InvalidLimitEpoch.selector);
|
|
|
|
stakeManager.executeAccount(address(userVault), currentEpoch + 2);
|
|
|
|
|
|
|
|
stakeManager.executeAccount(address(userVault), currentEpoch + 1);
|
|
|
|
stakeManager.executeEpoch(currentEpoch + 1);
|
|
|
|
|
|
|
|
currentEpoch++;
|
|
|
|
|
|
|
|
vm.expectRevert(StakeManager.StakeManager__InvalidLimitEpoch.selector);
|
|
|
|
stakeManager.executeEpoch(currentEpoch + 1);
|
|
|
|
|
2023-10-10 15:49:34 +02:00
|
|
|
vm.expectRevert(StakeManager.StakeManager__InvalidLimitEpoch.selector);
|
|
|
|
stakeManager.executeAccount(address(userVault), currentEpoch + 1);
|
|
|
|
}
|
2024-02-24 16:35:34 -03:00
|
|
|
|
2024-09-11 08:55:51 -03:00
|
|
|
//Stake, pass epoch, execute epoch, passEpoch, executeEpoch, executeAccount, verify amount of MPs, totalMPs, etc.
|
|
|
|
function test_ExecuteAccountLimit() public {
|
|
|
|
uint256 stakeAmount = 10_000_000;
|
|
|
|
deal(stakeToken, testUser, stakeAmount);
|
|
|
|
userVaults.push(_createStakingAccount(makeAddr("testUser"), stakeAmount, 0));
|
|
|
|
|
|
|
|
(,,, uint256 totalMP, uint256 lastMint,, uint256 epoch,) = stakeManager.accounts(address(userVaults[0]));
|
|
|
|
vm.warp(stakeManager.epochEnd());
|
|
|
|
stakeManager.executeEpoch();
|
|
|
|
vm.warp(stakeManager.epochEnd());
|
|
|
|
|
|
|
|
//expected MP is, the starting totalMP + the calculatedMPToMint of user balance for one EPOCH_SIZE multiplied by
|
|
|
|
// 2.
|
2024-11-07 10:50:13 -03:00
|
|
|
uint256 expectedMP = totalMP + (stakeManager.calculateMP(stakeAmount, stakeManager.EPOCH_SIZE()) * 2);
|
2024-09-11 08:55:51 -03:00
|
|
|
stakeManager.executeAccount(address(userVaults[0]), stakeManager.currentEpoch() + 1);
|
|
|
|
(,,, totalMP, lastMint,, epoch,) = stakeManager.accounts(address(userVaults[0]));
|
|
|
|
|
|
|
|
assertEq(totalMP, expectedMP, "totalMP is not what the contract itself predicted");
|
|
|
|
}
|
|
|
|
|
2024-02-24 16:35:34 -03:00
|
|
|
function test_ExecuteAccountMintMP() public {
|
|
|
|
uint256 stakeAmount = 10_000_000;
|
|
|
|
deal(stakeToken, testUser, stakeAmount);
|
|
|
|
|
|
|
|
userVaults.push(_createStakingAccount(makeAddr("testUser"), stakeAmount, 0));
|
|
|
|
userVaults.push(_createStakingAccount(makeAddr("testUser2"), stakeAmount, 0));
|
|
|
|
userVaults.push(_createStakingAccount(makeAddr("testUser3"), stakeAmount, 0));
|
|
|
|
|
|
|
|
console.log("######### NOW", block.timestamp);
|
|
|
|
console.log("# START EPOCH", stakeManager.currentEpoch());
|
|
|
|
console.log("# PND_REWARDS", stakeManager.pendingReward());
|
|
|
|
|
|
|
|
for (uint256 i = 0; i < 3; i++) {
|
|
|
|
deal(stakeToken, address(stakeManager), 100 ether);
|
|
|
|
vm.warp(stakeManager.epochEnd());
|
|
|
|
console.log("######### NOW", block.timestamp);
|
|
|
|
stakeManager.executeEpoch();
|
|
|
|
console.log("##### NEW EPOCH", stakeManager.currentEpoch());
|
|
|
|
console.log("# PND_REWARDS", stakeManager.pendingReward());
|
|
|
|
|
|
|
|
for (uint256 j = 0; j < userVaults.length; j++) {
|
2024-09-04 09:54:45 +02:00
|
|
|
//solhint-disable-next-line max-line-length
|
|
|
|
(address rewardAddress,,, uint256 totalMPBefore, uint256 lastMintBefore,, uint256 epochBefore,) =
|
2024-02-24 16:35:34 -03:00
|
|
|
stakeManager.accounts(address(userVaults[j]));
|
|
|
|
uint256 rewardsBefore = ERC20(stakeToken).balanceOf(rewardAddress);
|
|
|
|
console.log("-Vault number", j);
|
|
|
|
console.log("--=====BEFORE=====");
|
2024-06-25 12:48:23 +02:00
|
|
|
console.log("---### totalMP :", totalMPBefore);
|
2024-02-24 16:35:34 -03:00
|
|
|
console.log("---#### lastMint :", lastMintBefore);
|
|
|
|
console.log("---## user_epoch :", epochBefore);
|
|
|
|
console.log("---##### rewards :", rewardsBefore);
|
|
|
|
console.log("--=====AFTER======");
|
|
|
|
stakeManager.executeAccount(address(userVaults[j]), epochBefore + 1);
|
2024-09-04 09:54:45 +02:00
|
|
|
//solhint-disable-next-line max-line-length
|
|
|
|
(,,, uint256 totalMP, uint256 lastMint,, uint256 epoch,) = stakeManager.accounts(address(userVaults[j]));
|
2024-02-24 16:35:34 -03:00
|
|
|
uint256 rewards = ERC20(stakeToken).balanceOf(rewardAddress);
|
|
|
|
console.log("---### deltaTime :", lastMint - lastMintBefore);
|
2024-06-25 12:48:23 +02:00
|
|
|
console.log("---### totalMP :", totalMP);
|
2024-02-24 16:35:34 -03:00
|
|
|
console.log("---#### lastMint :", lastMint);
|
|
|
|
console.log("---## user_epoch :", epoch);
|
|
|
|
console.log("---##### rewards :", rewards);
|
|
|
|
console.log("--=======#=======");
|
|
|
|
console.log("--# TOTAL_SUPPLY", stakeManager.totalSupply());
|
|
|
|
console.log("--# PND_REWARDS", stakeManager.pendingReward());
|
|
|
|
assertEq(lastMint, lastMintBefore + stakeManager.EPOCH_SIZE(), "must increaase lastMint");
|
|
|
|
assertEq(epoch, epochBefore + 1, "must increase epoch");
|
2024-06-25 12:48:23 +02:00
|
|
|
assertGt(totalMP, totalMPBefore, "must increase MPs");
|
2024-02-24 16:35:34 -03:00
|
|
|
assertGt(rewards, rewardsBefore, "must increase rewards");
|
|
|
|
lastMintBefore = lastMint;
|
|
|
|
epochBefore = epoch;
|
2024-06-25 12:48:23 +02:00
|
|
|
totalMPBefore = totalMP;
|
2024-02-24 16:35:34 -03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2024-02-28 00:03:35 -03:00
|
|
|
|
2024-03-04 10:47:09 -03:00
|
|
|
function test_ShouldNotMintMoreThanCap() public {
|
2024-09-04 09:54:45 +02:00
|
|
|
uint256 stakeAmount = 10_000_000_000;
|
|
|
|
|
2024-11-07 10:50:13 -03:00
|
|
|
uint256 epochsAmountToReachCap = stakeManager.calculateMP(
|
|
|
|
stakeAmount, stakeManager.MAX_MULTIPLIER() * stakeManager.YEAR()
|
|
|
|
) / stakeManager.calculateMP(stakeAmount, stakeManager.EPOCH_SIZE());
|
2024-09-04 09:54:45 +02:00
|
|
|
|
2024-03-04 10:47:09 -03:00
|
|
|
deal(stakeToken, testUser, stakeAmount);
|
|
|
|
|
|
|
|
userVaults.push(_createStakingAccount(makeAddr("testUser"), stakeAmount, 0));
|
|
|
|
|
2024-09-04 09:54:45 +02:00
|
|
|
vm.warp(stakeManager.epochEnd() - (stakeManager.EPOCH_SIZE() - 1));
|
|
|
|
userVaults.push(_createStakingAccount(makeAddr("testUser2"), stakeAmount, 0));
|
|
|
|
|
|
|
|
vm.warp(stakeManager.epochEnd() - (stakeManager.EPOCH_SIZE() - 2));
|
|
|
|
userVaults.push(_createStakingAccount(makeAddr("testUser3"), stakeAmount, 0));
|
|
|
|
|
|
|
|
vm.warp(stakeManager.epochEnd() - ((stakeManager.EPOCH_SIZE() / 4) * 3));
|
|
|
|
userVaults.push(_createStakingAccount(makeAddr("testUser4"), stakeAmount, 0));
|
|
|
|
|
|
|
|
vm.warp(stakeManager.epochEnd() - ((stakeManager.EPOCH_SIZE() / 4) * 2));
|
|
|
|
userVaults.push(_createStakingAccount(makeAddr("testUser5"), stakeAmount, 0));
|
|
|
|
|
|
|
|
vm.warp(stakeManager.epochEnd() - ((stakeManager.EPOCH_SIZE() / 4) * 1));
|
|
|
|
userVaults.push(_createStakingAccount(makeAddr("testUser6"), stakeAmount, 0));
|
|
|
|
|
|
|
|
vm.warp(stakeManager.epochEnd() - 2);
|
|
|
|
userVaults.push(_createStakingAccount(makeAddr("testUser7"), stakeAmount, 0));
|
|
|
|
|
|
|
|
vm.warp(stakeManager.epochEnd() - 1);
|
|
|
|
userVaults.push(_createStakingAccount(makeAddr("testUser8"), stakeAmount, 0));
|
|
|
|
|
|
|
|
for (uint256 i = 0; i <= epochsAmountToReachCap; i++) {
|
2024-03-04 10:47:09 -03:00
|
|
|
deal(stakeToken, address(stakeManager), 100 ether);
|
|
|
|
vm.warp(stakeManager.epochEnd());
|
|
|
|
stakeManager.executeEpoch();
|
|
|
|
for (uint256 j = 0; j < userVaults.length; j++) {
|
2024-09-04 09:54:45 +02:00
|
|
|
(address rewardAddress,,, uint256 totalMPBefore, uint256 lastMintBefore,, uint256 epochBefore,) =
|
2024-03-04 10:47:09 -03:00
|
|
|
stakeManager.accounts(address(userVaults[j]));
|
|
|
|
uint256 rewardsBefore = ERC20(stakeToken).balanceOf(rewardAddress);
|
|
|
|
|
|
|
|
stakeManager.executeAccount(address(userVaults[j]), epochBefore + 1);
|
2024-09-04 09:54:45 +02:00
|
|
|
//solhint-disable-next-line max-line-length
|
|
|
|
(,,, uint256 totalMP, uint256 lastMint,, uint256 epoch,) = stakeManager.accounts(address(userVaults[j]));
|
2024-03-04 10:47:09 -03:00
|
|
|
uint256 rewards = ERC20(stakeToken).balanceOf(rewardAddress);
|
|
|
|
assertEq(epoch, epochBefore + 1, "must increase epoch");
|
2024-06-25 12:48:23 +02:00
|
|
|
assertGt(totalMP, totalMPBefore, "must increase MPs");
|
2024-03-04 10:47:09 -03:00
|
|
|
assertGt(rewards, rewardsBefore, "must increase rewards");
|
|
|
|
lastMintBefore = lastMint;
|
|
|
|
epochBefore = epoch;
|
2024-06-25 12:48:23 +02:00
|
|
|
totalMPBefore = totalMP;
|
2024-03-04 10:47:09 -03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (uint256 i = 0; i < 100; i++) {
|
|
|
|
deal(stakeToken, address(stakeManager), 100 ether);
|
|
|
|
vm.warp(stakeManager.epochEnd());
|
|
|
|
stakeManager.executeEpoch();
|
|
|
|
for (uint256 j = 0; j < userVaults.length; j++) {
|
2024-09-04 09:54:45 +02:00
|
|
|
(address rewardAddress,,, uint256 totalMPBefore, uint256 lastMintBefore,, uint256 epochBefore,) =
|
2024-03-04 10:47:09 -03:00
|
|
|
stakeManager.accounts(address(userVaults[j]));
|
|
|
|
uint256 rewardsBefore = ERC20(stakeToken).balanceOf(rewardAddress);
|
|
|
|
|
|
|
|
stakeManager.executeAccount(address(userVaults[j]), epochBefore + 1);
|
2024-09-04 09:54:45 +02:00
|
|
|
//solhint-disable-next-line max-line-length
|
|
|
|
(,,, uint256 totalMP, uint256 lastMint,, uint256 epoch,) = stakeManager.accounts(address(userVaults[j]));
|
2024-03-04 10:47:09 -03:00
|
|
|
uint256 rewards = ERC20(stakeToken).balanceOf(rewardAddress);
|
|
|
|
assertEq(lastMint, lastMintBefore + stakeManager.EPOCH_SIZE(), "must increaase lastMint");
|
|
|
|
assertEq(epoch, epochBefore + 1, "must increase epoch");
|
2024-09-04 09:54:45 +02:00
|
|
|
// MPs will still be minted in mpLimitEpoch + 1 when accounts
|
|
|
|
// started staking at any point *inside* of an epoch, so we
|
|
|
|
// only perform the assert below one epoch after *that*
|
|
|
|
if (i > 0) {
|
|
|
|
assertEq(totalMP, totalMPBefore, "must NOT increase MPs");
|
|
|
|
}
|
2024-03-04 10:47:09 -03:00
|
|
|
assertGt(rewards, rewardsBefore, "must increase rewards");
|
|
|
|
lastMintBefore = lastMint;
|
|
|
|
epochBefore = epoch;
|
2024-06-25 12:48:23 +02:00
|
|
|
totalMPBefore = totalMP;
|
2024-03-04 10:47:09 -03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2023-10-10 15:32:46 +02:00
|
|
|
}
|
2024-01-12 13:07:31 +01:00
|
|
|
|
|
|
|
contract UserFlowsTest is StakeManagerTest {
|
2024-09-04 09:54:45 +02:00
|
|
|
StakeVault[] private userVaults;
|
|
|
|
|
2024-01-12 13:07:31 +01:00
|
|
|
function test_StakedSupplyShouldIncreaseAndDecreaseAgain() public {
|
2024-02-28 00:03:35 -03:00
|
|
|
uint256 lockTime = 0;
|
|
|
|
uint256 stakeAmount = 100;
|
|
|
|
uint256 mintAmount = stakeAmount * 10;
|
2024-01-12 13:07:31 +01:00
|
|
|
|
2024-02-28 00:03:35 -03:00
|
|
|
StakeVault userVault = _createStakingAccount(testUser, stakeAmount, lockTime, mintAmount);
|
|
|
|
StakeVault user2Vault = _createStakingAccount(testUser2, stakeAmount, lockTime, mintAmount);
|
2024-01-12 13:07:31 +01:00
|
|
|
|
|
|
|
assertEq(ERC20(stakeToken).balanceOf(address(userVault)), 100);
|
|
|
|
assertEq(ERC20(stakeToken).balanceOf(address(user2Vault)), 100);
|
2024-11-07 10:50:13 -03:00
|
|
|
assertEq(stakeManager.totalStaked(), 200);
|
2024-01-12 13:07:31 +01:00
|
|
|
|
|
|
|
vm.startPrank(testUser);
|
|
|
|
userVault.unstake(100);
|
|
|
|
assertEq(ERC20(stakeToken).balanceOf(address(userVault)), 0);
|
2024-11-07 10:50:13 -03:00
|
|
|
assertEq(stakeManager.totalStaked(), 100);
|
2024-01-12 13:07:31 +01:00
|
|
|
|
|
|
|
vm.startPrank(testUser2);
|
|
|
|
user2Vault.unstake(100);
|
|
|
|
assertEq(ERC20(stakeToken).balanceOf(address(user2Vault)), 0);
|
2024-11-07 10:50:13 -03:00
|
|
|
assertEq(stakeManager.totalStaked(), 0);
|
2024-01-12 13:07:31 +01:00
|
|
|
}
|
2024-01-12 13:36:13 +01:00
|
|
|
|
|
|
|
function test_StakeWithLockUpTimeLocksStake() public {
|
2024-02-28 00:03:35 -03:00
|
|
|
uint256 lockTime = stakeManager.MIN_LOCKUP_PERIOD();
|
|
|
|
uint256 stakeAmount = 100;
|
|
|
|
uint256 mintAmount = stakeAmount * 10;
|
|
|
|
StakeVault userVault = _createStakingAccount(testUser, stakeAmount, lockTime, mintAmount);
|
2024-01-12 13:36:13 +01:00
|
|
|
vm.startPrank(testUser);
|
|
|
|
|
|
|
|
// unstaking should fail as lockup time isn't over yet
|
2024-11-07 10:50:13 -03:00
|
|
|
vm.expectRevert(IStakeManager.StakeManager__FundsLocked.selector);
|
2024-01-12 13:36:13 +01:00
|
|
|
userVault.unstake(100);
|
|
|
|
|
|
|
|
// fast forward 12 weeks
|
2024-02-28 00:03:35 -03:00
|
|
|
skip(lockTime + 1);
|
2024-01-12 13:36:13 +01:00
|
|
|
|
|
|
|
userVault.unstake(100);
|
|
|
|
assertEq(ERC20(stakeToken).balanceOf(address(userVault)), 0);
|
2024-11-07 10:50:13 -03:00
|
|
|
assertEq(stakeManager.totalStaked(), 0);
|
2024-01-12 13:36:13 +01:00
|
|
|
}
|
2024-09-04 09:54:45 +02:00
|
|
|
|
2024-09-25 08:22:16 -03:00
|
|
|
function test_PendingMPToBeMintedCannotBeGreaterThanTotalSupplyMP(
|
|
|
|
uint8 randomStakeMultiplier,
|
|
|
|
uint128 randomStakeAddition
|
|
|
|
)
|
|
|
|
public
|
|
|
|
{
|
|
|
|
uint8 accountNum = 5;
|
|
|
|
uint256 minimumPossibleStake = 53; //less than this the stake per epoch of the account would be 0
|
|
|
|
uint256 baseStakeAmount =
|
|
|
|
(minimumPossibleStake * (uint256(randomStakeMultiplier) + 1)) + uint256(randomStakeAddition);
|
|
|
|
uint256 epochsAmountToReachCap = 0;
|
2024-09-04 09:54:45 +02:00
|
|
|
for (uint256 i = 0; i <= accountNum; i++) {
|
2024-09-25 08:22:16 -03:00
|
|
|
uint256 thisAccStake = baseStakeAmount + (i * (10 ^ i * baseStakeAmount)); //changes the amount of stake of
|
|
|
|
// each account so we have mixed amounts of stake
|
2024-09-04 09:54:45 +02:00
|
|
|
userVaults.push(
|
2024-09-25 08:22:16 -03:00
|
|
|
_createStakingAccount(makeAddr(string(abi.encode(keccak256(abi.encode(accountNum))))), thisAccStake, 0)
|
2024-09-04 09:54:45 +02:00
|
|
|
);
|
2024-11-07 10:50:13 -03:00
|
|
|
uint256 thisAccReachCapIn = stakeManager.calculateMP(
|
|
|
|
thisAccStake, stakeManager.MAX_MULTIPLIER() * stakeManager.YEAR()
|
|
|
|
) / stakeManager.calculateMP(thisAccStake, stakeManager.EPOCH_SIZE());
|
2024-09-25 08:22:16 -03:00
|
|
|
if (thisAccReachCapIn > epochsAmountToReachCap) {
|
|
|
|
epochsAmountToReachCap = thisAccReachCapIn; //uses the amount to reach cap from the account that takes
|
|
|
|
// longer to reach cap
|
|
|
|
}
|
2024-09-04 09:54:45 +02:00
|
|
|
}
|
|
|
|
|
2024-11-07 10:50:13 -03:00
|
|
|
//tests up to epochs to reach MAX_MULTIPLIER + 10 epochs
|
2024-09-25 08:22:16 -03:00
|
|
|
for (uint256 i = 0; i < epochsAmountToReachCap + 10; i++) {
|
2024-09-04 09:54:45 +02:00
|
|
|
vm.warp(stakeManager.epochEnd());
|
|
|
|
stakeManager.executeEpoch();
|
2024-11-07 10:50:13 -03:00
|
|
|
uint256 pendingMPToBeMintedBefore = stakeManager.potentialMP();
|
|
|
|
uint256 totalMP = stakeManager.totalMP();
|
2024-09-04 09:54:45 +02:00
|
|
|
for (uint256 j = 0; j < userVaults.length; j++) {
|
|
|
|
(,,, uint256 totalMPBefore, uint256 lastMintBefore,, uint256 epochBefore,) =
|
|
|
|
stakeManager.accounts(address(userVaults[j]));
|
|
|
|
|
|
|
|
stakeManager.executeAccount(address(userVaults[j]), epochBefore + 1);
|
|
|
|
}
|
2024-11-07 10:50:13 -03:00
|
|
|
uint256 pendingMPToBeMintedAfter = stakeManager.potentialMP();
|
2024-09-04 09:54:45 +02:00
|
|
|
|
2024-11-07 10:50:13 -03:00
|
|
|
assertEq(pendingMPToBeMintedBefore + totalMP, stakeManager.totalMP());
|
2024-09-04 09:54:45 +02:00
|
|
|
assertEq(pendingMPToBeMintedAfter, 0);
|
|
|
|
}
|
|
|
|
}
|
2024-01-12 13:07:31 +01:00
|
|
|
}
|
2024-02-28 13:46:50 +01:00
|
|
|
|
2024-03-04 11:50:41 -03:00
|
|
|
contract MigrationStakeManagerTest is StakeManagerTest {
|
2024-02-28 13:46:50 +01:00
|
|
|
StakeManager internal newStakeManager;
|
|
|
|
|
|
|
|
function setUp() public virtual override {
|
|
|
|
super.setUp();
|
|
|
|
DeployMigrationStakeManager deployment = new DeployMigrationStakeManager(address(stakeManager), stakeToken);
|
|
|
|
newStakeManager = deployment.run();
|
|
|
|
}
|
|
|
|
|
|
|
|
function testNewDeployment() public {
|
|
|
|
assertEq(newStakeManager.owner(), deployer);
|
|
|
|
assertEq(newStakeManager.currentEpoch(), 0);
|
|
|
|
assertEq(newStakeManager.pendingReward(), 0);
|
2024-11-07 10:50:13 -03:00
|
|
|
assertEq(newStakeManager.totalMP(), 0);
|
|
|
|
assertEq(newStakeManager.totalStaked(), 0);
|
2024-11-08 00:56:09 -03:00
|
|
|
assertEq(address(newStakeManager.REWARD_TOKEN()), stakeToken);
|
2024-06-19 11:26:03 +02:00
|
|
|
assertEq(address(newStakeManager.previousManager()), address(stakeManager));
|
2024-02-28 13:46:50 +01:00
|
|
|
assertEq(newStakeManager.totalSupply(), 0);
|
|
|
|
}
|
2024-02-28 00:03:35 -03:00
|
|
|
|
|
|
|
function test_ExecuteEpochShouldNotIncreaseEpochInMigration() public {
|
|
|
|
assertEq(stakeManager.currentEpoch(), 0);
|
|
|
|
assertEq(address(stakeManager.migration()), address(0));
|
|
|
|
vm.prank(deployer);
|
2024-03-04 11:53:44 -03:00
|
|
|
|
2024-02-28 00:03:35 -03:00
|
|
|
stakeManager.startMigration(newStakeManager);
|
|
|
|
assertEq(address(stakeManager.migration()), address(newStakeManager));
|
|
|
|
|
|
|
|
vm.warp(stakeManager.epochEnd());
|
|
|
|
vm.expectRevert(StakeManager.StakeManager__PendingMigration.selector);
|
|
|
|
stakeManager.executeEpoch();
|
|
|
|
assertEq(stakeManager.currentEpoch(), 0);
|
|
|
|
}
|
2024-03-04 11:53:44 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
contract ExecuteEpochTest is MigrationStakeManagerTest {
|
2024-09-25 12:53:59 -03:00
|
|
|
function test_ExecuteEpochNewEpoch() public {
|
|
|
|
uint256 firstEpochEnd = stakeManager.epochEnd();
|
|
|
|
assertEq(stakeManager.currentEpoch(), 0, "Epoch not 0 at start of test");
|
|
|
|
assertEq(stakeManager.newEpoch(), 0, "New epoch not 0 at start of test");
|
|
|
|
stakeManager.executeEpoch();
|
|
|
|
assertEq(stakeManager.currentEpoch(), 0, "Epoch should not increase if no time passed since start");
|
|
|
|
|
|
|
|
vm.warp(firstEpochEnd - 1);
|
|
|
|
assertEq(stakeManager.newEpoch(), 0, "New epoch not 0 if 1 second before epoch end");
|
|
|
|
stakeManager.executeEpoch();
|
|
|
|
assertEq(stakeManager.currentEpoch(), 0, "Epoch should not increase if 1 second before epoch end");
|
|
|
|
|
|
|
|
vm.warp(firstEpochEnd);
|
|
|
|
assertEq(stakeManager.newEpoch(), 1, "New epoch should be 1 if exactly at epochEnd");
|
|
|
|
stakeManager.executeEpoch();
|
|
|
|
assertEq(stakeManager.currentEpoch(), 1, "Current epoch should increased to 1 if exactly at epochEnd of 0");
|
|
|
|
|
|
|
|
vm.warp(firstEpochEnd + 1);
|
|
|
|
assertEq(stakeManager.newEpoch(), 1, "New epoch should be 1 if 1 second after epochend of 1");
|
|
|
|
stakeManager.executeEpoch();
|
|
|
|
assertEq(stakeManager.currentEpoch(), 1, "Current epoch should increase if 1 second after epochend of 1");
|
|
|
|
|
|
|
|
vm.warp(firstEpochEnd + (stakeManager.EPOCH_SIZE() * 99));
|
|
|
|
assertEq(stakeManager.newEpoch(), 100);
|
|
|
|
stakeManager.executeEpoch();
|
|
|
|
assertEq(stakeManager.currentEpoch(), 100);
|
|
|
|
}
|
|
|
|
|
2024-09-25 12:45:59 -03:00
|
|
|
function test_ExecuteEpochExecuteEpochAfterEnd() public {
|
|
|
|
StakeVault userVault = _createStakingAccount(makeAddr("testUser"), 100_000, 0);
|
|
|
|
|
|
|
|
vm.warp(stakeManager.epochEnd() + (stakeManager.EPOCH_SIZE() / 2));
|
|
|
|
stakeManager.executeEpoch();
|
|
|
|
stakeManager.executeAccount(address(userVault), stakeManager.currentEpoch());
|
|
|
|
vm.warp(stakeManager.epochEnd());
|
|
|
|
|
|
|
|
stakeManager.executeEpoch();
|
|
|
|
stakeManager.executeAccount(address(userVault), stakeManager.currentEpoch());
|
|
|
|
|
|
|
|
vm.warp(stakeManager.epochEnd() + (stakeManager.EPOCH_SIZE() * 2));
|
|
|
|
stakeManager.executeEpoch();
|
|
|
|
stakeManager.executeAccount(address(userVault), stakeManager.currentEpoch());
|
|
|
|
}
|
|
|
|
|
|
|
|
function test_ExecuteEpochExecuteEpochExecuteAccountAfterManyEpochs() public {
|
|
|
|
StakeVault userVault = _createStakingAccount(makeAddr("testUser"), 100_000, 0);
|
|
|
|
|
|
|
|
for (uint256 i = 0; i < 10; i++) {
|
|
|
|
vm.warp(stakeManager.epochEnd());
|
|
|
|
stakeManager.executeEpoch();
|
|
|
|
}
|
|
|
|
stakeManager.executeAccount(address(userVault), stakeManager.currentEpoch());
|
|
|
|
}
|
|
|
|
|
2024-09-25 12:53:59 -03:00
|
|
|
function test_ExecuteEpochExecuteEpochExecuteAccountAfterManyEpochsJumoMany() public {
|
|
|
|
StakeVault userVault = _createStakingAccount(makeAddr("testUser"), 100_000, 0);
|
|
|
|
|
|
|
|
for (uint256 i = 0; i < 10; i++) {
|
|
|
|
vm.warp(stakeManager.epochEnd());
|
|
|
|
}
|
|
|
|
stakeManager.executeEpoch();
|
|
|
|
stakeManager.executeAccount(address(userVault), stakeManager.currentEpoch());
|
|
|
|
}
|
|
|
|
|
|
|
|
function test_ExecuteEpochExecuteAccountAfterManyEpochsJumoMany() public {
|
|
|
|
StakeVault userVault = _createStakingAccount(makeAddr("testUser"), 100_000, 0);
|
|
|
|
|
|
|
|
for (uint256 i = 0; i < 10; i++) {
|
|
|
|
vm.warp(stakeManager.epochEnd());
|
|
|
|
}
|
|
|
|
stakeManager.executeAccount(address(userVault));
|
|
|
|
}
|
|
|
|
|
2024-09-25 12:45:59 -03:00
|
|
|
function test_ExecuteEpochExecuteEpochExecuteAccountAfterManyEpochsWithBrokenTime() public {
|
|
|
|
StakeVault userVault = _createStakingAccount(makeAddr("testUser"), 100_000, 0);
|
|
|
|
|
|
|
|
for (uint256 i = 0; i < 10; i++) {
|
|
|
|
vm.warp(stakeManager.epochEnd() + (stakeManager.EPOCH_SIZE() / 10 - i));
|
|
|
|
stakeManager.executeEpoch();
|
|
|
|
}
|
|
|
|
stakeManager.executeAccount(address(userVault), stakeManager.currentEpoch());
|
|
|
|
}
|
|
|
|
|
2024-09-25 12:53:59 -03:00
|
|
|
function test_ExecuteEpochExecuteEpochExecuteAccountAfterManyEpochsWithBrokenTimeJumpMany() public {
|
|
|
|
StakeVault userVault = _createStakingAccount(makeAddr("testUser"), 100_000, 0);
|
|
|
|
|
|
|
|
for (uint256 i = 0; i < 10; i++) {
|
|
|
|
vm.warp(stakeManager.epochEnd() + (stakeManager.EPOCH_SIZE() / 10 - i));
|
|
|
|
}
|
|
|
|
stakeManager.executeEpoch();
|
|
|
|
stakeManager.executeAccount(address(userVault), stakeManager.currentEpoch());
|
|
|
|
}
|
|
|
|
|
|
|
|
function test_ExecuteEpochExecuteAccountAfterManyEpochsWithBrokenTimeJumpMany() public {
|
|
|
|
StakeVault userVault = _createStakingAccount(makeAddr("testUser"), 100_000, 0);
|
|
|
|
|
|
|
|
for (uint256 i = 0; i < 10; i++) {
|
|
|
|
vm.warp(stakeManager.epochEnd() + (stakeManager.EPOCH_SIZE() / 10 - i));
|
|
|
|
}
|
|
|
|
stakeManager.executeAccount(address(userVault));
|
|
|
|
}
|
|
|
|
|
2024-09-25 12:45:59 -03:00
|
|
|
function test_ExecuteEpochExecuteAccountAfterEpochEnd() public {
|
|
|
|
StakeVault userVault = _createStakingAccount(makeAddr("testUser"), 100_000, 0);
|
|
|
|
|
|
|
|
vm.warp(stakeManager.epochEnd() + (stakeManager.EPOCH_SIZE() / 2));
|
|
|
|
stakeManager.executeAccount(address(userVault), stakeManager.currentEpoch());
|
|
|
|
|
|
|
|
vm.warp(stakeManager.epochEnd());
|
|
|
|
stakeManager.executeAccount(address(userVault), stakeManager.currentEpoch());
|
|
|
|
|
|
|
|
vm.warp(stakeManager.epochEnd() + (stakeManager.EPOCH_SIZE() * 2));
|
|
|
|
stakeManager.executeAccount(address(userVault), stakeManager.currentEpoch());
|
|
|
|
}
|
|
|
|
|
|
|
|
function test_ExecuteEpochExecuteAccountAfterManyEpochsWithBrokenTime() public {
|
|
|
|
StakeVault userVault = _createStakingAccount(makeAddr("testUser"), 100_000, 0);
|
|
|
|
|
|
|
|
for (uint256 i = 0; i < 10; i++) {
|
|
|
|
vm.warp(stakeManager.epochEnd() + (stakeManager.EPOCH_SIZE() / 10 - i));
|
|
|
|
stakeManager.executeAccount(address(userVault), stakeManager.currentEpoch());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-03-04 11:53:44 -03:00
|
|
|
function test_ExecuteEpochShouldNotIncreaseEpochBeforeEnd() public {
|
|
|
|
assertEq(stakeManager.currentEpoch(), 0);
|
2024-02-28 00:03:35 -03:00
|
|
|
|
2024-03-04 11:53:44 -03:00
|
|
|
vm.warp(stakeManager.epochEnd() - 1);
|
|
|
|
stakeManager.executeEpoch();
|
|
|
|
assertEq(stakeManager.currentEpoch(), 0);
|
|
|
|
}
|
|
|
|
|
2024-02-28 00:03:35 -03:00
|
|
|
function test_ExecuteEpochShouldIncreaseEpoch() public {
|
|
|
|
assertEq(stakeManager.currentEpoch(), 0);
|
|
|
|
|
|
|
|
vm.warp(stakeManager.epochEnd());
|
|
|
|
stakeManager.executeEpoch();
|
|
|
|
assertEq(stakeManager.currentEpoch(), 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
//invariant: stakeManager balanceOf stakeToken > pendingReward
|
|
|
|
function test_ExecuteEpochShouldIncreasePendingReward() public {
|
|
|
|
assertEq(stakeManager.pendingReward(), 0);
|
|
|
|
assertEq(stakeManager.epochReward(), 0);
|
|
|
|
deal(stakeToken, address(stakeManager), 1);
|
|
|
|
assertEq(stakeManager.pendingReward(), 0);
|
|
|
|
assertEq(stakeManager.epochReward(), 1);
|
|
|
|
vm.warp(stakeManager.epochEnd());
|
|
|
|
stakeManager.executeEpoch();
|
|
|
|
assertEq(stakeManager.pendingReward(), 1);
|
|
|
|
assertEq(stakeManager.epochReward(), 0);
|
|
|
|
}
|
|
|
|
}
|