diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml index 4ca28d4..eb694c9 100644 --- a/.github/workflows/ci.yml +++ b/.github/workflows/ci.yml @@ -53,16 +53,15 @@ jobs: # https://twitter.com/PaulRBerg/status/1611116650664796166 - name: Generate fuzz seed with 1 day TTL - run: > - echo "FOUNDRY_FUZZ_SEED=$( - echo $(($EPOCHSECONDS - $EPOCHSECONDS % 86400)) - )" >> $GITHUB_ENV + run: | + echo "FOUNDRY_FUZZ_SEED=$(( $(date +%s) - $(date +%s) % 86400 ))" >> $GITHUB_ENV - name: Run coverage - run: forge coverage --report summary --report lcov --ir-minimum + run: | + forge coverage --report summary --report lcov --nmc IntegrationTest --ir-minimum - # To ignore coverage for certain directories modify the paths in this step as needed. The - # below default ignores coverage results for the test and script directories. Alternatively, + # To ignore coverage for certain directories modify the paths in this step as needed. + # The below default ignores coverage results for the test and script directories. Alternatively, # to include coverage in all directories, comment out this step. Note that because this # filtering applies to the lcov file, the summary table generated in the previous step will # still include all files and directories. @@ -72,8 +71,7 @@ jobs: - name: Filter directories run: | sudo apt update && sudo apt install -y lcov - lcov --remove lcov.info 'test/*' 'script/*' 'src/libraries/*' \ - --output-file lcov.info --rc lcov_branch_coverage=1 + lcov --remove lcov.info 'test/*' 'script/*' 'src/libraries/*' --output-file lcov.info --rc lcov_branch_coverage=0 # This step posts a detailed coverage report as a comment and deletes previous comments on # each push. The below step is used to fail coverage if the specified coverage threshold is @@ -126,35 +124,35 @@ jobs: echo "✅ Passed or warnings found" >> $GITHUB_STEP_SUMMARY fi - slither-analyze: - runs-on: "ubuntu-latest" - permissions: - actions: "read" - contents: "read" - security-events: "write" - steps: - - name: "Check out the repo" - uses: "actions/checkout@v4" - - - name: "Install Bun" - uses: "oven-sh/setup-bun@v1" - - - name: "Install the Node.js dependencies" - run: "bun install" - - - name: "Run Slither analysis" - uses: "crytic/slither-action@v0.3.0" - id: "slither" - with: - fail-on: "none" - sarif: "results.sarif" - - - name: "Upload SARIF file to GitHub code scanning" - uses: "github/codeql-action/upload-sarif@v2" - with: - sarif_file: ${{ steps.slither.outputs.sarif }} - - - name: "Add summary" - run: | - echo "## Slither result" >> $GITHUB_STEP_SUMMARY - echo "✅ Uploaded to GitHub code scanning" >> $GITHUB_STEP_SUMMARY + # slither-analyze: + # runs-on: "ubuntu-latest" + # permissions: + # actions: "read" + # contents: "read" + # security-events: "write" + # steps: + # - name: "Check out the repo" + # uses: "actions/checkout@v4" + + # - name: "Install Bun" + # uses: "oven-sh/setup-bun@v1" + + # - name: "Install the Node.js dependencies" + # run: "bun install" + + # - name: "Run Slither analysis" + # uses: "crytic/slither-action@v0.3.0" + # id: "slither" + # with: + # fail-on: "none" + # sarif: "results.sarif" + + # - name: "Upload SARIF file to GitHub code scanning" + # uses: "github/codeql-action/upload-sarif@v2" + # with: + # sarif_file: ${{ steps.slither.outputs.sarif }} + + # - name: "Add summary" + # run: | + # echo "## Slither result" >> $GITHUB_STEP_SUMMARY + # echo "✅ Uploaded to GitHub code scanning" >> $GITHUB_STEP_SUMMARY diff --git a/src/BaseStaking.sol b/src/BaseStaking.sol index 005a280..f220352 100644 --- a/src/BaseStaking.sol +++ b/src/BaseStaking.sol @@ -71,9 +71,6 @@ abstract contract BaseStaking is OwnableUpgradeable, ERC20VotesUpgradeable { /// @notice Thrown when transfer/tranferFrom is called error TransferDisabled(); - /// @notice Thrown when a user has no shares - error UserHasNoShares(); - /// @notice Thrown when a user try to claim rewards but has no rewards to /// claim error NoRewardsToClaim(); @@ -81,6 +78,9 @@ abstract contract BaseStaking is OwnableUpgradeable, ERC20VotesUpgradeable { /// @notice Thrown when the argument is the zero address error AddressZero(); + /// @notice Thrown when the amount of shares is 0 + error SharesMustBeGreaterThanZero(); + /*////////////////////////////////////////////////////////////// MODIFIERS //////////////////////////////////////////////////////////////*/ @@ -181,6 +181,7 @@ abstract contract BaseStaking is OwnableUpgradeable, ERC20VotesUpgradeable { uint256 assets ) public view virtual returns (uint256) { // sum + 1 on both sides to prevent donation attack + // this is the same as OZ ERC4626 prevetion to inflation attack with decimal offset = 0 return assets.mulDivDown(totalSupply() + 1, _totalAssets() + 1); } @@ -190,6 +191,7 @@ abstract contract BaseStaking is OwnableUpgradeable, ERC20VotesUpgradeable { uint256 shares ) public view virtual returns (uint256) { // sum + 1 on both sides to prevent donation attack + // this is the same as OZ ERC4626 prevetion to inflation attack with decimal offset = 0 return shares.mulDivDown(_totalAssets() + 1, totalSupply() + 1); } @@ -215,6 +217,15 @@ abstract contract BaseStaking is OwnableUpgradeable, ERC20VotesUpgradeable { // Calculate the amount of shares to mint uint256 shares = convertToShares(amount); + // A first deposit donation attack may result in shares being 0 if the + // contract has very high assets balance but a very low total supply. + // Although this attack is not profitable for the attacker, as they will + // spend more tokens than they will receive, it can still be used to perform a DDOS attack + // against a specific user. The targeted user can still withdraw their SHU, + // but this is only guaranteed if someone mints to increase the total supply of shares, + // because previewWithdraw rounds up and their shares will be less than the burn amount. + require(shares > 0, SharesMustBeGreaterThanZero()); + // Update the total locked amount totalLocked[user] += amount; @@ -253,6 +264,7 @@ abstract contract BaseStaking is OwnableUpgradeable, ERC20VotesUpgradeable { /// @param assets The amount of assets function _previewWithdraw(uint256 assets) internal view returns (uint256) { // sum + 1 on both sides to prevent donation attack + // this is the same as OZ ERC4626 prevetion to inflation attack with decimal offset = 0 return assets.mulDivUp(totalSupply() + 1, _totalAssets() + 1); } diff --git a/src/DelegateStaking.sol b/src/DelegateStaking.sol index 8ea67c6..b9e9767 100644 --- a/src/DelegateStaking.sol +++ b/src/DelegateStaking.sol @@ -25,10 +25,15 @@ interface IStaking { * A user's SHU balance is calculated as: * balanceOf(user) * totalSupply() / totalShares() * + * Staking, unstaking, and claiming rewards are based on shares, not the balance directly. + * This method ensures the balance can change over time without needing too many storage updates. + * * When staking, you must specify a keyper address. This symbolically demonstrates your support * for that keyper. The keyper address must be a valid keyper in the staking contract. - * Staking, unstaking, and claiming rewards are based on shares rather than the balance directly. - * This method ensures the balance can change over time without needing too many storage updates. + * + * @dev SHU tokens transferred into the contract without using the `stake` function will be included + * in the rewards distribution and shared among all stakers. This contract only supports SHU + * tokens. Any non-SHU tokens transferred into the contract will be permanently lost. * */ contract DelegateStaking is BaseStaking { @@ -67,6 +72,9 @@ contract DelegateStaking is BaseStaking { /// @notice stores the metadata associated with a given stake mapping(uint256 id => Stake _stake) public stakes; + /// @notice stores the amount delegated to a keyper + mapping(address keyper => uint256 totalDelegated) public totalDelegated; + /*////////////////////////////////////////////////////////////// EVENTS //////////////////////////////////////////////////////////////*/ @@ -89,6 +97,9 @@ contract DelegateStaking is BaseStaking { ERRORS //////////////////////////////////////////////////////////////*/ + /// @notice Thrown when a user has no shares + error UserHasNoShares(); + /// @notice Trown when amount is zero error ZeroAmount(); @@ -160,6 +171,9 @@ contract DelegateStaking is BaseStaking { stakes[stakeId].timestamp = block.timestamp; stakes[stakeId].lockPeriod = lockPeriod; + // Increase the keyper total delegated amount + totalDelegated[keyper] += amount; + _deposit(user, amount); emit Staked(user, keyper, amount, lockPeriod); @@ -183,7 +197,7 @@ contract DelegateStaking is BaseStaking { function unstake( uint256 stakeId, uint256 _amount - ) external returns (uint256 amount) { + ) external updateRewards returns (uint256 amount) { address user = msg.sender; require(userStakes[user].contains(stakeId), StakeDoesNotBelongToUser()); Stake memory userStake = stakes[stakeId]; @@ -208,6 +222,9 @@ contract DelegateStaking is BaseStaking { // Decrease the amount from the stake stakes[stakeId].amount -= amount; + // Decrease the total delegated amount + totalDelegated[userStake.keyper] -= amount; + // If the stake is empty, remove it if (stakes[stakeId].amount == 0) { // Remove the stake from the stakes mapping diff --git a/src/Staking.sol b/src/Staking.sol index a571508..9124c4f 100644 --- a/src/Staking.sol +++ b/src/Staking.sol @@ -27,8 +27,13 @@ import {IRewardsDistributor} from "./interfaces/IRewardsDistributor.sol"; * Please be aware that the contract's Owner can change the minimum stake amount. * If the Owner is compromised, they could set the minimum stake amount very high, * making it impossible for keypers to unstake their SHU. - * The Owner of this contract is the Shutter DAO multisig. By staking SHU, you trust - * the Owner not to set the minimum stake amount to an unreasonably high value. + * The Owner of this contract is the Shutter DAO multisig with a Azorius module. + * By staking SHU, you trust the Owner not to set the minimum stake amount to + * an unreasonably high value. + * + * @dev SHU tokens transferred into the contract without using the `stake` function will be included + * in the rewards distribution and shared among all stakers. This contract only supports SHU + * tokens. Any non-SHU tokens transferred into the contract will be permanently lost. * */ contract Staking is BaseStaking { @@ -84,6 +89,8 @@ contract Staking is BaseStaking { /*////////////////////////////////////////////////////////////// ERRORS //////////////////////////////////////////////////////////////*/ + /// @notice Thrown when a user has no shares + error UserHasNoShares(); /// @notice Thrown when a non-keyper attempts a call for which only keypers are allowed error OnlyKeyper(); @@ -97,7 +104,7 @@ contract Staking is BaseStaking { /// @notice Thrown when someone try to unstake a stake that doesn't belong /// to the keyper in question - error StakeDoesNotBelongToKeyper(); + error StakeDoesNotBelongToUser(); /// @notice Thrown when someone try to unstake a stake that doesn't exist error StakeDoesNotExist(); @@ -115,11 +122,6 @@ contract Staking is BaseStaking { _; } - /// @notice Ensure logic contract is unusable - constructor() { - _disableInitializers(); - } - /// @notice Initialize the contract /// @param _owner The owner of the contract, i.e. the DAO contract address /// @param _stakingToken The address of the staking token, i.e. SHU @@ -211,10 +213,10 @@ contract Staking is BaseStaking { address keyper, uint256 stakeId, uint256 _amount - ) external returns (uint256 amount) { + ) external updateRewards returns (uint256 amount) { require( userStakes[keyper].contains(stakeId), - StakeDoesNotBelongToKeyper() + StakeDoesNotBelongToUser() ); Stake memory keyperStake = stakes[stakeId]; @@ -233,7 +235,6 @@ contract Staking is BaseStaking { // must be locked for the lock period // If the global lock period is greater than the stake lock period, // the stake must be locked for the stake lock period - uint256 lock = keyperStake.lockPeriod > lockPeriod ? lockPeriod : keyperStake.lockPeriod; @@ -280,7 +281,6 @@ contract Staking is BaseStaking { /// @param _minStake The minimum stake amount function setMinStake(uint256 _minStake) external onlyOwner { minStake = _minStake; - emit NewMinStake(_minStake); } @@ -320,7 +320,7 @@ contract Staking is BaseStaking { /// - if the keyper sSHU balance is less or equal than the minimum /// stake or the total locked amount, the function will return 0 /// @param keyper The keyper address - /// @param unlockedAmount The amount of assets to unlock + /// @param unlockedAmount The amount of unlocked assets /// @return amount The maximum amount of assets that a keyper can withdraw after unlocking a certain amount function _maxWithdraw( address keyper, diff --git a/test/DelegateStaking.t.sol b/test/DelegateStaking.t.sol index 7f27709..44a5705 100644 --- a/test/DelegateStaking.t.sol +++ b/test/DelegateStaking.t.sol @@ -2,11 +2,15 @@ pragma solidity 0.8.26; import "@forge-std/Test.sol"; +import {console} from "@forge-std/console.sol"; import {TransparentUpgradeableProxy, ITransparentUpgradeableProxy} from "@openzeppelin/contracts/proxy/transparent/TransparentUpgradeableProxy.sol"; +import {Ownable} from "@openzeppelin/contracts/access/Ownable.sol"; import {IERC20Metadata} from "@openzeppelin/contracts/token/ERC20/extensions/IERC20Metadata.sol"; + import {FixedPointMathLib} from "src/libraries/FixedPointMathLib.sol"; import {Staking} from "src/Staking.sol"; +import {BaseStaking} from "src/BaseStaking.sol"; import {DelegateStaking} from "src/DelegateStaking.sol"; import {RewardsDistributor} from "src/RewardsDistributor.sol"; import {IRewardsDistributor} from "src/interfaces/IRewardsDistributor.sol"; @@ -41,6 +45,7 @@ contract DelegateStakingTest is Test { // deploy staking address stakingImpl = address(new Staking()); + vm.label(stakingImpl, "stakingImpl"); staking = Staking( address( @@ -58,13 +63,14 @@ contract DelegateStakingTest is Test { ); address delegateImpl = address(new DelegateStakingHarness()); + vm.label(delegateImpl, "delegateImpl"); delegate = DelegateStakingHarness( address( new TransparentUpgradeableProxy(delegateImpl, address(this), "") ) ); - vm.label(address(delegate), "staking"); + vm.label(address(delegate), "delegate"); delegate.initialize( address(this), // owner @@ -138,7 +144,8 @@ contract DelegateStakingTest is Test { _user != address(this) && _user != address(delegate) && _user != ProxyUtils.getAdminAddress(address(delegate)) && - _user != address(rewardsDistributor) + _user != address(rewardsDistributor) && + _user != address(staking) ); vm.startPrank(_user); @@ -422,6 +429,7 @@ contract Stake is DelegateStakingTest { address _depositor2, uint256 _amount ) public { + vm.assume(_depositor1 != _depositor2); _amount = _boundToRealisticStake(_amount); _mintGovToken(_depositor1, _amount); @@ -453,6 +461,7 @@ contract Stake is DelegateStakingTest { uint256 _amount, uint256 _jump ) public { + vm.assume(_depositor1 != _depositor2); _amount = _boundToRealisticStake(_amount); _jump = _boundRealisticTimeAhead(_jump); @@ -651,59 +660,1041 @@ contract Stake is DelegateStakingTest { delegate.stake(_keyper, 0); } - // function test_DonationAttackNoRewards( - // address keyper, - // address bob, - // address alice, - // uint256 bobAmount - // ) public { - // vm.assume(bob != alice); - // rewardsDistributor.removeRewardConfiguration(address(delegate)); + function test_DonationAttackNoRewards( + address keyper, + address bob, + address alice, + uint256 bobAmount + ) public { + vm.assume(bob != alice); + rewardsDistributor.removeRewardConfiguration(address(delegate)); + + _setKeyper(keyper, true); + + bobAmount = _boundToRealisticStake(bobAmount); + + // alice deposits 1 + _mintGovToken(alice, 1); + _stake(alice, keyper, 1); + + // simulate donation + govToken.mint(address(delegate), bobAmount); + + // bob stake + _mintGovToken(bob, bobAmount); + uint256 bobStakeId = _stake(bob, keyper, bobAmount); + + _jumpAhead(vm.getBlockTimestamp() + LOCK_PERIOD); + + // alice withdraw rewards (bob stake) even when there is no rewards distributed + vm.startPrank(alice); + //delegate.unstake(aliceStakeId, 0); + uint256 aliceRewards = delegate.claimRewards(0); + vm.stopPrank(); + + uint256 aliceBalanceAfterAttack = govToken.balanceOf(alice); + + // attack should not be profitable for alice + assertGtDecimal( + bobAmount + 1, // amount alice has spend in total + aliceBalanceAfterAttack, + 1e18, + "Alice receive more than expend for the attack" + ); + + // as previewWithdraw rounds up, someone needs to stake again to have a dSHU total supply > 1 + // so bob can unstake + _mintGovToken(bob, aliceRewards + 10e18); + _stake(bob, keyper, aliceRewards + 10e18); + + vm.prank(bob); + delegate.unstake(bobStakeId, 0); + + uint256 bobBalanceAfterAttack = govToken.balanceOf(bob); + + // Alice earn less than bob + assertGt( + bobBalanceAfterAttack, + aliceBalanceAfterAttack, + "Alice earn more than Bob after the attack" + ); + } + + function testFuzz_KeyperCanDelegateToHimself( + address _keyper, + uint256 _amount + ) public { + _amount = _boundToRealisticStake(_amount); + + _mintGovToken(_keyper, _amount); + _setKeyper(_keyper, true); + + uint256 stakeId = _stake(_keyper, _keyper, _amount); + + (address keyper, , , ) = delegate.stakes(stakeId); + + assertEq(keyper, _keyper, "Wrong keyper"); + } +} + +contract ClaimRewards is DelegateStakingTest { + function testFuzz_UpdateStakerGovTokenBalanceWhenClaimingRewards( + address _keyper, + address _depositor, + uint256 _amount, + uint256 _jump + ) public { + _amount = _boundToRealisticStake(_amount); + _jump = _boundRealisticTimeAhead(_jump); + + _mintGovToken(_depositor, _amount); + _setKeyper(_keyper, true); + + _stake(_depositor, _keyper, _amount); + + _jumpAhead(_jump); + + vm.startPrank(_depositor); + delegate.claimRewards(0); + + uint256 expectedRewards = REWARD_RATE * (_jump); + + // need to accept a small error due to the donation attack prevention + assertApproxEqAbs( + govToken.balanceOf(_depositor), + expectedRewards, + 1e18, + "Wrong balance" + ); + } + + function testFuzz_GovTokenBalanceUnchangedWhenClaimingRewardsOnlyStaker( + address _keyper, + address _depositor, + uint256 _amount, + uint256 _jump + ) public { + _amount = _boundToRealisticStake(_amount); + _jump = _boundRealisticTimeAhead(_jump); + + _mintGovToken(_depositor, _amount); + _setKeyper(_keyper, true); + + _stake(_depositor, _keyper, _amount); + + uint256 contractBalanceBefore = govToken.balanceOf(address(delegate)); + + _jumpAhead(_jump); + + vm.prank(_depositor); + delegate.claimRewards(0); + + uint256 contractBalanceAfter = govToken.balanceOf(address(delegate)); + + // small percentage lost to the vault due to the donation attack prevention + assertApproxEqAbs( + contractBalanceAfter - contractBalanceBefore, + 0, + 1e18, + "Wrong balance" + ); + } + + function testFuzz_EmitRewardsClaimedEventWhenClaimingRewards( + address _keyper, + address _depositor, + uint256 _amount, + uint256 _jump + ) public { + _amount = _boundToRealisticStake(_amount); + _jump = _boundRealisticTimeAhead(_jump); + + _mintGovToken(_depositor, _amount); + _setKeyper(_keyper, true); + + _stake(_depositor, _keyper, _amount); + + _jumpAhead(_jump); + + vm.expectEmit(true, true, false, false); + emit BaseStaking.RewardsClaimed(_depositor, REWARD_RATE * _jump); + + vm.prank(_depositor); + delegate.claimRewards(0); + } + + function testFuzz_ClaimAllRewardsOnlyStaker( + address _keyper, + address _depositor, + uint256 _amount, + uint256 _jump + ) public { + _amount = _boundToRealisticStake(_amount); + _jump = _boundRealisticTimeAhead(_jump); + + _mintGovToken(_depositor, _amount); + _setKeyper(_keyper, true); + + _stake(_depositor, _keyper, _amount); + + _jumpAhead(_jump); + + vm.prank(_depositor); + uint256 rewards = delegate.claimRewards(0); + + uint256 expectedRewards = REWARD_RATE * _jump; + + // need to accept a small error due to the donation attack prevention + assertApproxEqAbs(rewards, expectedRewards, 1e18, "Wrong rewards"); + } + + function testFuzz_ClaimRewardBurnShares( + address _keyper, + address _depositor, + uint256 _amount, + uint256 _jump + ) public { + _amount = _boundToRealisticStake(_amount); + _jump = _boundRealisticTimeAhead(_jump); + + _mintGovToken(_depositor, _amount); + _setKeyper(_keyper, true); + + _stake(_depositor, _keyper, _amount); + + uint256 sharesBefore = delegate.balanceOf(_depositor); + + _jumpAhead(_jump); + + uint256 expectedRewards = REWARD_RATE * _jump; + + uint256 burnShares = _previewWithdrawIncludeRewardsDistributed( + expectedRewards, + expectedRewards + ); + + vm.prank(_depositor); + delegate.claimRewards(0); + + uint256 sharesAfter = delegate.balanceOf(_depositor); + + // need to accept a small error due to the donation attack prevention + assertApproxEqAbs( + sharesBefore - sharesAfter, + burnShares, + 1, + "Wrong shares burned" + ); + } + + function testFuzz_UpdateTotalSupplyWhenClaimingRewards( + address _keyper, + address _depositor, + uint256 _amount, + uint256 _jump + ) public { + _amount = _boundToRealisticStake(_amount); + _jump = _boundRealisticTimeAhead(_jump); + + _mintGovToken(_depositor, _amount); + _setKeyper(_keyper, true); + + _stake(_depositor, _keyper, _amount); + + uint256 totalSupplyBefore = delegate.totalSupply(); + + _jumpAhead(_jump); + + uint256 expectedRewards = REWARD_RATE * _jump; + + uint256 burnShares = _previewWithdrawIncludeRewardsDistributed( + expectedRewards, + expectedRewards + ); + + vm.prank(_depositor); + delegate.claimRewards(0); + + uint256 totalSupplyAfter = delegate.totalSupply(); + + assertApproxEqAbs( + totalSupplyAfter, + totalSupplyBefore - burnShares, + 1, + "Wrong total supply" + ); + } + + function testFuzz_Depositor1GetsMoreRewardsThanDepositor2WhenStakingFirst( + address _keyper, + address _depositor1, + address _depositor2, + uint256 _amount, + uint256 _jump1, + uint256 _jump2 + ) public { + _amount = _boundToRealisticStake(_amount); + _jump1 = _boundRealisticTimeAhead(_jump1); + _jump2 = _boundRealisticTimeAhead(_jump2); + + vm.assume(_depositor1 != _depositor2); + + _mintGovToken(_depositor1, _amount); + _mintGovToken(_depositor2, _amount); + + _setKeyper(_keyper, true); + + _stake(_depositor1, _keyper, _amount); + + _jumpAhead(_jump1); + + _stake(_depositor2, _keyper, _amount); + + _jumpAhead(_jump2); + + vm.prank(_depositor1); + uint256 rewards1 = delegate.claimRewards(0); + + vm.prank(_depositor2); + uint256 rewards2 = delegate.claimRewards(0); + + assertGt(rewards1, rewards2, "Wrong rewards"); + } + + function testFuzz_DepositorsGetApproxSameRewardAmountWhenStakingSameAmountInSameBlock( + address _keyper, + address _depositor1, + address _depositor2, + uint256 _amount, + uint256 _jump + ) public { + _amount = _boundToRealisticStake(_amount); + _jump = _boundRealisticTimeAhead(_jump); + + vm.assume(_depositor1 != _depositor2); + + _mintGovToken(_depositor1, _amount); + _mintGovToken(_depositor2, _amount); + + _setKeyper(_keyper, true); + + _stake(_depositor1, _keyper, _amount); + + _stake(_depositor2, _keyper, _amount); + + _jumpAhead(_jump); + + vm.prank(_depositor1); + uint256 rewards1 = delegate.claimRewards(0); + + vm.prank(_depositor2); + uint256 rewards2 = delegate.claimRewards(0); + + assertApproxEqAbs(rewards1, rewards2, 1e18, "Wrong rewards"); + } + + function testFuzz_DepositorGetExactSpecifiedAmountWhenClaimingRewards( + address _keyper, + address _depositor, + uint256 _amount, + uint256 _jump + ) public { + _amount = _boundToRealisticStake(_amount); + _jump = _boundRealisticTimeAhead(_jump); + + _mintGovToken(_depositor, _amount); + _setKeyper(_keyper, true); + + _stake(_depositor, _keyper, _amount); + + _jumpAhead(_jump); + + uint256 expectedRewards = REWARD_RATE * _jump; + vm.prank(_depositor); + uint256 rewards = delegate.claimRewards(expectedRewards / 2); - // _setKeyper(keyper, true); + assertEq(rewards, expectedRewards / 2, "Wrong rewards"); + } - // bobAmount = _boundToRealisticStake(bobAmount); + function testFuzz_OnlyBurnTheCorrespondedAmountOfSharesSpecifiedWhenClaimingRewards( + address _keyper, + address _depositor, + uint256 _amount, + uint256 _jump + ) public { + _amount = _boundToRealisticStake(_amount); + _jump = _boundRealisticTimeAhead(_jump); - // // alice deposits 1 - // _mintGovToken(alice, 1); - // uint256 aliceStakeId = _stake(alice, keyper, 1); + _mintGovToken(_depositor, _amount); + _setKeyper(_keyper, true); - // // simulate donation - // govToken.mint(address(delegate), bobAmount); + _stake(_depositor, _keyper, _amount); - // // bob stake - // _mintGovToken(bob, bobAmount); - // uint256 bobStakeId = _stake(bob, keyper, bobAmount); + uint256 sharesBefore = delegate.balanceOf(_depositor); - // _jumpAhead(vm.getBlockTimestamp() + LOCK_PERIOD); + _jumpAhead(_jump); - // // alice withdraw rewards (bob stake) even when there is no rewards distributed - // vm.startPrank(alice); - // //delegate.unstake(aliceStakeId, 0); - // delegate.claimRewards(0); - // vm.stopPrank(); + uint256 expectedRewards = REWARD_RATE * _jump; + uint256 burnShares = _previewWithdrawIncludeRewardsDistributed( + expectedRewards / 2, + expectedRewards + ); - // uint256 aliceBalanceAfterAttack = govToken.balanceOf(alice); + vm.prank(_depositor); + delegate.claimRewards(expectedRewards / 2); - // // attack should not be profitable for alice - // assertGtDecimal( - // bobAmount + 1, // amount alice has spend in total - // aliceBalanceAfterAttack, - // 1e18, - // "Alice receive more than expend for the attack" - // ); + uint256 sharesAfter = delegate.balanceOf(_depositor); - // vm.startPrank(bob); - // delegate.unstake(bobStakeId, 0); - // delegate.claimRewards(0); + assertEq(sharesBefore - sharesAfter, burnShares, "Wrong shares burned"); + } - // uint256 bobBalanceAfterAttack = govToken.balanceOf(bob); + function testFuzz_RevertIf_NoRewardsToClaim( + address _keyper, + address _depositor, + uint256 _amount + ) public { + _amount = _boundToRealisticStake(_amount); - // // at the end Alice still earn less than bob - // assertGt( - // bobBalanceAfterAttack, - // aliceBalanceAfterAttack, - // "Alice earn more than Bob after the attack" - // ); - // } + _mintGovToken(_depositor, _amount); + _setKeyper(_keyper, true); + + _stake(_depositor, _keyper, _amount); + + vm.prank(_depositor); + vm.expectRevert(BaseStaking.NoRewardsToClaim.selector); + delegate.claimRewards(0); + } + + function testFuzz_RevertIf_UserHasNoShares(address _depositor) public { + vm.assume( + _depositor != address(0) && + _depositor != ProxyUtils.getAdminAddress(address(staking)) + ); + + vm.prank(_depositor); + vm.expectRevert(DelegateStaking.UserHasNoShares.selector); + staking.claimRewards(0); + } + + function testFuzz_RevertIf_NoRewardsToClaimForThatUser( + address _keyper, + address _depositor1, + address _depositor2, + uint256 _amount1, + uint256 _amount2, + uint256 _jump + ) public { + _amount1 = _boundToRealisticStake(_amount1); + _amount2 = _boundToRealisticStake(_amount2); + _jump = _boundRealisticTimeAhead(_jump); + + vm.assume(_depositor1 != _depositor2); + + _mintGovToken(_depositor1, _amount1); + _mintGovToken(_depositor2, _amount2); + + _setKeyper(_keyper, true); + + _stake(_depositor1, _keyper, _amount1); + + _jumpAhead(_jump); + + _stake(_depositor2, _keyper, _amount2); + + vm.prank(_depositor2); + vm.expectRevert(BaseStaking.NoRewardsToClaim.selector); + delegate.claimRewards(0); + } +} + +contract Unstake is DelegateStakingTest { + function testFuzz_UnstakeUpdateStakerGovTokenBalanceWhenUnstaking( + address _keyper, + address _depositor, + uint256 _amount, + uint256 _jump + ) public { + _amount = _boundToRealisticStake(_amount); + _jump = _boundUnlockedTime(_jump); + + _mintGovToken(_depositor, _amount); + _setKeyper(_keyper, true); + + uint256 stakeId = _stake(_depositor, _keyper, _amount); + + _jumpAhead(_jump); + + vm.prank(_depositor); + delegate.unstake(stakeId, 0); + + assertEq(govToken.balanceOf(_depositor), _amount, "Wrong balance"); + } + + function testFuzz_UpdateTotalSupplyWhenUnstaking( + address _keyper, + address _depositor, + uint256 _amount, + uint256 _jump + ) public { + _amount = _boundToRealisticStake(_amount); + _jump = _boundUnlockedTime(_jump); + + _mintGovToken(_depositor, _amount); + _setKeyper(_keyper, true); + + uint256 stakeId = _stake(_depositor, _keyper, _amount); + + uint256 totalSupplyBefore = delegate.totalSupply(); + + _jumpAhead(_jump); + + uint256 expectedRewards = REWARD_RATE * _jump; + + uint256 sharesToBurn = _previewWithdrawIncludeRewardsDistributed( + _amount, + expectedRewards + ); + + vm.prank(_depositor); + delegate.unstake(stakeId, 0); + + assertEq( + delegate.totalSupply(), + totalSupplyBefore - sharesToBurn, + "Wrong total supply" + ); + + uint256 expectedSharesRemaining = delegate.convertToShares( + expectedRewards + ); + + assertEq(delegate.totalSupply(), expectedSharesRemaining); + } + + function testFuzz_UnstakeShouldNotTransferRewards( + address _keyper, + address _depositor, + uint256 _amount, + uint256 _jump + ) public { + _amount = _boundToRealisticStake(_amount); + _jump = _boundUnlockedTime(_jump); + + _mintGovToken(_depositor, _amount); + _setKeyper(_keyper, true); + + uint256 stakeId = _stake(_depositor, _keyper, _amount); + + _jumpAhead(_jump); + + uint256 expectedRewards = REWARD_RATE * _jump; + + vm.prank(_depositor); + uint256 unstakeAmount = delegate.unstake(stakeId, 0); + + assertEq( + govToken.balanceOf(address(delegate)), + expectedRewards, + "Wrong balance" + ); + assertEq( + govToken.balanceOf(_depositor), + unstakeAmount, + "Wrong balance" + ); + } + + function testFuzz_EmitUnstakeEventWhenUnstaking( + address _keyper, + address _depositor, + uint256 _amount, + uint256 _jump + ) public { + _amount = _boundToRealisticStake(_amount); + _jump = _boundUnlockedTime(_jump); + + _mintGovToken(_depositor, _amount); + _setKeyper(_keyper, true); + + uint256 stakeId = _stake(_depositor, _keyper, _amount); + + _jumpAhead(_jump); + + uint256 shares = _previewWithdrawIncludeRewardsDistributed( + _amount, + REWARD_RATE * _jump + ); + vm.expectEmit(); + emit Staking.Unstaked(_depositor, _amount, shares); + + vm.prank(_depositor); + delegate.unstake(stakeId, 0); + } + + function testFuzz_DepositorHasMultipleStakesUnstakeCorrectStake( + address _keyper, + address _depositor, + uint256 _amount1, + uint256 _amount2, + uint256 _jump + ) public { + _amount1 = _boundToRealisticStake(_amount1); + _amount2 = _boundToRealisticStake(_amount2); + _jump = _boundUnlockedTime(_jump); + + _mintGovToken(_depositor, _amount1 + _amount2); + _setKeyper(_keyper, true); + + uint256 stakeId1 = _stake(_depositor, _keyper, _amount1); + uint256 stakeId2 = _stake(_depositor, _keyper, _amount2); + assertEq(govToken.balanceOf(_depositor), 0, "Wrong balance"); + + _jumpAhead(_jump); + + vm.prank(_depositor); + delegate.unstake(stakeId1, 0); + + assertEq(govToken.balanceOf(_depositor), _amount1, "Wrong balance"); + + vm.prank(_depositor); + delegate.unstake(stakeId2, 0); + + assertEq( + govToken.balanceOf(_depositor), + _amount1 + _amount2, + "Wrong balance" + ); + } + + function testFuzz_UnstakeOnlyAmountSpecified( + address _keyper, + address _depositor, + uint256 _amount1, + uint256 _amount2, + uint256 _jump + ) public { + _amount1 = _boundToRealisticStake(_amount1); + _amount2 = _boundToRealisticStake(_amount2); + _jump = _boundUnlockedTime(_jump); + + vm.assume(_amount1 > _amount2); + _jump = _boundUnlockedTime(_jump); + + _mintGovToken(_depositor, _amount1); + + _setKeyper(_keyper, true); + + uint256 stakeId = _stake(_depositor, _keyper, _amount1); + assertEq(govToken.balanceOf(_depositor), 0, "Wrong balance"); + + _jumpAhead(_jump); + + vm.prank(_depositor); + delegate.unstake(stakeId, _amount2); + + assertEq(govToken.balanceOf(_depositor), _amount2, "Wrong balance"); + + uint256[] memory stakeIds = delegate.getUserStakeIds(_depositor); + assertEq(stakeIds.length, 1, "Wrong stake ids"); + + (, uint256 amount, , ) = delegate.stakes(stakeIds[0]); + + assertEq(amount, _amount1 - _amount2, "Wrong amount"); + } + + function testFuzz_RevertIf_StakeIsStillLocked( + address _keyper, + address _depositor, + uint256 _amount, + uint256 _jump + ) public { + _amount = _boundToRealisticStake(_amount); + _jump = bound(_jump, vm.getBlockTimestamp(), LOCK_PERIOD); + + _mintGovToken(_depositor, _amount); + _setKeyper(_keyper, true); + + uint256 stakeId = _stake(_depositor, _keyper, _amount); + + _jumpAhead(_jump); + + vm.prank(_depositor); + vm.expectRevert(DelegateStaking.StakeIsStillLocked.selector); + delegate.unstake(stakeId, 0); + } + + function testFuzz_RevertIf_StakeIsStillLockedAfterLockPeriodChangedToLessThanCurrent( + address _keyper, + address _depositor, + uint256 _amount, + uint256 _jump + ) public { + _amount = _boundToRealisticStake(_amount); + _jump = bound(_jump, vm.getBlockTimestamp(), LOCK_PERIOD - 1); + + _mintGovToken(_depositor, _amount); + _setKeyper(_keyper, true); + + uint256 stakeId = _stake(_depositor, _keyper, _amount); + + delegate.setLockPeriod(_jump); + + _jumpAhead(_jump); + + vm.prank(_depositor); + vm.expectRevert(DelegateStaking.StakeIsStillLocked.selector); + delegate.unstake(stakeId, 0); + } + + function testFuzz_RevertIf_StakeDoesNotBelongToUser( + address _keyper, + address _depositor1, + address _depositor2, + uint256 _amount1 + ) public { + vm.assume(_depositor1 != _depositor2); + vm.assume( + _depositor1 != address(0) && + _depositor1 != ProxyUtils.getAdminAddress(address(delegate)) + ); + vm.assume( + _depositor2 != address(0) && + _depositor2 != ProxyUtils.getAdminAddress(address(delegate)) + ); + _amount1 = _boundToRealisticStake(_amount1); + + _mintGovToken(_depositor1, _amount1); + + _setKeyper(_keyper, true); + + uint256 stakeId = _stake(_depositor1, _keyper, _amount1); + + vm.prank(_depositor2); + vm.expectRevert(DelegateStaking.StakeDoesNotBelongToUser.selector); + delegate.unstake(stakeId, 0); + } + + function testFuzz_RevertIf_AmountGreaterThanStakeAmount( + address _keyper, + address _depositor, + uint256 _amount + ) public { + _amount = _boundToRealisticStake(_amount); + + _mintGovToken(_depositor, _amount); + _setKeyper(_keyper, true); + + uint256 stakeId = _stake(_depositor, _keyper, _amount); + + vm.prank(_depositor); + vm.expectRevert(BaseStaking.WithdrawAmountTooHigh.selector); + delegate.unstake(stakeId, _amount + 1); + } +} + +contract OwnableFunctions is DelegateStakingTest { + function testFuzz_setRewardsDistributor( + address _newRewardsDistributor + ) public { + vm.assume( + _newRewardsDistributor != address(0) && + _newRewardsDistributor != address(delegate) && + _newRewardsDistributor != address(govToken) + ); + + delegate.setRewardsDistributor(_newRewardsDistributor); + + assertEq( + address(delegate.rewardsDistributor()), + _newRewardsDistributor, + "Wrong rewards distributor" + ); + } + + function testFuzz_setLockPeriod(uint256 _newLockPeriod) public { + vm.expectEmit(); + + emit BaseStaking.NewLockPeriod(_newLockPeriod); + + delegate.setLockPeriod(_newLockPeriod); + + assertEq(delegate.lockPeriod(), _newLockPeriod, "Wrong lock period"); + } + + function testFuzz_setStakingContract(address _newStaking) public { + vm.assume( + _newStaking != address(0) && + _newStaking != address(delegate) && + _newStaking != address(govToken) + ); + + vm.expectEmit(); + emit DelegateStaking.NewStakingContract(_newStaking); + delegate.setStakingContract(_newStaking); + + assertEq( + address(delegate.staking()), + _newStaking, + "Wrong staking contract" + ); + } + + function testFuzz_RevertIf_NonOwnerSetRewardsDistributor( + address _newRewardsDistributor, + address _nonOwner + ) public { + vm.assume( + _newRewardsDistributor != address(0) && + _newRewardsDistributor != address(delegate) && + _newRewardsDistributor != address(govToken) + ); + + vm.assume( + _nonOwner != address(0) && + _nonOwner != ProxyUtils.getAdminAddress(address(delegate)) && + _nonOwner != address(this) + ); + + vm.prank(_nonOwner); + vm.expectRevert( + abi.encodeWithSelector( + Ownable.OwnableUnauthorizedAccount.selector, + _nonOwner + ) + ); + delegate.setRewardsDistributor(_newRewardsDistributor); + } + + function testFuzz_RevertIf_NonOwnerSetLockPeriod( + uint256 _newLockPeriod, + address _nonOwner + ) public { + vm.assume( + _nonOwner != address(0) && + _nonOwner != ProxyUtils.getAdminAddress(address(delegate)) && + _nonOwner != address(this) + ); + + vm.prank(_nonOwner); + vm.expectRevert( + abi.encodeWithSelector( + Ownable.OwnableUnauthorizedAccount.selector, + _nonOwner + ) + ); + delegate.setLockPeriod(_newLockPeriod); + } + + function testFuzz_RevertIf_NonOwnerSetStakingContract( + address _newStaking, + address _nonOwner + ) public { + vm.assume( + _newStaking != address(0) && + _newStaking != address(delegate) && + _newStaking != address(govToken) + ); + + vm.assume( + _nonOwner != address(0) && + _nonOwner != ProxyUtils.getAdminAddress(address(delegate)) && + _nonOwner != address(this) + ); + + vm.prank(_nonOwner); + vm.expectRevert( + abi.encodeWithSelector( + Ownable.OwnableUnauthorizedAccount.selector, + _nonOwner + ) + ); + delegate.setStakingContract(_newStaking); + } +} + +contract ViewFunctions is DelegateStakingTest { + function testFuzz_Revertif_MaxWithdrawDepositorHasNoStakes( + address _depositor + ) public { + vm.expectRevert(DelegateStaking.UserHasNoShares.selector); + delegate.maxWithdraw(_depositor); + } + + function testFuzz_MaxWithdrawDepositorHasLockedStakeNoRewards( + address _keyper, + address _depositor, + uint256 _amount + ) public { + _amount = _boundToRealisticStake(_amount); + + _mintGovToken(_depositor, _amount); + _setKeyper(_keyper, true); + + _stake(_depositor, _keyper, _amount); + + uint256 maxWithdraw = delegate.maxWithdraw(_depositor); + assertEq(maxWithdraw, 0, "Wrong max withdraw"); + } + + function testFuzz_MaxWithdrawDepositorHasLockedStakeAndReward( + address _keyper, + address _depositor1, + uint256 _amount1, + uint256 _jump + ) public { + _amount1 = _boundToRealisticStake(_amount1); + + _jump = _boundUnlockedTime(_jump); + + _mintGovToken(_depositor1, _amount1); + _setKeyper(_keyper, true); + + _stake(_depositor1, _keyper, _amount1); + + _jumpAhead(_jump); + + rewardsDistributor.collectRewardsTo(address(delegate)); + + uint256 rewards = REWARD_RATE * _jump; + + uint256 maxWithdraw = delegate.maxWithdraw(_depositor1); + + assertApproxEqAbs(maxWithdraw, rewards, 0.1e18, "Wrong max withdraw"); + } + + function testFuzz_MaxWithdrawDepositorHasMultipleLockedStakes( + address _keyper, + address _depositor, + uint256 _amount1, + uint256 _amount2, + uint256 _jump + ) public { + _amount1 = _boundToRealisticStake(_amount1); + _amount2 = _boundToRealisticStake(_amount2); + _jump = _boundUnlockedTime(_jump); + + _mintGovToken(_depositor, _amount1 + _amount2); + _setKeyper(_keyper, true); + + _stake(_depositor, _keyper, _amount1); + _stake(_depositor, _keyper, _amount2); + + uint256 maxWithdraw = delegate.maxWithdraw(_depositor); + assertEq(maxWithdraw, 0, "Wrong max withdraw"); + } + + function testFuzz_ConvertToSharesNoSupply(uint256 assets) public view { + assertEq(delegate.convertToShares(assets), assets); + } + + function testFuzz_ConvertToSharesHasSupplySameBlock( + address _keyper, + address _depositor, + uint256 _assets + ) public { + _assets = _boundToRealisticStake(_assets); + + _mintGovToken(_depositor, _assets); + _setKeyper(_keyper, true); + + _stake(_depositor, _keyper, _assets); + + uint256 shares = delegate.convertToShares(_assets); + + assertEq(shares, _assets, "Wrong shares"); + } + + function testFuzz_ConvertToAssetsHasSupplySameBlock( + address _keyper, + address _depositor, + uint256 _assets + ) public { + _assets = _boundToRealisticStake(_assets); + + _mintGovToken(_depositor, _assets); + _setKeyper(_keyper, true); + + _stake(_depositor, _keyper, _assets); + + uint256 shares = delegate.convertToShares(_assets); + uint256 assets = delegate.convertToAssets(shares); + + assertEq(assets, _assets, "Wrong assets"); + } + + function testFuzz_GetUserStakeIds( + address _keyper, + address _depositor, + uint256 _amount1, + uint256 _amount2 + ) public { + _amount1 = _boundToRealisticStake(_amount1); + _amount2 = _boundToRealisticStake(_amount2); + + _mintGovToken(_depositor, _amount1 + _amount2); + _setKeyper(_keyper, true); + + uint256 stakeId1 = _stake(_depositor, _keyper, _amount1); + uint256 stakeId2 = _stake(_depositor, _keyper, _amount2); + + uint256[] memory stakeIds = delegate.getUserStakeIds(_depositor); + + assertEq(stakeIds.length, 2, "Wrong stake ids"); + assertEq(stakeIds[0], stakeId1, "Wrong stake id"); + assertEq(stakeIds[1], stakeId2, "Wrong stake id"); + } + + function testFuzz_CalculateWithdrawAmountReturnsAmount( + address _keyper, + address _depositor, + uint256 _amount + ) public { + _amount = _boundToRealisticStake(_amount); + + _mintGovToken(_depositor, _amount); + _setKeyper(_keyper, true); + + uint256 stakeId = _stake(_depositor, _keyper, _amount); + + uint256 withdrawAmount = delegate.exposed_calculateWithdrawAmount( + _amount / 2, + _amount + ); + + assertEq(withdrawAmount, _amount / 2, "Wrong withdraw amount"); + } +} + +contract Transfer is DelegateStakingTest { + function testFuzz_RevertWith_transferDisabled( + address _from, + address _to, + uint256 _amount + ) public { + _amount = _boundToRealisticStake(_amount); + + _mintGovToken(_from, _amount); + _setKeyper(_from, true); + + _stake(_from, _from, _amount); + + vm.expectRevert(BaseStaking.TransferDisabled.selector); + delegate.transfer(_to, _amount); + } + + function testFuzz_RevertWith_transferFromDisabled( + address _from, + address _to, + uint256 _amount + ) public { + _amount = _boundToRealisticStake(_amount); + + _mintGovToken(_from, _amount); + _setKeyper(_from, true); + + _stake(_from, _from, _amount); + + vm.expectRevert(BaseStaking.TransferDisabled.selector); + delegate.transferFrom(_from, _to, _amount); + } } diff --git a/test/RewardsDistributor.t.sol b/test/RewardsDistributor.t.sol index 778962c..1170d10 100644 --- a/test/RewardsDistributor.t.sol +++ b/test/RewardsDistributor.t.sol @@ -136,6 +136,7 @@ contract OwnableFunctions is RewardsDistributorTest { function testFuzz_RevertIf_SetRewardConfigurationEmissionRateZero( address _receiver ) public { + vm.assume(_receiver != address(0)); vm.expectRevert(RewardsDistributor.EmissionRateZero.selector); rewardsDistributor.setRewardConfiguration(_receiver, 0); } diff --git a/test/Staking.t.sol b/test/Staking.t.sol index 520f4e0..29c0765 100644 --- a/test/Staking.t.sol +++ b/test/Staking.t.sol @@ -4,9 +4,10 @@ pragma solidity 0.8.26; import "@forge-std/Test.sol"; import {IERC20Metadata} from "@openzeppelin/contracts/token/ERC20/extensions/IERC20Metadata.sol"; -import {FixedPointMathLib} from "src/libraries/FixedPointMathLib.sol"; import {TransparentUpgradeableProxy, ITransparentUpgradeableProxy} from "@openzeppelin/contracts/proxy/transparent/TransparentUpgradeableProxy.sol"; import {Ownable} from "@openzeppelin/contracts/access/Ownable.sol"; + +import {FixedPointMathLib} from "src/libraries/FixedPointMathLib.sol"; import {Staking} from "src/Staking.sol"; import {BaseStaking} from "src/BaseStaking.sol"; import {RewardsDistributor} from "src/RewardsDistributor.sol"; @@ -439,6 +440,7 @@ contract Stake is StakingTest { uint256 _amount, uint256 _jump ) public { + vm.assume(_depositor1 != _depositor2); _amount = _boundToRealisticStake(_amount); _jump = _boundRealisticTimeAhead(_jump); @@ -657,81 +659,6 @@ contract Stake is StakingTest { vm.stopPrank(); } - // function test_DonationAttack(address bob, address alice) public { - // uint256 initialStake = MIN_STAKE; - // uint256 donationAmount = MIN_STAKE * 100; - // uint256 bobStake = MIN_STAKE * 100; - - // // first alice mints - // _mintGovToken(alice, initialStake); - // _setKeyper(alice, true); - // _stake(alice, initialStake); - - // assertEq(staking.totalSupply(), initialStake); - - // // simulate donation - // govToken.mint(address(staking), donationAmount); - - // assertEq(staking.totalSupply(), initialStake); - // assertEq( - // govToken.balanceOf(address(staking)), - // initialStake + donationAmount - // ); - - // // bob mints - // _mintGovToken(bob, bobStake); - // _setKeyper(bob, true); - // uint256 bobStakeId = _stake(bob, bobStake); - - // // bob shares - // uint256 bobShares = staking.balanceOf(bob); - // console.log("bob shares", bobShares); - - // //vm.prank(alice); - // // uint256 aliceUnstake = staking.unstake(alice, 1, 0); - // // assertEq(aliceUnstake, initialStake); - - // // alice claim rewards withdrawing donation - // vm.prank(alice); - // uint256 aliceRewards = staking.claimRewards(0); - - // // attacker cost is greater than expected gains - // assertGt( - // donationAmount, - // aliceRewards, - // "Alice receive more than expend for the attack" - // ); - - // _jumpAhead(vm.getBlockTimestamp() + LOCK_PERIOD); - // // bob unstake maximum he can unstake - // uint256 maxBobCanWithdraw = staking.exposed_maxWithdraw(bob, bobStake); - // vm.prank(bob); - // staking.unstake(bob, bobStakeId, maxBobCanWithdraw); - - // uint256 bobBalance = govToken.balanceOf(bob); - // uint256 aliceBalance = govToken.balanceOf(alice); - - // vm.prank(bob); - // uint256 bobRewards = staking.claimRewards(0); - // console.log("bob rewards", bobRewards); - - // // bob lost a small amount maximum lost is 1% - // assertApproxEqRel( - // bobBalance, - // bobStake + bobRewards, - // 0.01e18, - // "Bob lost more than 1%" - // ); - - // // at the end Alice still lost more than bob - // assertGtDecimal( - // donationAmount - aliceRewards, - // bobStake - bobBalance, - // 1e18, - // "Alice receive more than bob" - // ); - // } - function testFuzz_DonationAttackNoRewards( address bob, address alice, @@ -919,7 +846,7 @@ contract ClaimRewards is StakingTest { assertApproxEqAbs(rewards, expectedRewards, 1e18, "Wrong rewards"); } - function testFuzz_claimRewardBurnShares( + function testFuzz_ClaimRewardBurnShares( address _depositor, uint256 _amount, uint256 _jump @@ -1145,7 +1072,7 @@ contract ClaimRewards is StakingTest { ); vm.prank(_depositor); - vm.expectRevert(BaseStaking.UserHasNoShares.selector); + vm.expectRevert(Staking.UserHasNoShares.selector); staking.claimRewards(0); } @@ -1259,7 +1186,10 @@ contract Unstake is StakingTest { _jumpAhead(_jump); uint256 unstakeAmount = _amount - MIN_STAKE; - uint256 shares = staking.previewWithdraw(unstakeAmount); + uint256 shares = _previewWithdrawIncludeRewardsDistributed( + unstakeAmount, + REWARD_RATE * _jump + ); vm.expectEmit(); emit Staking.Unstaked(_depositor, unstakeAmount, shares); @@ -1305,23 +1235,37 @@ contract Unstake is StakingTest { _mintGovToken(_depositor, _amount + MIN_STAKE); _setKeyper(_depositor, true); - _stake(_depositor, MIN_STAKE); - uint256 stakeId = _stake(_depositor, _amount); + uint256 totalSupplyBefore = staking.totalSupply(); + _jumpAhead(_jump); + uint256 expectedRewards = REWARD_RATE * _jump; + uint256 sharesToBurn = _previewWithdrawIncludeRewardsDistributed( + _amount, + expectedRewards + ); + vm.prank(_depositor); staking.unstake(_depositor, stakeId, 0); - uint256 expectedSharesRemaining = staking.convertToShares(MIN_STAKE); + assertEq( + staking.totalSupply(), + totalSupplyBefore - sharesToBurn, + "Wrong total supply" + ); - uint256 totalSupplyAfter = staking.totalSupply(); + uint256 expectedSharesRemaining = staking.convertToShares( + MIN_STAKE + expectedRewards + ); - assertEq( - totalSupplyAfter, + // TODO review this + assertApproxEqRel( + staking.totalSupply(), expectedSharesRemaining, - "Wrong total supply" + 0.1e18, + "Wrong total supply with remaing shares" ); } @@ -1506,7 +1450,7 @@ contract Unstake is StakingTest { staking.unstake(depositor, stakeId, MIN_STAKE); } - function testFuzz_RevertIf_StakeDoesNotBelongToKeyper( + function testFuzz_RevertIf_StakeDoesNotBelongToUser( address _depositor1, address _depositor2, uint256 _amount1 @@ -1529,7 +1473,7 @@ contract Unstake is StakingTest { uint256 stakeId = _stake(_depositor1, _amount1); vm.prank(_depositor2); - vm.expectRevert(Staking.StakeDoesNotBelongToKeyper.selector); + vm.expectRevert(Staking.StakeDoesNotBelongToUser.selector); staking.unstake(_depositor2, stakeId, 0); } @@ -1588,8 +1532,6 @@ contract OwnableFunctions is StakingTest { _newRewardsDistributor != address(govToken) ); - vm.expectEmit(); - emit BaseStaking.NewRewardsDistributor(_newRewardsDistributor); staking.setRewardsDistributor(_newRewardsDistributor); assertEq( @@ -1715,7 +1657,7 @@ contract ViewFunctions is StakingTest { function testFuzz_Revertif_MaxWithdrawDepositorHasNoStakes( address _depositor ) public { - vm.expectRevert(BaseStaking.UserHasNoShares.selector); + vm.expectRevert(Staking.UserHasNoShares.selector); staking.maxWithdraw(_depositor); } @@ -1829,7 +1771,7 @@ contract ViewFunctions is StakingTest { assertEq(assets, _assets, "Wrong assets"); } - function testFuzz_GetKeyperStakeIds( + function testFuzz_GetUserStakeIds( address _depositor, uint256 _amount1, uint256 _amount2 diff --git a/test/helpers/DelegateStakingHarness.sol b/test/helpers/DelegateStakingHarness.sol index 738570b..b1348f5 100644 --- a/test/helpers/DelegateStakingHarness.sol +++ b/test/helpers/DelegateStakingHarness.sol @@ -7,4 +7,17 @@ contract DelegateStakingHarness is DelegateStaking { function exposed_nextStakeId() external view returns (uint256) { return nextStakeId; } + + function exposed_previewWithdraw( + uint256 amount + ) external view returns (uint256) { + return _previewWithdraw(amount); + } + + function exposed_calculateWithdrawAmount( + uint256 _amount, + uint256 _maxWithdrawAmount + ) external view returns (uint256) { + return _calculateWithdrawAmount(_amount, _maxWithdrawAmount); + } } diff --git a/test/helpers/StakingHarness.sol b/test/helpers/StakingHarness.sol index 81e44a8..0a99125 100644 --- a/test/helpers/StakingHarness.sol +++ b/test/helpers/StakingHarness.sol @@ -14,4 +14,10 @@ contract StakingHarness is Staking { ) external view virtual returns (uint256) { return _maxWithdraw(keyper, unlockedAmount); } + + function exposed_previewWithdraw( + uint256 amount + ) external view returns (uint256) { + return _previewWithdraw(amount); + } }