2020-08-20 12:32:20 +00:00
|
|
|
// SPDX-License-Identifier: agpl-3.0
|
2020-06-08 15:36:40 +00:00
|
|
|
pragma solidity ^0.6.8;
|
|
|
|
pragma experimental ABIEncoderV2;
|
2020-06-08 19:06:26 +00:00
|
|
|
|
2020-07-13 08:54:08 +00:00
|
|
|
import {ILendingPoolAddressesProvider} from '../interfaces/ILendingPoolAddressesProvider.sol';
|
|
|
|
import {IERC20Detailed} from '../interfaces/IERC20Detailed.sol';
|
2020-08-20 12:32:20 +00:00
|
|
|
import {ILendingPool} from '../interfaces/ILendingPool.sol';
|
2020-10-12 12:25:03 +00:00
|
|
|
import {ReserveLogic} from '../libraries/logic/ReserveLogic.sol';
|
|
|
|
import {ReserveConfiguration} from '../libraries/configuration/ReserveConfiguration.sol';
|
2020-10-12 13:10:16 +00:00
|
|
|
import {UserConfiguration} from '../libraries/configuration/UserConfiguration.sol';
|
2020-10-12 12:25:03 +00:00
|
|
|
import {IStableDebtToken} from '../tokenization/interfaces/IStableDebtToken.sol';
|
2020-10-12 13:10:16 +00:00
|
|
|
import {IVariableDebtToken} from '../tokenization/interfaces/IVariableDebtToken.sol';
|
2020-06-08 15:36:40 +00:00
|
|
|
|
|
|
|
contract AaveProtocolTestHelpers {
|
2020-10-12 12:25:03 +00:00
|
|
|
using ReserveConfiguration for ReserveConfiguration.Map;
|
2020-10-12 13:10:16 +00:00
|
|
|
using UserConfiguration for UserConfiguration.Map;
|
2020-10-12 12:25:03 +00:00
|
|
|
|
2020-07-13 08:54:08 +00:00
|
|
|
struct TokenData {
|
|
|
|
string symbol;
|
|
|
|
address tokenAddress;
|
|
|
|
}
|
2020-06-08 15:36:40 +00:00
|
|
|
|
2020-07-13 08:54:08 +00:00
|
|
|
ILendingPoolAddressesProvider public immutable ADDRESSES_PROVIDER;
|
2020-06-08 15:36:40 +00:00
|
|
|
|
2020-07-13 08:54:08 +00:00
|
|
|
constructor(ILendingPoolAddressesProvider addressesProvider) public {
|
|
|
|
ADDRESSES_PROVIDER = addressesProvider;
|
|
|
|
}
|
2020-06-08 15:36:40 +00:00
|
|
|
|
2020-07-13 08:54:08 +00:00
|
|
|
function getAllReservesTokens() external view returns (TokenData[] memory) {
|
2020-08-20 12:32:20 +00:00
|
|
|
ILendingPool pool = ILendingPool(ADDRESSES_PROVIDER.getLendingPool());
|
2020-10-06 13:51:48 +00:00
|
|
|
address[] memory reserves = pool.getReservesList();
|
2020-07-13 08:54:08 +00:00
|
|
|
TokenData[] memory reservesTokens = new TokenData[](reserves.length);
|
|
|
|
for (uint256 i = 0; i < reserves.length; i++) {
|
|
|
|
reservesTokens[i] = TokenData({
|
|
|
|
symbol: (reserves[i] == 0xEeeeeEeeeEeEeeEeEeEeeEEEeeeeEeeeeeeeEEeE)
|
|
|
|
? 'ETH'
|
|
|
|
: IERC20Detailed(reserves[i]).symbol(),
|
|
|
|
tokenAddress: reserves[i]
|
|
|
|
});
|
2020-06-09 09:11:19 +00:00
|
|
|
}
|
2020-07-13 08:54:08 +00:00
|
|
|
return reservesTokens;
|
|
|
|
}
|
2020-06-09 09:11:19 +00:00
|
|
|
|
2020-07-13 08:54:08 +00:00
|
|
|
function getAllATokens() external view returns (TokenData[] memory) {
|
2020-08-20 12:32:20 +00:00
|
|
|
ILendingPool pool = ILendingPool(ADDRESSES_PROVIDER.getLendingPool());
|
2020-10-06 13:51:48 +00:00
|
|
|
address[] memory reserves = pool.getReservesList();
|
2020-07-13 08:54:08 +00:00
|
|
|
TokenData[] memory aTokens = new TokenData[](reserves.length);
|
|
|
|
for (uint256 i = 0; i < reserves.length; i++) {
|
2020-10-12 12:25:03 +00:00
|
|
|
ReserveLogic.ReserveData memory reserveData = pool.getReserveData(reserves[i]);
|
2020-07-13 10:24:36 +00:00
|
|
|
aTokens[i] = TokenData({
|
2020-10-12 12:25:03 +00:00
|
|
|
symbol: IERC20Detailed(reserveData.aTokenAddress).symbol(),
|
|
|
|
tokenAddress: reserveData.aTokenAddress
|
2020-07-13 10:24:36 +00:00
|
|
|
});
|
2020-06-08 15:36:40 +00:00
|
|
|
}
|
2020-07-13 08:54:08 +00:00
|
|
|
return aTokens;
|
|
|
|
}
|
2020-10-12 12:25:03 +00:00
|
|
|
|
|
|
|
function getReserveConfigurationData(address asset)
|
|
|
|
external
|
|
|
|
view
|
|
|
|
returns (
|
|
|
|
uint256 decimals,
|
|
|
|
uint256 ltv,
|
|
|
|
uint256 liquidationThreshold,
|
|
|
|
uint256 liquidationBonus,
|
|
|
|
uint256 reserveFactor,
|
|
|
|
bool usageAsCollateralEnabled,
|
|
|
|
bool borrowingEnabled,
|
|
|
|
bool stableBorrowRateEnabled,
|
|
|
|
bool isActive,
|
|
|
|
bool isFrozen
|
|
|
|
)
|
|
|
|
{
|
|
|
|
ReserveConfiguration.Map memory configuration = ILendingPool(
|
|
|
|
ADDRESSES_PROVIDER.getLendingPool()
|
|
|
|
)
|
|
|
|
.getConfiguration(asset);
|
|
|
|
|
|
|
|
(ltv, liquidationThreshold, liquidationBonus, decimals, reserveFactor) = configuration
|
|
|
|
.getParamsMemory();
|
|
|
|
|
|
|
|
(isActive, isFrozen, borrowingEnabled, stableBorrowRateEnabled) = configuration
|
|
|
|
.getFlagsMemory();
|
|
|
|
|
2020-10-12 18:07:17 +00:00
|
|
|
usageAsCollateralEnabled = liquidationThreshold > 0;
|
2020-10-12 12:25:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
function getReserveData(address asset)
|
|
|
|
external
|
|
|
|
view
|
|
|
|
returns (
|
|
|
|
uint256 availableLiquidity,
|
|
|
|
uint256 totalStableDebt,
|
|
|
|
uint256 totalVariableDebt,
|
|
|
|
uint256 liquidityRate,
|
|
|
|
uint256 variableBorrowRate,
|
|
|
|
uint256 stableBorrowRate,
|
|
|
|
uint256 averageStableBorrowRate,
|
|
|
|
uint256 liquidityIndex,
|
|
|
|
uint256 variableBorrowIndex,
|
|
|
|
uint40 lastUpdateTimestamp
|
|
|
|
)
|
|
|
|
{
|
|
|
|
ReserveLogic.ReserveData memory reserve = ILendingPool(ADDRESSES_PROVIDER.getLendingPool())
|
|
|
|
.getReserveData(asset);
|
|
|
|
|
|
|
|
return (
|
|
|
|
IERC20Detailed(asset).balanceOf(reserve.aTokenAddress),
|
|
|
|
IERC20Detailed(reserve.stableDebtTokenAddress).totalSupply(),
|
|
|
|
IERC20Detailed(reserve.variableDebtTokenAddress).totalSupply(),
|
|
|
|
reserve.currentLiquidityRate,
|
|
|
|
reserve.currentVariableBorrowRate,
|
|
|
|
reserve.currentStableBorrowRate,
|
|
|
|
IStableDebtToken(reserve.stableDebtTokenAddress).getAverageStableRate(),
|
|
|
|
reserve.liquidityIndex,
|
|
|
|
reserve.variableBorrowIndex,
|
|
|
|
reserve.lastUpdateTimestamp
|
|
|
|
);
|
|
|
|
}
|
2020-10-12 13:10:16 +00:00
|
|
|
|
2020-10-12 12:25:03 +00:00
|
|
|
function getUserReserveData(address asset, address user)
|
|
|
|
external
|
|
|
|
view
|
|
|
|
returns (
|
|
|
|
uint256 currentATokenBalance,
|
|
|
|
uint256 currentStableDebt,
|
|
|
|
uint256 currentVariableDebt,
|
|
|
|
uint256 principalStableDebt,
|
|
|
|
uint256 scaledVariableDebt,
|
|
|
|
uint256 stableBorrowRate,
|
|
|
|
uint256 liquidityRate,
|
|
|
|
uint40 stableRateLastUpdated,
|
|
|
|
bool usageAsCollateralEnabled
|
|
|
|
)
|
|
|
|
{
|
2020-10-12 13:10:16 +00:00
|
|
|
ReserveLogic.ReserveData memory reserve = ILendingPool(ADDRESSES_PROVIDER.getLendingPool())
|
2020-10-12 12:25:03 +00:00
|
|
|
.getReserveData(asset);
|
2020-10-12 13:10:16 +00:00
|
|
|
|
|
|
|
UserConfiguration.Map memory userConfig = ILendingPool(ADDRESSES_PROVIDER.getLendingPool())
|
|
|
|
.getUserConfiguration(user);
|
|
|
|
|
|
|
|
currentATokenBalance = IERC20Detailed(reserve.aTokenAddress).balanceOf(user);
|
|
|
|
currentVariableDebt = IERC20Detailed(reserve.variableDebtTokenAddress).balanceOf(user);
|
|
|
|
currentStableDebt = IERC20Detailed(reserve.stableDebtTokenAddress).balanceOf(user);
|
2020-10-12 12:25:03 +00:00
|
|
|
principalStableDebt = IStableDebtToken(reserve.stableDebtTokenAddress).principalBalanceOf(user);
|
|
|
|
scaledVariableDebt = IVariableDebtToken(reserve.variableDebtTokenAddress).scaledBalanceOf(user);
|
|
|
|
liquidityRate = reserve.currentLiquidityRate;
|
|
|
|
stableBorrowRate = IStableDebtToken(reserve.stableDebtTokenAddress).getUserStableRate(user);
|
|
|
|
stableRateLastUpdated = IStableDebtToken(reserve.stableDebtTokenAddress).getUserLastUpdated(
|
|
|
|
user
|
|
|
|
);
|
2020-10-12 13:10:16 +00:00
|
|
|
usageAsCollateralEnabled = userConfig.isUsingAsCollateral(reserve.id);
|
2020-10-12 12:25:03 +00:00
|
|
|
}
|
2020-10-12 18:07:17 +00:00
|
|
|
|
|
|
|
function getReserveTokensAddresses(address asset)
|
|
|
|
external
|
|
|
|
view
|
|
|
|
returns (
|
|
|
|
address aTokenAddress,
|
|
|
|
address stableDebtTokenAddress,
|
|
|
|
address variableDebtTokenAddress
|
|
|
|
)
|
|
|
|
{
|
|
|
|
ReserveLogic.ReserveData memory reserve = ILendingPool(ADDRESSES_PROVIDER.getLendingPool())
|
|
|
|
.getReserveData(asset);
|
|
|
|
|
|
|
|
return (
|
|
|
|
reserve.aTokenAddress,
|
|
|
|
reserve.stableDebtTokenAddress,
|
|
|
|
reserve.variableDebtTokenAddress
|
|
|
|
);
|
|
|
|
}
|
2020-07-13 08:54:08 +00:00
|
|
|
}
|