The YieldAggregator smart contract is designed to manage user deposits to maximize yield across popular DeFi protocols, such as Aave and Compound. The contract dynamically allocates funds to the protocol with the highest yield, provides fee management, and allows emergency withdrawals for enhanced security.
- Overview
- Features
- Contract Structure
- Usage
- Roles and Permissions
- Fee Structure
- Events
- Security Considerations
- Dependencies
- License
YieldAggregator is a secure, flexible smart contract for managing user funds across DeFi protocols. It allows users to deposit Wrapped ETH (WETH), dynamically shifting the funds between Aave and Compound to achieve optimal returns. The contract features management and performance fees, emergency withdrawal, and rebalance functionalities, all governed by the owner and assigned emergency admins.
- Dynamic Yield Optimization: Switches between Aave and Compound based on the higher APY.
- Flexible Fee Structure: Annual management and performance fees are applied and can be updated by the owner.
- Emergency Controls: Enables admins to manage emergency withdrawals and halt operations if necessary.
- Automated Rebalance: Adjusts fund allocation based on updated APY data.
- Secure WETH Handling: Utilizes SafeERC20 for safe WETH transfers.
- Access Control: Admin and owner roles for secure operation.
- Solidity 0.8.26 or later
- Node.js & npm (for running tests and compiling)
- OpenZeppelin for access control, security, and token standards
- Aave V3 and Compound Protocols (integrated within the contract)
The contract is organized as follows:
- UserDeposit and Fees Structs: Data structures to track user deposits and fee settings.
- Events: Emitted for deposits, withdrawals, fee collection, and other key actions.
- Modifiers: Access controls for emergency actions and emergency checks.
- Deposit and Withdraw Functions: Main user-facing functions for depositing and withdrawing funds.
- Internal Protocol Management: Functions for depositing, withdrawing, and rebalancing between protocols.
- Admin Functions: For emergency actions and protocol configuration.
-
Prerequisites: Deploy the contract with valid addresses for:
- WETH_ADDRESS: WETH token address.
- AAVE_WETH_ADDRESS: Address of Aave's WETH reserve.
- COMPOUND_PROXY_ADDRESS: Compound protocol’s proxy contract.
- AAVE_POOL_PROVIDER: Aave’s address provider for the pool.
- AAVE_POOL_ADDRESS: A fallback address for Aave's lending pool
- feeCollector: Address for collecting fees.
-
Constructor Parameters:
_wethAddress
: Address of the WETH token._aaveWethAddress
: Address of the Aave WETH reserve._compoundProxy
: Address of the Compound proxy._aavePoolProvider
: Address provider for Aave’s pool._aavePoolAddress
: Fallback Address for Aave’s pool._feeCollector
: Address for collecting protocol fees.
Example:
YieldAggregator(
_wethAddress,
_aaveWethAddress,
_compoundProxy,
_aavePoolProvider,
_aavePoolAddress,
_feeCollector
)
Function: deposit
Deposits the user’s WETH into the yield aggregator, which allocates funds to the protocol offering the highest APY.
Parameters:
amount
: Amount of WETH to deposit.compAPY
: Current APY from Compound.aaveAPY
: Current APY from Aave.
Example Call:
deposit(amount, compAPY, aaveAPY)
Function: withdraw
Allows users to withdraw their deposited WETH along with any accrued yield, after fees.
Example Call:
withdraw()
Function: rebalance
This function shifts funds from one protocol to another based on updated APY data. Only accessible by emergency admins and the owner.
Parameters:
compAPY
: Current APY from Compound.aaveAPY
: Current APY from Aave.
Example Call:
rebalance(compAPY, aaveAPY)
Function: emergencyWithdraw
Allows emergency admins or the owner to withdraw all deposited funds to the owner’s address, effectively pausing the protocol and setting emergencyExitEnabled
to true.
Example Call:
emergencyWithdraw()
- Owner: Has full control, including configuration updates and fee adjustments.
- Emergency Admins: Assigned addresses authorized to perform emergency actions, such as calling
emergencyWithdraw
andrebalance
.
The contract implements both management and performance fees:
- Annual Management Fee: Deducted based on the annual rate specified in basis points (1% by default).
- Performance Fee: Calculated on the yield accrued and also in basis points (10% by default).
Fees can be updated using the updateProtocolConfiguration
function, but cannot exceed a set maximum:
MAX_MANAGEMENT_FEE
: 5% (500 basis points)MAX_PERFORMANCE_FEE
: 30% (3000 basis points)
The contract emits several key events to notify external systems of state changes:
Deposit
: Emitted on a successful deposit.Withdrawal
: Emitted upon a successful withdrawal.Rebalance
: Emitted when funds are rebalanced between protocols.EmergencyWithdrawal
: Emitted during an emergency withdrawal.FeesCollected
: Emitted when fees are collected.ProtocolConfigurationUpdated
: Emitted upon protocol configuration updates.
-
Clone Repository:
git clone https://github.com/obinnafranklinduru/defi-yield-aggregator-hardhat.git cd defi-yield-aggregator-hardhat
-
Create a
.env
file use.env.example
template to make it. You will also need to get an Alchemy API key and add that to the ENV file. -
Start a local node Hardhat is a blockchain development toolkit used to compile your solidity files, run tests and run a local blockchain node. Open a new terminal and start the node.
npm install npx hardhat node
-
Open a new terminal and deploy the smart contract in the
localhost
networknpx hardhat run --network hardhat scripts/DeployYieldAggregator.js
-
Get WETH in your wallet. Running this script will turn 100 ETH to 100 WETH
npx hardhat run --network localhost scripts/GetWETH.js
- Emergency Controls: The contract includes emergency withdrawal functionality and only allows certain actions when
emergencyExitEnabled
is false. - SafeERC20: Uses SafeERC20 to manage WETH transfers, preventing potential token transfer errors.
- Custom Errors: Uses custom error messages to minimize gas usage and improve debugging.
- Non-Reentrant: All functions interacting with external contracts are protected by
nonReentrant
. - Fallback and Receive Functions: Fallback functions are set to revert on unexpected direct transfers.
The contract relies on several external libraries and interfaces:
- @openzeppelin/contracts/access/Ownable: Provides ownership control.
- @openzeppelin/contracts/utils/Pausable: Allows the owner to pause functions.
- @openzeppelin/contracts/token/ERC20/IERC20: ERC20 token interface for WETH.
- @openzeppelin/contracts/token/ERC20/utils/SafeERC20: Ensures safe token transfers.
- @aave/core-v3: For interaction with Aave's Pool and PoolAddressesProvider.
- Compound Protocol: For interaction with Compound’s lending platform.
This project is licensed under the MIT License.
This project was made possible by:
- OpenZeppelin for their security libraries
- Aave Protocol for yield generation
- Compound Protocol for lending and borrowing
- Depth-Hoar Repo for inspiration on getting started