From e8167f777736bdce27272938192f53644759327d Mon Sep 17 00:00:00 2001 From: Cheyenne Atapour Date: Wed, 26 Jun 2024 04:46:42 -0700 Subject: [PATCH] Add AIP for V4 AL Service Provider Proposal (#357) --- ...LServiceProviderProposal_20240614_after.md | 5 + ...m_V4ALServiceProviderProposal_20240614.sol | 41 +++ ...V4ALServiceProviderProposal_20240614.t.sol | 311 ++++++++++++++++++ .../V4ALServiceProviderProposal.md | 112 +++++++ ...V4ALServiceProviderProposal_20240614.s.sol | 60 ++++ .../config.ts | 14 + 6 files changed, 543 insertions(+) create mode 100644 diffs/AaveV3Ethereum_V4ALServiceProviderProposal_20240614_before_AaveV3Ethereum_V4ALServiceProviderProposal_20240614_after.md create mode 100644 src/20240614_AaveV3Ethereum_V4ALServiceProviderProposal/AaveV3Ethereum_V4ALServiceProviderProposal_20240614.sol create mode 100644 src/20240614_AaveV3Ethereum_V4ALServiceProviderProposal/AaveV3Ethereum_V4ALServiceProviderProposal_20240614.t.sol create mode 100644 src/20240614_AaveV3Ethereum_V4ALServiceProviderProposal/V4ALServiceProviderProposal.md create mode 100644 src/20240614_AaveV3Ethereum_V4ALServiceProviderProposal/V4ALServiceProviderProposal_20240614.s.sol create mode 100644 src/20240614_AaveV3Ethereum_V4ALServiceProviderProposal/config.ts diff --git a/diffs/AaveV3Ethereum_V4ALServiceProviderProposal_20240614_before_AaveV3Ethereum_V4ALServiceProviderProposal_20240614_after.md b/diffs/AaveV3Ethereum_V4ALServiceProviderProposal_20240614_before_AaveV3Ethereum_V4ALServiceProviderProposal_20240614_after.md new file mode 100644 index 000000000..c15d3e2bc --- /dev/null +++ b/diffs/AaveV3Ethereum_V4ALServiceProviderProposal_20240614_before_AaveV3Ethereum_V4ALServiceProviderProposal_20240614_after.md @@ -0,0 +1,5 @@ +## Raw diff + +```json +{} +``` \ No newline at end of file diff --git a/src/20240614_AaveV3Ethereum_V4ALServiceProviderProposal/AaveV3Ethereum_V4ALServiceProviderProposal_20240614.sol b/src/20240614_AaveV3Ethereum_V4ALServiceProviderProposal/AaveV3Ethereum_V4ALServiceProviderProposal_20240614.sol new file mode 100644 index 000000000..6d9005b5c --- /dev/null +++ b/src/20240614_AaveV3Ethereum_V4ALServiceProviderProposal/AaveV3Ethereum_V4ALServiceProviderProposal_20240614.sol @@ -0,0 +1,41 @@ +// SPDX-License-Identifier: MIT +pragma solidity ^0.8.0; + +import {IProposalGenericExecutor} from 'aave-helpers/interfaces/IProposalGenericExecutor.sol'; +import {AaveV3Ethereum, AaveV3EthereumAssets} from 'aave-address-book/AaveV3Ethereum.sol'; + +/** + * @title V4 AL Service Provider Proposal + * @author Aave Labs + * - Snapshot: https://snapshot.org/#/aave.eth/proposal/0x70dfd865b78c4c391e2b0729b907d152e6e8a0da683416d617d8f84782036349 + * - Discussion: https://governance.aave.com/t/arfc-al-service-provider-proposal/17974 + */ +contract AaveV3Ethereum_V4ALServiceProviderProposal_20240614 is IProposalGenericExecutor { + address public constant AAVE_LABS = 0x1c037b3C22240048807cC9d7111be5d455F640bd; + + // 3 million GHO upfront + uint256 public constant GHO_UPFRONT_AMOUNT = 3_000_000 ether; + + // 9 million GHO streamed over the year + uint256 public constant GHO_STREAM_AMOUNT = 9_000_000 ether; + uint256 public constant GHO_STREAM_DURATION = 365 days; + + uint256 public constant ACTUAL_STREAM = + (GHO_STREAM_AMOUNT / GHO_STREAM_DURATION) * GHO_STREAM_DURATION; + + function execute() external { + AaveV3Ethereum.COLLECTOR.createStream( + AAVE_LABS, + ACTUAL_STREAM, + AaveV3EthereumAssets.GHO_UNDERLYING, + block.timestamp, + block.timestamp + GHO_STREAM_DURATION + ); + + AaveV3Ethereum.COLLECTOR.transfer( + AaveV3EthereumAssets.GHO_UNDERLYING, + AAVE_LABS, + GHO_UPFRONT_AMOUNT + ); + } +} diff --git a/src/20240614_AaveV3Ethereum_V4ALServiceProviderProposal/AaveV3Ethereum_V4ALServiceProviderProposal_20240614.t.sol b/src/20240614_AaveV3Ethereum_V4ALServiceProviderProposal/AaveV3Ethereum_V4ALServiceProviderProposal_20240614.t.sol new file mode 100644 index 000000000..0b0132243 --- /dev/null +++ b/src/20240614_AaveV3Ethereum_V4ALServiceProviderProposal/AaveV3Ethereum_V4ALServiceProviderProposal_20240614.t.sol @@ -0,0 +1,311 @@ +// SPDX-License-Identifier: MIT +pragma solidity ^0.8.0; + +import {AaveV3Ethereum, AaveV3EthereumAssets} from 'aave-address-book/AaveV3Ethereum.sol'; + +import 'forge-std/Test.sol'; +import 'forge-std/StdUtils.sol'; +import {ProtocolV3TestBase, ReserveConfig} from 'aave-helpers/ProtocolV3TestBase.sol'; +import {AaveV3Ethereum_V4ALServiceProviderProposal_20240614} from './AaveV3Ethereum_V4ALServiceProviderProposal_20240614.sol'; +import {IERC20} from 'solidity-utils/contracts/oz-common/interfaces/IERC20.sol'; + +/** + * @dev Test for AaveV3Ethereum_V4ALServiceProviderProposal_20240614 + * command: FOUNDRY_PROFILE=mainnet forge test --match-path=src/20240614_AaveV3Ethereum_V4ALServiceProviderProposal/AaveV3Ethereum_V4ALServiceProviderProposal_20240614.t.sol -vv + */ +contract AaveV3Ethereum_V4ALServiceProviderProposal_20240614_Test is ProtocolV3TestBase { + AaveV3Ethereum_V4ALServiceProviderProposal_20240614 internal proposal; + + address public constant AAVE_LABS = 0x1c037b3C22240048807cC9d7111be5d455F640bd; + uint256 public constant GHO_UPFRONT_AMOUNT = 3_000_000 ether; + uint256 public constant GHO_STREAM_AMOUNT = 9_000_000 ether; + uint256 public constant GHO_STREAM_DURATION = 365 days; + uint256 public constant ACTUAL_STREAM_AMOUNT_GHO = + (GHO_STREAM_AMOUNT / GHO_STREAM_DURATION) * GHO_STREAM_DURATION; + + function setUp() public { + vm.createSelectFork(vm.rpcUrl('mainnet'), 20092863); + proposal = new AaveV3Ethereum_V4ALServiceProviderProposal_20240614(); + } + + /** + * @dev executes the generic test suite including e2e and config snapshots + */ + function test_defaultProposalExecution() public { + defaultTest( + 'AaveV3Ethereum_V4ALServiceProviderProposal_20240614', + AaveV3Ethereum.POOL, + address(proposal) + ); + } + + function testProposalExecution() public { + uint256 nextCollectorStreamID = AaveV3Ethereum.COLLECTOR.getNextStreamId(); + uint256 ALGHOBalanceBefore = IERC20(AaveV3EthereumAssets.GHO_UNDERLYING).balanceOf(AAVE_LABS); + uint256 CollectorV3GHOBalanceBefore = IERC20(AaveV3EthereumAssets.GHO_UNDERLYING).balanceOf( + address(AaveV3Ethereum.COLLECTOR) + ); + + executePayload(vm, address(proposal)); + + // Check balances directly after proposal execution (upfront payment distributed to Aave Labs) + assertEq( + IERC20(AaveV3EthereumAssets.GHO_UNDERLYING).balanceOf(AAVE_LABS), + ALGHOBalanceBefore + GHO_UPFRONT_AMOUNT + ); + assertEq( + IERC20(AaveV3EthereumAssets.GHO_UNDERLYING).balanceOf(address(AaveV3Ethereum.COLLECTOR)), + CollectorV3GHOBalanceBefore - GHO_UPFRONT_AMOUNT + ); + + // Checking if the streams have been created properly + // Scoping to avoid "stack too deep" error + { + ( + address senderGHO, + address recipientGHO, + uint256 depositGHO, + address tokenAddressGHO, + uint256 startTimeGHO, + uint256 stopTimeGHO, + uint256 remainingBalanceGHO, + + ) = AaveV3Ethereum.COLLECTOR.getStream(nextCollectorStreamID); + + assertEq(senderGHO, address(AaveV3Ethereum.COLLECTOR)); + assertEq(recipientGHO, AAVE_LABS); + assertEq(depositGHO, ACTUAL_STREAM_AMOUNT_GHO); + assertEq(tokenAddressGHO, AaveV3EthereumAssets.GHO_UNDERLYING); + assertEq(stopTimeGHO - startTimeGHO, GHO_STREAM_DURATION); + assertEq(remainingBalanceGHO, ACTUAL_STREAM_AMOUNT_GHO); + } + + // Checking if Aave Labs can withdraw from the stream + vm.startPrank(AAVE_LABS); + vm.warp(block.timestamp + GHO_STREAM_DURATION); + + // Currently Collector has less funds than stream amount + assertLe( + IERC20(AaveV3EthereumAssets.GHO_UNDERLYING).balanceOf(address(AaveV3Ethereum.COLLECTOR)), + ACTUAL_STREAM_AMOUNT_GHO + ); + + // Partial withdrawal of Collector's remaining balance + AaveV3Ethereum.COLLECTOR.withdrawFromStream( + nextCollectorStreamID, + CollectorV3GHOBalanceBefore - GHO_UPFRONT_AMOUNT + ); + uint256 nextALGHOBalance = IERC20(AaveV3EthereumAssets.GHO_UNDERLYING).balanceOf(AAVE_LABS); + + // Aave Labs received the entirety of Collector's balance + assertEq(ALGHOBalanceBefore, nextALGHOBalance - CollectorV3GHOBalanceBefore); + + // Check Collector balance after stream withdrawal + uint256 CollectorV3GHOBalanceAfter = IERC20(AaveV3EthereumAssets.GHO_UNDERLYING).balanceOf( + address(AaveV3Ethereum.COLLECTOR) + ); + + assertEq(CollectorV3GHOBalanceAfter, 0); + + vm.stopPrank(); + } + + // Test giving V2 Collector more funds and full withdrawing + function testProposalExecutionPrankFunds() public { + uint256 nextCollectorStreamID = AaveV3Ethereum.COLLECTOR.getNextStreamId(); + uint256 ALGHOBalanceBefore = IERC20(AaveV3EthereumAssets.GHO_UNDERLYING).balanceOf(AAVE_LABS); + + // Giving the Collector enough funds to cover the stream + deal( + AaveV3EthereumAssets.GHO_UNDERLYING, + address(AaveV3Ethereum.COLLECTOR), + GHO_UPFRONT_AMOUNT + GHO_STREAM_AMOUNT + ); + + uint256 CollectorV3GHOBalanceBefore = IERC20(AaveV3EthereumAssets.GHO_UNDERLYING).balanceOf( + address(AaveV3Ethereum.COLLECTOR) + ); + + executePayload(vm, address(proposal)); + + // Check balances directly after proposal execution (upfront payment distributed to Aave Labs) + assertEq( + IERC20(AaveV3EthereumAssets.GHO_UNDERLYING).balanceOf(AAVE_LABS), + ALGHOBalanceBefore + GHO_UPFRONT_AMOUNT + ); + assertEq( + IERC20(AaveV3EthereumAssets.GHO_UNDERLYING).balanceOf(address(AaveV3Ethereum.COLLECTOR)), + CollectorV3GHOBalanceBefore - GHO_UPFRONT_AMOUNT + ); + + // Checking if the streams have been created properly + // Scoping to avoid "stack too deep" error + { + ( + address senderGHO, + address recipientGHO, + uint256 depositGHO, + address tokenAddressGHO, + uint256 startTimeGHO, + uint256 stopTimeGHO, + uint256 remainingBalanceGHO, + + ) = AaveV3Ethereum.COLLECTOR.getStream(nextCollectorStreamID); + + assertEq(senderGHO, address(AaveV3Ethereum.COLLECTOR)); + assertEq(recipientGHO, AAVE_LABS); + assertEq(depositGHO, ACTUAL_STREAM_AMOUNT_GHO); + assertEq(tokenAddressGHO, AaveV3EthereumAssets.GHO_UNDERLYING); + assertEq(stopTimeGHO - startTimeGHO, GHO_STREAM_DURATION); + assertEq(remainingBalanceGHO, ACTUAL_STREAM_AMOUNT_GHO); + } + + // Checking if Aave Labs can withdraw from the stream + vm.startPrank(AAVE_LABS); + vm.warp(block.timestamp + GHO_STREAM_DURATION); + + assertGe( + IERC20(AaveV3EthereumAssets.GHO_UNDERLYING).balanceOf(address(AaveV3Ethereum.COLLECTOR)), + ACTUAL_STREAM_AMOUNT_GHO + ); + + AaveV3Ethereum.COLLECTOR.withdrawFromStream(nextCollectorStreamID, ACTUAL_STREAM_AMOUNT_GHO); + uint256 nextALGHOBalance = IERC20(AaveV3EthereumAssets.GHO_UNDERLYING).balanceOf(AAVE_LABS); + + assertEq( + ALGHOBalanceBefore, + nextALGHOBalance - (ACTUAL_STREAM_AMOUNT_GHO + GHO_UPFRONT_AMOUNT) + ); + + // Check Collector balance after stream withdrawal + uint256 CollectorV3GHOBalanceAfter = IERC20(AaveV3EthereumAssets.GHO_UNDERLYING).balanceOf( + address(AaveV3Ethereum.COLLECTOR) + ); + + assertEq( + CollectorV3GHOBalanceAfter, + CollectorV3GHOBalanceBefore - (ACTUAL_STREAM_AMOUNT_GHO + GHO_UPFRONT_AMOUNT) + ); + + vm.stopPrank(); + } + + // Test showing V2 Collector currently doesn't have enough funds + function testProposalExecutionFailureFunding() public { + uint256 nextCollectorStreamID = AaveV3Ethereum.COLLECTOR.getNextStreamId(); + uint256 ALGHOBalanceBefore = IERC20(AaveV3EthereumAssets.GHO_UNDERLYING).balanceOf(AAVE_LABS); + + uint256 CollectorV3GHOBalanceBefore = IERC20(AaveV3EthereumAssets.GHO_UNDERLYING).balanceOf( + address(AaveV3Ethereum.COLLECTOR) + ); + + executePayload(vm, address(proposal)); + + // Check balances directly after proposal execution (upfront payment distributed to Aave Labs) + assertEq( + IERC20(AaveV3EthereumAssets.GHO_UNDERLYING).balanceOf(AAVE_LABS), + ALGHOBalanceBefore + GHO_UPFRONT_AMOUNT + ); + assertEq( + IERC20(AaveV3EthereumAssets.GHO_UNDERLYING).balanceOf(address(AaveV3Ethereum.COLLECTOR)), + CollectorV3GHOBalanceBefore - GHO_UPFRONT_AMOUNT + ); + + // Checking if the streams have been created properly + // Scoping to avoid "stack too deep" error + { + ( + address senderGHO, + address recipientGHO, + uint256 depositGHO, + address tokenAddressGHO, + uint256 startTimeGHO, + uint256 stopTimeGHO, + uint256 remainingBalanceGHO, + + ) = AaveV3Ethereum.COLLECTOR.getStream(nextCollectorStreamID); + + assertEq(senderGHO, address(AaveV3Ethereum.COLLECTOR)); + assertEq(recipientGHO, AAVE_LABS); + assertEq(depositGHO, ACTUAL_STREAM_AMOUNT_GHO); + assertEq(tokenAddressGHO, AaveV3EthereumAssets.GHO_UNDERLYING); + assertEq(stopTimeGHO - startTimeGHO, GHO_STREAM_DURATION); + assertEq(remainingBalanceGHO, ACTUAL_STREAM_AMOUNT_GHO); + } + + // Checking if Aave Labs can withdraw from the stream + vm.startPrank(AAVE_LABS); + vm.warp(block.timestamp + GHO_STREAM_DURATION); + + assertLe( + IERC20(AaveV3EthereumAssets.GHO_UNDERLYING).balanceOf(address(AaveV3Ethereum.COLLECTOR)), + ACTUAL_STREAM_AMOUNT_GHO + ); + + vm.expectRevert(stdError.arithmeticError); + AaveV3Ethereum.COLLECTOR.withdrawFromStream(nextCollectorStreamID, ACTUAL_STREAM_AMOUNT_GHO); + + vm.stopPrank(); + } + + function testProposalExecutionFuzzWithdrawalAmounts( + uint256 withdrawalAmount, + uint256 waitPeriod + ) public { + waitPeriod = bound(waitPeriod, 1, GHO_STREAM_DURATION); + withdrawalAmount = bound( + withdrawalAmount, + 1, + waitPeriod * (ACTUAL_STREAM_AMOUNT_GHO / GHO_STREAM_DURATION) + ); + + // Giving the Collector enough funds to cover the stream + deal( + AaveV3EthereumAssets.GHO_UNDERLYING, + address(AaveV3Ethereum.COLLECTOR), + GHO_UPFRONT_AMOUNT + GHO_STREAM_AMOUNT + ); + + assertGe( + IERC20(AaveV3EthereumAssets.GHO_UNDERLYING).balanceOf(address(AaveV3Ethereum.COLLECTOR)), + ACTUAL_STREAM_AMOUNT_GHO + ); + + uint256 ALGHOBalanceBefore = IERC20(AaveV3EthereumAssets.GHO_UNDERLYING).balanceOf(AAVE_LABS); + uint256 CollectorV3GHOBalanceBefore = IERC20(AaveV3EthereumAssets.GHO_UNDERLYING).balanceOf( + address(AaveV3Ethereum.COLLECTOR) + ); + + uint256 nextCollectorStreamID = AaveV3Ethereum.COLLECTOR.getNextStreamId(); + executePayload(vm, address(proposal)); + + // Check balances directly after proposal execution (upfront payment distributed to Aave Labs) + assertEq( + IERC20(AaveV3EthereumAssets.GHO_UNDERLYING).balanceOf(AAVE_LABS), + ALGHOBalanceBefore + GHO_UPFRONT_AMOUNT + ); + assertEq( + IERC20(AaveV3EthereumAssets.GHO_UNDERLYING).balanceOf(address(AaveV3Ethereum.COLLECTOR)), + CollectorV3GHOBalanceBefore - GHO_UPFRONT_AMOUNT + ); + + vm.startPrank(AAVE_LABS); + vm.warp(block.timestamp + waitPeriod); + + // Withdraw fuzzed amount from stream + AaveV3Ethereum.COLLECTOR.withdrawFromStream(nextCollectorStreamID, withdrawalAmount); + + uint256 ALGHOBalanceAfter = IERC20(AaveV3EthereumAssets.GHO_UNDERLYING).balanceOf(AAVE_LABS); + uint256 CollectorV3GHOBalanceAfter = IERC20(AaveV3EthereumAssets.GHO_UNDERLYING).balanceOf( + address(AaveV3Ethereum.COLLECTOR) + ); + + assertEq(ALGHOBalanceAfter - ALGHOBalanceBefore, withdrawalAmount + GHO_UPFRONT_AMOUNT); + assertEq( + CollectorV3GHOBalanceAfter, + CollectorV3GHOBalanceBefore - (withdrawalAmount + GHO_UPFRONT_AMOUNT) + ); + + vm.stopPrank(); + } +} diff --git a/src/20240614_AaveV3Ethereum_V4ALServiceProviderProposal/V4ALServiceProviderProposal.md b/src/20240614_AaveV3Ethereum_V4ALServiceProviderProposal/V4ALServiceProviderProposal.md new file mode 100644 index 000000000..bc06bd09e --- /dev/null +++ b/src/20240614_AaveV3Ethereum_V4ALServiceProviderProposal/V4ALServiceProviderProposal.md @@ -0,0 +1,112 @@ +--- +title: "AL Service Provider Proposal" +author: "Aave Labs" +discussions: "https://governance.aave.com/t/arfc-al-service-provider-proposal/17974" +snapshot: "https://snapshot.org/#/aave.eth/proposal/0x70dfd865b78c4c391e2b0729b907d152e6e8a0da683416d617d8f84782036349" +--- + +## Summary + +This AIP proposes a one-year technical contributor engagement for the Aave Labs to act as a service provider for development of Aave V4 and a new visual identity. + +## Motivation + +Aave Labs seeks to be onboarded as a service provider to be one of the technical contributors to the Aave DAO. The specification below outlines the scope of the technical contributions of Aave Labs. + +## Specification + +The following are the key deliverables within the grant: + +### Aave Protocol V4 + +Key deliverables for the 1 year scope of development include: + +- New Modular Architecture + + - An innovative design to enhance flexibility and efficiency, minimizing disruption for integrators. + +- Unified Liquidity Layer + + - A flexible liquidity management approach that allows for module modifications without needing to migrate liquidity. + +- Fuzzy-controlled Interest Rates + + - Automated rate adjustments based on market conditions, optimizing for both suppliers and borrowers. Utilizing Chainlink for precise data feeds, setting new standards in capital efficiency. + +- Liquidity Premiums + + - Adjusted borrowing costs based on collateral risk, ensuring fairer pricing. Higher risks incur higher premiums, while lower risks reduce costs. + +- Dynamic Risk Configuration + + - Allows dynamic adjustments to risk parameters based on market conditions without governance overhead. + +- Smart Accounts and Vaults + + - Simplifies user interactions with the protocol by allowing multiple smart accounts per wallet and enabling borrowing without direct collateral supply. + +- Automated Assets Offboarding + + - Streamlines the process of offboarding assets, reducing governance workload and ensuring predictable offboarding plans. + +- Enhanced Liquidation Engine + + - Introduces a variable liquidation factor, liquidation strategies, and batch liquidations to improve efficiency and borrower experience. + +- Automated Treasury Management + + - Implements a reverse auction mechanism for reserve factor assets, reducing governance overhead. + +- GHO Features + + - Native GHO Minting + + - Allows for more efficient minting of GHO directly from the liquidity layer. + + - GHO “Soft” Liquidations + + - Utilizes a LLAMM model to ease liquidations and manage market downturns, providing options for users to choose which collateral to liquidate to GHO. + + - Stablecoin Interest Paid in GHO + + - Suppliers can opt to receive interest payments in GHO, enhancing capital efficiency and providing additional benefits to the protocol. + + - Emergency Redemption Mechanism + - A feature to handle prolonged and heavy depegging of GHO, ensuring stability and reliability. + +### Visual Identity Assets + +[Aave Visual Identity Guidelines | 2024](https://www.youtube.com/watch?v=TQHLCACwnbE) + +Deliverables include: + +- Provision of Visual Identity Assets: Compiling and delivering a package of visual identity assets, including, but are not limited to, logos, color schemes, typography guidelines, various Ronnie illustrations and other graphical elements. + +- Usage Guidelines: Instructions on how to implement the visual identity assets across various platforms and media, including digital and print formats. Practical examples and best practice scenarios demonstrating the potential usage of the visual identity assets. + +- Delivery: The assets will be delivered and packaged in a Github repository accessible to everyone and hosted within the Aave DAO Origin organization. + +## Pospective Second and Third Year Scope + +The DAO may renew the engagement with Aave Labs as a service provider for a technical contributor for a second and third year with that scope of work to be determined at a later date and subject to community input and ARFC approval. + +## Grant + +$12m GHO, including $3m upfront and $9m streamed over the year. + +## Evaluation Metrics + +- Monthly reporting of contributions to the Aave DAO. +- Solicitation of feedback from other service providers working as technical contributors and DAO members. +- Direct and transparent engagement with the DAO via the governance forum. + +## Principles + +- Aave Labs shall strictly adhere to the scope of its designated role as one of the service providers working as a technical contributor, ensuring all decisions and actions remain within our areas of the engagement. +- Aave Labs shall prioritize the security and robustness of its technical contributions. +- Any production-ready code written within the scope of the proposal will be moved to the Aave DAO’s GitHub repository. +- Aave Labs shall engage with the DAO like all other contributors within the established working relationships and procedures of the Aave DAO. For example, decisions concerning risk configurations, DAO treasury, code audits, and other matters not covered by this proposal will need to be separately approved by the Aave DAO via TEMP CHECKs and Snapshots. + +## Copyright + +The text of this AIP is released under the [CC0 license](https://creativecommons.org/publicdomain/zero/1.0/). The visuals and the New Visual identity are subject to and governed by the license specified in the approved governance proposal by the Aave DAO. diff --git a/src/20240614_AaveV3Ethereum_V4ALServiceProviderProposal/V4ALServiceProviderProposal_20240614.s.sol b/src/20240614_AaveV3Ethereum_V4ALServiceProviderProposal/V4ALServiceProviderProposal_20240614.s.sol new file mode 100644 index 000000000..7e2a18682 --- /dev/null +++ b/src/20240614_AaveV3Ethereum_V4ALServiceProviderProposal/V4ALServiceProviderProposal_20240614.s.sol @@ -0,0 +1,60 @@ +// SPDX-License-Identifier: MIT +pragma solidity ^0.8.0; + +import {GovV3Helpers, IPayloadsControllerCore, PayloadsControllerUtils} from 'aave-helpers/GovV3Helpers.sol'; +import {GovernanceV3Ethereum} from 'aave-address-book/GovernanceV3Ethereum.sol'; +import {EthereumScript} from 'aave-helpers/ScriptUtils.sol'; +import {AaveV3Ethereum_V4ALServiceProviderProposal_20240614} from './AaveV3Ethereum_V4ALServiceProviderProposal_20240614.sol'; + +/** + * @dev Deploy Ethereum + * deploy-command: make deploy-ledger contract=src/20240614_AaveV3Ethereum_V4ALServiceProviderProposal/V4ALServiceProviderProposal_20240614.s.sol:DeployEthereum chain=mainnet + * verify-command: FOUNDRY_PROFILE=mainnet npx catapulta-verify -b broadcast/V4ALServiceProviderProposal_20240614.s.sol/1/run-latest.json + */ +contract DeployEthereum is EthereumScript { + function run() external broadcast { + // deploy payloads + address payload0 = GovV3Helpers.deployDeterministic( + type(AaveV3Ethereum_V4ALServiceProviderProposal_20240614).creationCode + ); + + // compose action + IPayloadsControllerCore.ExecutionAction[] + memory actions = new IPayloadsControllerCore.ExecutionAction[](1); + actions[0] = GovV3Helpers.buildAction(payload0); + + // register action at payloadsController + GovV3Helpers.createPayload(actions); + } +} + +/** + * @dev Create Proposal + * command: make deploy-ledger contract=src/20240614_AaveV3Ethereum_V4ALServiceProviderProposal/V4ALServiceProviderProposal_20240614.s.sol:CreateProposal chain=mainnet + */ +contract CreateProposal is EthereumScript { + function run() external { + // create payloads + PayloadsControllerUtils.Payload[] memory payloads = new PayloadsControllerUtils.Payload[](1); + + // compose actions for validation + IPayloadsControllerCore.ExecutionAction[] + memory actionsEthereum = new IPayloadsControllerCore.ExecutionAction[](1); + actionsEthereum[0] = GovV3Helpers.buildAction( + type(AaveV3Ethereum_V4ALServiceProviderProposal_20240614).creationCode + ); + payloads[0] = GovV3Helpers.buildMainnetPayload(vm, actionsEthereum); + + // create proposal + vm.startBroadcast(); + GovV3Helpers.createProposal( + vm, + payloads, + GovernanceV3Ethereum.VOTING_PORTAL_ETH_POL, + GovV3Helpers.ipfsHashFile( + vm, + 'src/20240614_AaveV3Ethereum_V4ALServiceProviderProposal/ALServiceProviderProposal.md' + ) + ); + } +} diff --git a/src/20240614_AaveV3Ethereum_V4ALServiceProviderProposal/config.ts b/src/20240614_AaveV3Ethereum_V4ALServiceProviderProposal/config.ts new file mode 100644 index 000000000..c2c7a9c21 --- /dev/null +++ b/src/20240614_AaveV3Ethereum_V4ALServiceProviderProposal/config.ts @@ -0,0 +1,14 @@ +import {ConfigFile} from '../../generator/types'; +export const config: ConfigFile = { + rootOptions: { + title: 'V4 AL Service Provider Proposal', + author: 'Aave Labs', + discussion: 'https://governance.aave.com/t/arfc-al-service-provider-proposal/17974', + snapshot: + 'https://snapshot.org/#/aave.eth/proposal/0x70dfd865b78c4c391e2b0729b907d152e6e8a0da683416d617d8f84782036349', + pools: ['AaveV3Ethereum'], + shortName: 'V4ALServiceProviderProposal', + date: '20240614', + }, + poolOptions: {AaveV3Ethereum: {configs: {}, cache: {blockNumber: 20092863}}}, +};