aave-protocol-v2/contracts/libraries/ValidationLogic.sol

347 lines
12 KiB
Solidity
Raw Normal View History

2020-06-20 23:40:03 +00:00
// SPDX-License-Identifier: agpl-3.0
pragma solidity ^0.6.8;
2020-06-30 12:09:28 +00:00
import {SafeMath} from '@openzeppelin/contracts/math/SafeMath.sol';
import {IERC20} from '@openzeppelin/contracts/token/ERC20/IERC20.sol';
2020-06-20 23:40:03 +00:00
2020-06-30 12:09:28 +00:00
import {ReserveLogic} from './ReserveLogic.sol';
import {UserLogic} from './UserLogic.sol';
import {GenericLogic} from './GenericLogic.sol';
import {WadRayMath} from './WadRayMath.sol';
import {PercentageMath} from './PercentageMath.sol';
2020-06-30 12:09:28 +00:00
import {UniversalERC20} from './UniversalERC20.sol';
2020-07-23 15:18:06 +00:00
import {ReserveConfiguration} from './ReserveConfiguration.sol';
2020-06-30 12:09:28 +00:00
import {IPriceOracleGetter} from '../interfaces/IPriceOracleGetter.sol';
import {IFeeProvider} from '../interfaces/IFeeProvider.sol';
import '@nomiclabs/buidler/console.sol';
2020-06-20 23:40:03 +00:00
2020-06-27 02:13:32 +00:00
/**
2020-06-30 12:09:28 +00:00
* @title ReserveLogic library
* @author Aave
* @notice Implements functions to validate specific action on the protocol.
*/
2020-06-20 23:40:03 +00:00
library ValidationLogic {
2020-07-08 22:16:05 +00:00
using ReserveLogic for ReserveLogic.ReserveData;
using UserLogic for UserLogic.UserReserveData;
2020-06-30 12:09:28 +00:00
using SafeMath for uint256;
using WadRayMath for uint256;
using PercentageMath for uint256;
2020-06-30 12:09:28 +00:00
using UniversalERC20 for IERC20;
2020-07-23 15:18:06 +00:00
using ReserveConfiguration for ReserveConfiguration.Map;
2020-06-30 12:09:28 +00:00
/**
* @dev validates a deposit.
* @param _reserve the reserve state on which the user is depositing
* @param _amount the amount to be deposited
*/
2020-07-08 22:16:05 +00:00
function validateDeposit(ReserveLogic.ReserveData storage _reserve, uint256 _amount)
2020-06-30 12:09:28 +00:00
external
view
{
2020-07-23 15:18:06 +00:00
(bool isActive, bool isFreezed, , ) = _reserve.configuration.getFlags();
2020-06-30 12:09:28 +00:00
internalValidateReserveStateAndAmount(_reserve, _amount);
}
/**
* @dev validates a redeem.
* @param _reserve the reserve state from which the user is redeeming
* @param _reserveAddress the address of the reserve
* @param _amount the amount to be redeemed
*/
function validateRedeem(
2020-07-08 22:16:05 +00:00
ReserveLogic.ReserveData storage _reserve,
2020-06-30 12:09:28 +00:00
address _reserveAddress,
uint256 _amount
) external view {
internalValidateReserveStateAndAmount(_reserve, _amount);
require(msg.sender == _reserve.aTokenAddress, '31');
uint256 currentAvailableLiquidity = IERC20(_reserveAddress).universalBalanceOf(
address(_reserve.aTokenAddress)
);
2020-07-10 17:16:04 +00:00
2020-06-30 12:09:28 +00:00
require(currentAvailableLiquidity >= _amount, '4');
}
struct ValidateBorrowLocalVars {
uint256 principalBorrowBalance;
uint256 currentLtv;
uint256 currentLiquidationThreshold;
uint256 requestedBorrowAmountETH;
uint256 amountOfCollateralNeededETH;
uint256 userCollateralBalanceETH;
uint256 userBorrowBalanceETH;
uint256 borrowBalanceIncrease;
uint256 currentReserveStableRate;
uint256 availableLiquidity;
uint256 finalUserBorrowRate;
uint256 healthFactor;
2020-07-08 22:16:05 +00:00
ReserveLogic.InterestRateMode rateMode;
2020-06-30 12:09:28 +00:00
bool healthFactorBelowThreshold;
2020-07-23 15:18:06 +00:00
bool isActive;
bool isFreezed;
bool borrowingEnabled;
bool stableRateBorrowingEnabled;
2020-06-30 12:09:28 +00:00
}
/**
* @dev validates a borrow.
* @param _reserve the reserve state from which the user is borrowing
* @param _user the state of the user for the specific reserve
* @param _reserveAddress the address of the reserve
* @param _amount the amount to be borrowed
* @param _amountInETH the amount to be borrowed, in ETH
* @param _interestRateMode the interest rate mode at which the user is borrowing
* @param _maxStableLoanPercent the max amount of the liquidity that can be borrowed at stable rate, in percentage
* @param _reservesData the state of all the reserves
* @param _usersData the state of all the users for all the reserves
* @param _reserves the addresses of all the active reserves
* @param _oracle the price oracle
*/
function validateBorrow(
2020-07-08 22:16:05 +00:00
ReserveLogic.ReserveData storage _reserve,
UserLogic.UserReserveData storage _user,
2020-06-30 12:09:28 +00:00
address _reserveAddress,
uint256 _amount,
uint256 _amountInETH,
uint256 _interestRateMode,
uint256 _maxStableLoanPercent,
2020-07-08 22:16:05 +00:00
mapping(address => ReserveLogic.ReserveData) storage _reservesData,
mapping(address => mapping(address => UserLogic.UserReserveData)) storage _usersData,
2020-06-30 12:09:28 +00:00
address[] calldata _reserves,
address _oracle
) external view {
ValidateBorrowLocalVars memory vars;
2020-07-23 15:18:06 +00:00
//internalValidateReserveStateAndAmount(_reserve, _amount);
(
vars.isActive,
vars.isFreezed,
vars.borrowingEnabled,
vars.stableRateBorrowingEnabled
) = _reserve.configuration.getFlags();
require(vars.isActive, 'Action requires an active reserve');
require(!vars.isFreezed, 'Action requires an unfreezed reserve');
2020-06-30 12:09:28 +00:00
2020-07-23 15:18:06 +00:00
require(vars.borrowingEnabled, '5');
2020-06-30 12:09:28 +00:00
//validate interest rate mode
require(
2020-07-08 22:16:05 +00:00
uint256(ReserveLogic.InterestRateMode.VARIABLE) == _interestRateMode ||
uint256(ReserveLogic.InterestRateMode.STABLE) == _interestRateMode,
2020-06-30 12:09:28 +00:00
'Invalid interest rate mode selected'
);
//check that the amount is available in the reserve
vars.availableLiquidity = IERC20(_reserveAddress).universalBalanceOf(
address(_reserve.aTokenAddress)
);
2020-06-30 12:09:28 +00:00
require(vars.availableLiquidity >= _amount, '7');
(
vars.userCollateralBalanceETH,
vars.userBorrowBalanceETH,
vars.currentLtv,
vars.currentLiquidationThreshold,
vars.healthFactor
) = GenericLogic.calculateUserAccountData(
msg.sender,
_reservesData,
_usersData,
_reserves,
_oracle
);
require(vars.userCollateralBalanceETH > 0, 'The collateral balance is 0');
require(vars.healthFactor > GenericLogic.HEALTH_FACTOR_LIQUIDATION_THRESHOLD, '8');
//add the current already borrowed amount to the amount requested to calculate the total collateral needed.
vars.amountOfCollateralNeededETH = vars.userBorrowBalanceETH.add(_amountInETH).percentDiv(
2020-07-23 15:18:06 +00:00
vars.currentLtv
); //LTV is calculated in percentage
2020-06-30 12:09:28 +00:00
require(
vars.amountOfCollateralNeededETH <= vars.userCollateralBalanceETH,
'There is not enough collateral to cover a new borrow'
);
2020-06-27 02:13:32 +00:00
/**
2020-06-30 12:09:28 +00:00
* Following conditions need to be met if the user is borrowing at a stable rate:
* 1. Reserve must be enabled for stable rate borrowing
* 2. Users cannot borrow from the reserve if their collateral is (mostly) the same currency
* they are borrowing, to prevent abuses.
* 3. Users will be able to borrow only a relatively small, configurable amount of the total
* liquidity
**/
2020-07-08 22:16:05 +00:00
if (vars.rateMode == ReserveLogic.InterestRateMode.STABLE) {
2020-06-30 12:09:28 +00:00
//check if the borrow mode is stable and if stable rate borrowing is enabled on this reserve
2020-07-23 15:18:06 +00:00
require(vars.stableRateBorrowingEnabled, '11');
2020-06-30 12:09:28 +00:00
require(
!_user.useAsCollateral ||
2020-07-23 15:18:06 +00:00
_reserve.configuration.getLtv() == 0 ||
2020-06-30 12:09:28 +00:00
_amount > IERC20(_reserve.aTokenAddress).balanceOf(msg.sender),
'12'
);
//calculate the max available loan size in stable rate mode as a percentage of the
//available liquidity
uint256 maxLoanSizeStable = vars.availableLiquidity.percentMul(_maxStableLoanPercent);
2020-06-30 12:09:28 +00:00
require(_amount <= maxLoanSizeStable, '13');
2020-06-20 23:40:03 +00:00
}
2020-06-30 12:09:28 +00:00
}
/**
* @dev validates a repay.
* @param _reserve the reserve state from which the user is repaying
* @param _reserveAddress the address of the reserve
* @param _amountSent the amount sent for the repayment. Can be an actual value or uint(-1)
* @param _onBehalfOf the address of the user msg.sender is repaying for
* @param _stableBorrowBalance the borrow balance of the user
* @param _variableBorrowBalance the borrow balance of the user
* @param _actualPaybackAmount the actual amount being repaid
* @param _msgValue the value passed to the repay() function
*/
function validateRepay(
2020-07-08 22:16:05 +00:00
ReserveLogic.ReserveData storage _reserve,
2020-06-30 12:09:28 +00:00
address _reserveAddress,
uint256 _amountSent,
2020-07-08 22:16:05 +00:00
ReserveLogic.InterestRateMode _rateMode,
2020-06-30 12:09:28 +00:00
address _onBehalfOf,
uint256 _stableBorrowBalance,
uint256 _variableBorrowBalance,
uint256 _actualPaybackAmount,
uint256 _msgValue
) external view {
bool isActive = _reserve.configuration.getActive();
2020-07-23 15:18:06 +00:00
require(isActive, 'Action requires an active reserve');
2020-06-30 12:09:28 +00:00
require(_amountSent > 0, 'Amount must be greater than 0');
require(
(_stableBorrowBalance > 0 &&
2020-07-08 22:16:05 +00:00
ReserveLogic.InterestRateMode(_rateMode) == ReserveLogic.InterestRateMode.STABLE) ||
2020-06-30 12:09:28 +00:00
(_variableBorrowBalance > 0 &&
2020-07-08 22:16:05 +00:00
ReserveLogic.InterestRateMode(_rateMode) == ReserveLogic.InterestRateMode.VARIABLE),
2020-06-30 12:09:28 +00:00
'16'
);
require(
_amountSent != uint256(-1) || msg.sender == _onBehalfOf,
'To repay on behalf of an user an explicit amount to repay is needed'
);
require(
!IERC20(_reserveAddress).isETH() || _msgValue >= _actualPaybackAmount,
'Invalid msg.value sent for the repayment'
);
}
/**
* @dev validates a swap of borrow rate mode.
* @param _reserve the reserve state on which the user is swapping the rate
* @param _user the user state for the reserve on which user is swapping the rate
* @param _stableBorrowBalance the stable borrow balance of the user
* @param _variableBorrowBalance the stable borrow balance of the user
2020-06-30 12:09:28 +00:00
* @param _currentRateMode the rate mode of the borrow
*/
function validateSwapRateMode(
2020-07-08 22:16:05 +00:00
ReserveLogic.ReserveData storage _reserve,
UserLogic.UserReserveData storage _user,
uint256 _stableBorrowBalance,
uint256 _variableBorrowBalance,
2020-07-08 22:16:05 +00:00
ReserveLogic.InterestRateMode _currentRateMode
2020-06-30 12:09:28 +00:00
) external view {
2020-07-23 15:18:06 +00:00
(bool isActive, bool isFreezed, , bool stableRateEnabled) = _reserve.configuration.getFlags();
require(isActive, 'Action requires an active reserve');
require(!isFreezed, 'Action requires an unfreezed reserve');
2020-06-30 12:09:28 +00:00
2020-07-08 22:16:05 +00:00
if (_currentRateMode == ReserveLogic.InterestRateMode.STABLE) {
2020-07-07 10:07:31 +00:00
require(
_stableBorrowBalance > 0,
'User does not have a stable rate loan in progress on this reserve'
);
2020-07-08 22:16:05 +00:00
} else if (_currentRateMode == ReserveLogic.InterestRateMode.VARIABLE) {
2020-07-07 10:07:31 +00:00
require(
_variableBorrowBalance > 0,
'User does not have a variable rate loan in progress on this reserve'
);
2020-06-30 12:09:28 +00:00
/**
* user wants to swap to stable, before swapping we need to ensure that
* 1. stable borrow rate is enabled on the reserve
* 2. user is not trying to abuse the reserve by depositing
* more collateral than he is borrowing, artificially lowering
* the interest rate, borrowing at variable, and switching to stable
**/
2020-07-23 15:18:06 +00:00
require(stableRateEnabled, '11');
2020-06-30 12:09:28 +00:00
require(
!_user.useAsCollateral ||
2020-07-23 15:18:06 +00:00
_reserve.configuration.getLtv() == 0 ||
_stableBorrowBalance.add(_variableBorrowBalance) >
IERC20(_reserve.aTokenAddress).balanceOf(msg.sender),
2020-06-30 12:09:28 +00:00
'12'
);
2020-07-13 08:54:08 +00:00
} else {
revert('Invalid interest rate mode selected');
2020-07-07 10:07:31 +00:00
}
2020-06-30 12:09:28 +00:00
}
/**
* @dev validates the choice of a user of setting (or not) an asset as collateral
* @param _reserve the state of the reserve that the user is enabling or disabling as collateral
* @param _reserveAddress the address of the reserve
* @param _reservesData the data of all the reserves
* @param _usersData the data of all the users
*/
function validateSetUseReserveAsCollateral(
2020-07-08 22:16:05 +00:00
ReserveLogic.ReserveData storage _reserve,
2020-06-30 12:09:28 +00:00
address _reserveAddress,
2020-07-08 22:16:05 +00:00
mapping(address => ReserveLogic.ReserveData) storage _reservesData,
mapping(address => mapping(address => UserLogic.UserReserveData)) storage _usersData,
2020-06-30 12:09:28 +00:00
address[] calldata _reserves,
address _oracle
) external view {
uint256 underlyingBalance = IERC20(_reserve.aTokenAddress).balanceOf(msg.sender);
require(underlyingBalance > 0, '22');
require(
GenericLogic.balanceDecreaseAllowed(
_reserveAddress,
msg.sender,
underlyingBalance,
_reservesData,
_usersData,
_reserves,
_oracle
),
'User deposit is already being used as collateral'
);
}
/**
* @dev validates that the reserve is active and the amount is greater than 0
* @param _reserve the state of the reserve being validated
* @param _amount the amount being validated
*/
function internalValidateReserveStateAndAmount(
2020-07-08 22:16:05 +00:00
ReserveLogic.ReserveData storage _reserve,
2020-06-30 12:09:28 +00:00
uint256 _amount
) internal view {
require(_amount > 0, 'Amount must be greater than 0');
}
2020-06-20 23:40:03 +00:00
}