Address Contract Verified
Address
0x4dbcb0cFf525B91E8b9D18b224c1B45feF008549
Balance
0 ETH
Nonce
1
Code Size
5930 bytes
Creator
0x49E3C589...4d2B at tx 0xe18dc641...065fae
Indexed Transactions
0
Contract Bytecode
5930 bytes
0x6080604052600436106100e25760003560e01c80638da5cb5b116100855780638da5cb5b146102bb5780638f32d59b146102d957806398d99ee614610304578063a3f4df7e14610324578063aac1846f14610367578063b7f8cf9b14610387578063f07424e9146103a7578063f2fde38b146103bc578063f6ff93d6146103dc57600080fd5b80630b622ab2146101a45780631529a639146101da5780632a9043ec146101f957806342c31a231461021957806348d399e71461023b5780635a4d28bb1461025b578063625021691461027b5780636cfb6bf91461029b57600080fd5b3661019f576100ef6103fc565b6003546001600160a01b0316156101625760405162461bcd60e51b815260206004820152602c60248201527f416374697665506f6f6c3a20455243323020636f6c6c61746572616c206e656560448201526b0c8cac85840dcdee8408aa8960a31b60648201526084015b60405180910390fd5b3460076000828254610174919061148d565b90915550506007546040519081526000805160206116d58339815191529060200160405180910390a1005b600080fd5b3480156101b057600080fd5b506005546101c4906001600160a01b031681565b6040516101d191906114a6565b60405180910390f35b3480156101e657600080fd5b506007545b6040519081526020016101d1565b34801561020557600080fd5b506004546101c4906001600160a01b031681565b34801561022557600080fd5b506102396102343660046114ba565b61048b565b005b34801561024757600080fd5b506003546101c4906001600160a01b031681565b34801561026757600080fd5b506006546101c4906001600160a01b031681565b34801561028757600080fd5b506102396102963660046114e8565b6104e4565b3480156102a757600080fd5b506102396102b6366004611514565b6106a8565b3480156102c757600080fd5b506000546001600160a01b03166101c4565b3480156102e557600080fd5b506000546001600160a01b0316331460405190151581526020016101d1565b34801561031057600080fd5b5061023961031f3660046114ba565b610d30565b34801561033057600080fd5b5061035a6040518060400160405280600a8152602001691058dd1a5d99541bdbdb60b21b81525081565b6040516101d191906115ba565b34801561037357600080fd5b506001546101c4906001600160a01b031681565b34801561039357600080fd5b506002546101c4906001600160a01b031681565b3480156103b357600080fd5b506008546101eb565b3480156103c857600080fd5b506102396103d73660046115ed565b610ddd565b3480156103e857600080fd5b506102396103f73660046114ba565b610e78565b6002546001600160a01b031633148061041f57506001546001600160a01b031633145b6104895760405162461bcd60e51b815260206004820152604160248201526000805160206116b583398151915260448201527f72726f7765724f7065726174696f6e73206e6f722044656661756c7420506f6f6064820152601b60fa1b608482015260a401610159565b565b610493610e92565b80600860008282546104a5919061148d565b90915550506008546040519081527f71e50c7de40c802fa2869e1059dd72c32a02b63f03dd59f5aabd4af46f0f51da906020015b60405180910390a150565b6104ec610f1f565b80600760008282546104fe9190611611565b90915550506007546040519081526000805160206116d58339815191529060200160405180910390a1604080516001600160a01b0384168152602081018390527f342693d2465f6f44931e41128424a0227e0cbc69d1c3917a839e6de71696d44c910160405180910390a1600354610580906001600160a01b03168383610fd3565b6003546001600160a01b0316610594575050565b6001546001600160a01b039081169083160361060957604051634c6ccf7360e11b8152600481018290526001600160a01b038316906398d99ee6906024015b600060405180830381600087803b1580156105ed57600080fd5b505af1158015610601573d6000803e3d6000fd5b505050505050565b6004546001600160a01b039081169083160361064c57604051634c6ccf7360e11b8152600481018290526001600160a01b038316906398d99ee6906024016105d3565b6005546001600160a01b03908116908316036106a457604051634c6ccf7360e11b8152600481018290526001600160a01b038316906398d99ee690602401600060405180830381600087803b1580156105ed57600080fd5b5050565b6000546001600160a01b031633146106d25760405162461bcd60e51b815260040161015990611624565b6106db86611098565b6106e485611098565b6106ed84611098565b6106f683611098565b6106ff82611098565b6001600160a01b038116156107175761071781611098565b600280546001600160a01b03199081166001600160a01b03898116919091179092556006805482168884161790556005805482168784161790556001805482168684169081179091556003805483168585161790556004805490921692851692909217815560408051638da5cb5b60e01b8152905160009392638da5cb5b928082019260209290918290030181865afa1580156107b8573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906107dc9190611659565b6001600160a01b03161415806108645750806001600160a01b0316836001600160a01b03166348d399e76040518163ffffffff1660e01b8152600401602060405180830381865afa158015610835573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906108599190611659565b6001600160a01b0316145b8015610960575060006001600160a01b0316866001600160a01b0316638da5cb5b6040518163ffffffff1660e01b8152600401602060405180830381865afa1580156108b4573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906108d89190611659565b6001600160a01b03161415806109605750806001600160a01b0316866001600160a01b03166348d399e76040518163ffffffff1660e01b8152600401602060405180830381865afa158015610931573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906109559190611659565b6001600160a01b0316145b8015610a5f575060006001600160a01b0316846001600160a01b0316638da5cb5b6040518163ffffffff1660e01b8152600401602060405180830381865afa1580156109b0573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906109d49190611659565b6001600160a01b0316141580610a5f5750600554604080516348d399e760e01b815290516001600160a01b038085169316916348d399e79160048083019260209291908290030181865afa158015610a30573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610a549190611659565b6001600160a01b0316145b8015610b5b575060006001600160a01b0316826001600160a01b0316638da5cb5b6040518163ffffffff1660e01b8152600401602060405180830381865afa158015610aaf573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610ad39190611659565b6001600160a01b0316141580610b5b5750806001600160a01b0316826001600160a01b03166348d399e76040518163ffffffff1660e01b8152600401602060405180830381865afa158015610b2c573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610b509190611659565b6001600160a01b0316145b610bde5760405162461bcd60e51b815260206004820152604860248201527f5468652073616d6520636f6c6c61746572616c2061646472657373206d75737460448201527f206265207573656420666f722074686520656e7469726520736574206f6620636064820152676f6e74726163747360c01b608482015260a401610159565b7f3ca631ffcd2a9b5d9ae18543fc82f58eb4ca33af9e6ab01b7a8e95331e6ed98586604051610c0d91906114a6565b60405180910390a17f143219c9e69b09e07e095fcc889b43d8f46ca892bba65f08dc3a0050869a567885604051610c4491906114a6565b60405180910390a17f82966d27eea39b038ee0fa30cd16532bb24f6e65d31cb58fb227aa5766cdcc7f84604051610c7b91906114a6565b60405180910390a17f5ee0cae2f063ed938bb55046f6a932fb6ae792bf43624806bb90abe68a50be9b83604051610cb291906114a6565b60405180910390a17ffba7421f3d1a98e80d72491a6e0523133444a5842cc7310951d8b82d075a7dff81604051610ce991906114a6565b60405180910390a17fe67f36a6e961157d6eff83b91f3af5a62131ceb6f04954ef74f51c1c05e7f88d82604051610d2091906114a6565b60405180910390a161060161113d565b610d386103fc565b6003546001600160a01b0316610da55760405162461bcd60e51b815260206004820152602c60248201527f416374697665506f6f6c3a2045544820636f6c6c61746572616c206e6565646560448201526b0642c206e6f742045524332360a41b6064820152608401610159565b8060076000828254610db7919061148d565b90915550506007546040519081526000805160206116d5833981519152906020016104d9565b6000546001600160a01b03163314610e075760405162461bcd60e51b815260040161015990611624565b6001600160a01b038116610e6c5760405162461bcd60e51b815260206004820152602660248201527f4f776e61626c653a206e6577206f776e657220697320746865207a65726f206160448201526564647265737360d01b6064820152608401610159565b610e7581611187565b50565b610e80610f1f565b80600860008282546104a59190611611565b6002546001600160a01b0316331480610eb557506006546001600160a01b031633145b6104895760405162461bcd60e51b815260206004820152604160248201526000805160206116b583398151915260448201527f72726f7765724f7065726174696f6e73206e6f722054726f76654d616e6167656064820152603960f91b608482015260a401610159565b6002546001600160a01b0316331480610f4257506006546001600160a01b031633145b80610f5757506005546001600160a01b031633145b6104895760405162461bcd60e51b815260206004820152605360248201526000805160206116b583398151915260448201527f72726f7765724f7065726174696f6e73206e6f722054726f76654d616e6167656064820152721c881b9bdc8814dd18589a5b1a5d1e541bdbdb606a1b608482015260a401610159565b6001600160a01b03831661107f576000826001600160a01b03168260405160006040518083038185875af1925050503d806000811461102e576040519150601f19603f3d011682016040523d82523d6000602084013e611033565b606091505b50509050806110795760405162461bcd60e51b815260206004820152601260248201527114d95b991a5b99c81155120819985a5b195960721b6044820152606401610159565b50505050565b6110936001600160a01b03841683836111d7565b505050565b6001600160a01b0381166110ee5760405162461bcd60e51b815260206004820152601e60248201527f4163636f756e742063616e6e6f74206265207a65726f206164647265737300006044820152606401610159565b803b806106a45760405162461bcd60e51b815260206004820181905260248201527f4163636f756e7420636f64652073697a652063616e6e6f74206265207a65726f6044820152606401610159565b600080546040516001600160a01b03909116907f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e0908390a3600080546001600160a01b0319169055565b600080546001600160a01b038381166001600160a01b0319831681178455604051919092169283917f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e09190a35050565b604080516001600160a01b03848116602483015260448083018590528351808403909101815260649092018352602080830180516001600160e01b031663a9059cbb60e01b17905283518085019094528084527f5361666545524332303a206c6f772d6c6576656c2063616c6c206661696c656490840152611093928692916000916112679185169084906112e7565b90508051600014806112885750808060200190518101906112889190611676565b6110935760405162461bcd60e51b815260206004820152602a60248201527f5361666545524332303a204552433230206f7065726174696f6e20646964206e6044820152691bdd081cdd58d8d9595960b21b6064820152608401610159565b60606112f684846000856112fe565b949350505050565b60608247101561135f5760405162461bcd60e51b815260206004820152602660248201527f416464726573733a20696e73756666696369656e742062616c616e636520666f6044820152651c8818d85b1b60d21b6064820152608401610159565b600080866001600160a01b0316858760405161137b9190611698565b60006040518083038185875af1925050503d80600081146113b8576040519150601f19603f3d011682016040523d82523d6000602084013e6113bd565b606091505b50915091506113ce878383876113d9565b979650505050505050565b60608315611448578251600003611441576001600160a01b0385163b6114415760405162461bcd60e51b815260206004820152601d60248201527f416464726573733a2063616c6c20746f206e6f6e2d636f6e74726163740000006044820152606401610159565b50816112f6565b6112f6838381511561145d5781518083602001fd5b8060405162461bcd60e51b815260040161015991906115ba565b634e487b7160e01b600052601160045260246000fd5b808201808211156114a0576114a0611477565b92915050565b6001600160a01b0391909116815260200190565b6000602082840312156114cc57600080fd5b5035919050565b6001600160a01b0381168114610e7557600080fd5b600080604083850312156114fb57600080fd5b8235611506816114d3565b946020939093013593505050565b60008060008060008060c0878903121561152d57600080fd5b8635611538816114d3565b95506020870135611548816114d3565b94506040870135611558816114d3565b93506060870135611568816114d3565b92506080870135611578816114d3565b915060a0870135611588816114d3565b809150509295509295509295565b60005b838110156115b1578181015183820152602001611599565b50506000910152565b60208152600082518060208401526115d9816040850160208701611596565b601f01601f19169190910160400192915050565b6000602082840312156115ff57600080fd5b813561160a816114d3565b9392505050565b818103818111156114a0576114a0611477565b6020808252818101527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572604082015260600190565b60006020828403121561166b57600080fd5b815161160a816114d3565b60006020828403121561168857600080fd5b8151801515811461160a57600080fd5b600082516116aa818460208701611596565b919091019291505056fe416374697665506f6f6c3a2043616c6c6572206973206e65697468657220426f51e7c30439c76308c020a84ca2a666735d3baa69d070e13beaf83e15bb697eb3a2646970667358221220912b2e19793c88c4d0765d788287cfb330db45a5564a18fdce6da0655a9d9ee264736f6c63430008110033
Verified Source Code Full Match
Compiler: v0.8.17+commit.8df45f5f
EVM: london
Optimization: Yes (100 runs)
Ownable.sol 85 lines
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.17;
/**
* Based on OpenZeppelin's Ownable contract:
* https://github.com/OpenZeppelin/openzeppelin-contracts/blob/master/contracts/access/Ownable.sol
*
* @dev Contract module which provides a basic access control mechanism, where
* there is an account (an owner) that can be granted exclusive access to
* specific functions.
*
* This module is used through inheritance. It will make available the modifier
* `onlyOwner`, which can be applied to your functions to restrict their use to
* the owner.
*/
contract Ownable {
address private _owner;
event OwnershipTransferred(address indexed previousOwner, address indexed newOwner);
/**
* @dev Initializes the contract setting the deployer as the initial owner.
*/
constructor () {
_owner = msg.sender;
emit OwnershipTransferred(address(0), msg.sender);
}
/**
* @dev Returns the address of the current owner.
*/
function owner() public view returns (address) {
return _owner;
}
/**
* @dev Throws if called by any account other than the owner.
*/
modifier onlyOwner() {
require(isOwner(), "Ownable: caller is not the owner");
_;
}
/**
* @dev Returns true if the caller is the current owner.
*/
function isOwner() public view returns (bool) {
return msg.sender == _owner;
}
/**
* @dev Leaves the contract without owner. It will not be possible to call
* `onlyOwner` functions anymore.
*
* NOTE: Renouncing ownership will leave the contract without an owner,
* thereby removing any functionality that is only available to the owner.
*
* NOTE: This function is not safe, as it doesn’t check owner is calling it.
* Make sure you check it before calling it.
*/
function _renounceOwnership() internal {
emit OwnershipTransferred(_owner, address(0));
_owner = address(0);
}
/**
* @dev Transfers ownership of the contract to a new account (`newOwner`).
* Can only be called by the current owner.
*/
function transferOwnership(address newOwner) public virtual onlyOwner {
require(newOwner != address(0), "Ownable: new owner is the zero address");
_transferOwnership(newOwner);
}
/**
* @dev Transfers ownership of the contract to a new account (`newOwner`).
* Internal function without access restriction.
*/
function _transferOwnership(address newOwner) internal virtual {
address oldOwner = _owner;
_owner = newOwner;
emit OwnershipTransferred(oldOwner, newOwner);
}
}
ActivePool.sol 176 lines
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.17;
import "@openzeppelin/contracts/token/ERC20/IERC20.sol";
import './Interfaces/IActivePool.sol';
import './Interfaces/ICollSurplusPool.sol';
import './Interfaces/IDefaultPool.sol';
import './Interfaces/IStabilityPool.sol';
import './Interfaces/IBorrowerOperations.sol';
import "./Dependencies/Ownable.sol";
import "./Dependencies/CheckContract.sol";
// import "./Dependencies/console.sol";
import "./Dependencies/SendCollateral.sol";
/*
* The Active Pool holds the collateral and THUSD debt (but not THUSD tokens) for all active troves.
*
* When a trove is liquidated, it's collateral and THUSD debt are transferred from the Active Pool, to either the
* Stability Pool, the Default Pool, or both, depending on the liquidation conditions.
*
*/
contract ActivePool is Ownable, CheckContract, SendCollateral, IActivePool {
string constant public NAME = "ActivePool";
address public defaultPoolAddress;
address public borrowerOperationsAddress;
address public collateralAddress;
address public collSurplusPoolAddress;
address public stabilityPoolAddress;
address public troveManagerAddress;
uint256 internal collateral; // deposited collateral tracker
uint256 internal THUSDDebt;
// --- Contract setters ---
function setAddresses(
address _borrowerOperationsAddress,
address _troveManagerAddress,
address _stabilityPoolAddress,
address _defaultPoolAddress,
address _collSurplusPoolAddress,
address _collateralAddress
)
external
onlyOwner
{
checkContract(_borrowerOperationsAddress);
checkContract(_troveManagerAddress);
checkContract(_stabilityPoolAddress);
checkContract(_defaultPoolAddress);
checkContract(_collSurplusPoolAddress);
if (_collateralAddress != address(0)) {
checkContract(_collateralAddress);
}
borrowerOperationsAddress = _borrowerOperationsAddress;
troveManagerAddress = _troveManagerAddress;
stabilityPoolAddress = _stabilityPoolAddress;
defaultPoolAddress = _defaultPoolAddress;
collateralAddress = _collateralAddress;
collSurplusPoolAddress = _collSurplusPoolAddress;
require(
(Ownable(_defaultPoolAddress).owner() != address(0) ||
IDefaultPool(_defaultPoolAddress).collateralAddress() == _collateralAddress) &&
(Ownable(_borrowerOperationsAddress).owner() != address(0) ||
IBorrowerOperations(_borrowerOperationsAddress).collateralAddress() == _collateralAddress) &&
(Ownable(_stabilityPoolAddress).owner() != address(0) ||
IStabilityPool(stabilityPoolAddress).collateralAddress() == _collateralAddress) &&
(Ownable(_collSurplusPoolAddress).owner() != address(0) ||
ICollSurplusPool(_collSurplusPoolAddress).collateralAddress() == _collateralAddress),
"The same collateral address must be used for the entire set of contracts"
);
emit BorrowerOperationsAddressChanged(_borrowerOperationsAddress);
emit TroveManagerAddressChanged(_troveManagerAddress);
emit StabilityPoolAddressChanged(_stabilityPoolAddress);
emit DefaultPoolAddressChanged(_defaultPoolAddress);
emit CollateralAddressChanged(_collateralAddress);
emit CollSurplusPoolAddressChanged(_collSurplusPoolAddress);
_renounceOwnership();
}
// --- Getters for public variables. Required by IPool interface ---
/*
* Returns the collateral state variable.
*
* Not necessarily equal to the the contract's raw collateral balance - collateral can be forcibly sent to contracts.
*/
function getCollateralBalance() external view override returns (uint) {
return collateral;
}
function getTHUSDDebt() external view override returns (uint) {
return THUSDDebt;
}
// --- Pool functionality ---
function sendCollateral(address _account, uint256 _amount) external override {
_requireCallerIsBOorTroveMorSP();
collateral -= _amount;
emit ActivePoolCollateralBalanceUpdated(collateral);
emit CollateralSent(_account, _amount);
sendCollateral(IERC20(collateralAddress), _account, _amount);
if (collateralAddress == address(0)) {
return;
}
if (_account == defaultPoolAddress) {
IDefaultPool(_account).updateCollateralBalance(_amount);
} else if (_account == collSurplusPoolAddress) {
ICollSurplusPool(_account).updateCollateralBalance(_amount);
} else if (_account == stabilityPoolAddress) {
IStabilityPool(_account).updateCollateralBalance(_amount);
}
}
function increaseTHUSDDebt(uint256 _amount) external override {
_requireCallerIsBOorTroveM();
THUSDDebt += _amount;
emit ActivePoolTHUSDDebtUpdated(THUSDDebt);
}
function decreaseTHUSDDebt(uint256 _amount) external override {
_requireCallerIsBOorTroveMorSP();
THUSDDebt -= _amount;
emit ActivePoolTHUSDDebtUpdated(THUSDDebt);
}
// --- 'require' functions ---
function _requireCallerIsBorrowerOperationsOrDefaultPool() internal view {
require(
msg.sender == borrowerOperationsAddress ||
msg.sender == defaultPoolAddress,
"ActivePool: Caller is neither BorrowerOperations nor Default Pool");
}
function _requireCallerIsBOorTroveMorSP() internal view {
require(
msg.sender == borrowerOperationsAddress ||
msg.sender == troveManagerAddress ||
msg.sender == stabilityPoolAddress,
"ActivePool: Caller is neither BorrowerOperations nor TroveManager nor StabilityPool");
}
function _requireCallerIsBOorTroveM() internal view {
require(
msg.sender == borrowerOperationsAddress ||
msg.sender == troveManagerAddress,
"ActivePool: Caller is neither BorrowerOperations nor TroveManager");
}
// When ERC20 token collateral is received this function needs to be called
function updateCollateralBalance(uint256 _amount) external override {
_requireCallerIsBorrowerOperationsOrDefaultPool();
require(collateralAddress != address(0), "ActivePool: ETH collateral needed, not ERC20");
collateral += _amount;
emit ActivePoolCollateralBalanceUpdated(collateral);
}
// --- Fallback function ---
// This executes when the contract recieves ETH
receive() external payable {
_requireCallerIsBorrowerOperationsOrDefaultPool();
require(collateralAddress == address(0), "ActivePool: ERC20 collateral needed, not ETH");
collateral += msg.value;
emit ActivePoolCollateralBalanceUpdated(collateral);
}
}
IPool.sol 26 lines
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.17;
// Common interface for the Pools.
interface IPool {
// --- Events ---
event CollateralBalanceUpdated(uint256 _newBalance);
event THUSDBalanceUpdated(uint256 _newBalance);
event ActivePoolAddressChanged(address _newActivePoolAddress);
event DefaultPoolAddressChanged(address _newDefaultPoolAddress);
event StabilityPoolAddressChanged(address _newStabilityPoolAddress);
event CollateralSent(address _to, uint256 _amount);
// --- Functions ---
function getCollateralBalance() external view returns (uint);
function getTHUSDDebt() external view returns (uint);
function increaseTHUSDDebt(uint256 _amount) external;
function decreaseTHUSDDebt(uint256 _amount) external;
}
IActivePool.sol 21 lines
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.17;
import "./IPool.sol";
interface IActivePool is IPool {
// --- Events ---
event BorrowerOperationsAddressChanged(address _newBorrowerOperationsAddress);
event TroveManagerAddressChanged(address _newTroveManagerAddress);
event ActivePoolTHUSDDebtUpdated(uint256 _THUSDDebt);
event ActivePoolCollateralBalanceUpdated(uint256 _collateral);
event CollateralAddressChanged(address _newCollateralAddress);
event CollSurplusPoolAddressChanged(address _newCollSurplusPoolAddress);
// --- Functions ---
function sendCollateral(address _account, uint256 _amount) external;
function updateCollateralBalance(uint256 _amount) external;
function collateralAddress() external view returns(address);
}
IDefaultPool.sol 19 lines
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.17;
import "./IPool.sol";
interface IDefaultPool is IPool {
// --- Events ---
event TroveManagerAddressChanged(address _newTroveManagerAddress);
event DefaultPoolTHUSDDebtUpdated(uint256 _THUSDDebt);
event DefaultPoolCollateralBalanceUpdated(uint256 _collateral);
event CollateralAddressChanged(address _newCollateralAddress);
// --- Functions ---
function sendCollateralToActivePool(uint256 _amount) external;
function updateCollateralBalance(uint256 _amount) external;
function collateralAddress() external view returns(address);
}
IStabilityPool.sol 142 lines
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.17;
/*
* The Stability Pool holds THUSD tokens deposited by Stability Pool depositors.
*
* When a trove is liquidated, then depending on system conditions, some of its THUSD debt gets offset with
* THUSD in the Stability Pool: that is, the offset debt evaporates, and an equal amount of THUSD tokens in the Stability Pool is burned.
*
* Thus, a liquidation causes each depositor to receive a THUSD loss, in proportion to their deposit as a share of total deposits.
* They also receive an collateral gain, as the collateral of the liquidated trove is distributed among Stability depositors,
* in the same proportion.
*
* When a liquidation occurs, it depletes every deposit by the same fraction: for example, a liquidation that depletes 40%
* of the total THUSD in the Stability Pool, depletes 40% of each deposit.
*
* A deposit that has experienced a series of liquidations is termed a "compounded deposit": each liquidation depletes the deposit,
* multiplying it by some factor in range ]0,1[
*
* Please see the implementation spec in the proof document, which closely follows on from the compounded deposit / collateral gain derivations:
* https://github.com/liquity/liquity/blob/master/papers/Scalable_Reward_Distribution_with_Compounding_Stakes.pdf
*
*/
interface IStabilityPool {
// --- Events ---
event StabilityPoolCollateralBalanceUpdated(uint256 _newBalance);
event StabilityPoolTHUSDBalanceUpdated(uint256 _newBalance);
event BorrowerOperationsAddressChanged(address _newBorrowerOperationsAddress);
event TroveManagerAddressChanged(address _newTroveManagerAddress);
event ActivePoolAddressChanged(address _newActivePoolAddress);
event DefaultPoolAddressChanged(address _newDefaultPoolAddress);
event THUSDTokenAddressChanged(address _newTHUSDTokenAddress);
event SortedTrovesAddressChanged(address _newSortedTrovesAddress);
event PriceFeedAddressChanged(address _newPriceFeedAddress);
event CollateralAddressChanged(address _newCollateralAddress);
event P_Updated(uint256 _P);
event S_Updated(uint256 _S, uint128 _epoch, uint128 _scale);
event EpochUpdated(uint128 _currentEpoch);
event ScaleUpdated(uint128 _currentScale);
event DepositSnapshotUpdated(address indexed _depositor, uint256 _P, uint256 _S);
event UserDepositChanged(address indexed _depositor, uint256 _newDeposit);
event CollateralGainWithdrawn(address indexed _depositor, uint256 _collateral, uint256 _THUSDLoss);
event CollateralSent(address _to, uint256 _amount);
// --- Functions ---
/*
* Called only once on init, to set addresses of other Liquity contracts
* Callable only by owner, renounces ownership at the end
*/
function setAddresses(
address _borrowerOperationsAddress,
address _troveManagerAddress,
address _activePoolAddress,
address _thusdTokenAddress,
address _sortedTrovesAddress,
address _priceFeedAddress,
address _collateralAddress
) external;
/*
* Initial checks:
* - _amount is not zero
* ---
* - Sends depositor's accumulated gains (collateral) to depositor
*/
function provideToSP(uint256 _amount) external;
/*
* Initial checks:
* - _amount is zero or there are no under collateralized troves left in the system
* - User has a non zero deposit
* ---
* - Sends all depositor's accumulated gains (collateral) to depositor
* - Decreases deposit stake, and takes new snapshot.
*
* If _amount > userDeposit, the user withdraws all of their compounded deposit.
*/
function withdrawFromSP(uint256 _amount) external;
/*
* Initial checks:
* - User has a non zero deposit
* - User has an open trove
* - User has some collateral gain
* ---
* - Transfers the depositor's entire collateral gain from the Stability Pool to the caller's trove
* - Leaves their compounded deposit in the Stability Pool
* - Updates snapshots for deposit
*/
function withdrawCollateralGainToTrove(address _upperHint, address _lowerHint) external;
/*
* Initial checks:
* - Caller is TroveManager
* ---
* Cancels out the specified debt against the THUSD contained in the Stability Pool (as far as possible)
* and transfers the Trove's collateral from ActivePool to StabilityPool.
* Only called by liquidation functions in the TroveManager.
*/
function offset(uint256 _debt, uint256 _coll) external;
/*
* Returns the total amount of collateral held by the pool, accounted in an internal variable instead of `balance`,
* to exclude edge cases like collateral received from a self-destruct.
*/
function getCollateralBalance() external view returns (uint);
/*
* Returns THUSD held in the pool. Changes when users deposit/withdraw, and when Trove debt is offset.
*/
function getTotalTHUSDDeposits() external view returns (uint);
/*
* Calculates the collateral gain earned by the deposit since its last snapshots were taken.
*/
function getDepositorCollateralGain(address _depositor) external view returns (uint);
/*
* Return the user's compounded deposit.
*/
function getCompoundedTHUSDDeposit(address _depositor) external view returns (uint);
/*
* Only callable by Active Pool, updates ERC20 tokens recieved
*/
function updateCollateralBalance(uint256 _amount) external;
/*
* Fallback function
* Only callable by Active Pool, it just accounts for ETH received
* receive() external payable;
*/
function collateralAddress() external view returns(address);
}
CheckContract.sol 19 lines
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.17;
contract CheckContract {
/**
* Check that the account is an already deployed non-destroyed contract.
* See: https://github.com/OpenZeppelin/openzeppelin-contracts/blob/master/contracts/utils/Address.sol#L12
*/
function checkContract(address _account) internal view {
require(_account != address(0), "Account cannot be zero address");
uint256 size;
// solhint-disable-next-line no-inline-assembly
assembly { size := extcodesize(_account) }
require(size > 0, "Account code size cannot be zero");
}
}
Address.sol 244 lines
// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts (last updated v4.9.0) (utils/Address.sol)
pragma solidity ^0.8.1;
/**
* @dev Collection of functions related to the address type
*/
library Address {
/**
* @dev Returns true if `account` is a contract.
*
* [IMPORTANT]
* ====
* It is unsafe to assume that an address for which this function returns
* false is an externally-owned account (EOA) and not a contract.
*
* Among others, `isContract` will return false for the following
* types of addresses:
*
* - an externally-owned account
* - a contract in construction
* - an address where a contract will be created
* - an address where a contract lived, but was destroyed
*
* Furthermore, `isContract` will also return true if the target contract within
* the same transaction is already scheduled for destruction by `SELFDESTRUCT`,
* which only has an effect at the end of a transaction.
* ====
*
* [IMPORTANT]
* ====
* You shouldn't rely on `isContract` to protect against flash loan attacks!
*
* Preventing calls from contracts is highly discouraged. It breaks composability, breaks support for smart wallets
* like Gnosis Safe, and does not provide security since it can be circumvented by calling from a contract
* constructor.
* ====
*/
function isContract(address account) internal view returns (bool) {
// This method relies on extcodesize/address.code.length, which returns 0
// for contracts in construction, since the code is only stored at the end
// of the constructor execution.
return account.code.length > 0;
}
/**
* @dev Replacement for Solidity's `transfer`: sends `amount` wei to
* `recipient`, forwarding all available gas and reverting on errors.
*
* https://eips.ethereum.org/EIPS/eip-1884[EIP1884] increases the gas cost
* of certain opcodes, possibly making contracts go over the 2300 gas limit
* imposed by `transfer`, making them unable to receive funds via
* `transfer`. {sendValue} removes this limitation.
*
* https://consensys.net/diligence/blog/2019/09/stop-using-soliditys-transfer-now/[Learn more].
*
* IMPORTANT: because control is transferred to `recipient`, care must be
* taken to not create reentrancy vulnerabilities. Consider using
* {ReentrancyGuard} or the
* https://solidity.readthedocs.io/en/v0.8.0/security-considerations.html#use-the-checks-effects-interactions-pattern[checks-effects-interactions pattern].
*/
function sendValue(address payable recipient, uint256 amount) internal {
require(address(this).balance >= amount, "Address: insufficient balance");
(bool success, ) = recipient.call{value: amount}("");
require(success, "Address: unable to send value, recipient may have reverted");
}
/**
* @dev Performs a Solidity function call using a low level `call`. A
* plain `call` is an unsafe replacement for a function call: use this
* function instead.
*
* If `target` reverts with a revert reason, it is bubbled up by this
* function (like regular Solidity function calls).
*
* Returns the raw returned data. To convert to the expected return value,
* use https://solidity.readthedocs.io/en/latest/units-and-global-variables.html?highlight=abi.decode#abi-encoding-and-decoding-functions[`abi.decode`].
*
* Requirements:
*
* - `target` must be a contract.
* - calling `target` with `data` must not revert.
*
* _Available since v3.1._
*/
function functionCall(address target, bytes memory data) internal returns (bytes memory) {
return functionCallWithValue(target, data, 0, "Address: low-level call failed");
}
/**
* @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], but with
* `errorMessage` as a fallback revert reason when `target` reverts.
*
* _Available since v3.1._
*/
function functionCall(
address target,
bytes memory data,
string memory errorMessage
) internal returns (bytes memory) {
return functionCallWithValue(target, data, 0, errorMessage);
}
/**
* @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
* but also transferring `value` wei to `target`.
*
* Requirements:
*
* - the calling contract must have an ETH balance of at least `value`.
* - the called Solidity function must be `payable`.
*
* _Available since v3.1._
*/
function functionCallWithValue(address target, bytes memory data, uint256 value) internal returns (bytes memory) {
return functionCallWithValue(target, data, value, "Address: low-level call with value failed");
}
/**
* @dev Same as {xref-Address-functionCallWithValue-address-bytes-uint256-}[`functionCallWithValue`], but
* with `errorMessage` as a fallback revert reason when `target` reverts.
*
* _Available since v3.1._
*/
function functionCallWithValue(
address target,
bytes memory data,
uint256 value,
string memory errorMessage
) internal returns (bytes memory) {
require(address(this).balance >= value, "Address: insufficient balance for call");
(bool success, bytes memory returndata) = target.call{value: value}(data);
return verifyCallResultFromTarget(target, success, returndata, errorMessage);
}
/**
* @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
* but performing a static call.
*
* _Available since v3.3._
*/
function functionStaticCall(address target, bytes memory data) internal view returns (bytes memory) {
return functionStaticCall(target, data, "Address: low-level static call failed");
}
/**
* @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`],
* but performing a static call.
*
* _Available since v3.3._
*/
function functionStaticCall(
address target,
bytes memory data,
string memory errorMessage
) internal view returns (bytes memory) {
(bool success, bytes memory returndata) = target.staticcall(data);
return verifyCallResultFromTarget(target, success, returndata, errorMessage);
}
/**
* @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
* but performing a delegate call.
*
* _Available since v3.4._
*/
function functionDelegateCall(address target, bytes memory data) internal returns (bytes memory) {
return functionDelegateCall(target, data, "Address: low-level delegate call failed");
}
/**
* @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`],
* but performing a delegate call.
*
* _Available since v3.4._
*/
function functionDelegateCall(
address target,
bytes memory data,
string memory errorMessage
) internal returns (bytes memory) {
(bool success, bytes memory returndata) = target.delegatecall(data);
return verifyCallResultFromTarget(target, success, returndata, errorMessage);
}
/**
* @dev Tool to verify that a low level call to smart-contract was successful, and revert (either by bubbling
* the revert reason or using the provided one) in case of unsuccessful call or if target was not a contract.
*
* _Available since v4.8._
*/
function verifyCallResultFromTarget(
address target,
bool success,
bytes memory returndata,
string memory errorMessage
) internal view returns (bytes memory) {
if (success) {
if (returndata.length == 0) {
// only check isContract if the call was successful and the return data is empty
// otherwise we already know that it was a contract
require(isContract(target), "Address: call to non-contract");
}
return returndata;
} else {
_revert(returndata, errorMessage);
}
}
/**
* @dev Tool to verify that a low level call was successful, and revert if it wasn't, either by bubbling the
* revert reason or using the provided one.
*
* _Available since v4.3._
*/
function verifyCallResult(
bool success,
bytes memory returndata,
string memory errorMessage
) internal pure returns (bytes memory) {
if (success) {
return returndata;
} else {
_revert(returndata, errorMessage);
}
}
function _revert(bytes memory returndata, string memory errorMessage) private pure {
// Look for revert reason and bubble it up if present
if (returndata.length > 0) {
// The easiest way to bubble the revert reason is using memory via assembly
/// @solidity memory-safe-assembly
assembly {
let returndata_size := mload(returndata)
revert(add(32, returndata), returndata_size)
}
} else {
revert(errorMessage);
}
}
}
SendCollateral.sol 44 lines
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.17;
import "@openzeppelin/contracts/token/ERC20/IERC20.sol";
import "@openzeppelin/contracts/token/ERC20/utils/SafeERC20.sol";
contract SendCollateral {
using SafeERC20 for IERC20;
/**
* Sends collateral to recipient
*/
function sendCollateral(IERC20 _collateralERC20, address _recipient, uint256 _amount) internal {
if (address(_collateralERC20) == address(0)) {
// ETH
// require(_amount <= address(this).balance, "Not enough ETH");
(bool success, ) = _recipient.call{ value: _amount }(""); // re-entry is fine here
require(success, "Sending ETH failed");
} else {
// ERC20
// require(_amount <= _collateralERC20.balanceOf(address(this)), "Not enough collateral");
_collateralERC20.safeTransfer(_recipient, _amount);
}
}
/**
* Sends collateral to recipient
*/
function sendCollateralFrom(IERC20 _collateralERC20, address _from, address _recipient, uint256 _amount) internal {
if (address(_collateralERC20) == address(0)) {
// ETH
// require(_amount <= address(this).balance, "Not enough ETH");
(bool success, ) = _recipient.call{ value: _amount }(""); // re-entry is fine here
require(success, "Sending ETH failed");
} else {
// ERC20
// require(_amount <= _collateralERC20.balanceOf(address(this)), "Not enough collateral");
_collateralERC20.safeTransferFrom(_from, _recipient, _amount);
}
}
}
ICollSurplusPool.sol 38 lines
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.17;
interface ICollSurplusPool {
// --- Events ---
event BorrowerOperationsAddressChanged(address _newBorrowerOperationsAddress);
event TroveManagerAddressChanged(address _newTroveManagerAddress);
event ActivePoolAddressChanged(address _newActivePoolAddress);
event CollateralAddressChanged(address _newCollateralAddress);
event CollBalanceUpdated(address indexed _account, uint256 _newBalance);
event CollateralSent(address _to, uint256 _amount);
// --- Contract setters ---
function setAddresses(
address _borrowerOperationsAddress,
address _troveManagerAddress,
address _activePoolAddress,
address _collateralAddress
) external;
function getCollateralBalance() external view returns (uint);
function getCollateral(address _account) external view returns (uint);
function accountSurplus(address _account, uint256 _amount) external;
function claimColl(address _account) external;
function updateCollateralBalance(uint256 _amount) external;
function collateralAddress() external view returns(address);
}
IBorrowerOperations.sol 64 lines
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.17;
// Common interface for the Trove Manager.
interface IBorrowerOperations {
// --- Events ---
event TroveManagerAddressChanged(address _newTroveManagerAddress);
event ActivePoolAddressChanged(address _activePoolAddress);
event DefaultPoolAddressChanged(address _defaultPoolAddress);
event StabilityPoolAddressChanged(address _stabilityPoolAddress);
event GasPoolAddressChanged(address _gasPoolAddress);
event CollSurplusPoolAddressChanged(address _collSurplusPoolAddress);
event PriceFeedAddressChanged(address _newPriceFeedAddress);
event SortedTrovesAddressChanged(address _sortedTrovesAddress);
event THUSDTokenAddressChanged(address _thusdTokenAddress);
event PCVAddressChanged(address _pcvAddress);
event CollateralAddressChanged(address _newCollateralAddress);
event TroveCreated(address indexed _borrower, uint256 arrayIndex);
event TroveUpdated(address indexed _borrower, uint256 _debt, uint256 _coll, uint256 stake, uint8 operation);
event THUSDBorrowingFeePaid(address indexed _borrower, uint256 _THUSDFee);
// --- Functions ---
function setAddresses(
address _troveManagerAddress,
address _activePoolAddress,
address _defaultPoolAddress,
address _stabilityPoolAddress,
address _gasPoolAddress,
address _collSurplusPoolAddress,
address _priceFeedAddress,
address _sortedTrovesAddress,
address _thusdTokenAddress,
address _pcvAddress,
address _collateralAddress
) external;
function openTrove(uint256 _maxFee, uint256 _THUSDAmount, uint256 _assetAmount, address _upperHint, address _lowerHint) external payable;
function addColl(uint256 _assetAmount, address _upperHint, address _lowerHint) external payable;
function moveCollateralGainToTrove(address _user, uint256 _assetAmount, address _upperHint, address _lowerHint) external payable;
function withdrawColl(uint256 _amount, address _upperHint, address _lowerHint) external;
function withdrawTHUSD(uint256 _maxFee, uint256 _amount, address _upperHint, address _lowerHint) external;
function repayTHUSD(uint256 _amount, address _upperHint, address _lowerHint) external;
function closeTrove() external;
function adjustTrove(uint256 _maxFee, uint256 _collWithdrawal, uint256 _debtChange, bool isDebtIncrease, uint256 _assetAmount, address _upperHint, address _lowerHint) external payable;
function claimCollateral() external;
function getCompositeDebt(uint256 _debt) external pure returns (uint);
function collateralAddress() external view returns(address);
}
IERC20.sol 78 lines
// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts (last updated v4.9.0) (token/ERC20/IERC20.sol)
pragma solidity ^0.8.0;
/**
* @dev Interface of the ERC20 standard as defined in the EIP.
*/
interface IERC20 {
/**
* @dev Emitted when `value` tokens are moved from one account (`from`) to
* another (`to`).
*
* Note that `value` may be zero.
*/
event Transfer(address indexed from, address indexed to, uint256 value);
/**
* @dev Emitted when the allowance of a `spender` for an `owner` is set by
* a call to {approve}. `value` is the new allowance.
*/
event Approval(address indexed owner, address indexed spender, uint256 value);
/**
* @dev Returns the amount of tokens in existence.
*/
function totalSupply() external view returns (uint256);
/**
* @dev Returns the amount of tokens owned by `account`.
*/
function balanceOf(address account) external view returns (uint256);
/**
* @dev Moves `amount` tokens from the caller's account to `to`.
*
* Returns a boolean value indicating whether the operation succeeded.
*
* Emits a {Transfer} event.
*/
function transfer(address to, uint256 amount) external returns (bool);
/**
* @dev Returns the remaining number of tokens that `spender` will be
* allowed to spend on behalf of `owner` through {transferFrom}. This is
* zero by default.
*
* This value changes when {approve} or {transferFrom} are called.
*/
function allowance(address owner, address spender) external view returns (uint256);
/**
* @dev Sets `amount` as the allowance of `spender` over the caller's tokens.
*
* Returns a boolean value indicating whether the operation succeeded.
*
* IMPORTANT: Beware that changing an allowance with this method brings the risk
* that someone may use both the old and the new allowance by unfortunate
* transaction ordering. One possible solution to mitigate this race
* condition is to first reduce the spender's allowance to 0 and set the
* desired value afterwards:
* https://github.com/ethereum/EIPs/issues/20#issuecomment-263524729
*
* Emits an {Approval} event.
*/
function approve(address spender, uint256 amount) external returns (bool);
/**
* @dev Moves `amount` tokens from `from` to `to` using the
* allowance mechanism. `amount` is then deducted from the caller's
* allowance.
*
* Returns a boolean value indicating whether the operation succeeded.
*
* Emits a {Transfer} event.
*/
function transferFrom(address from, address to, uint256 amount) external returns (bool);
}
SafeERC20.sol 143 lines
// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts (last updated v4.9.0) (token/ERC20/utils/SafeERC20.sol)
pragma solidity ^0.8.0;
import "../IERC20.sol";
import "../extensions/IERC20Permit.sol";
import "../../../utils/Address.sol";
/**
* @title SafeERC20
* @dev Wrappers around ERC20 operations that throw on failure (when the token
* contract returns false). Tokens that return no value (and instead revert or
* throw on failure) are also supported, non-reverting calls are assumed to be
* successful.
* To use this library you can add a `using SafeERC20 for IERC20;` statement to your contract,
* which allows you to call the safe operations as `token.safeTransfer(...)`, etc.
*/
library SafeERC20 {
using Address for address;
/**
* @dev Transfer `value` amount of `token` from the calling contract to `to`. If `token` returns no value,
* non-reverting calls are assumed to be successful.
*/
function safeTransfer(IERC20 token, address to, uint256 value) internal {
_callOptionalReturn(token, abi.encodeWithSelector(token.transfer.selector, to, value));
}
/**
* @dev Transfer `value` amount of `token` from `from` to `to`, spending the approval given by `from` to the
* calling contract. If `token` returns no value, non-reverting calls are assumed to be successful.
*/
function safeTransferFrom(IERC20 token, address from, address to, uint256 value) internal {
_callOptionalReturn(token, abi.encodeWithSelector(token.transferFrom.selector, from, to, value));
}
/**
* @dev Deprecated. This function has issues similar to the ones found in
* {IERC20-approve}, and its usage is discouraged.
*
* Whenever possible, use {safeIncreaseAllowance} and
* {safeDecreaseAllowance} instead.
*/
function safeApprove(IERC20 token, address spender, uint256 value) internal {
// safeApprove should only be called when setting an initial allowance,
// or when resetting it to zero. To increase and decrease it, use
// 'safeIncreaseAllowance' and 'safeDecreaseAllowance'
require(
(value == 0) || (token.allowance(address(this), spender) == 0),
"SafeERC20: approve from non-zero to non-zero allowance"
);
_callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, value));
}
/**
* @dev Increase the calling contract's allowance toward `spender` by `value`. If `token` returns no value,
* non-reverting calls are assumed to be successful.
*/
function safeIncreaseAllowance(IERC20 token, address spender, uint256 value) internal {
uint256 oldAllowance = token.allowance(address(this), spender);
_callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, oldAllowance + value));
}
/**
* @dev Decrease the calling contract's allowance toward `spender` by `value`. If `token` returns no value,
* non-reverting calls are assumed to be successful.
*/
function safeDecreaseAllowance(IERC20 token, address spender, uint256 value) internal {
unchecked {
uint256 oldAllowance = token.allowance(address(this), spender);
require(oldAllowance >= value, "SafeERC20: decreased allowance below zero");
_callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, oldAllowance - value));
}
}
/**
* @dev Set the calling contract's allowance toward `spender` to `value`. If `token` returns no value,
* non-reverting calls are assumed to be successful. Compatible with tokens that require the approval to be set to
* 0 before setting it to a non-zero value.
*/
function forceApprove(IERC20 token, address spender, uint256 value) internal {
bytes memory approvalCall = abi.encodeWithSelector(token.approve.selector, spender, value);
if (!_callOptionalReturnBool(token, approvalCall)) {
_callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, 0));
_callOptionalReturn(token, approvalCall);
}
}
/**
* @dev Use a ERC-2612 signature to set the `owner` approval toward `spender` on `token`.
* Revert on invalid signature.
*/
function safePermit(
IERC20Permit token,
address owner,
address spender,
uint256 value,
uint256 deadline,
uint8 v,
bytes32 r,
bytes32 s
) internal {
uint256 nonceBefore = token.nonces(owner);
token.permit(owner, spender, value, deadline, v, r, s);
uint256 nonceAfter = token.nonces(owner);
require(nonceAfter == nonceBefore + 1, "SafeERC20: permit did not succeed");
}
/**
* @dev Imitates a Solidity high-level call (i.e. a regular function call to a contract), relaxing the requirement
* on the return value: the return value is optional (but if data is returned, it must not be false).
* @param token The token targeted by the call.
* @param data The call data (encoded using abi.encode or one of its variants).
*/
function _callOptionalReturn(IERC20 token, bytes memory data) private {
// We need to perform a low level call here, to bypass Solidity's return data size checking mechanism, since
// we're implementing it ourselves. We use {Address-functionCall} to perform this call, which verifies that
// the target address contains contract code and also asserts for success in the low-level call.
bytes memory returndata = address(token).functionCall(data, "SafeERC20: low-level call failed");
require(returndata.length == 0 || abi.decode(returndata, (bool)), "SafeERC20: ERC20 operation did not succeed");
}
/**
* @dev Imitates a Solidity high-level call (i.e. a regular function call to a contract), relaxing the requirement
* on the return value: the return value is optional (but if data is returned, it must not be false).
* @param token The token targeted by the call.
* @param data The call data (encoded using abi.encode or one of its variants).
*
* This is a variant of {_callOptionalReturn} that silents catches all reverts and returns a bool instead.
*/
function _callOptionalReturnBool(IERC20 token, bytes memory data) private returns (bool) {
// We need to perform a low level call here, to bypass Solidity's return data size checking mechanism, since
// we're implementing it ourselves. We cannot use {Address-functionCall} here since this should return false
// and not revert is the subcall reverts.
(bool success, bytes memory returndata) = address(token).call(data);
return
success && (returndata.length == 0 || abi.decode(returndata, (bool))) && Address.isContract(address(token));
}
}
IERC20Permit.sol 60 lines
// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts (last updated v4.9.0) (token/ERC20/extensions/IERC20Permit.sol)
pragma solidity ^0.8.0;
/**
* @dev Interface of the ERC20 Permit extension allowing approvals to be made via signatures, as defined in
* https://eips.ethereum.org/EIPS/eip-2612[EIP-2612].
*
* Adds the {permit} method, which can be used to change an account's ERC20 allowance (see {IERC20-allowance}) by
* presenting a message signed by the account. By not relying on {IERC20-approve}, the token holder account doesn't
* need to send a transaction, and thus is not required to hold Ether at all.
*/
interface IERC20Permit {
/**
* @dev Sets `value` as the allowance of `spender` over ``owner``'s tokens,
* given ``owner``'s signed approval.
*
* IMPORTANT: The same issues {IERC20-approve} has related to transaction
* ordering also apply here.
*
* Emits an {Approval} event.
*
* Requirements:
*
* - `spender` cannot be the zero address.
* - `deadline` must be a timestamp in the future.
* - `v`, `r` and `s` must be a valid `secp256k1` signature from `owner`
* over the EIP712-formatted function arguments.
* - the signature must use ``owner``'s current nonce (see {nonces}).
*
* For more information on the signature format, see the
* https://eips.ethereum.org/EIPS/eip-2612#specification[relevant EIP
* section].
*/
function permit(
address owner,
address spender,
uint256 value,
uint256 deadline,
uint8 v,
bytes32 r,
bytes32 s
) external;
/**
* @dev Returns the current nonce for `owner`. This value must be
* included whenever a signature is generated for {permit}.
*
* Every successful call to {permit} increases ``owner``'s nonce by one. This
* prevents a signature from being used multiple times.
*/
function nonces(address owner) external view returns (uint256);
/**
* @dev Returns the domain separator used in the encoding of the signature for {permit}, as defined by {EIP712}.
*/
// solhint-disable-next-line func-name-mixedcase
function DOMAIN_SEPARATOR() external view returns (bytes32);
}
Read Contract
NAME 0xa3f4df7e → string
borrowerOperationsAddress 0xb7f8cf9b → address
collSurplusPoolAddress 0x2a9043ec → address
collateralAddress 0x48d399e7 → address
defaultPoolAddress 0xaac1846f → address
getCollateralBalance 0x1529a639 → uint256
getTHUSDDebt 0xf07424e9 → uint256
isOwner 0x8f32d59b → bool
owner 0x8da5cb5b → address
stabilityPoolAddress 0x0b622ab2 → address
troveManagerAddress 0x5a4d28bb → address
Write Contract 6 functions
These functions modify contract state and require a wallet transaction to execute.
decreaseTHUSDDebt 0xf6ff93d6
uint256 _amount
increaseTHUSDDebt 0x42c31a23
uint256 _amount
sendCollateral 0x62502169
address _account
uint256 _amount
setAddresses 0x6cfb6bf9
address _borrowerOperationsAddress
address _troveManagerAddress
address _stabilityPoolAddress
address _defaultPoolAddress
address _collSurplusPoolAddress
address _collateralAddress
transferOwnership 0xf2fde38b
address newOwner
updateCollateralBalance 0x98d99ee6
uint256 _amount
Recent Transactions
No transactions found for this address