2020-05-29 16:45:37 +00:00
|
|
|
// 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";
|
2020-06-02 13:49:24 +00:00
|
|
|
import "../libraries/UniversalERC20.sol";
|
2020-05-29 16:45:37 +00:00
|
|
|
|
|
|
|
|
|
|
|
/// @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;
|
2020-06-02 13:49:24 +00:00
|
|
|
using UniversalERC20 for IERC20;
|
2020-05-29 16:45:37 +00:00
|
|
|
|
|
|
|
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++) {
|
2020-06-02 13:49:24 +00:00
|
|
|
uint256 _balanceToDistribute = _tokens[i].universalBalanceOf(address(this));
|
|
|
|
|
2020-05-29 16:45:37 +00:00
|
|
|
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++) {
|
2020-06-02 13:49:24 +00:00
|
|
|
uint256 _amountToDistribute = _tokens[i].universalBalanceOf(address(this)).mul(_percentages[i]).div(100);
|
|
|
|
|
2020-05-29 16:45:37 +00:00
|
|
|
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)) {
|
2020-06-02 13:49:24 +00:00
|
|
|
_token.universalTransfer(_distribution.receivers[j], _amount);
|
2020-05-29 16:45:37 +00:00
|
|
|
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));
|
|
|
|
}
|
2020-07-09 09:23:30 +00:00
|
|
|
_burn(_amountToBurn);
|
2020-05-29 16:45:37 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// @notice Internal function to send _amount of tokenToBurn to the 0x0 address
|
|
|
|
/// @param _amount The amount to burn
|
2020-07-09 09:23:30 +00:00
|
|
|
function _burn(uint256 _amount) internal {
|
2020-05-29 16:45:37 +00:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2020-07-03 09:15:30 +00:00
|
|
|
}
|