diff --git a/DEPLOYMENT_ADDRESSES.md b/DEPLOYMENT_ADDRESSES.md new file mode 100644 index 0000000..105d78b --- /dev/null +++ b/DEPLOYMENT_ADDRESSES.md @@ -0,0 +1,31 @@ +# Deployment Addresses Puffer Mainnet + +```yaml + Deploying from: 0x36b6fE474dAD8e822d3133B76E9adA671E75eC86 + + Chain ID: 1 + Factory: 0xfbF3B5Fa2380C77a6ac255A8e19a142490601767 + + DAO: 0x5dEA8E499b05de8F86E7521F039770268055b23F + + Plugins + - Multisig plugin: 0xA303C435563a4544a84E26501F4666346Ff73a0d + + Gauge voter plugin: 0x69E8D5151d71d4cde35b5076aF3023C7D54d379E + Curve: 0xAaAb5528aFf964CEAc972E39c2357c2D503A9196 + Exit Queue: 0xD9C2d314E29F1940d2a65A691881F0950fE4A455 + Voting Escrow: 0xA55eD5808aeCDF23AE3782C1443185f5D2363ce7 + Clock: 0x8BCDf6291F251cF8eCf5ac06bFc4A2B02Ecc26eB + NFT Lock: 0x1b6ec227ceBeC25118270efbb4b67642fc29965E + + Plugin repositories + - Multisig plugin repository (existing): 0x8c278e37D0817210E18A7958524b7D0a1fAA6F7b + - Gauge voter plugin repository: 0xa10192700F6c3A0ee0deAA2EE72A91916a4CBb70 + + Router + - 0xD33340fb9C8c2aEBC6922552bf3bC268a3682696 + + +``` + +The deployment addresses can also be verified [on the factory contract via Etherscan](https://etherscan.io/address/0xfbf3b5fa2380c77a6ac255a8e19a142490601767#readContract) diff --git a/Makefile b/Makefile index a760692..abcfc72 100644 --- a/Makefile +++ b/Makefile @@ -37,8 +37,31 @@ ft-sepolia-fork :; forge test --match-contract TestE2EV2 \ --rpc-url https://sepolia.drpc.org \ -vvvvv +# Fork testing - mainnet +ft-mainnet-fork :; forge test --match-contract TestE2EV2 \ + --rpc-url https://eth-mainnet.g.alchemy.com/v2/$(ALCHEMY_API_KEY) \ + -vvvvv + #### Deployments #### +deploy-preview-mainnet :; forge script script/Deploy.s.sol:Deploy \ + --rpc-url https://eth-mainnet.g.alchemy.com/v2/$(ALCHEMY_API_KEY) \ + --private-key $(DEPLOYMENT_PRIVATE_KEY) \ + -vvvvv + +deploy-mainnet :; forge script script/Deploy.s.sol:Deploy \ + --rpc-url https://eth-mainnet.g.alchemy.com/v2/$(ALCHEMY_API_KEY) \ + --slow \ + --resume \ + --private-key $(DEPLOYMENT_PRIVATE_KEY) \ + --verify \ + --etherscan-api-key $(ETHERSCAN_API_KEY) \ + -vvvvv + +log-mainnet :; forge script Logger \ + --rpc-url https://eth-mainnet.g.alchemy.com/v2/$(ALCHEMY_API_KEY) \ + -vvvvv + deploy-preview-mode-sepolia :; forge script script/Deploy.s.sol:Deploy \ --rpc-url https://sepolia.mode.network \ --private-key $(DEPLOYMENT_PRIVATE_KEY) \ @@ -66,3 +89,42 @@ deploy-mode :; forge script script/Deploy.s.sol:Deploy \ --etherscan-api-key $(ETHERSCAN_API_KEY) \ -vvv +deploy-preview-holesky :; forge script script/Deploy.s.sol:Deploy \ + --rpc-url https://holesky.drpc.org \ + --private-key $(DEPLOYMENT_PRIVATE_KEY) \ + -vvvvv + +deploy-holesky :; forge script script/Deploy.s.sol:Deploy \ + --rpc-url https://holesky.drpc.org \ + --private-key $(DEPLOYMENT_PRIVATE_KEY) \ + --broadcast \ + --verify \ + --etherscan-api-key $(ETHERSCAN_API_KEY) \ + -vvv + +deploy-preview-sepolia :; forge script script/Deploy.s.sol:Deploy \ + --rpc-url https://1rpc.io/sepolia \ + --private-key $(DEPLOYMENT_PRIVATE_KEY) \ + -vvvvv + +router-preview-sepolia :; forge script DeployRouter \ + --rpc-url https://1rpc.io/sepolia \ + --private-key $(DEPLOYMENT_PRIVATE_KEY) \ + -vvvvv + +router-sepolia :; forge script DeployRouter \ + --rpc-url https://1rpc.io/sepolia \ + --private-key $(DEPLOYMENT_PRIVATE_KEY) \ + --broadcast \ + --verify \ + --etherscan-api-key $(ETHERSCAN_API_KEY) \ + -vvvvv + +deploy-sepolia :; forge script script/Deploy.s.sol:Deploy \ + --rpc-url https://1rpc.io/sepolia \ + --private-key $(DEPLOYMENT_PRIVATE_KEY) \ + --broadcast \ + --verify \ + --etherscan-api-key $(ETHERSCAN_API_KEY) \ + -vvvvv + diff --git a/README.md b/README.md index 9df21cf..2b1b9a5 100644 --- a/README.md +++ b/README.md @@ -2,6 +2,8 @@ Welcome to Aragon's veGovernance Plugin - a flexible, modular and secure system which can be used to create custom DAOs that foster a strong alignment between token holders and capital flows. +The mainnet deployment addresses can be found [here](./DEPLOYMENT_ADDRESSES.md). + ## Setup To get started, ensure that [Foundry](https://getfoundry.sh/) is installed on your computer, then copy `.env.example` into `.env` and define the parameters @@ -71,19 +73,22 @@ Check the `Makefile` for examples of deployments on different networks. ### Deployment Checklist -- [] I have reviewed the parameters for the veDAO I want to deploy -- [] I have reviewed the multisig file for the correct addresses - - [] I have ensured all multisig members have undergone a proper security review and are aware of the security implications of being on said multisig -- [] I have updated the `.env` with these parameters -- [] I have updated the `CurveConstantLib` and `Clock` with any new constants. -- [] All my unit tests pass -- [] I have run a fork test in `fork-deploy` mode against the OSx contracts on my target testnet -- [] I have deployed my contracts successfully to a target testnet +- [x] I have reviewed the parameters for the veDAO I want to deploy +- [x] I have reviewed the multisig file for the correct addresses + - [x] I have ensured all multisig members have undergone a proper security review and are aware of the security implications of being on said multisig +- [x] I have updated the `.env` with these parameters +- [x] I have updated the `CurveConstantLib` and `Clock` with any new constants. +- [x] All my unit tests pass +- [x] I have run a fork test in `fork-deploy` mode against the OSx contracts on my target testnet +- [x] I have deployed my contracts successfully to a target testnet +- [x] I have previewed my deploy +- [x] My deployer address is a fresh wallet or setup for repeat production deploys in a safe manner. +- [x] My wallet has sufficient native token for gas - [] I have confirmed my tests still work in `fork-existing` mode with the live tokens and the factory. -- [] I have run the same workflow against the mainnet I wish to deploy on -- [] I have previewed my deploy -- [] My deployer address is a fresh wallet or setup for repeat production deploys in a safe manner. -- [] My wallet has sufficient native token for gas + +**Puffer Only** + +- [] I have deployed the Router contract ### Manual from the command line diff --git a/script/DeployRouter.s.sol b/script/DeployRouter.s.sol new file mode 100644 index 0000000..b4dff2f --- /dev/null +++ b/script/DeployRouter.s.sol @@ -0,0 +1,52 @@ +// SPDX-License-Identifier: UNLICENSED +pragma solidity ^0.8.17; + +import {Script, console} from "forge-std/Script.sol"; +import {VotingEscrow} from "src/voting/SimpleGaugeVoterSetup.sol"; +import {GaugesDaoFactory, GaugePluginSet, Deployment} from "src/factory/GaugesDaoFactory.sol"; + +contract Router { + address public escrow; + + constructor(address _escrow) { + escrow = _escrow; + } + + function getLocked(uint _tokenId) public view returns (uint256) { + return VotingEscrow(escrow).locked(_tokenId).amount; + } + + function getTotalLocked(address _user) public view returns (uint256) { + uint256[] memory ids = VotingEscrow(escrow).ownedTokens(_user); + + uint256 totalLocked = 0; + + for (uint i = 0; i < ids.length; i++) { + totalLocked += VotingEscrow(escrow).locked(ids[i]).amount; + } + return totalLocked; + } +} + +contract DeployRouter is Script { + function run() public { + address factoryAddress = vm.envAddress("FACTORY"); + if (factoryAddress == address(0)) { + revert("Factory address not set"); + } + GaugesDaoFactory factory = GaugesDaoFactory(factoryAddress); + + // set our contracts + Deployment memory deployment = factory.getDeployment(); + + // if deploying multiple tokens, you can adjust the index here + GaugePluginSet memory pluginSet = deployment.gaugeVoterPluginSets[0]; + + vm.startBroadcast(vm.envUint("DEPLOYMENT_PRIVATE_KEY")); + { + Router router = new Router({_escrow: address(pluginSet.votingEscrow)}); + console.log("Router deployed at:", address(router)); + } + vm.stopBroadcast(); + } +} diff --git a/src/libs/CurveConstantLib.sol b/src/libs/CurveConstantLib.sol index 9f4fbde..06b63eb 100644 --- a/src/libs/CurveConstantLib.sol +++ b/src/libs/CurveConstantLib.sol @@ -3,20 +3,20 @@ pragma solidity ^0.8.0; /// @title CurveConstantLib /// @notice Precomputed coefficients for escrow curve -/// This curve implementation is a quadratic curve of the form y = (1/7)t^2 + (2/7)t + 1 -/// Which is a transformation of the quadratic curve y = (x^2 + 6)/7 -/// That starts with 1 unit of voting in period 1, and max 6 in period 6. -/// To use this in zero indexed time, with a per-second rate of increase, -/// we transform this to the polynomial y = (1/7)t^2 + (2/7)t + 1 -/// where t = timestamp / 2_weeks (2 weeks is one period) /// Below are the shared coefficients for the linear and quadratic terms +/// @dev This curve goes from 1x -> 2x voting power over a 2 year time horizon +/// Epochs are still 2 weeks long library CurveConstantLib { int256 internal constant SHARED_CONSTANT_COEFFICIENT = 1e18; - /// @dev 2 / (7 * 2_weeks) - expressed in fixed point - int256 internal constant SHARED_LINEAR_COEFFICIENT = 236205593348; - /// @dev 1 / (7 * (2_weeks)^2) - expressed in fixed point - int256 internal constant SHARED_QUADRATIC_COEFFICIENT = 97637; + + /// @dev straight line so the curve is increasing only in the linear term + /// 1 / (52 * SECONDS_IN_2_WEEKS) + int256 internal constant SHARED_LINEAR_COEFFICIENT = 15898453398; + + /// @dev this curve is linear + int256 internal constant SHARED_QUADRATIC_COEFFICIENT = 0; /// @dev the maxiumum number of epochs the cure can keep increasing - uint256 internal constant MAX_EPOCHS = 5; + /// 26 epochs in a year, 2 years = 52 epochs + uint256 internal constant MAX_EPOCHS = 52; } diff --git a/test/escrow/curve/QuadraticCurveMath.t.sol b/test/escrow/curve/QuadraticCurveMath.t.sol index 93711bb..e89a880 100644 --- a/test/escrow/curve/QuadraticCurveMath.t.sol +++ b/test/escrow/curve/QuadraticCurveMath.t.sol @@ -1,3 +1,4 @@ +// SPDX-License-Identifier: UNLICENSED pragma solidity ^0.8.17; import {console2 as console} from "forge-std/console2.sol"; @@ -8,16 +9,7 @@ import {IVotingEscrowIncreasing, ILockedBalanceIncreasing} from "src/escrow/incr import {QuadraticCurveBase} from "./QuadraticCurveBase.t.sol"; contract TestQuadraticIncreasingCurve is QuadraticCurveBase { - function test_votingPowerComputesCorrect() public { - /** - Period Result - 1 1 - 2 1.428571429 - 3 2.142857143 - 4 3.142857143 - 5 4.428571429 - 6 6 - */ + function test_votingPowerComputesCorrect() public view { uint256 amount = 100e18; int256[3] memory coefficients = curve.getCoefficients(100e18); @@ -30,7 +22,7 @@ contract TestQuadraticIncreasingCurve is QuadraticCurveBase { console.log("Coefficients: %st^2 + %st + %s", quadratic, linear, const); - for (uint i; i <= 6; i++) { + for (uint i; i <= 53; i++) { uint period = 2 weeks * i; console.log( "Period: %d Voting Power : %s", @@ -58,6 +50,35 @@ contract TestQuadraticIncreasingCurve is QuadraticCurveBase { } // write a new checkpoint + /* + * for the 1000 tokens (Python) (extend to 1bn with more zeros) + * 0 Voting Power: 1000000000000000000000 + * 1 minute Voting Power: 1000000953907203932160 + * 1 hour Voting Power: 1000057234432234487808 + * 1 day Voting Power: 1001373626373626396672 + * WARMUP_PERIOD (3 days) Voting Power: 1004120879120879058944 + * WARMUP_PERIOD + 1s Voting Power: 1004120895019332534272 + * 1 week Voting Power: 1009615384615384645632 + * 1 period (2 weeks) Voting Power: 1019230769230769160192 + * 10 periods (10 * PERIOD) Voting Power: 1192307692307692388352 + * 50% periods (26 * PERIOD) Voting Power: 1500000000000000000000 + * 35 periods (35 * PERIOD) Voting Power: 1673076923076923097088 + * PERIOD_END (26 * PERIOD) Voting Power: 2000000000000000000000 + + * for the 420.69 tokens (Python) + * 0 Voting Power: 420690000000000000000 + * 1 minute Voting Power: 420690401299221577728 + * 1 hour Voting Power: 420714077953296695296 + * 1 day Voting Power: 421267870879120883712 + * WARMUP_PERIOD (3 days) Voting Power: 422423612637362651136 + * WARMUP_PERIOD + 1s Voting Power: 422423619325683040256 + * 1 week Voting Power: 424735096153846120448 + * 1 period (2 weeks) Voting Power: 428780192307692306432 + * 10 periods (10 * PERIOD) Voting Power: 501591923076923064320 + * 50% periods (26 * PERIOD) Voting Power: 631035000000000032768 + * 35 periods (35 * PERIOD) Voting Power: 703846730769230856192 + * PERIOD_END (26 * PERIOD) Voting Power: 841380000000000000000 + */ function testWritesCheckpoint() public { uint tokenIdFirst = 1; uint tokenIdSecond = 2; @@ -104,52 +125,43 @@ contract TestQuadraticIncreasingCurve is QuadraticCurveBase { // warmup complete vm.warp(block.timestamp + 1); - // python: 449.206279554928541696 - // solmate (optimized): 449.206254284606635135 assertEq( curve.votingPowerAt(tokenIdFirst, block.timestamp), - 449206254284606635135, + 422423619325633557508, "Balance incorrect after warmup" ); assertEq(curve.isWarm(tokenIdFirst), true, "Still warming up"); - // python: 1067784543380942056100724736 - // solmate: 1067784483312193385000000000 assertEq( curve.votingPowerAt(tokenIdSecond, block.timestamp), - 1067784483312193385000000000, + 1004120895019214998000000000, "Balance incorrect after warmup II" ); // warp to the start of period 2 vm.warp(start + clock.epochDuration()); - // excel: 600.985714300000000000 - // PRB: 600.985163959347100568 - // solmate: 600.985163959347101852 - // python : 600.985714285714341888 - // solmate2: 600.985163959347101952 assertEq( curve.votingPowerAt(tokenIdFirst, block.timestamp), - 600985163959347101952, + 428780192307461588352, "Balance incorrect after p1" ); - uint256 expectedMaxI = 2524126241845405205760; - uint256 expectedMaxII = 5999967296216704000000000000; + uint256 expectedMaxI = 841379999988002594304; + uint256 expectedMaxII = 1999999999971481600000000000; // warp to the final period - // TECHNICALLY, this should finish at exactly 5 periodd and 6 * voting power - // but FP arithmetic has a small rounding error - vm.warp(start + clock.epochDuration() * 5); + // TECHNICALLY, this should round to a whole max + // but FP arithmetic has a small rounding error and it finishes just below + vm.warp(start + clock.epochDuration() * 52); assertEq( curve.votingPowerAt(tokenIdFirst, block.timestamp), expectedMaxI, - "Balance incorrect after p6" + "Balance incorrect after pend" ); assertEq( curve.votingPowerAt(tokenIdSecond, block.timestamp), expectedMaxII, - "Balance incorrect after p6 II " + "Balance incorrect after pend II " ); // warp to the future and balance should be the same diff --git a/test/fork/e2eV2.t.sol b/test/fork/e2eV2.t.sol index ca7468d..cf92ea5 100644 --- a/test/fork/e2eV2.t.sol +++ b/test/fork/e2eV2.t.sol @@ -53,10 +53,14 @@ contract TestE2EV2 is Test, IWithdrawalQueueErrors, IGaugeVote, IEscrowCurveToke error VotingInactive(); error OnlyEscrow(); error GaugeDoesNotExist(address _pool); + error AmountTooSmall(); error NotApprovedOrOwner(); error NoVotingPower(); error NotWhitelisted(); error NothingToSweep(); + error MinLockNotReached(uint256 tokenId, uint48 minLock, uint48 earliestExitDate); + + uint constant MONTH = 2592000; GaugesDaoFactory factory; @@ -519,9 +523,9 @@ contract TestE2EV2 is Test, IWithdrawalQueueErrors, IGaugeVote, IEscrowCurveToke } // carlos goes first and makes the first deposit, it's at the start of the - // week, so we would expect him to be warm by the end of the week if using <6 day + // week, so we would expect him to be warm after 1 week // we wait a couple of days and he makes a deposit for javi - // we expect his warmup to carryover to the next week + // we expect his warmup to carryover also to the next week // we expect both of their locks to start accruing voting power on the same day { goToEpochStartPlus(1 days); @@ -530,6 +534,10 @@ contract TestE2EV2 is Test, IWithdrawalQueueErrors, IGaugeVote, IEscrowCurveToke { token.approve(address(escrow), balanceCarlos); + // we also check he can't create too small a lock + vm.expectRevert(AmountTooSmall.selector); + escrow.createLock(100 ether - 1); + escrow.createLock(depositCarlos0); goToEpochStartPlus(6 days); @@ -552,7 +560,7 @@ contract TestE2EV2 is Test, IWithdrawalQueueErrors, IGaugeVote, IEscrowCurveToke assertEq( tp1_1.checkpointTs, epochStartTime + clock.checkpointInterval(), - "Carlos point should have the correct checkpoint" + "carlos point should have the correct checkpoint" ); assertEq( tp2_1.checkpointTs, @@ -611,13 +619,35 @@ contract TestE2EV2 is Test, IWithdrawalQueueErrors, IGaugeVote, IEscrowCurveToke // fast forward to the checkpoint interval carlos is warm and has voting power, javi is not goToEpochStartPlus(clock.checkpointInterval()); - assertEq(escrow.votingPower(1), depositCarlos0, "Carlos should have voting power"); + assertEq( + escrow.votingPower(1), + curve.getBias(0, depositCarlos0), + "Carlos should not yet have voting power" + ); assertTrue(curve.isWarm(1), "Carlos should not be warm"); assertEq(escrow.votingPower(2), 0, "Javi should not have the correct voting power"); assertFalse(curve.isWarm(2), "Javi should not be warm"); } + // carlos can't even begin an exit because of the min lock + { + vm.startPrank(carlos); + { + lock.approve(address(escrow), 1); + + TokenPoint memory tp1_1 = curve.tokenPointHistory(1, 1); + + uint expectedMinLock = tp1_1.checkpointTs + MONTH; + + vm.expectRevert( + abi.encodeWithSelector(MinLockNotReached.selector, 1, MONTH, expectedMinLock) + ); + escrow.beginWithdrawal(1); + } + vm.stopPrank(); + } + // we fast forward 4 weeks and check the expected balances { goToEpochStartPlus(4 weeks); @@ -686,7 +716,7 @@ contract TestE2EV2 is Test, IWithdrawalQueueErrors, IGaugeVote, IEscrowCurveToke assertEq( escrow.votingPowerForAccount(carlos), curve.getBias(timeElapsedSinceFirstLock, depositCarlos0) + depositCarlos1, - "Carlos should now have the correct aggregate voting power" + "Carlos should have extra voting power" ); } @@ -699,7 +729,13 @@ contract TestE2EV2 is Test, IWithdrawalQueueErrors, IGaugeVote, IEscrowCurveToke vm.expectRevert(OnlyEscrow.selector); queue.queueExit(i, jordan); - vm.expectRevert(erc721ownererr); + // lingering permissions from carlos' approval + if (i == 1) { + vm.expectRevert(bytes("ERC721: transfer from incorrect owner")); + } else { + vm.expectRevert(erc721ownererr); + } + escrow.beginWithdrawal(i); } } @@ -1024,10 +1060,11 @@ contract TestE2EV2 is Test, IWithdrawalQueueErrors, IGaugeVote, IEscrowCurveToke assertEq(lock.ownerOf(1), address(escrow), "Carlos should not own the nft"); assertEq(queue.queue(1).holder, carlos, "Carlos should be in the queue"); - // exit date should be the next checkpoint + // expected exit date is: + // now + cooldown given that it crosses the cp boundary assertEq( queue.queue(1).exitDate, - epochStartTime + 8 weeks + clock.checkpointInterval(), + epochStartTime + 8 weeks + 1 hours + queue.cooldown(), "Carlos should be able to exit at the next checkpoint" ); @@ -1052,16 +1089,15 @@ contract TestE2EV2 is Test, IWithdrawalQueueErrors, IGaugeVote, IEscrowCurveToke vm.expectRevert(CannotExit.selector); escrow.withdraw(1); - // he waits till the end of the week to exit - goToEpochStartPlus(9 weeks); + // go to cooldown end + goToEpochStartPlus(8 weeks + 1 hours + MONTH); // can't exit yet vm.expectRevert(CannotExit.selector); escrow.withdraw(1); - // + 1s he can - - goToEpochStartPlus(9 weeks + 1); + // + 1s he can (1738616401) + goToEpochStartPlus(8 weeks + 1 hours + MONTH + 1); escrow.withdraw(1); } @@ -1080,6 +1116,9 @@ contract TestE2EV2 is Test, IWithdrawalQueueErrors, IGaugeVote, IEscrowCurveToke depositCarlos1 + depositCarlosJavi + balanceJordi, "Total locked should be the sum of the two deposits" ); + + // there are no fees in our contract + assertEq(token.balanceOf(address(queue)), 0, "Queue should have no fees"); } // governance changes some params: warmup is now one day, cooldown is a week @@ -1246,8 +1285,8 @@ contract TestE2EV2 is Test, IWithdrawalQueueErrors, IGaugeVote, IEscrowCurveToke // we get all the guys to exit and unwind their positions { - // warp to a voting window - goToEpochStartPlus(12 weeks + 2 hours); + // warp to a voting window - must pass the min lock + goToEpochStartPlus(16 weeks + 2 hours); vm.startPrank(carlos); { @@ -1266,8 +1305,8 @@ contract TestE2EV2 is Test, IWithdrawalQueueErrors, IGaugeVote, IEscrowCurveToke } vm.stopPrank(); - // fast forward like 5 weeks - goToEpochStartPlus(16 weeks); + // fast forward a month + goToEpochStartPlus(16 weeks + 2 hours + MONTH); // carlos exits vm.startPrank(carlos); diff --git a/test/python/crosscheck.py b/test/python/crosscheck.py index 86f544e..a40b634 100644 --- a/test/python/crosscheck.py +++ b/test/python/crosscheck.py @@ -1,40 +1,39 @@ # time utils HOUR = 60 * 60 -DAY = 24 * HOUR +DAY = 24 * HOUR WEEK = 7 * DAY # Variables -AMOUNT = 1_000_000_000 # example amount to deposit +AMOUNT = 420.69 # example amount to deposit PERIOD_LENGTH = 2 * WEEK # example period length in seconds (1 week) WARMUP_PERIOD = 3 * DAY # warmup period in days -MAX_PERIODS = 5 # maximum periods +MAX_PERIODS = 52 # maximum periods -QUADRATIC_COEFFICIENT = 1/7 -LINEAR_COEFFICIENT = 2/7 +QUADRATIC_COEFFICIENT = 0 +LINEAR_COEFFICIENT = 1 / 52 CONSTANT = 1 # Scale amount amount_scaled = AMOUNT * 1e18 + # Function to evaluate y def evaluate_y(secondsElapsed, PeriodLength, amount_scaled): x = secondsElapsed / PeriodLength y = amount_scaled * ( - QUADRATIC_COEFFICIENT * (x**2) - + LINEAR_COEFFICIENT * x - + CONSTANT + QUADRATIC_COEFFICIENT * (x**2) + LINEAR_COEFFICIENT * x + CONSTANT ) return y + def evaluate_y_v2(secondsElapsed, PeriodLength, amount_scaled): x = secondsElapsed / PeriodLength y = amount_scaled * ( - QUADRATIC_COEFFICIENT * (x*x) - + LINEAR_COEFFICIENT * x - + CONSTANT + QUADRATIC_COEFFICIENT * (x * x) + LINEAR_COEFFICIENT * x + CONSTANT ) return y + # Time points to evaluate, using tuples with optional labels time_points = [ ("0", 0), @@ -42,13 +41,13 @@ def evaluate_y_v2(secondsElapsed, PeriodLength, amount_scaled): ("1 hour", 60 * 60), ("1 day", 60 * 60 * 24), (f"WARMUP_PERIOD ({WARMUP_PERIOD//DAY} days)", WARMUP_PERIOD), - (f"WARMUP_PERIOD + 1s", (WARMUP_PERIOD) + 1), + (f"WARMUP_PERIOD + 1s", (WARMUP_PERIOD) + 1), ("1 week", 60 * 60 * 24 * 7), (f"1 period ({PERIOD_LENGTH // (WEEK)} weeks)", PERIOD_LENGTH), - (f"2 periods (2 * PERIOD)", 2 * PERIOD_LENGTH), - (f"3 periods (3 * PERIOD)", 3 * PERIOD_LENGTH), - (f"4 periods (4 * PERIOD)", 4 * PERIOD_LENGTH), - (f"PERIOD_END (5 * PERIOD)", MAX_PERIODS * PERIOD_LENGTH) + (f"10 periods (10 * PERIOD)", 10 * PERIOD_LENGTH), + (f"50% periods (26 * PERIOD)", 26 * PERIOD_LENGTH), + (f"35 periods (35 * PERIOD)", 35 * PERIOD_LENGTH), + (f"PERIOD_END (26 * PERIOD)", MAX_PERIODS * PERIOD_LENGTH), ] # Evaluate and print results @@ -56,5 +55,3 @@ def evaluate_y_v2(secondsElapsed, PeriodLength, amount_scaled): y_value = evaluate_y(t, PERIOD_LENGTH, amount_scaled) # Avoid scientific notation by formatting with commas and align values vertically print(f"{label:<30} Voting Power: {y_value:>20.0f}") - -