From dfbdf63cef045322023e4cb39a2ec4f25077bf88 Mon Sep 17 00:00:00 2001 From: DrZoltanFazekas Date: Mon, 2 Dec 2024 17:07:50 +0100 Subject: [PATCH 1/5] Extend state.sh logging --- state.sh | 28 +++++++++++++++++++++------- 1 file changed, 21 insertions(+), 7 deletions(-) diff --git a/state.sh b/state.sh index 41674c6..c0745c5 100755 --- a/state.sh +++ b/state.sh @@ -1,7 +1,7 @@ #!/bin/bash -if [ $# -ne 2 ]; then - echo "Provide the delegation contract address and a staker address as arguments." +if [ $# -lt 2 ]; then + echo "Provide the delegation contract address, a staker address and optionally, a block number as arguments." exit 1 fi @@ -12,8 +12,13 @@ if [[ "$variant" == "$temp" ]]; then exit 1 fi -block=$(cast rpc eth_blockNumber --rpc-url http://localhost:4201) -block_num=$(echo $block | tr -d '"' | cast to-dec --base-in 16) +if [ $# -eq 3 ]; then + block_num=$3 + block=$(echo $block_num | cast to-hex --base-in 10) +else + block=$(cast rpc eth_blockNumber --rpc-url http://localhost:4201) + block_num=$(echo $block | tr -d '"' | cast to-dec --base-in 16) +fi echo $(date +"%T,%3N") $block_num owner=$(cast call $1 "owner()(address)" --block $block_num --rpc-url http://localhost:4201) @@ -71,6 +76,15 @@ claimable=$(cast call $1 "getClaimable()(uint256)" --from $2 --block $block_num echo staker claimable: $(cast to-unit $claimable ether) ZIL echo validator deposit: $(cast to-unit $stake ether) ZIL -balance=$(cast rpc eth_getBalance $1 $block --rpc-url http://localhost:4201 | tr -d '"' | cast to-dec --base-in 16) -echo validator balance: $(cast to-unit $balance ether) ZIL -echo pending withdrawals: $(cast call $1 "getTotalWithdrawals()(uint256)" --block $block_num --rpc-url http://localhost:4201 | sed 's/\[[^]]*\]//g') wei \ No newline at end of file + +validatorBalance=$(cast rpc eth_getBalance $1 $block --rpc-url http://localhost:4201 | tr -d '"' | cast to-dec --base-in 16) +echo validator balance: $(cast to-unit $validatorBalance ether) ZIL + +pendingWithdrawals=$(cast call $1 "getTotalWithdrawals()(uint256)" --block $block_num --rpc-url http://localhost:4201 | sed 's/\[[^]]*\]//g') +echo pending withdrawals: $(cast to-unit $pendingWithdrawals ether) ZIL + +totalStake=$(cast call 0x000000000000000000005a494C4445504F534954 "getTotalStake()(uint256)" --block $block_num --rpc-url http://localhost:4201 | sed 's/\[[^]]*\]//g') +echo total stake: $(cast to-unit $totalStake ether) ZIL + +depositBalance=$(cast rpc eth_getBalance 0x000000000000000000005a494C4445504F534954 $block --rpc-url http://localhost:4201 | tr -d '"' | cast to-dec --base-in 16) +echo deposit balance: $(cast to-unit $depositBalance ether) ZIL From 8a07d80475bf6543f5cd85bd512a9462ef403647 Mon Sep 17 00:00:00 2001 From: DrZoltanFazekas Date: Tue, 3 Dec 2024 06:44:34 +0100 Subject: [PATCH 2/5] Improve readme file --- README.md | 51 ++++++++++++++++++++++++++++++--------------------- state.sh | 2 +- 2 files changed, 31 insertions(+), 22 deletions(-) diff --git a/README.md b/README.md index 0eceb33..179bf98 100644 --- a/README.md +++ b/README.md @@ -1,25 +1,27 @@ # Delegated Staking -This repo contains the contracts and scripts needed to activate a validator that users can stake ZIL with. Currently, there are two variants of the contracts. When delegating stake to the liquid variant, users receive a non-rebasing **liquid staking token** (LST) that anyone can send to the validator's delegation contract later on to withdraw the staked ZIL plus the corresponding share of the validator rewards. When delegating stake to the non-liquid variant, the delegator can withdraw rewards. +This repository contains the contracts and scripts needed to activate a validator that users can delegate stake to. Currently, there are two variants of the contracts: +1. When delegating stake to the liquid variant, users receive a non-rebasing **liquid staking token** (LST) that anyone can send to the validator's contract later on to withdraw the stake plus the corresponding share of the validator rewards. +1. When delegating stake to the non-liquid variant, the users can regularly withdraw their share of the rewards without withdrawing their stake. +## Prerequisites Install Foundry (https://book.getfoundry.sh/getting-started/installation) and the OpenZeppelin contracts before proceeding with the deployment: ``` forge install OpenZeppelin/openzeppelin-contracts-upgradeable --no-commit forge install OpenZeppelin/openzeppelin-contracts --no-commit ``` -The Zilliqa 2.0 deposit contract must be compiled for the tests in this repository to work. Specify the folder containing the `deposit.sol` file in `remappings.txt`: +The Zilliqa 2.0 deposit contract must be compiled for the tests included in this repository to work. Specify the folder containing the `deposit.sol` file in `remappings.txt`: ``` @zilliqa/zq2/=/home/user/zq2/zilliqa/src/contracts/ ``` ## Contract Deployment -The delegation contract is used by delegators to stake and unstake ZIL with the respective validator. It acts as the validator node's control address and interacts with the `Deposit` system contract. +The delegation contract is used by delegators to stake and unstake ZIL with the respective validator. It acts as the validator node's control address and interacts with the deposit contract. `BaseDelegation` is an abstract contract that concrete implementations inherit from. +`LiquidDelegation` is the initial version of the liquid staking variant of the delegation contract that creates a `NonRebasingLST` contract when it is initialized. `LiquidDelegationV2` contains the full implementation including the LST price calculation and other features. `NonLiquidDelegation` is the initial version of the non-liquid staking variant of the delegation contract. `NonLiquidDelegationV2` contains the full implementation that allows delegators to withdraw rewards. -`LiquidDelegation` is the initial implementation of the liquid staking variant of the delegation contract that creates a `NonRebasingLST` contract when it is initialized. `LiquidDelegationV2` implements all other features. `NonLiquidDelegation` is the initial implementation of the non-liquid staking variant of the delegation contract that allows delegators to withdraw rewards. - -Before running the deployment script, set the `PRIVATE_KEY` environment variable to the private key of the contract owner. Note that only the contract owner will be able to upgrade the contract, change the commission and activate the node as a validator. +Before running the deployment script, set the `PRIVATE_KEY` environment variable to the private key of the contract owner. Note that only the contract owner will be able to upgrade the contract, change the commission rate and activate the node as a validator. To deploy `LiquidDelegation` run ```bash @@ -40,13 +42,13 @@ You will see an output like this: Owner is 0x15fc323DFE5D5DCfbeEdc25CEcbf57f676634d77 ``` -You will need the proxy address from the above output in all commands below. If you have the address of a proxy contract but don't know which variant of staking it supports, run +You and your delegators will need the proxy address from the above output in all commands below. If you know the address of a proxy contract but don't know which variant of staking it supports, run ```bash forge script script/variant_Delegation.s.sol --rpc-url http://localhost:4201 --sig "run(address payable)" 0x7A0b7e6D24eDe78260c9ddBD98e828B0e11A8EA2 ``` The output will be `ILiquidStaking`, `INonLiquidStaking` or none of them if the address is not a valid delegation contract. -To upgrade the contract to `LiquidDelegationV2` or `NonLiquidDelegationV2` depending on the staking model it implements, run +To use the delegation contract, upgrade it to the latest version of `LiquidDelegationV2` or `NonLiquidDelegationV2` depending on the staking model it implements, by running ```bash forge script script/upgrade_Delegation.s.sol --rpc-url http://localhost:4201 --broadcast --legacy --sig "run(address payable)" 0x7A0b7e6D24eDe78260c9ddBD98e828B0e11A8EA2 ``` @@ -60,6 +62,9 @@ The output will look like this: Upgraded to version: 2 ``` +To adapt the contract to your needs, create your own copy of `LiquidDelegationV2` or `NonLiquidDelegationV2` and run the above upgrade script again. + + ## Contract Configuration Now or at a later time you can set the commission on the rewards the validator earns to e.g. 10% as follows: @@ -85,6 +90,7 @@ forge script script/commission_Delegation.s.sol --rpc-url http://localhost:4201 ``` using `same` for the second argument to leave the commission percentage unchanged and `true` for the third argument. Replacing the second argument with `same` and the third argument with `false` only displays the current commission rate. + ## Validator Activation If your node's account has enough ZIL for the minimum stake required, you can activate your node as a validator with a deposit of e.g. 10 million ZIL. Run ```bash @@ -110,12 +116,13 @@ to deposit all of it. Note that the deposit will not take effect and the node will not start earning rewards until the epoch after next. + ## Staking and Unstaking -If the delegation contract has been deployed and upgraded to the latest version, the validator can accept delegations. In order to stake e.g. 200 ZIL, run +Once the delegation contract has been deployed and upgraded to the latest version, your node can accept delegations. In order to stake e.g. 200 ZIL, your delegators must run ```bash forge script script/stake_Delegation.s.sol --rpc-url http://localhost:4201 --broadcast --legacy --sig "run(address payable, uint256)" 0x7A0b7e6D24eDe78260c9ddBD98e828B0e11A8EA2 200000000000000000000 --private-key 0x... ``` -with the private key of the delegator account. Make sure the account's balance can cover the transaction fees plus the 200 ZIL to be delegated. +with the private key of their account. It's important to make sure the account's balance can cover the transaction fees plus the 200 ZIL to be delegated. The output will look like this for liquid staking: ``` @@ -140,18 +147,18 @@ Due to the fact that the above output was generated based on the local script ex cast call 0x9e5c257D1c6dF74EaA54e58CdccaCb924669dc83 "balanceOf(address)(uint256)" 0xd819fFcE7A58b1E835c25617Db7b46a00888B013 --rpc-url http://localhost:4201 | sed 's/\[[^]]*\]//g' ``` -You can copy the LST address from the above output and add it to your wallet to transfer your liquid staking tokens to another account if you want to. +Your delegators can copy the LST address from the above output and add it to their wallet to transfer their liquid staking tokens to another account if they want to. To query the current price of an LST, run ```bash cast to-unit $(cast call 0x7A0b7e6D24eDe78260c9ddBD98e828B0e11A8EA2 "getPrice()(uint256)" --block latest --rpc-url http://localhost:4201 | sed 's/\[[^]]*\]//g') ether ``` -To unstake e.g. 100 LST, run +To unstake e.g. 100 LST (liquid variant) or 100 ZIL (non-liquid variant), run ```bash forge script script/unstake_Delegation.s.sol --rpc-url http://localhost:4201 --broadcast --legacy --sig "run(address payable, uint256)" 0x7A0b7e6D24eDe78260c9ddBD98e828B0e11A8EA2 100000000000000000000 --private-key 0x... ``` -with the private key of an account that holds some LST. +using the private key of an account that holds some LST in case of the liquid variant or using the private key of the delegator account in case of the non-liquid variant. The output will look like this for liquid staking: ``` @@ -175,7 +182,7 @@ The ZIL balance hasn't increased because the unstaked amount can not be transfer ```bash forge script script/claim_Delegation.s.sol --rpc-url http://localhost:4201 --broadcast --legacy --sig "run(address payable)" 0x7A0b7e6D24eDe78260c9ddBD98e828B0e11A8EA2 --private-key 0x... ``` -with the private key of an account that unstaked some LST. +with the private key of the account that unstaked in the previous step above. The output will look like this: ``` @@ -184,10 +191,12 @@ The output will look like this: Staker balance after: 99798095485861371162343 wei ``` -To query how much ZIL you can already claim, run +To query how much ZIL a user can already claim, run ```bash cast to-unit $(cast call 0x7A0b7e6D24eDe78260c9ddBD98e828B0e11A8EA2 "getClaimable()(uint256)" --from 0xd819fFcE7A58b1E835c25617Db7b46a00888B013 --block latest --rpc-url http://localhost:4201 | sed 's/\[[^]]*\]//g') ether ``` +with the user's address as an argument. + ## Staking and Withdrawing Rewards In the liquid staking variant, only you as the node operator can stake the rewards accrued by the node. To do so, run @@ -195,25 +204,25 @@ In the liquid staking variant, only you as the node operator can stake the rewar forge script script/stakeRewards_Delegation.s.sol --rpc-url http://localhost:4201 --broadcast --legacy --sig "run(address payable)" 0x7A0b7e6D24eDe78260c9ddBD98e828B0e11A8EA2 --private-key 0x... ``` -In the non-liquid variant of staking, delegators can stake or withdraw their share of the rewards earned by the validator. To query the amount of rewards available, run +In the non-liquid variant of staking, your delegators can stake or withdraw their share of the rewards. To query the amount of rewards available, run ```bash cast to-unit $(cast call 0x7A0b7e6D24eDe78260c9ddBD98e828B0e11A8EA2 "rewards()(uint256)" --from 0xd819fFcE7A58b1E835c25617Db7b46a00888B013 --block latest --rpc-url http://localhost:4201 | sed 's/\[[^]]*\]//g') ether ``` -In case you haven't withdrawn rewards for a long time during which many delegators staked or unstaked, the gas used by the above function might hit the block limit. In this case you can withdraw rewards from the period between the (un)staking until which you withdrew rewards last time and the `n`th subsequent (un)staking. Choose a number `0 <= n <= 11000` e.g. `100` and run +In case users haven't withdrawn rewards for a long time during which many delegators staked or unstaked, the gas used by the above function might hit the block limit. In this case rewards can be withdrawn from the period between the (un)staking until which they were withdrawn last time and the `n`th subsequent (un)staking. This can be repeated several times to withdraw all rewards using multiple transactions. To calculate the rewards that can be withdrawn in the next transaction, choose a number `0 <= n <= 11000` e.g. `100` and run ```bash cast to-unit $(cast call 0x7A0b7e6D24eDe78260c9ddBD98e828B0e11A8EA2 "rewards(uint64)(uint256)" 100 --from 0xd819fFcE7A58b1E835c25617Db7b46a00888B013 --block latest --rpc-url http://localhost:4201 | sed 's/\[[^]]*\]//g') ether ``` Note that `n` actually denotes the number of additional (un)stakings so that at least one is always reflected in the result, even if you specified `n = 0`. -To withdraw 1 ZIL of rewards using `n = 100`, run +To withdraw e.g. 1 ZIL of rewards using `n = 100`, run ```bash forge script script/withdrawRewards_Delegation.s.sol --rpc-url http://localhost:4201 --broadcast --legacy --sig "run(address payable, string, string)" 0x7A0b7e6D24eDe78260c9ddBD98e828B0e11A8EA2 1000000000000000000 100 --private-key 0x... ``` -with the private key of a staked account. To withdrawn as much as possible with the given value of `n` set the amount to `all`. To withdraw the chosen amount without setting `n` replace `n` with `all`. To withdraw all rewards replace both the amount and `n` with `all`. +with the private key of a delegator account. To withdraw as much as possible with the given value of `n` set the amount to `all`. To withdraw the chosen amount without setting `n` replace `n` with `all`. To withdraw all rewards replace both the amount and `n` with `all`. -Last but not least, in order to stake rewards instead of withdrawing them, run +Last but not least, in order to stake rewards instead of withdrawing them, your delegators can run ```bash forge script script/stakeRewards_Delegation.s.sol --rpc-url http://localhost:4201 --broadcast --legacy --sig "run(address payable)" 0x7A0b7e6D24eDe78260c9ddBD98e828B0e11A8EA2 --private-key 0x... ``` -with the private key of a staked account. +using the private key of their account. diff --git a/state.sh b/state.sh index c0745c5..98ce980 100755 --- a/state.sh +++ b/state.sh @@ -83,7 +83,7 @@ echo validator balance: $(cast to-unit $validatorBalance ether) ZIL pendingWithdrawals=$(cast call $1 "getTotalWithdrawals()(uint256)" --block $block_num --rpc-url http://localhost:4201 | sed 's/\[[^]]*\]//g') echo pending withdrawals: $(cast to-unit $pendingWithdrawals ether) ZIL -totalStake=$(cast call 0x000000000000000000005a494C4445504F534954 "getTotalStake()(uint256)" --block $block_num --rpc-url http://localhost:4201 | sed 's/\[[^]]*\]//g') +totalStake=$(cast call 0x000000000000000000005a494C4445504F534954 "getFutureTotalStake()(uint256)" --block $block_num --rpc-url http://localhost:4201 | sed 's/\[[^]]*\]//g') echo total stake: $(cast to-unit $totalStake ether) ZIL depositBalance=$(cast rpc eth_getBalance 0x000000000000000000005a494C4445504F534954 $block --rpc-url http://localhost:4201 | tr -d '"' | cast to-dec --base-in 16) From f0b4d20fa92fb63ec4f6e3a1d003f53d278e0da9 Mon Sep 17 00:00:00 2001 From: DrZoltanFazekas Date: Wed, 4 Dec 2024 13:31:01 +0100 Subject: [PATCH 3/5] Remove duplicate code from the test contracts --- src/BaseDelegation.sol | 12 ++ src/LiquidDelegation.sol | 16 ++ src/LiquidDelegationV2.sol | 4 +- src/NonLiquidDelegation.sol | 16 ++ src/NonLiquidDelegationV2.sol | 4 +- test/BaseDelegation.t.sol | 174 ++++++++++++++++++++++ test/LiquidDelegation.t.sol | 240 +++++++----------------------- test/NonLiquidDelegation.t.sol | 259 +++++++-------------------------- 8 files changed, 331 insertions(+), 394 deletions(-) create mode 100644 test/BaseDelegation.t.sol diff --git a/src/BaseDelegation.sol b/src/BaseDelegation.sol index 8e56c79..c9d0fb3 100644 --- a/src/BaseDelegation.sol +++ b/src/BaseDelegation.sol @@ -119,6 +119,18 @@ abstract contract BaseDelegation is Delegation, PausableUpgradeable, Ownable2Ste require(success, "deposit failed"); } + function deposit( + bytes calldata blsPubKey, + bytes calldata peerId, + bytes calldata signature + ) public virtual payable; + + function deposit2( + bytes calldata blsPubKey, + bytes calldata peerId, + bytes calldata signature + ) public virtual; + function _increaseDeposit(uint256 amount) internal virtual { // topup the deposit only if already activated as a validator if (_isActivated()) { diff --git a/src/LiquidDelegation.sol b/src/LiquidDelegation.sol index b2ba7ce..3d7e068 100644 --- a/src/LiquidDelegation.sol +++ b/src/LiquidDelegation.sol @@ -44,6 +44,22 @@ contract LiquidDelegation is BaseDelegation, ILiquidDelegation { return $.lst; } + function deposit( + bytes calldata blsPubKey, + bytes calldata peerId, + bytes calldata signature + ) public override payable { + revert("not implemented"); + } + + function deposit2( + bytes calldata blsPubKey, + bytes calldata peerId, + bytes calldata signature + ) public override { + revert("not implemented"); + } + function stake() external override payable { revert("not implemented"); } diff --git a/src/LiquidDelegationV2.sol b/src/LiquidDelegationV2.sol index 8d2debb..61bd05a 100644 --- a/src/LiquidDelegationV2.sol +++ b/src/LiquidDelegationV2.sol @@ -51,7 +51,7 @@ contract LiquidDelegationV2 is BaseDelegation, ILiquidDelegation { bytes calldata blsPubKey, bytes calldata peerId, bytes calldata signature - ) public onlyOwner { + ) public override onlyOwner { _deposit( blsPubKey, peerId, @@ -67,7 +67,7 @@ contract LiquidDelegationV2 is BaseDelegation, ILiquidDelegation { bytes calldata blsPubKey, bytes calldata peerId, bytes calldata signature - ) public payable onlyOwner { + ) public override payable onlyOwner { _deposit( blsPubKey, peerId, diff --git a/src/NonLiquidDelegation.sol b/src/NonLiquidDelegation.sol index 9723257..efbd65b 100644 --- a/src/NonLiquidDelegation.sol +++ b/src/NonLiquidDelegation.sol @@ -39,6 +39,22 @@ contract NonLiquidDelegation is BaseDelegation, INonLiquidDelegation { __BaseDelegation_init(initialOwner); } + function deposit( + bytes calldata blsPubKey, + bytes calldata peerId, + bytes calldata signature + ) public override payable { + revert("not implemented"); + } + + function deposit2( + bytes calldata blsPubKey, + bytes calldata peerId, + bytes calldata signature + ) public override { + revert("not implemented"); + } + function stake() external payable override { revert("not implemented"); } diff --git a/src/NonLiquidDelegationV2.sol b/src/NonLiquidDelegationV2.sol index 3febd65..f2f01a1 100644 --- a/src/NonLiquidDelegationV2.sol +++ b/src/NonLiquidDelegationV2.sol @@ -120,7 +120,7 @@ contract NonLiquidDelegationV2 is BaseDelegation, INonLiquidDelegation { bytes calldata blsPubKey, bytes calldata peerId, bytes calldata signature - ) public onlyOwner { + ) public override onlyOwner { _deposit( blsPubKey, peerId, @@ -136,7 +136,7 @@ contract NonLiquidDelegationV2 is BaseDelegation, INonLiquidDelegation { bytes calldata blsPubKey, bytes calldata peerId, bytes calldata signature - ) public payable onlyOwner { + ) public payable override onlyOwner { _deposit( blsPubKey, peerId, diff --git a/test/BaseDelegation.t.sol b/test/BaseDelegation.t.sol new file mode 100644 index 0000000..da7fc54 --- /dev/null +++ b/test/BaseDelegation.t.sol @@ -0,0 +1,174 @@ +// SPDX-License-Identifier: MIT OR Apache-2.0 +pragma solidity ^0.8.26; + +import {BaseDelegation} from "src/BaseDelegation.sol"; +import {Delegation} from "src/Delegation.sol"; +import {Deposit, InitialStaker} from "@zilliqa/zq2/deposit.sol"; +import {Console} from "src/Console.sol"; +import {ERC1967Proxy} from "@openzeppelin/contracts/proxy/ERC1967/ERC1967Proxy.sol"; +import {Test, Vm} from "forge-std/Test.sol"; +import "forge-std/console.sol"; + +contract PopVerifyPrecompile { + function popVerify(bytes memory, bytes memory) public pure returns(bool) { + return true; + } +} + +abstract contract BaseDelegationTest is Test { + address payable proxy; + address oldImplementation; + bytes initializerCall; + address payable newImplementation; + bytes reinitializerCall; + address owner; + address[4] stakers = [ + 0xd819fFcE7A58b1E835c25617Db7b46a00888B013, + 0x092E5E57955437876dA9Df998C96e2BE19341670, + 0xeA78aAE5Be606D2D152F00760662ac321aB8F017, + 0x6603A37980DF7ef6D44E994B3183A15D0322B7bF + ]; + + constructor() { + uint256 deployerPrivateKey = vm.envUint("PRIVATE_KEY"); + owner = vm.addr(deployerPrivateKey); + //console.log("Signer is %s", owner); + } + + function storeDelegation() internal virtual; + + function setUp() public { + vm.chainId(33469); + vm.deal(owner, 100_000 ether); + vm.startPrank(owner); + + proxy = payable( + new ERC1967Proxy(oldImplementation, initializerCall) + ); + /* + console.log( + "Proxy deployed: %s \r\n Implementation deployed: %s", + proxy, + oldImplementation + ); + //*/ + + BaseDelegation oldDelegation = BaseDelegation( + proxy + ); + /* + console.log("Deployed version: %s", + oldDelegation.version() + ); + + console.log("Owner is %s", + oldDelegation.owner() + ); + //*/ + + /* + console.log("New implementation deployed: %s", + newImplementation + ); + //*/ + + oldDelegation.upgradeToAndCall( + newImplementation, + reinitializerCall + ); + + storeDelegation(); + BaseDelegation delegation = BaseDelegation( + proxy + ); + + /* + console.log("Upgraded to version: %s", + delegation.version() + ); + //*/ + /* + Console.log("Old commission rate: %s.%s%s%%", + delegation.getCommissionNumerator(), + 2 + ); + //*/ + uint256 commissionNumerator = 1_000; + delegation.setCommissionNumerator(commissionNumerator); + /* + Console.log("New commission rate: %s.%s%s%%", + delegation.getCommissionNumerator(), + 2 + ); + //*/ + + InitialStaker[] memory initialStakers = new InitialStaker[](0); + //vm.deployCodeTo("Deposit.sol", delegation.DEPOSIT_CONTRACT()); + vm.etch( + delegation.DEPOSIT_CONTRACT(), //0x000000000000000000005a494C4445504F534954, + address(new Deposit(10_000_000 ether, 256, 10, initialStakers)).code + ); + vm.store(delegation.DEPOSIT_CONTRACT(), bytes32(uint256(11)), bytes32(uint256(block.number / 10))); + vm.store(delegation.DEPOSIT_CONTRACT(), bytes32(uint256(12)), bytes32(uint256(10_000_000 ether))); + vm.store(delegation.DEPOSIT_CONTRACT(), bytes32(uint256(13)), bytes32(uint256(256))); + vm.store(delegation.DEPOSIT_CONTRACT(), bytes32(uint256(14)), bytes32(uint256(10))); + /* + console.log("Deposit.minimimStake() =", Deposit(delegation.DEPOSIT_CONTRACT()).minimumStake()); + console.log("Deposit.maximumStakers() =", Deposit(delegation.DEPOSIT_CONTRACT()).maximumStakers()); + console.log("Deposit.blocksPerEpoch() =", Deposit(delegation.DEPOSIT_CONTRACT()).blocksPerEpoch()); + //*/ + + vm.etch(address(0x5a494c80), address(new PopVerifyPrecompile()).code); + + vm.stopPrank(); + } + + function deposit( + BaseDelegation delegation, + uint256 depositAmount, + bool initialDeposit + ) internal { + if (initialDeposit) { + vm.deal(owner, owner.balance + depositAmount); + vm.startPrank(owner); + + delegation.deposit{ + value: depositAmount + }( + bytes(hex"92fbe50544dce63cfdcc88301d7412f0edea024c91ae5d6a04c7cd3819edfc1b9d75d9121080af12e00f054d221f876c"), + bytes(hex"002408011220d5ed74b09dcbe84d3b32a56c01ab721cf82809848b6604535212a219d35c412f"), + bytes(hex"b14832a866a49ddf8a3104f8ee379d29c136f29aeb8fccec9d7fb17180b99e8ed29bee2ada5ce390cb704bc6fd7f5ce814f914498376c4b8bc14841a57ae22279769ec8614e2673ba7f36edc5a4bf5733aa9d70af626279ee2b2cde939b4bd8a") + ); + } else { + vm.deal(stakers[0], stakers[0].balance + depositAmount); + vm.startPrank(stakers[0]); + + vm.expectEmit( + true, + false, + false, + false, + address(delegation) + ); + emit Delegation.Staked( + stakers[0], + depositAmount, + "" + ); + + delegation.stake{ + value: depositAmount + }(); + + vm.startPrank(owner); + + delegation.deposit2( + bytes(hex"92fbe50544dce63cfdcc88301d7412f0edea024c91ae5d6a04c7cd3819edfc1b9d75d9121080af12e00f054d221f876c"), + bytes(hex"002408011220d5ed74b09dcbe84d3b32a56c01ab721cf82809848b6604535212a219d35c412f"), + bytes(hex"b14832a866a49ddf8a3104f8ee379d29c136f29aeb8fccec9d7fb17180b99e8ed29bee2ada5ce390cb704bc6fd7f5ce814f914498376c4b8bc14841a57ae22279769ec8614e2673ba7f36edc5a4bf5733aa9d70af626279ee2b2cde939b4bd8a") + ); + } + // wait 2 epochs for the change to the deposit to take affect + vm.roll(block.number + Deposit(delegation.DEPOSIT_CONTRACT()).blocksPerEpoch() * 2); + } +} \ No newline at end of file diff --git a/test/LiquidDelegation.t.sol b/test/LiquidDelegation.t.sol index c376ad7..51d01f8 100644 --- a/test/LiquidDelegation.t.sol +++ b/test/LiquidDelegation.t.sol @@ -1,133 +1,43 @@ // SPDX-License-Identifier: MIT OR Apache-2.0 pragma solidity ^0.8.26; +import {BaseDelegationTest, PopVerifyPrecompile} from "test/BaseDelegation.t.sol"; import {LiquidDelegation} from "src/LiquidDelegation.sol"; import {LiquidDelegationV2} from "src/LiquidDelegationV2.sol"; import {NonRebasingLST} from "src/NonRebasingLST.sol"; -import {WithdrawalQueue} from "src/BaseDelegation.sol"; +import {BaseDelegation, WithdrawalQueue} from "src/BaseDelegation.sol"; import {Delegation} from "src/Delegation.sol"; -import {Deposit, InitialStaker} from "@zilliqa/zq2/deposit.sol"; +import {Deposit} from "@zilliqa/zq2/deposit.sol"; import {Console} from "src/Console.sol"; -import {ERC1967Proxy} from "@openzeppelin/contracts/proxy/ERC1967/ERC1967Proxy.sol"; -import {Test, Vm} from "forge-std/Test.sol"; +import {Vm} from "forge-std/Test.sol"; import "forge-std/console.sol"; -contract PopVerifyPrecompile { - function popVerify(bytes memory, bytes memory) public pure returns(bool) { - return true; - } -} - -contract LiquidDelegationTest is Test { - address payable proxy; +contract LiquidDelegationTest is BaseDelegationTest { LiquidDelegationV2 delegation; NonRebasingLST lst; - address owner; - address staker = 0xd819fFcE7A58b1E835c25617Db7b46a00888B013; - - function setUp() public { - vm.chainId(33469); - uint256 deployerPrivateKey = vm.envUint("PRIVATE_KEY"); - owner = vm.addr(deployerPrivateKey); - //console.log("Signer is %s", owner); - vm.deal(owner, 100_000 ether); - vm.startPrank(owner); - - address oldImplementation = address( - new LiquidDelegation() - ); - bytes memory initializerCall = abi.encodeWithSelector( + constructor() BaseDelegationTest() { + oldImplementation = address(new LiquidDelegation()); + newImplementation = payable(new LiquidDelegationV2()); + initializerCall = abi.encodeWithSelector( LiquidDelegation.initialize.selector, owner ); - - proxy = payable( - new ERC1967Proxy(oldImplementation, initializerCall) - ); - /* - console.log( - "Proxy deployed: %s \r\n Implementation deployed: %s", - proxy, - oldImplementation - ); - //*/ - LiquidDelegation oldDelegation = LiquidDelegation( - proxy - ); - /* - console.log("Deployed version: %s", - oldDelegation.version() - ); - - console.log("Owner is %s", - oldDelegation.owner() - ); - //*/ - address payable newImplementation = payable( - new LiquidDelegationV2() - ); - /* - console.log("New implementation deployed: %s", - newImplementation - ); - //*/ - bytes memory reinitializerCall = abi.encodeWithSelector( + reinitializerCall = abi.encodeWithSelector( LiquidDelegationV2.reinitialize.selector ); + } - oldDelegation.upgradeToAndCall( - newImplementation, - reinitializerCall - ); - + function storeDelegation() internal override { delegation = LiquidDelegationV2( - proxy - ); - /* - console.log("Upgraded to version: %s", - delegation.version() + proxy ); - //*/ lst = NonRebasingLST(delegation.getLST()); /* console.log("LST address: %s", address(lst) ); - - Console.log("Old commission rate: %s.%s%s%%", - delegation.getCommissionNumerator(), - 2 - ); //*/ - uint256 commissionNumerator = 1_000; - delegation.setCommissionNumerator(commissionNumerator); - /* - Console.log("New commission rate: %s.%s%s%%", - delegation.getCommissionNumerator(), - 2 - ); - //*/ - - InitialStaker[] memory initialStakers = new InitialStaker[](0); - //vm.deployCodeTo("Deposit.sol", delegation.DEPOSIT_CONTRACT()); - vm.etch( - delegation.DEPOSIT_CONTRACT(), //0x000000000000000000005a494C4445504F534954, - address(new Deposit(10_000_000 ether, 256, 10, initialStakers)).code - ); - vm.store(delegation.DEPOSIT_CONTRACT(), bytes32(uint256(11)), bytes32(uint256(block.number / 10))); - vm.store(delegation.DEPOSIT_CONTRACT(), bytes32(uint256(12)), bytes32(uint256(10_000_000 ether))); - vm.store(delegation.DEPOSIT_CONTRACT(), bytes32(uint256(13)), bytes32(uint256(256))); - vm.store(delegation.DEPOSIT_CONTRACT(), bytes32(uint256(14)), bytes32(uint256(10))); - /* - console.log("Deposit.minimimStake() =", Deposit(delegation.DEPOSIT_CONTRACT()).minimumStake()); - console.log("Deposit.maximumStakers() =", Deposit(delegation.DEPOSIT_CONTRACT()).maximumStakers()); - console.log("Deposit.blocksPerEpoch() =", Deposit(delegation.DEPOSIT_CONTRACT()).blocksPerEpoch()); - //*/ - - vm.etch(address(0x5a494c80), address(new PopVerifyPrecompile()).code); - - vm.stopPrank(); } function run( @@ -144,53 +54,12 @@ contract LiquidDelegationTest is Test { delegation = LiquidDelegationV2(proxy); lst = NonRebasingLST(delegation.getLST()); - if (initialDeposit) { - vm.deal(owner, owner.balance + depositAmount); - vm.startPrank(owner); - - delegation.deposit{ - value: depositAmount - }( - bytes(hex"92fbe50544dce63cfdcc88301d7412f0edea024c91ae5d6a04c7cd3819edfc1b9d75d9121080af12e00f054d221f876c"), - bytes(hex"002408011220d5ed74b09dcbe84d3b32a56c01ab721cf82809848b6604535212a219d35c412f"), - bytes(hex"b14832a866a49ddf8a3104f8ee379d29c136f29aeb8fccec9d7fb17180b99e8ed29bee2ada5ce390cb704bc6fd7f5ce814f914498376c4b8bc14841a57ae22279769ec8614e2673ba7f36edc5a4bf5733aa9d70af626279ee2b2cde939b4bd8a") - ); - } else { - vm.deal(staker, staker.balance + depositAmount); - vm.startPrank(staker); - - vm.expectEmit( - true, - false, - false, - true, - address(delegation) - ); - emit Delegation.Staked( - staker, - depositAmount, - abi.encode(depositAmount) - ); - - delegation.stake{ - value: depositAmount - }(); - - vm.startPrank(owner); - - delegation.deposit2( - bytes(hex"92fbe50544dce63cfdcc88301d7412f0edea024c91ae5d6a04c7cd3819edfc1b9d75d9121080af12e00f054d221f876c"), - bytes(hex"002408011220d5ed74b09dcbe84d3b32a56c01ab721cf82809848b6604535212a219d35c412f"), - bytes(hex"b14832a866a49ddf8a3104f8ee379d29c136f29aeb8fccec9d7fb17180b99e8ed29bee2ada5ce390cb704bc6fd7f5ce814f914498376c4b8bc14841a57ae22279769ec8614e2673ba7f36edc5a4bf5733aa9d70af626279ee2b2cde939b4bd8a") - ); - } - // wait 2 epochs for the change to the deposit to take affect - vm.roll(block.number + Deposit(delegation.DEPOSIT_CONTRACT()).blocksPerEpoch() * 2); + deposit(BaseDelegation(delegation), depositAmount, initialDeposit); vm.store(address(delegation), 0xfa57cbed4b267d0bc9f2cbdae86b4d1d23ca818308f873af9c968a23afadfd01, bytes32(taxedRewardsBeforeStaking)); vm.deal(address(delegation), rewardsBeforeStaking); - vm.deal(staker, 100_000 ether); - vm.startPrank(staker); + vm.deal(stakers[0], 100_000 ether); + vm.startPrank(stakers[0]); Console.log("Deposit before staking: %s.%s%s ZIL", delegation.getStake() @@ -205,11 +74,11 @@ contract LiquidDelegationTest is Test { ); Console.log("Staker balance before staking: %s.%s%s ZIL", - staker.balance + stakers[0].balance ); Console.log("Staker balance before staking: %s.%s%s LST", - lst.balanceOf(staker) + lst.balanceOf(stakers[0]) ); Console.log("Total supply before staking: %s.%s%s LST", @@ -217,7 +86,6 @@ contract LiquidDelegationTest is Test { ); uint256[2] memory ownerZIL = [uint256(0), 0]; - uint256 ownerZILAfter; uint256 loggedAmount; uint256 loggedShares; uint256 totalShares; @@ -239,7 +107,7 @@ contract LiquidDelegationTest is Test { address(delegation) ); emit Delegation.Staked( - staker, + stakers[0], delegatedAmount, abi.encode(lst.totalSupply() * delegatedAmount / (delegation.getStake() + delegation.getRewards())) ); @@ -286,11 +154,11 @@ contract LiquidDelegationTest is Test { ); Console.log("Staker balance after staking: %s.%s%s ZIL", - staker.balance + stakers[0].balance ); Console.log("Staker balance after staking: %s.%s%s LST", - lst.balanceOf(staker) + lst.balanceOf(stakers[0]) ); Console.log("Total supply after staking: %s.%s%s LST", @@ -323,15 +191,15 @@ contract LiquidDelegationTest is Test { address(delegation) ); emit Delegation.Unstaked( - staker, - (delegation.getStake() + delegation.getRewards()) * lst.balanceOf(staker) / lst.totalSupply(), - abi.encode(lst.balanceOf(staker)) + stakers[0], + (delegation.getStake() + delegation.getRewards()) * lst.balanceOf(stakers[0]) / lst.totalSupply(), + abi.encode(lst.balanceOf(stakers[0])) ); - uint256[2] memory stakerLST = [lst.balanceOf(staker), 0]; + uint256[2] memory stakerLST = [lst.balanceOf(stakers[0]), 0]; ownerZIL[0] = delegation.owner().balance; - uint256 shares = initialDeposit ? lst.balanceOf(staker) : lst.balanceOf(staker) - depositAmount; + uint256 shares = initialDeposit ? lst.balanceOf(stakers[0]) : lst.balanceOf(stakers[0]) - depositAmount; assertEq(totalShares, shares, "staked shares balance mismatch"); delegation.unstake( @@ -341,7 +209,7 @@ contract LiquidDelegationTest is Test { // wait 2 epochs for the change to the deposit to take affect vm.roll(block.number + Deposit(delegation.DEPOSIT_CONTRACT()).blocksPerEpoch() * 2); - stakerLST[1] = lst.balanceOf(staker); + stakerLST[1] = lst.balanceOf(stakers[0]); ownerZIL[1] = delegation.owner().balance; assertEq((rewardsBeforeUnstaking - taxedRewardsAfterStaking) * delegation.getCommissionNumerator() / delegation.DENOMINATOR(), ownerZIL[1] - ownerZIL[0], "commission mismatch after unstaking"); @@ -377,13 +245,13 @@ contract LiquidDelegationTest is Test { taxedRewardsAfterUnstaking ); - uint256 stakerBalanceAfterUnstaking = staker.balance; + uint256 stakerBalanceAfterUnstaking = stakers[0].balance; Console.log("Staker balance after unstaking: %s.%s%s ZIL", stakerBalanceAfterUnstaking ); Console.log("Staker balance after unstaking: %s.%s%s LST", - lst.balanceOf(staker) + lst.balanceOf(stakers[0]) ); Console.log("Total supply after unstaking: %s.%s%s LST", @@ -407,17 +275,17 @@ contract LiquidDelegationTest is Test { address(delegation) ); emit Delegation.Claimed( - staker, + stakers[0], unstakedAmount, "" ); - uint256[2] memory stakerZIL = [staker.balance, 0]; + uint256[2] memory stakerZIL = [stakers[0].balance, 0]; ownerZIL[0] = delegation.owner().balance; delegation.claim(); - stakerZIL[1] = staker.balance; + stakerZIL[1] = stakers[0].balance; ownerZIL[1] = delegation.owner().balance; entries = vm.getRecordedLogs(); @@ -448,12 +316,12 @@ contract LiquidDelegationTest is Test { ); Console.log("Staker balance after claiming: %s.%s%s ZIL", - staker.balance + stakers[0].balance ); - assertEq(staker.balance, stakerBalanceAfterUnstaking + unstakedAmount, "final staker balance mismatch"); + assertEq(stakers[0].balance, stakerBalanceAfterUnstaking + unstakedAmount, "final staker balance mismatch"); Console.log("Staker balance after claiming: %s.%s%s LST", - lst.balanceOf(staker) + lst.balanceOf(stakers[0]) ); Console.log("Total supply after claiming: %s.%s%s LST", @@ -463,7 +331,7 @@ contract LiquidDelegationTest is Test { } function test_1a_LargeStake_Late_NoRewards_UnstakeAll() public { - staker = 0x092E5E57955437876dA9Df998C96e2BE19341670; + stakers[0] = 0x092E5E57955437876dA9Df998C96e2BE19341670; uint256 depositAmount = 10_000_000 ether; uint256 totalDeposit = 5_200_000_000 ether; uint256 delegatedAmount = 10_000 ether; @@ -488,7 +356,7 @@ contract LiquidDelegationTest is Test { //TODO: remove the test once https://github.com/Zilliqa/zq2/issues/1761 is fixed function test_DepositContract() public { vm.deal(owner, 10_000_000 ether + 1_000_000 ether + 0 ether); - vm.deal(staker, 0); + vm.deal(stakers[0], 0); vm.startPrank(owner); Deposit(delegation.DEPOSIT_CONTRACT()).deposit{ value: 10_000_000 ether @@ -496,7 +364,7 @@ contract LiquidDelegationTest is Test { bytes(hex"92fbe50544dce63cfdcc88301d7412f0edea024c91ae5d6a04c7cd3819edfc1b9d75d9121080af12e00f054d221f876c"), bytes(hex"002408011220d5ed74b09dcbe84d3b32a56c01ab721cf82809848b6604535212a219d35c412f"), bytes(hex"b14832a866a49ddf8a3104f8ee379d29c136f29aeb8fccec9d7fb17180b99e8ed29bee2ada5ce390cb704bc6fd7f5ce814f914498376c4b8bc14841a57ae22279769ec8614e2673ba7f36edc5a4bf5733aa9d70af626279ee2b2cde939b4bd8a"), - address(staker) + address(stakers[0]) ); console.log("validator deposited"); console.log("validator stake: %s", Deposit(delegation.DEPOSIT_CONTRACT()).getStake( @@ -542,7 +410,7 @@ contract LiquidDelegationTest is Test { } function test_1b_LargeStake_Early_NoRewards_UnstakeAll() public { - staker = 0x092E5E57955437876dA9Df998C96e2BE19341670; + stakers[0] = 0x092E5E57955437876dA9Df998C96e2BE19341670; uint256 depositAmount = 10_000_000 ether; uint256 totalDeposit = 5_200_000_000 ether; uint256 delegatedAmount = 10_000 ether; @@ -565,7 +433,7 @@ contract LiquidDelegationTest is Test { } function test_2a_LargeStake_Late_SmallValidator_OwnDeposit_OneYearOfRewards_UnstakeAll() public { - staker = 0x092E5E57955437876dA9Df998C96e2BE19341670; + stakers[0] = 0x092E5E57955437876dA9Df998C96e2BE19341670; uint256 depositAmount = 10_000_000 ether; uint256 totalDeposit = 5_200_000_000 ether; uint256 delegatedAmount = 10_000 ether; @@ -588,7 +456,7 @@ contract LiquidDelegationTest is Test { } function test_3a_SmallStake_Late_SmallValidator_OwnDeposit_OneYearOfRewards_UnstakeAll() public { - staker = 0x092E5E57955437876dA9Df998C96e2BE19341670; + stakers[0] = 0x092E5E57955437876dA9Df998C96e2BE19341670; uint256 depositAmount = 10_000_000 ether; uint256 totalDeposit = 5_200_000_000 ether; uint256 delegatedAmount = 100 ether; @@ -611,7 +479,7 @@ contract LiquidDelegationTest is Test { } function test_4a_LargeStake_Late_LargeValidator_OwnDeposit_OneYearOfRewards_UnstakeAll() public { - staker = 0x092E5E57955437876dA9Df998C96e2BE19341670; + stakers[0] = 0x092E5E57955437876dA9Df998C96e2BE19341670; uint256 depositAmount = 100_000_000 ether; uint256 totalDeposit = 5_200_000_000 ether; uint256 delegatedAmount = 10_000 ether; @@ -634,7 +502,7 @@ contract LiquidDelegationTest is Test { } function test_5a_SmallStake_Late_LargeValidator_OwnDeposit_OneYearOfRewards_UnstakeAll() public { - staker = 0x092E5E57955437876dA9Df998C96e2BE19341670; + stakers[0] = 0x092E5E57955437876dA9Df998C96e2BE19341670; uint256 depositAmount = 100_000_000 ether; uint256 totalDeposit = 5_200_000_000 ether; uint256 delegatedAmount = 100 ether; @@ -657,7 +525,7 @@ contract LiquidDelegationTest is Test { } function test_2b_LargeStake_Late_SmallValidator_DelegatedDeposit_OneYearOfRewards_UnstakeAll() public { - staker = 0x092E5E57955437876dA9Df998C96e2BE19341670; + stakers[0] = 0x092E5E57955437876dA9Df998C96e2BE19341670; uint256 depositAmount = 10_000_000 ether; uint256 totalDeposit = 5_200_000_000 ether; uint256 delegatedAmount = 10_000 ether; @@ -680,7 +548,7 @@ contract LiquidDelegationTest is Test { } function test_3b_SmallStake_Late_SmallValidator_DelegatedDeposit_OneYearOfRewards_UnstakeAll() public { - staker = 0x092E5E57955437876dA9Df998C96e2BE19341670; + stakers[0] = 0x092E5E57955437876dA9Df998C96e2BE19341670; uint256 depositAmount = 10_000_000 ether; uint256 totalDeposit = 5_200_000_000 ether; uint256 delegatedAmount = 100 ether; @@ -703,7 +571,7 @@ contract LiquidDelegationTest is Test { } function test_4b_LargeStake_Late_LargeValidator_DelegatedDeposit_OneYearOfRewards_UnstakeAll() public { - staker = 0x092E5E57955437876dA9Df998C96e2BE19341670; + stakers[0] = 0x092E5E57955437876dA9Df998C96e2BE19341670; uint256 depositAmount = 100_000_000 ether; uint256 totalDeposit = 5_200_000_000 ether; uint256 delegatedAmount = 10_000 ether; @@ -726,7 +594,7 @@ contract LiquidDelegationTest is Test { } function test_5b_SmallStake_Late_LargeValidator_DelegatedDeposit_OneYearOfRewards_UnstakeAll() public { - staker = 0x092E5E57955437876dA9Df998C96e2BE19341670; + stakers[0] = 0x092E5E57955437876dA9Df998C96e2BE19341670; uint256 depositAmount = 100_000_000 ether; uint256 totalDeposit = 5_200_000_000 ether; uint256 delegatedAmount = 100 ether; @@ -749,7 +617,7 @@ contract LiquidDelegationTest is Test { } function test_2c_LargeStake_Early_SmallValidator_OwnDeposit_OneYearOfRewards_UnstakeAll() public { - staker = 0x092E5E57955437876dA9Df998C96e2BE19341670; + stakers[0] = 0x092E5E57955437876dA9Df998C96e2BE19341670; uint256 depositAmount = 10_000_000 ether; uint256 totalDeposit = 5_200_000_000 ether; uint256 delegatedAmount = 10_000 ether; @@ -772,7 +640,7 @@ contract LiquidDelegationTest is Test { } function test_3c_SmallStake_Early_SmallValidator_OwnDeposit_OneYearOfRewards_UnstakeAll() public { - staker = 0x092E5E57955437876dA9Df998C96e2BE19341670; + stakers[0] = 0x092E5E57955437876dA9Df998C96e2BE19341670; uint256 depositAmount = 10_000_000 ether; uint256 totalDeposit = 5_200_000_000 ether; uint256 delegatedAmount = 100 ether; @@ -795,7 +663,7 @@ contract LiquidDelegationTest is Test { } function test_4c_LargeStake_Early_LargeValidator_OwnDeposit_OneYearOfRewards_UnstakeAll() public { - staker = 0x092E5E57955437876dA9Df998C96e2BE19341670; + stakers[0] = 0x092E5E57955437876dA9Df998C96e2BE19341670; uint256 depositAmount = 100_000_000 ether; uint256 totalDeposit = 5_200_000_000 ether; uint256 delegatedAmount = 10_000 ether; @@ -818,7 +686,7 @@ contract LiquidDelegationTest is Test { } function test_5c_SmallStake_Early_LargeValidator_OwnDeposit_OneYearOfRewards_UnstakeAll() public { - staker = 0x092E5E57955437876dA9Df998C96e2BE19341670; + stakers[0] = 0x092E5E57955437876dA9Df998C96e2BE19341670; uint256 depositAmount = 100_000_000 ether; uint256 totalDeposit = 5_200_000_000 ether; uint256 delegatedAmount = 100 ether; @@ -841,7 +709,7 @@ contract LiquidDelegationTest is Test { } function test_6a_ManyVsOneStake_UnstakeAll() public { - staker = 0x092E5E57955437876dA9Df998C96e2BE19341670; + stakers[0] = 0x092E5E57955437876dA9Df998C96e2BE19341670; uint256 depositAmount = 10_000_000 ether; uint256 totalDeposit = 110_000_000 ether; uint256 delegatedAmount = 10_000 ether; @@ -866,7 +734,7 @@ contract LiquidDelegationTest is Test { } function test_6b_OneVsManyStakes_UnstakeAll() public { - staker = 0x092E5E57955437876dA9Df998C96e2BE19341670; + stakers[0] = 0x092E5E57955437876dA9Df998C96e2BE19341670; uint256 depositAmount = 10_000_000 ether; uint256 totalDeposit = 110_000_000 ether; uint256 delegatedAmount = 90_000 ether; @@ -911,7 +779,7 @@ contract LiquidDelegationTest is Test { */ //TODO: update the values based on the devnet and fix the failing test (typo intentional) function est_0_ReproduceRealNetwork() public { - staker = 0xd819fFcE7A58b1E835c25617Db7b46a00888B013; + stakers[0] = 0xd819fFcE7A58b1E835c25617Db7b46a00888B013; uint256 delegatedAmount = 10_000 ether; // Insert the following values output by the STATE script below uint256 rewardsBeforeStaking = 197818620596390326580; diff --git a/test/NonLiquidDelegation.t.sol b/test/NonLiquidDelegation.t.sol index 6a5282f..7acb8c8 100644 --- a/test/NonLiquidDelegation.t.sol +++ b/test/NonLiquidDelegation.t.sol @@ -1,190 +1,41 @@ // SPDX-License-Identifier: MIT OR Apache-2.0 pragma solidity ^0.8.26; +import {BaseDelegationTest, PopVerifyPrecompile} from "test/BaseDelegation.t.sol"; import {NonLiquidDelegation} from "src/NonLiquidDelegation.sol"; import {NonLiquidDelegationV2} from "src/NonLiquidDelegationV2.sol"; -import {WithdrawalQueue} from "src/BaseDelegation.sol"; +import {BaseDelegation, WithdrawalQueue} from "src/BaseDelegation.sol"; import {Delegation} from "src/Delegation.sol"; -import {Deposit, InitialStaker} from "@zilliqa/zq2/deposit.sol"; +import {Deposit} from "@zilliqa/zq2/deposit.sol"; import {Console} from "src/Console.sol"; import {Strings} from "@openzeppelin/contracts/utils/Strings.sol"; -import {ERC1967Proxy} from "@openzeppelin/contracts/proxy/ERC1967/ERC1967Proxy.sol"; -import {Test, Vm} from "forge-std/Test.sol"; +import {Vm} from "forge-std/Test.sol"; import "forge-std/console.sol"; -contract PopVerifyPrecompile { - function popVerify(bytes memory, bytes memory) public pure returns(bool) { - return true; - } -} - -contract NonLiquidDelegationTest is Test { - address payable proxy; - address owner; +contract NonLiquidDelegationTest is BaseDelegationTest { NonLiquidDelegationV2 delegation; - address[4] staker = [ - 0xd819fFcE7A58b1E835c25617Db7b46a00888B013, - 0x092E5E57955437876dA9Df998C96e2BE19341670, - 0xeA78aAE5Be606D2D152F00760662ac321aB8F017, - 0x6603A37980DF7ef6D44E994B3183A15D0322B7bF - ]; - - function setUp() public { - vm.chainId(33469); - uint256 deployerPrivateKey = vm.envUint("PRIVATE_KEY"); - owner = vm.addr(deployerPrivateKey); - //console.log("Signer is %s", owner); - vm.deal(owner, 100_000 ether); - vm.startPrank(owner); - - address oldImplementation = address( - new NonLiquidDelegation() - ); - bytes memory initializerCall = abi.encodeWithSelector( + constructor() BaseDelegationTest() { + oldImplementation = address(new NonLiquidDelegation()); + newImplementation = payable(new NonLiquidDelegationV2()); + initializerCall = abi.encodeWithSelector( NonLiquidDelegation.initialize.selector, owner ); - - proxy = payable( - new ERC1967Proxy(oldImplementation, initializerCall) - ); - /* - console.log( - "Proxy deployed: %s \r\n Implementation deployed: %s", - proxy, - oldImplementation - ); - //*/ - - NonLiquidDelegation oldDelegation = NonLiquidDelegation( - proxy - ); - /* - console.log("Deployed version: %s", - oldDelegation.version() - ); - - console.log("Owner is %s", - oldDelegation.owner() - ); - //*/ - - address payable newImplementation = payable( - new NonLiquidDelegationV2() - ); - - /* - console.log("New implementation deployed: %s", - newImplementation - ); - //*/ - - bytes memory reinitializerCall = abi.encodeWithSelector( + reinitializerCall = abi.encodeWithSelector( NonLiquidDelegationV2.reinitialize.selector ); + } - oldDelegation.upgradeToAndCall( - newImplementation, - reinitializerCall - ); - + function storeDelegation() internal override { delegation = NonLiquidDelegationV2( proxy ); - - /* - console.log("Upgraded to version: %s", - delegation.version() - ); - //*/ - /* - Console.log("Old commission rate: %s.%s%s%%", - delegation.getCommissionNumerator(), - 2 - ); - //*/ - uint256 commissionNumerator = 1_000; - delegation.setCommissionNumerator(commissionNumerator); - /* - Console.log("New commission rate: %s.%s%s%%", - delegation.getCommissionNumerator(), - 2 - ); - //*/ - - InitialStaker[] memory initialStakers = new InitialStaker[](0); - //vm.deployCodeTo("Deposit.sol", delegation.DEPOSIT_CONTRACT()); - vm.etch( - delegation.DEPOSIT_CONTRACT(), //0x000000000000000000005a494C4445504F534954, - address(new Deposit(10_000_000 ether, 256, 10, initialStakers)).code - ); - vm.store(delegation.DEPOSIT_CONTRACT(), bytes32(uint256(11)), bytes32(uint256(block.number / 10))); - vm.store(delegation.DEPOSIT_CONTRACT(), bytes32(uint256(12)), bytes32(uint256(10_000_000 ether))); - vm.store(delegation.DEPOSIT_CONTRACT(), bytes32(uint256(13)), bytes32(uint256(256))); - vm.store(delegation.DEPOSIT_CONTRACT(), bytes32(uint256(14)), bytes32(uint256(10))); - /* - console.log("Deposit.minimimStake() =", Deposit(delegation.DEPOSIT_CONTRACT()).minimumStake()); - console.log("Deposit.maximumStakers() =", Deposit(delegation.DEPOSIT_CONTRACT()).maximumStakers()); - console.log("Deposit.blocksPerEpoch() =", Deposit(delegation.DEPOSIT_CONTRACT()).blocksPerEpoch()); - //*/ - - vm.etch(address(0x5a494c80), address(new PopVerifyPrecompile()).code); - - vm.stopPrank(); - } - - function deposit( - uint256 depositAmount, - bool initialDeposit - ) internal { - if (initialDeposit) { - vm.deal(owner, owner.balance + depositAmount); - vm.startPrank(owner); - - delegation.deposit{ - value: depositAmount - }( - bytes(hex"92fbe50544dce63cfdcc88301d7412f0edea024c91ae5d6a04c7cd3819edfc1b9d75d9121080af12e00f054d221f876c"), - bytes(hex"002408011220d5ed74b09dcbe84d3b32a56c01ab721cf82809848b6604535212a219d35c412f"), - bytes(hex"b14832a866a49ddf8a3104f8ee379d29c136f29aeb8fccec9d7fb17180b99e8ed29bee2ada5ce390cb704bc6fd7f5ce814f914498376c4b8bc14841a57ae22279769ec8614e2673ba7f36edc5a4bf5733aa9d70af626279ee2b2cde939b4bd8a") - ); - } else { - vm.deal(staker[0], staker[0].balance + depositAmount); - vm.startPrank(staker[0]); - - vm.expectEmit( - true, - false, - false, - true, - address(delegation) - ); - emit Delegation.Staked( - staker[0], - depositAmount, - "" - ); - - delegation.stake{ - value: depositAmount - }(); - - vm.startPrank(owner); - - delegation.deposit2( - bytes(hex"92fbe50544dce63cfdcc88301d7412f0edea024c91ae5d6a04c7cd3819edfc1b9d75d9121080af12e00f054d221f876c"), - bytes(hex"002408011220d5ed74b09dcbe84d3b32a56c01ab721cf82809848b6604535212a219d35c412f"), - bytes(hex"b14832a866a49ddf8a3104f8ee379d29c136f29aeb8fccec9d7fb17180b99e8ed29bee2ada5ce390cb704bc6fd7f5ce814f914498376c4b8bc14841a57ae22279769ec8614e2673ba7f36edc5a4bf5733aa9d70af626279ee2b2cde939b4bd8a") - ); - } - // wait 2 epochs for the change to the deposit to take affect - vm.roll(block.number + Deposit(delegation.DEPOSIT_CONTRACT()).blocksPerEpoch() * 2); } function findStaker(address a) internal view returns(uint256) { - for (uint256 i = 0; i < staker.length; i++) - if (staker[i] == a) + for (uint256 i = 0; i < stakers.length; i++) + if (stakers[i] == a) return i; revert("staker not found"); } @@ -192,7 +43,7 @@ contract NonLiquidDelegationTest is Test { function snapshot(string memory s, uint256 i, uint256 x) internal view { console.log("-----------------------------------------------"); console.log(s, i, x); - uint256[] memory shares = new uint256[](staker.length); + uint256[] memory shares = new uint256[](stakers.length); NonLiquidDelegationV2.Staking[] memory stakings = delegation.getStakingHistory(); for (i = 0; i < stakings.length; i++) //i = stakings.length - 1; @@ -201,7 +52,7 @@ contract NonLiquidDelegationTest is Test { shares[stakerIndex] = stakings[i].amount; s = string.concat("index: ", Strings.toString(i)); s = string.concat(s, "\tstaker "); - assertEq(stakings[i].staker, staker[stakerIndex], "found staker mismatch"); + assertEq(stakings[i].staker, stakers[stakerIndex], "found staker mismatch"); s = string.concat(s, Strings.toString(stakerIndex + 1)); s = string.concat(s, ": "); s = string.concat(s, Strings.toHexString(stakings[i].staker)); @@ -264,12 +115,12 @@ contract NonLiquidDelegationTest is Test { if (initialDeposit) // otherwise snapshot() doesn't find the staker and reverts - staker[0] = owner; - deposit(depositAmount, initialDeposit); + stakers[0] = owner; + deposit(BaseDelegation(delegation), depositAmount, initialDeposit); - for (uint256 i = 0; i < staker.length; i++) { - vm.deal(staker[i], 10 * depositAmount); - console.log("staker %s: %s", i+1, staker[i]); + for (uint256 i = 0; i < stakers.length; i++) { + vm.deal(stakers[i], 10 * depositAmount); + console.log("staker %s: %s", i+1, stakers[i]); } delegation = NonLiquidDelegationV2(proxy); @@ -280,7 +131,7 @@ contract NonLiquidDelegationTest is Test { for (uint256 i = 0; i < stakerIndicesBeforeWithdrawals.length; i++) { vm.deal(address(delegation), address(delegation).balance + rewardsAccruedAfterEach); int256 x = relativeAmountsBeforeWithdrawals[i] * int256(depositAmount) / 10; - vm.startPrank(staker[stakerIndicesBeforeWithdrawals[i]-1]); + vm.startPrank(stakers[stakerIndicesBeforeWithdrawals[i]-1]); if (x >= 0) { delegation.stake{value: uint256(x)}(); //snapshot("staker %s staked %s", stakerIndices[i], uint256(x)); @@ -297,14 +148,14 @@ contract NonLiquidDelegationTest is Test { //vm.deal(address(delegation), address(delegation).balance + rewardsAccruedAfterEach); for (uint256 i = 1; i <= 2; i++) { - vm.startPrank(staker[i-1]); + vm.startPrank(stakers[i-1]); if (steps == 123_456_789) snapshot("staker %s withdrawing all, remaining rewards:", i, 0); else snapshot("staker %s withdrawing 1+%s times", i, steps); Console.log("rewards accrued until last staking: %s.%s%s", delegation.getTotalRewards()); Console.log("delegation contract balance: %s.%s%s", address(delegation).balance); - //Console.log("staker balance: %s.%s%s", staker[i-1].balance); + //Console.log("staker balance: %s.%s%s", stakers[i-1].balance); Console.log("staker rewards: %s.%s%s", delegation.rewards()); if (steps == 123_456_789) Console.log("staker withdrew: %s.%s%s", delegation.withdrawAllRewards()); @@ -312,14 +163,14 @@ contract NonLiquidDelegationTest is Test { Console.log("staker withdrew: %s.%s%s", delegation.withdrawRewards(delegation.rewards(steps), steps)); Console.log("rewards accrued until last staking: %s.%s%s", delegation.getTotalRewards()); Console.log("delegation contract balance: %s.%s%s", address(delegation).balance); - //Console.log("staker balance: %s.%s%s", staker[i-1].balance); + //Console.log("staker balance: %s.%s%s", stakers[i-1].balance); vm.stopPrank(); } for (uint256 i = 0; i < stakerIndicesAfterWithdrawals.length; i++) { vm.deal(address(delegation), address(delegation).balance + rewardsAccruedAfterEach); int256 x = relativeAmountsAfterWithdrawals[i] * int256(depositAmount) / 10; - vm.startPrank(staker[stakerIndicesAfterWithdrawals[i]-1]); + vm.startPrank(stakers[stakerIndicesAfterWithdrawals[i]-1]); if (x >= 0) { delegation.stake{value: uint256(x)}(); //snapshot("staker %s staked %s", stakerIndices[i], uint256(x)); @@ -335,15 +186,15 @@ contract NonLiquidDelegationTest is Test { //further rewards accrued since the last staking vm.deal(address(delegation), address(delegation).balance + rewardsAccruedAfterEach); - for (uint256 i = 1; i <= staker.length; i++) { - vm.startPrank(staker[i-1]); + for (uint256 i = 1; i <= stakers.length; i++) { + vm.startPrank(stakers[i-1]); if (steps == 123_456_789) snapshot("staker %s withdrawing all, remaining rewards:", i, 0); else snapshot("staker %s withdrawing 1+%s times", i, steps); Console.log("rewards accrued until last staking: %s.%s%s", delegation.getTotalRewards()); Console.log("delegation contract balance: %s.%s%s", address(delegation).balance); - //Console.log("staker balance: %s.%s%s", staker[i-1].balance); + //Console.log("staker balance: %s.%s%s", stakers[i-1].balance); Console.log("staker rewards: %s.%s%s", delegation.rewards()); if (steps == 123_456_789) Console.log("staker withdrew: %s.%s%s", delegation.withdrawAllRewards()); @@ -352,22 +203,22 @@ contract NonLiquidDelegationTest is Test { Console.log("staker withdrew: %s.%s%s", delegation.withdrawRewards(delegation.rewards(steps), steps)); Console.log("rewards accrued until last staking: %s.%s%s", delegation.getTotalRewards()); Console.log("delegation contract balance: %s.%s%s", address(delegation).balance); - //Console.log("staker balance: %s.%s%s", staker[i-1].balance); + //Console.log("staker balance: %s.%s%s", stakers[i-1].balance); vm.stopPrank(); } // if we try to withdraw again immediately (in the same block), // the amount withdrawn must equal zero //* - for (uint256 i = 1; i <= staker.length; i++) { - vm.startPrank(staker[i-1]); + for (uint256 i = 1; i <= stakers.length; i++) { + vm.startPrank(stakers[i-1]); if (steps == 123_456_789) snapshot("staker %s withdrawing all, remaining rewards:", i, 0); else snapshot("staker %s withdrawing 1+%s times", i, steps); Console.log("rewards accrued until last staking: %s.%s%s", delegation.getTotalRewards()); Console.log("delegation contract balance: %s.%s%s", address(delegation).balance); - //Console.log("staker balance: %s.%s%s", staker[i-1].balance); + //Console.log("staker balance: %s.%s%s", stakers[i-1].balance); Console.log("staker rewards: %s.%s%s", delegation.rewards()); if (steps == 123_456_789) Console.log("staker withdrew: %s.%s%s", delegation.withdrawAllRewards()); @@ -376,7 +227,7 @@ contract NonLiquidDelegationTest is Test { Console.log("staker withdrew: %s.%s%s", delegation.withdrawRewards(delegation.rewards(steps), steps)); Console.log("rewards accrued until last staking: %s.%s%s", delegation.getTotalRewards()); Console.log("delegation contract balance: %s.%s%s", address(delegation).balance); - //Console.log("staker balance: %s.%s%s", staker[i-1].balance); + //Console.log("staker balance: %s.%s%s", stakers[i-1].balance); vm.stopPrank(); } //*/ @@ -529,14 +380,14 @@ contract NonLiquidDelegationTest is Test { uint256 x; uint64 steps = 11_000; - deposit(10_000_000 ether, true); + deposit(BaseDelegation(delegation), 10_000_000 ether, true); // wait 2 epochs for the change to the deposit to take affect vm.roll(block.number + Deposit(delegation.DEPOSIT_CONTRACT()).blocksPerEpoch() * 2); for (i = 0; i < 4; i++) { - vm.deal(staker[i], 100_000 ether); - console.log("staker %s: %s", i+1, staker[i]); + vm.deal(stakers[i], 100_000 ether); + console.log("staker %s: %s", i+1, stakers[i]); } delegation = NonLiquidDelegationV2(proxy); @@ -546,7 +397,7 @@ contract NonLiquidDelegationTest is Test { x = 50; for (uint256 j = 0; j < steps / 8; j++) { for (i = 1; i <= 4; i++) { - vm.startPrank(staker[i-1]); + vm.startPrank(stakers[i-1]); vm.recordLogs(); vm.expectEmit( true, @@ -556,7 +407,7 @@ contract NonLiquidDelegationTest is Test { address(delegation) ); emit Delegation.Staked( - staker[i-1], + stakers[i-1], x * 1 ether, "" ); @@ -568,7 +419,7 @@ contract NonLiquidDelegationTest is Test { vm.deal(address(delegation), address(delegation).balance + 10_000 ether); } for (i = 1; i <= 4; i++) { - vm.startPrank(staker[i-1]); + vm.startPrank(stakers[i-1]); vm.recordLogs(); vm.expectEmit( true, @@ -578,7 +429,7 @@ contract NonLiquidDelegationTest is Test { address(delegation) ); emit Delegation.Unstaked( - staker[i-1], + stakers[i-1], x * 1 ether, "" ); @@ -592,9 +443,9 @@ contract NonLiquidDelegationTest is Test { } i = 1; - vm.startPrank(staker[i-1]); + vm.startPrank(stakers[i-1]); //snapshot("staker %s withdrawing 1+%s times", i, steps); - //Console.log("staker balance: %s.%s%s", staker[i-1].balance); + //Console.log("staker balance: %s.%s%s", stakers[i-1].balance); //uint256 rewards = delegation.rewards(steps); uint256 rewards = delegation.rewards(); Console.log("staker rewards: %s.%s%s", rewards); @@ -607,7 +458,7 @@ contract NonLiquidDelegationTest is Test { rewards += delegation.withdrawRewards(1000000, 2000); //*/ Console.log("staker withdrew: %s.%s%s", rewards); - //Console.log("staker balance: %s.%s%s", staker[i-1].balance); + //Console.log("staker balance: %s.%s%s", stakers[i-1].balance); vm.stopPrank(); vm.roll(block.number + WithdrawalQueue.unbondingPeriod()); @@ -616,7 +467,7 @@ contract NonLiquidDelegationTest is Test { i = 1; - vm.startPrank(staker[i-1]); + vm.startPrank(stakers[i-1]); vm.recordLogs(); vm.expectEmit( true, @@ -626,7 +477,7 @@ contract NonLiquidDelegationTest is Test { address(delegation) ); emit Delegation.Claimed( - staker[i-1], + stakers[i-1], steps / 8 * x * 1 ether, "" ); @@ -639,14 +490,14 @@ contract NonLiquidDelegationTest is Test { uint256 x; // otherwise snapshot() doesn't find the staker and reverts - staker[0] = owner; - deposit(10_000_000 ether, true); + stakers[0] = owner; + deposit(BaseDelegation(delegation), 10_000_000 ether, true); // wait 2 epochs for the change to the deposit to take affect vm.roll(block.number + Deposit(delegation.DEPOSIT_CONTRACT()).blocksPerEpoch() * 2); for (i = 0; i < 4; i++) { - vm.deal(staker[i], 100_000 ether); + vm.deal(stakers[i], 100_000 ether); } delegation = NonLiquidDelegationV2(proxy); @@ -655,7 +506,7 @@ contract NonLiquidDelegationTest is Test { vm.deal(address(delegation), 50_000 ether); x = 50; i = 2; - vm.startPrank(staker[i-1]); + vm.startPrank(stakers[i-1]); vm.recordLogs(); vm.expectEmit( true, @@ -665,7 +516,7 @@ contract NonLiquidDelegationTest is Test { address(delegation) ); emit Delegation.Staked( - staker[i-1], + stakers[i-1], x * 1 ether, "" ); @@ -676,12 +527,12 @@ contract NonLiquidDelegationTest is Test { vm.stopPrank(); vm.deal(address(delegation), address(delegation).balance + 10_000 ether); - vm.startPrank(staker[i-1]); + vm.startPrank(stakers[i-1]); snapshot("staker %s withdrawing all, remaining rewards:", i, 0); console.log("-----------------------------------------------"); Console.log("contract balance: %s.%s%s", address(delegation).balance); - Console.log("staker balance: %s.%s%s", staker[i-1].balance); + Console.log("staker balance: %s.%s%s", stakers[i-1].balance); uint256 rewards = delegation.rewards(); Console.log("staker rewards: %s.%s%s", rewards); @@ -705,7 +556,7 @@ contract NonLiquidDelegationTest is Test { address(delegation) ); emit NonLiquidDelegationV2.RewardPaid( - staker[i-1], + stakers[i-1], rewards ); rewards = delegation.withdrawAllRewards(); @@ -722,7 +573,7 @@ contract NonLiquidDelegationTest is Test { console.log("allWithdrawnRewards = %s withdrawnAfterLastStaking = %s", allWithdrawnRewards, withdrawnAfterLastStaking); Console.log("contract balance: %s.%s%s", address(delegation).balance); - Console.log("staker balance: %s.%s%s", staker[i-1].balance); + Console.log("staker balance: %s.%s%s", stakers[i-1].balance); Console.log("staker rewards: %s.%s%s", delegation.rewards()); Console.log("staker should have received: %s.%s%s", rewards); vm.stopPrank(); From 753045064ab0049cc934a0e3c3d46a8321208db9 Mon Sep 17 00:00:00 2001 From: DrZoltanFazekas Date: Wed, 4 Dec 2024 15:29:50 +0100 Subject: [PATCH 4/5] Make the tests compatible with the upgradeable deposit contract --- test/BaseDelegation.t.sol | 16 +++++++++++++--- test/LiquidDelegation.t.sol | 22 ++++++++++++++++------ test/NonLiquidDelegation.t.sol | 30 ++++++++++++++++++++++-------- 3 files changed, 51 insertions(+), 17 deletions(-) diff --git a/test/BaseDelegation.t.sol b/test/BaseDelegation.t.sol index da7fc54..bd1d363 100644 --- a/test/BaseDelegation.t.sol +++ b/test/BaseDelegation.t.sol @@ -3,7 +3,7 @@ pragma solidity ^0.8.26; import {BaseDelegation} from "src/BaseDelegation.sol"; import {Delegation} from "src/Delegation.sol"; -import {Deposit, InitialStaker} from "@zilliqa/zq2/deposit.sol"; +import {Deposit, InitialStaker} from "@zilliqa/zq2/deposit_v2.sol"; import {Console} from "src/Console.sol"; import {ERC1967Proxy} from "@openzeppelin/contracts/proxy/ERC1967/ERC1967Proxy.sol"; import {Test, Vm} from "forge-std/Test.sol"; @@ -106,12 +106,20 @@ abstract contract BaseDelegationTest is Test { //vm.deployCodeTo("Deposit.sol", delegation.DEPOSIT_CONTRACT()); vm.etch( delegation.DEPOSIT_CONTRACT(), //0x000000000000000000005a494C4445504F534954, - address(new Deposit(10_000_000 ether, 256, 10, initialStakers)).code + // since the deposit contract is upgradeable, the constructor has no parameters + //address(new Deposit(10_000_000 ether, 256, 10, initialStakers)).code + address(new Deposit()).code ); + vm.store(delegation.DEPOSIT_CONTRACT(), bytes32(uint256(0x958a6cf6390bd7165e3519675caa670ab90f0161508a9ee714d3db7edc50740b)), bytes32(uint256(block.number / 10))); + vm.store(delegation.DEPOSIT_CONTRACT(), bytes32(uint256(0x958a6cf6390bd7165e3519675caa670ab90f0161508a9ee714d3db7edc50740c)), bytes32(uint256(10_000_000 ether))); + vm.store(delegation.DEPOSIT_CONTRACT(), bytes32(uint256(0x958a6cf6390bd7165e3519675caa670ab90f0161508a9ee714d3db7edc50740d)), bytes32(uint256(256))); + vm.store(delegation.DEPOSIT_CONTRACT(), bytes32(uint256(0x958a6cf6390bd7165e3519675caa670ab90f0161508a9ee714d3db7edc50740e)), bytes32(uint256(10))); + /* since the deposit contract is upgradeable, the storage locations changed too vm.store(delegation.DEPOSIT_CONTRACT(), bytes32(uint256(11)), bytes32(uint256(block.number / 10))); vm.store(delegation.DEPOSIT_CONTRACT(), bytes32(uint256(12)), bytes32(uint256(10_000_000 ether))); vm.store(delegation.DEPOSIT_CONTRACT(), bytes32(uint256(13)), bytes32(uint256(256))); vm.store(delegation.DEPOSIT_CONTRACT(), bytes32(uint256(14)), bytes32(uint256(10))); + */ /* console.log("Deposit.minimimStake() =", Deposit(delegation.DEPOSIT_CONTRACT()).minimumStake()); console.log("Deposit.maximumStakers() =", Deposit(delegation.DEPOSIT_CONTRACT()).maximumStakers()); @@ -169,6 +177,8 @@ abstract contract BaseDelegationTest is Test { ); } // wait 2 epochs for the change to the deposit to take affect - vm.roll(block.number + Deposit(delegation.DEPOSIT_CONTRACT()).blocksPerEpoch() * 2); + vm.roll(block.number + 3600 * 2); + // remove the previous line and comment out the next one once https://github.com/Zilliqa/zq2/issues/1956 is implemented + //vm.roll(block.number + Deposit(delegation.DEPOSIT_CONTRACT()).blocksPerEpoch() * 2); } } \ No newline at end of file diff --git a/test/LiquidDelegation.t.sol b/test/LiquidDelegation.t.sol index 51d01f8..7346168 100644 --- a/test/LiquidDelegation.t.sol +++ b/test/LiquidDelegation.t.sol @@ -7,7 +7,7 @@ import {LiquidDelegationV2} from "src/LiquidDelegationV2.sol"; import {NonRebasingLST} from "src/NonRebasingLST.sol"; import {BaseDelegation, WithdrawalQueue} from "src/BaseDelegation.sol"; import {Delegation} from "src/Delegation.sol"; -import {Deposit} from "@zilliqa/zq2/deposit.sol"; +import {Deposit} from "@zilliqa/zq2/deposit_v2.sol"; import {Console} from "src/Console.sol"; import {Vm} from "forge-std/Test.sol"; import "forge-std/console.sol"; @@ -119,7 +119,9 @@ contract LiquidDelegationTest is BaseDelegationTest { }(); // wait 2 epochs for the change to the deposit to take affect - vm.roll(block.number + Deposit(delegation.DEPOSIT_CONTRACT()).blocksPerEpoch() * 2); + vm.roll(block.number + 3600 * 2); + // remove the previous line and comment out the next one once https://github.com/Zilliqa/zq2/issues/1956 is implemented + //vm.roll(block.number + Deposit(delegation.DEPOSIT_CONTRACT()).blocksPerEpoch() * 2); ownerZIL[1] = delegation.owner().balance; @@ -207,7 +209,9 @@ contract LiquidDelegationTest is BaseDelegationTest { ); // wait 2 epochs for the change to the deposit to take affect - vm.roll(block.number + Deposit(delegation.DEPOSIT_CONTRACT()).blocksPerEpoch() * 2); + vm.roll(block.number + 3600 * 2); + // remove the previous line and comment out the next one once https://github.com/Zilliqa/zq2/issues/1956 is implemented + //vm.roll(block.number + Deposit(delegation.DEPOSIT_CONTRACT()).blocksPerEpoch() * 2); stakerLST[1] = lst.balanceOf(stakers[0]); ownerZIL[1] = delegation.owner().balance; @@ -370,7 +374,9 @@ contract LiquidDelegationTest is BaseDelegationTest { console.log("validator stake: %s", Deposit(delegation.DEPOSIT_CONTRACT()).getStake( bytes(hex"92fbe50544dce63cfdcc88301d7412f0edea024c91ae5d6a04c7cd3819edfc1b9d75d9121080af12e00f054d221f876c") )); - vm.roll(block.number + Deposit(delegation.DEPOSIT_CONTRACT()).blocksPerEpoch() * 2); + vm.roll(block.number + 3600 * 2); + // remove the previous line and comment out the next one once https://github.com/Zilliqa/zq2/issues/1956 is implemented + //vm.roll(block.number + Deposit(delegation.DEPOSIT_CONTRACT()).blocksPerEpoch() * 2); console.log("validator stake: %s", Deposit(delegation.DEPOSIT_CONTRACT()).getStake( bytes(hex"92fbe50544dce63cfdcc88301d7412f0edea024c91ae5d6a04c7cd3819edfc1b9d75d9121080af12e00f054d221f876c") )); @@ -381,7 +387,9 @@ contract LiquidDelegationTest is BaseDelegationTest { console.log("validator stake: %s", Deposit(delegation.DEPOSIT_CONTRACT()).getStake( bytes(hex"92fbe50544dce63cfdcc88301d7412f0edea024c91ae5d6a04c7cd3819edfc1b9d75d9121080af12e00f054d221f876c") )); - vm.roll(block.number + Deposit(delegation.DEPOSIT_CONTRACT()).blocksPerEpoch() * 2); + vm.roll(block.number + 3600 * 2); + // remove the previous line and comment out the next one once https://github.com/Zilliqa/zq2/issues/1956 is implemented + //vm.roll(block.number + Deposit(delegation.DEPOSIT_CONTRACT()).blocksPerEpoch() * 2); console.log("validator stake: %s", Deposit(delegation.DEPOSIT_CONTRACT()).getStake( bytes(hex"92fbe50544dce63cfdcc88301d7412f0edea024c91ae5d6a04c7cd3819edfc1b9d75d9121080af12e00f054d221f876c") )); @@ -392,7 +400,9 @@ contract LiquidDelegationTest is BaseDelegationTest { console.log("validator stake: %s", Deposit(delegation.DEPOSIT_CONTRACT()).getStake( bytes(hex"92fbe50544dce63cfdcc88301d7412f0edea024c91ae5d6a04c7cd3819edfc1b9d75d9121080af12e00f054d221f876c") )); - vm.roll(block.number + Deposit(delegation.DEPOSIT_CONTRACT()).blocksPerEpoch() * 2); + vm.roll(block.number + 3600 * 2); + // remove the previous line and comment out the next one once https://github.com/Zilliqa/zq2/issues/1956 is implemented + //vm.roll(block.number + Deposit(delegation.DEPOSIT_CONTRACT()).blocksPerEpoch() * 2); console.log("validator stake: %s", Deposit(delegation.DEPOSIT_CONTRACT()).getStake( bytes(hex"92fbe50544dce63cfdcc88301d7412f0edea024c91ae5d6a04c7cd3819edfc1b9d75d9121080af12e00f054d221f876c") )); diff --git a/test/NonLiquidDelegation.t.sol b/test/NonLiquidDelegation.t.sol index 7acb8c8..94165ca 100644 --- a/test/NonLiquidDelegation.t.sol +++ b/test/NonLiquidDelegation.t.sol @@ -6,7 +6,7 @@ import {NonLiquidDelegation} from "src/NonLiquidDelegation.sol"; import {NonLiquidDelegationV2} from "src/NonLiquidDelegationV2.sol"; import {BaseDelegation, WithdrawalQueue} from "src/BaseDelegation.sol"; import {Delegation} from "src/Delegation.sol"; -import {Deposit} from "@zilliqa/zq2/deposit.sol"; +import {Deposit} from "@zilliqa/zq2/deposit_v2.sol"; import {Console} from "src/Console.sol"; import {Strings} from "@openzeppelin/contracts/utils/Strings.sol"; import {Vm} from "forge-std/Test.sol"; @@ -141,7 +141,9 @@ contract NonLiquidDelegationTest is BaseDelegationTest { } vm.stopPrank(); // wait 2 epochs for the change to the deposit to take affect - vm.roll(block.number + Deposit(delegation.DEPOSIT_CONTRACT()).blocksPerEpoch() * 2); + vm.roll(block.number + 3600 * 2); + // remove the previous line and comment out the next one once https://github.com/Zilliqa/zq2/issues/1956 is implemented + //vm.roll(block.number + Deposit(delegation.DEPOSIT_CONTRACT()).blocksPerEpoch() * 2); } //no rewards if we withdraw in the same block as the last staking @@ -180,7 +182,9 @@ contract NonLiquidDelegationTest is BaseDelegationTest { } vm.stopPrank(); // wait 2 epochs for the change to the deposit to take affect - vm.roll(block.number + Deposit(delegation.DEPOSIT_CONTRACT()).blocksPerEpoch() * 2); + vm.roll(block.number + 3600 * 2); + // remove the previous line and comment out the next one once https://github.com/Zilliqa/zq2/issues/1956 is implemented + //vm.roll(block.number + Deposit(delegation.DEPOSIT_CONTRACT()).blocksPerEpoch() * 2); } //further rewards accrued since the last staking @@ -383,7 +387,9 @@ contract NonLiquidDelegationTest is BaseDelegationTest { deposit(BaseDelegation(delegation), 10_000_000 ether, true); // wait 2 epochs for the change to the deposit to take affect - vm.roll(block.number + Deposit(delegation.DEPOSIT_CONTRACT()).blocksPerEpoch() * 2); + vm.roll(block.number + 3600 * 2); + // remove the previous line and comment out the next one once https://github.com/Zilliqa/zq2/issues/1956 is implemented + //vm.roll(block.number + Deposit(delegation.DEPOSIT_CONTRACT()).blocksPerEpoch() * 2); for (i = 0; i < 4; i++) { vm.deal(stakers[i], 100_000 ether); @@ -413,7 +419,9 @@ contract NonLiquidDelegationTest is BaseDelegationTest { ); delegation.stake{value: x * 1 ether}(); // wait 2 epochs for the change to the deposit to take affect - vm.roll(block.number + Deposit(delegation.DEPOSIT_CONTRACT()).blocksPerEpoch() * 2); + vm.roll(block.number + 3600 * 2); + // remove the previous line and comment out the next one once https://github.com/Zilliqa/zq2/issues/1956 is implemented + //vm.roll(block.number + Deposit(delegation.DEPOSIT_CONTRACT()).blocksPerEpoch() * 2); //snapshot("staker %s staked %s", i, x); vm.stopPrank(); vm.deal(address(delegation), address(delegation).balance + 10_000 ether); @@ -435,7 +443,9 @@ contract NonLiquidDelegationTest is BaseDelegationTest { ); delegation.unstake(x * 1 ether); // wait 2 epochs for the change to the deposit to take affect - vm.roll(block.number + Deposit(delegation.DEPOSIT_CONTRACT()).blocksPerEpoch() * 2); + vm.roll(block.number + 3600 * 2); + // remove the previous line and comment out the next one once https://github.com/Zilliqa/zq2/issues/1956 is implemented + //vm.roll(block.number + Deposit(delegation.DEPOSIT_CONTRACT()).blocksPerEpoch() * 2); //snapshot("staker %s unstaked %s", i, x); vm.stopPrank(); vm.deal(address(delegation), address(delegation).balance + 10_000 ether); @@ -494,7 +504,9 @@ contract NonLiquidDelegationTest is BaseDelegationTest { deposit(BaseDelegation(delegation), 10_000_000 ether, true); // wait 2 epochs for the change to the deposit to take affect - vm.roll(block.number + Deposit(delegation.DEPOSIT_CONTRACT()).blocksPerEpoch() * 2); + vm.roll(block.number + 3600 * 2); + // remove the previous line and comment out the next one once https://github.com/Zilliqa/zq2/issues/1956 is implemented + //vm.roll(block.number + Deposit(delegation.DEPOSIT_CONTRACT()).blocksPerEpoch() * 2); for (i = 0; i < 4; i++) { vm.deal(stakers[i], 100_000 ether); @@ -523,7 +535,9 @@ contract NonLiquidDelegationTest is BaseDelegationTest { delegation.stake{value: x * 1 ether}(); // wait 2 epochs for the change to the deposit to take affect - vm.roll(block.number + Deposit(delegation.DEPOSIT_CONTRACT()).blocksPerEpoch() * 2); + vm.roll(block.number + 3600 * 2); + // remove the previous line and comment out the next one once https://github.com/Zilliqa/zq2/issues/1956 is implemented + //vm.roll(block.number + Deposit(delegation.DEPOSIT_CONTRACT()).blocksPerEpoch() * 2); vm.stopPrank(); vm.deal(address(delegation), address(delegation).balance + 10_000 ether); From e8bcaeadfda675d17bb25e7b7ccfdac424b6530c Mon Sep 17 00:00:00 2001 From: DrZoltanFazekas Date: Wed, 4 Dec 2024 16:22:38 +0100 Subject: [PATCH 5/5] Remove workaround for missing getter in deposit contract --- test/BaseDelegation.t.sol | 4 +--- test/LiquidDelegation.t.sol | 20 +++++--------------- test/NonLiquidDelegation.t.sol | 28 +++++++--------------------- 3 files changed, 13 insertions(+), 39 deletions(-) diff --git a/test/BaseDelegation.t.sol b/test/BaseDelegation.t.sol index bd1d363..7b72300 100644 --- a/test/BaseDelegation.t.sol +++ b/test/BaseDelegation.t.sol @@ -177,8 +177,6 @@ abstract contract BaseDelegationTest is Test { ); } // wait 2 epochs for the change to the deposit to take affect - vm.roll(block.number + 3600 * 2); - // remove the previous line and comment out the next one once https://github.com/Zilliqa/zq2/issues/1956 is implemented - //vm.roll(block.number + Deposit(delegation.DEPOSIT_CONTRACT()).blocksPerEpoch() * 2); + vm.roll(block.number + Deposit(delegation.DEPOSIT_CONTRACT()).blocksPerEpoch() * 2); } } \ No newline at end of file diff --git a/test/LiquidDelegation.t.sol b/test/LiquidDelegation.t.sol index 7346168..51a453f 100644 --- a/test/LiquidDelegation.t.sol +++ b/test/LiquidDelegation.t.sol @@ -119,9 +119,7 @@ contract LiquidDelegationTest is BaseDelegationTest { }(); // wait 2 epochs for the change to the deposit to take affect - vm.roll(block.number + 3600 * 2); - // remove the previous line and comment out the next one once https://github.com/Zilliqa/zq2/issues/1956 is implemented - //vm.roll(block.number + Deposit(delegation.DEPOSIT_CONTRACT()).blocksPerEpoch() * 2); + vm.roll(block.number + Deposit(delegation.DEPOSIT_CONTRACT()).blocksPerEpoch() * 2); ownerZIL[1] = delegation.owner().balance; @@ -209,9 +207,7 @@ contract LiquidDelegationTest is BaseDelegationTest { ); // wait 2 epochs for the change to the deposit to take affect - vm.roll(block.number + 3600 * 2); - // remove the previous line and comment out the next one once https://github.com/Zilliqa/zq2/issues/1956 is implemented - //vm.roll(block.number + Deposit(delegation.DEPOSIT_CONTRACT()).blocksPerEpoch() * 2); + vm.roll(block.number + Deposit(delegation.DEPOSIT_CONTRACT()).blocksPerEpoch() * 2); stakerLST[1] = lst.balanceOf(stakers[0]); ownerZIL[1] = delegation.owner().balance; @@ -374,9 +370,7 @@ contract LiquidDelegationTest is BaseDelegationTest { console.log("validator stake: %s", Deposit(delegation.DEPOSIT_CONTRACT()).getStake( bytes(hex"92fbe50544dce63cfdcc88301d7412f0edea024c91ae5d6a04c7cd3819edfc1b9d75d9121080af12e00f054d221f876c") )); - vm.roll(block.number + 3600 * 2); - // remove the previous line and comment out the next one once https://github.com/Zilliqa/zq2/issues/1956 is implemented - //vm.roll(block.number + Deposit(delegation.DEPOSIT_CONTRACT()).blocksPerEpoch() * 2); + vm.roll(block.number + Deposit(delegation.DEPOSIT_CONTRACT()).blocksPerEpoch() * 2); console.log("validator stake: %s", Deposit(delegation.DEPOSIT_CONTRACT()).getStake( bytes(hex"92fbe50544dce63cfdcc88301d7412f0edea024c91ae5d6a04c7cd3819edfc1b9d75d9121080af12e00f054d221f876c") )); @@ -387,9 +381,7 @@ contract LiquidDelegationTest is BaseDelegationTest { console.log("validator stake: %s", Deposit(delegation.DEPOSIT_CONTRACT()).getStake( bytes(hex"92fbe50544dce63cfdcc88301d7412f0edea024c91ae5d6a04c7cd3819edfc1b9d75d9121080af12e00f054d221f876c") )); - vm.roll(block.number + 3600 * 2); - // remove the previous line and comment out the next one once https://github.com/Zilliqa/zq2/issues/1956 is implemented - //vm.roll(block.number + Deposit(delegation.DEPOSIT_CONTRACT()).blocksPerEpoch() * 2); + vm.roll(block.number + Deposit(delegation.DEPOSIT_CONTRACT()).blocksPerEpoch() * 2); console.log("validator stake: %s", Deposit(delegation.DEPOSIT_CONTRACT()).getStake( bytes(hex"92fbe50544dce63cfdcc88301d7412f0edea024c91ae5d6a04c7cd3819edfc1b9d75d9121080af12e00f054d221f876c") )); @@ -400,9 +392,7 @@ contract LiquidDelegationTest is BaseDelegationTest { console.log("validator stake: %s", Deposit(delegation.DEPOSIT_CONTRACT()).getStake( bytes(hex"92fbe50544dce63cfdcc88301d7412f0edea024c91ae5d6a04c7cd3819edfc1b9d75d9121080af12e00f054d221f876c") )); - vm.roll(block.number + 3600 * 2); - // remove the previous line and comment out the next one once https://github.com/Zilliqa/zq2/issues/1956 is implemented - //vm.roll(block.number + Deposit(delegation.DEPOSIT_CONTRACT()).blocksPerEpoch() * 2); + vm.roll(block.number + Deposit(delegation.DEPOSIT_CONTRACT()).blocksPerEpoch() * 2); console.log("validator stake: %s", Deposit(delegation.DEPOSIT_CONTRACT()).getStake( bytes(hex"92fbe50544dce63cfdcc88301d7412f0edea024c91ae5d6a04c7cd3819edfc1b9d75d9121080af12e00f054d221f876c") )); diff --git a/test/NonLiquidDelegation.t.sol b/test/NonLiquidDelegation.t.sol index 94165ca..504afff 100644 --- a/test/NonLiquidDelegation.t.sol +++ b/test/NonLiquidDelegation.t.sol @@ -141,9 +141,7 @@ contract NonLiquidDelegationTest is BaseDelegationTest { } vm.stopPrank(); // wait 2 epochs for the change to the deposit to take affect - vm.roll(block.number + 3600 * 2); - // remove the previous line and comment out the next one once https://github.com/Zilliqa/zq2/issues/1956 is implemented - //vm.roll(block.number + Deposit(delegation.DEPOSIT_CONTRACT()).blocksPerEpoch() * 2); + vm.roll(block.number + Deposit(delegation.DEPOSIT_CONTRACT()).blocksPerEpoch() * 2); } //no rewards if we withdraw in the same block as the last staking @@ -182,9 +180,7 @@ contract NonLiquidDelegationTest is BaseDelegationTest { } vm.stopPrank(); // wait 2 epochs for the change to the deposit to take affect - vm.roll(block.number + 3600 * 2); - // remove the previous line and comment out the next one once https://github.com/Zilliqa/zq2/issues/1956 is implemented - //vm.roll(block.number + Deposit(delegation.DEPOSIT_CONTRACT()).blocksPerEpoch() * 2); + vm.roll(block.number + Deposit(delegation.DEPOSIT_CONTRACT()).blocksPerEpoch() * 2); } //further rewards accrued since the last staking @@ -387,9 +383,7 @@ contract NonLiquidDelegationTest is BaseDelegationTest { deposit(BaseDelegation(delegation), 10_000_000 ether, true); // wait 2 epochs for the change to the deposit to take affect - vm.roll(block.number + 3600 * 2); - // remove the previous line and comment out the next one once https://github.com/Zilliqa/zq2/issues/1956 is implemented - //vm.roll(block.number + Deposit(delegation.DEPOSIT_CONTRACT()).blocksPerEpoch() * 2); + vm.roll(block.number + Deposit(delegation.DEPOSIT_CONTRACT()).blocksPerEpoch() * 2); for (i = 0; i < 4; i++) { vm.deal(stakers[i], 100_000 ether); @@ -419,9 +413,7 @@ contract NonLiquidDelegationTest is BaseDelegationTest { ); delegation.stake{value: x * 1 ether}(); // wait 2 epochs for the change to the deposit to take affect - vm.roll(block.number + 3600 * 2); - // remove the previous line and comment out the next one once https://github.com/Zilliqa/zq2/issues/1956 is implemented - //vm.roll(block.number + Deposit(delegation.DEPOSIT_CONTRACT()).blocksPerEpoch() * 2); + vm.roll(block.number + Deposit(delegation.DEPOSIT_CONTRACT()).blocksPerEpoch() * 2); //snapshot("staker %s staked %s", i, x); vm.stopPrank(); vm.deal(address(delegation), address(delegation).balance + 10_000 ether); @@ -443,9 +435,7 @@ contract NonLiquidDelegationTest is BaseDelegationTest { ); delegation.unstake(x * 1 ether); // wait 2 epochs for the change to the deposit to take affect - vm.roll(block.number + 3600 * 2); - // remove the previous line and comment out the next one once https://github.com/Zilliqa/zq2/issues/1956 is implemented - //vm.roll(block.number + Deposit(delegation.DEPOSIT_CONTRACT()).blocksPerEpoch() * 2); + vm.roll(block.number + Deposit(delegation.DEPOSIT_CONTRACT()).blocksPerEpoch() * 2); //snapshot("staker %s unstaked %s", i, x); vm.stopPrank(); vm.deal(address(delegation), address(delegation).balance + 10_000 ether); @@ -504,9 +494,7 @@ contract NonLiquidDelegationTest is BaseDelegationTest { deposit(BaseDelegation(delegation), 10_000_000 ether, true); // wait 2 epochs for the change to the deposit to take affect - vm.roll(block.number + 3600 * 2); - // remove the previous line and comment out the next one once https://github.com/Zilliqa/zq2/issues/1956 is implemented - //vm.roll(block.number + Deposit(delegation.DEPOSIT_CONTRACT()).blocksPerEpoch() * 2); + vm.roll(block.number + Deposit(delegation.DEPOSIT_CONTRACT()).blocksPerEpoch() * 2); for (i = 0; i < 4; i++) { vm.deal(stakers[i], 100_000 ether); @@ -535,9 +523,7 @@ contract NonLiquidDelegationTest is BaseDelegationTest { delegation.stake{value: x * 1 ether}(); // wait 2 epochs for the change to the deposit to take affect - vm.roll(block.number + 3600 * 2); - // remove the previous line and comment out the next one once https://github.com/Zilliqa/zq2/issues/1956 is implemented - //vm.roll(block.number + Deposit(delegation.DEPOSIT_CONTRACT()).blocksPerEpoch() * 2); + vm.roll(block.number + Deposit(delegation.DEPOSIT_CONTRACT()).blocksPerEpoch() * 2); vm.stopPrank(); vm.deal(address(delegation), address(delegation).balance + 10_000 ether);