// SPDX-License-Identifier: agpl-3.0 pragma solidity ^0.6.8; import '@openzeppelin/contracts/math/SafeMath.sol'; import '@openzeppelin/contracts/token/ERC20/IERC20.sol'; import '@openzeppelin/contracts/token/ERC20/ERC20Burnable.sol'; import '@openzeppelin/contracts/utils/ReentrancyGuard.sol'; import '../libraries/openzeppelin-upgradeability/VersionedInitializable.sol'; import '../interfaces/IExchangeAdapter.sol'; import {SafeERC20} from '@openzeppelin/contracts/token/ERC20/SafeERC20.sol'; import {PercentageMath} from '../libraries/PercentageMath.sol'; /// @title TokenDistributor /// @author Aave /// @notice Receives tokens and manages the distribution amongst receivers /// The usage is as follows: /// - The distribution addresses and percentages are set up on construction /// - The Kyber Proxy is approved for a list of tokens in construction, which will be later burnt /// - At any moment, anyone can call distribute() with a list of token addresses in order to distribute /// the accumulated token amounts and/or ETH in this contract to all the receivers with percentages /// - If the address(0) is used as receiver, this contract will trade in Kyber to tokenToBurn (LEND) /// and burn it (sending to address(0) the tokenToBurn) contract TokenDistributor is ReentrancyGuard, VersionedInitializable { using SafeMath for uint256; using PercentageMath for uint256; using SafeERC20 for IERC20; struct Distribution { address[] receivers; uint256[] percentages; } event DistributionUpdated(address[] receivers, uint256[] percentages); event Distributed(address receiver, uint256 percentage, uint256 amount); event Setup(address tokenToBurn, IExchangeAdapter exchangeAdapter, address _recipientBurn); event Burn(uint256 amount); uint256 public constant IMPLEMENTATION_REVISION = 0x3; /// @notice DEPRECATED uint256 public constant MAX_UINT = 2**256 - 1; /// @notice DEPRECATED uint256 public constant MAX_UINT_MINUS_ONE = (2**256 - 1) - 1; /// @notice DEPRECATED uint256 public constant MIN_CONVERSION_RATE = 1; /// @notice DEPRECATED address public constant KYBER_ETH_MOCK_ADDRESS = address( 0x00eeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeee ); /// @notice Defines how tokens and ETH are distributed on each call to .distribute() Distribution private distribution; /// @notice Instead of using 100 for percentages, higher base to have more precision in the distribution uint256 public constant DISTRIBUTION_BASE = 10000; /// @notice The address of the token to burn (LEND token) address public tokenToBurn; /// @notice Address to send tokens to "burn". /// Because of limitations on OZ ERC20, on dev it's needed to use the 0x00000...1 address instead of address(0) /// So this param needs to be received on construction address public recipientBurn; /// @notice Smart contract implementing the logic to interact with a particular exchange. /// Will be called by DELEGATECALL IExchangeAdapter public exchangeAdapter; /// @notice Called by the proxy when setting this contract as implementation function initialize( address _recipientBurn, address _tokenToBurn, IExchangeAdapter _exchangeAdapter, address[] memory _receivers, uint256[] memory _percentages, IERC20[] memory _tokens ) public initializer { recipientBurn = _recipientBurn; tokenToBurn = _tokenToBurn; exchangeAdapter = _exchangeAdapter; internalSetTokenDistribution(_receivers, _percentages); approveExchange(_tokens); emit Setup(_tokenToBurn, _exchangeAdapter, _recipientBurn); } /// @notice In order to receive ETH transfers receive() external payable {} /// @notice "Infinite" approval for all the tokens initialized /// @param _tokens List of IERC20 to approve function approveExchange(IERC20[] memory _tokens) public { (bool _success, ) = address(exchangeAdapter).delegatecall( abi.encodeWithSelector(exchangeAdapter.approveExchange.selector, _tokens) ); } /// @notice Distributes the whole balance of a list of _tokens balances in this contract /// @param _tokens list of ERC20 tokens to distribute function distribute(IERC20[] memory _tokens) public { for (uint256 i = 0; i < _tokens.length; i++) { uint256 _balanceToDistribute = _tokens[i].balanceOf(address(this)); if (_balanceToDistribute <= 0) { continue; } internalDistributeTokenWithAmount(_tokens[i], _balanceToDistribute); } } /// @notice Distributes specific amounts of a list of _tokens /// @param _tokens list of ERC20 tokens to distribute /// @param _amounts list of amounts to distribute per token function distributeWithAmounts(IERC20[] memory _tokens, uint256[] memory _amounts) public { for (uint256 i = 0; i < _tokens.length; i++) { internalDistributeTokenWithAmount(_tokens[i], _amounts[i]); } } /// @notice Distributes specific total balance's percentages of a list of _tokens /// @param _tokens list of ERC20 tokens to distribute /// @param _percentages list of percentages to distribute per token function distributeWithPercentages(IERC20[] memory _tokens, uint256[] memory _percentages) public { for (uint256 i = 0; i < _tokens.length; i++) { uint256 _amountToDistribute = _tokens[i].balanceOf(address(this)).percentMul( _percentages[i] ); if (_amountToDistribute <= 0) { continue; } internalDistributeTokenWithAmount(_tokens[i], _amountToDistribute); } } /// @notice Sets _receivers addresses with _percentages for each one /// @param _receivers Array of addresses receiving a percentage of the distribution, both user addresses /// or contracts /// @param _percentages Array of percentages each _receivers member will get function internalSetTokenDistribution(address[] memory _receivers, uint256[] memory _percentages) internal { require(_receivers.length == _percentages.length, 'Array lengths should be equal'); distribution = Distribution({receivers: _receivers, percentages: _percentages}); emit DistributionUpdated(_receivers, _percentages); } /// @notice Distributes a specific amount of a token owned by this contract /// @param _token The ERC20 token to distribute /// @param _amountToDistribute The specific amount to distribute function internalDistributeTokenWithAmount(IERC20 _token, uint256 _amountToDistribute) internal { address _tokenAddress = address(_token); Distribution memory _distribution = distribution; for (uint256 j = 0; j < _distribution.receivers.length; j++) { uint256 _amount = _amountToDistribute.mul(_distribution.percentages[j]).div( DISTRIBUTION_BASE ); //avoid transfers/burns of 0 tokens if (_amount == 0) { continue; } if (_distribution.receivers[j] != address(0)) { _token.safeTransfer(_distribution.receivers[j], _amount); emit Distributed(_distribution.receivers[j], _distribution.percentages[j], _amount); } else { uint256 _amountToBurn = _amount; // If the token to burn is already tokenToBurn, we don't trade, burning directly if (_tokenAddress != tokenToBurn) { (bool _success, bytes memory _result) = address(exchangeAdapter).delegatecall( abi.encodeWithSelector( exchangeAdapter.exchange.selector, _tokenAddress, tokenToBurn, _amount, 10 ) ); require(_success, 'ERROR_ON_EXCHANGE'); _amountToBurn = abi.decode(_result, (uint256)); } _burn(_amountToBurn); } } } /// @notice Internal function to send _amount of tokenToBurn to the 0x0 address /// @param _amount The amount to burn function _burn(uint256 _amount) internal { require( IERC20(tokenToBurn).transfer(recipientBurn, _amount), 'INTERNAL_BURN. Reverted transfer to recipientBurn address' ); emit Burn(_amount); } /// @notice Returns the receivers and percentages of the contract Distribution /// @return receivers array of addresses and percentages array on uints function getDistribution() public view returns (address[] memory receivers, uint256[] memory percentages) { receivers = distribution.receivers; percentages = distribution.percentages; } /// @notice Gets the revision number of the contract /// @return The revision numeric reference function getRevision() internal override pure returns (uint256) { return IMPLEMENTATION_REVISION; } }