2020-05-29 16:45:37 +00:00
|
|
|
// SPDX-License-Identifier: agpl-3.0
|
|
|
|
pragma solidity ^0.6.8;
|
|
|
|
|
|
|
|
import "@openzeppelin/contracts/utils/Address.sol";
|
|
|
|
import "@openzeppelin/contracts/token/ERC20/IERC20.sol";
|
|
|
|
|
|
|
|
import "../configuration/LendingPoolAddressesProvider.sol";
|
|
|
|
import "../lendingpool/LendingPoolCore.sol";
|
2020-06-02 14:16:22 +00:00
|
|
|
import "../libraries/UniversalERC20.sol";
|
2020-05-29 16:45:37 +00:00
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @title WalletBalanceProvider contract
|
|
|
|
* @author Aave, influenced by https://github.com/wbobeirne/eth-balance-checker/blob/master/contracts/BalanceChecker.sol
|
|
|
|
* @notice Implements a logic of getting multiple tokens balance for one user address
|
|
|
|
* @dev NOTE: THIS CONTRACT IS NOT USED WITHIN THE AAVE PROTOCOL. It's an accessory contract used to reduce the number of calls
|
|
|
|
* towards the blockchain from the Aave backend.
|
|
|
|
**/
|
|
|
|
contract WalletBalanceProvider {
|
|
|
|
|
|
|
|
using Address for address;
|
2020-06-02 14:16:22 +00:00
|
|
|
using UniversalERC20 for IERC20;
|
2020-05-29 16:45:37 +00:00
|
|
|
|
|
|
|
LendingPoolAddressesProvider provider;
|
|
|
|
|
|
|
|
constructor(LendingPoolAddressesProvider _provider) public {
|
|
|
|
|
|
|
|
provider = _provider;
|
|
|
|
|
|
|
|
}
|
|
|
|
/**
|
|
|
|
@dev Fallback function, don't accept any ETH
|
|
|
|
**/
|
|
|
|
receive() external payable {
|
|
|
|
revert("WalletBalanceProvider does not accept payments");
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
@dev Check the token balance of a wallet in a token contract
|
|
|
|
|
|
|
|
Returns the balance of the token for user. Avoids possible errors:
|
|
|
|
- return 0 on non-contract address
|
|
|
|
**/
|
|
|
|
function balanceOf(address _user, address _token) public view returns (uint256) {
|
|
|
|
// check if token is actually a contract
|
|
|
|
if (_token.isContract()) {
|
|
|
|
return IERC20(_token).balanceOf(_user);
|
|
|
|
} else {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// @notice Fetches, for a list of _users and _tokens (ETH included with mock address), the balances
|
|
|
|
/// @param _users The list of users
|
|
|
|
/// @param _tokens The list of tokens
|
|
|
|
/// @return And array with the concatenation of, for each user, his/her balances
|
|
|
|
function batchBalanceOf(address[] memory _users, address[] memory _tokens) public view returns (uint256[] memory) {
|
|
|
|
uint256[] memory balances = new uint256[](_users.length * _tokens.length);
|
|
|
|
|
|
|
|
for (uint256 i = 0; i < _users.length; i++) {
|
|
|
|
for (uint256 j = 0; j < _tokens.length; j++) {
|
|
|
|
uint256 _offset = i * _tokens.length;
|
2020-06-02 14:16:22 +00:00
|
|
|
if (IERC20(_tokens[j]).isETH()) {
|
2020-05-29 16:45:37 +00:00
|
|
|
balances[_offset + j] = _users[i].balance; // ETH balance
|
|
|
|
} else {
|
|
|
|
if (!_tokens[j].isContract()) {
|
|
|
|
revert("INVALID_TOKEN");
|
|
|
|
} else {
|
|
|
|
balances[_offset + j] = balanceOf(_users[i], _tokens[j]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return balances;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
@dev provides balances of user wallet for all reserves available on the pool
|
|
|
|
*/
|
|
|
|
function getUserWalletBalances(address _user) public view returns (address[] memory, uint256[] memory) {
|
|
|
|
|
|
|
|
LendingPoolCore core = LendingPoolCore(provider.getLendingPoolCore());
|
|
|
|
|
|
|
|
address[] memory reserves = core.getReserves();
|
|
|
|
|
|
|
|
uint256[] memory balances = new uint256[](reserves.length);
|
|
|
|
|
|
|
|
for (uint256 j = 0; j < reserves.length; j++) {
|
|
|
|
if(!core.getReserveIsActive(reserves[j])){
|
|
|
|
balances[j] = 0;
|
|
|
|
continue;
|
|
|
|
}
|
2020-06-02 14:16:22 +00:00
|
|
|
if (!IERC20(reserves[j]).isETH()) {
|
2020-05-29 16:45:37 +00:00
|
|
|
balances[j] = balanceOf(_user, reserves[j]);
|
|
|
|
} else {
|
|
|
|
balances[j] = _user.balance; // ETH balance
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return (reserves, balances);
|
|
|
|
}
|
2020-06-02 14:16:22 +00:00
|
|
|
}
|