编译器
0.8.24+commit.e11b9ed9
文件 1 的 29:AccessControl.sol
pragma solidity ^0.8.0;
import "./IAccessControl.sol";
import "../utils/Context.sol";
import "../utils/Strings.sol";
import "../utils/introspection/ERC165.sol";
abstract contract AccessControl is Context, IAccessControl, ERC165 {
struct RoleData {
mapping(address => bool) members;
bytes32 adminRole;
}
mapping(bytes32 => RoleData) private _roles;
bytes32 public constant DEFAULT_ADMIN_ROLE = 0x00;
modifier onlyRole(bytes32 role) {
_checkRole(role);
_;
}
function supportsInterface(bytes4 interfaceId) public view virtual override returns (bool) {
return interfaceId == type(IAccessControl).interfaceId || super.supportsInterface(interfaceId);
}
function hasRole(bytes32 role, address account) public view virtual override returns (bool) {
return _roles[role].members[account];
}
function _checkRole(bytes32 role) internal view virtual {
_checkRole(role, _msgSender());
}
function _checkRole(bytes32 role, address account) internal view virtual {
if (!hasRole(role, account)) {
revert(
string(
abi.encodePacked(
"AccessControl: account ",
Strings.toHexString(account),
" is missing role ",
Strings.toHexString(uint256(role), 32)
)
)
);
}
}
function getRoleAdmin(bytes32 role) public view virtual override returns (bytes32) {
return _roles[role].adminRole;
}
function grantRole(bytes32 role, address account) public virtual override onlyRole(getRoleAdmin(role)) {
_grantRole(role, account);
}
function revokeRole(bytes32 role, address account) public virtual override onlyRole(getRoleAdmin(role)) {
_revokeRole(role, account);
}
function renounceRole(bytes32 role, address account) public virtual override {
require(account == _msgSender(), "AccessControl: can only renounce roles for self");
_revokeRole(role, account);
}
function _setupRole(bytes32 role, address account) internal virtual {
_grantRole(role, account);
}
function _setRoleAdmin(bytes32 role, bytes32 adminRole) internal virtual {
bytes32 previousAdminRole = getRoleAdmin(role);
_roles[role].adminRole = adminRole;
emit RoleAdminChanged(role, previousAdminRole, adminRole);
}
function _grantRole(bytes32 role, address account) internal virtual {
if (!hasRole(role, account)) {
_roles[role].members[account] = true;
emit RoleGranted(role, account, _msgSender());
}
}
function _revokeRole(bytes32 role, address account) internal virtual {
if (hasRole(role, account)) {
_roles[role].members[account] = false;
emit RoleRevoked(role, account, _msgSender());
}
}
}
文件 2 的 29:Address.sol
pragma solidity ^0.8.1;
library Address {
function isContract(address account) internal view returns (bool) {
return account.code.length > 0;
}
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");
}
function functionCall(address target, bytes memory data) internal returns (bytes memory) {
return functionCallWithValue(target, data, 0, "Address: low-level call failed");
}
function functionCall(
address target,
bytes memory data,
string memory errorMessage
) internal returns (bytes memory) {
return functionCallWithValue(target, data, 0, errorMessage);
}
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");
}
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);
}
function functionStaticCall(address target, bytes memory data) internal view returns (bytes memory) {
return functionStaticCall(target, data, "Address: low-level static call failed");
}
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);
}
function functionDelegateCall(address target, bytes memory data) internal returns (bytes memory) {
return functionDelegateCall(target, data, "Address: low-level delegate call failed");
}
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);
}
function verifyCallResultFromTarget(
address target,
bool success,
bytes memory returndata,
string memory errorMessage
) internal view returns (bytes memory) {
if (success) {
if (returndata.length == 0) {
require(isContract(target), "Address: call to non-contract");
}
return returndata;
} else {
_revert(returndata, errorMessage);
}
}
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 {
if (returndata.length > 0) {
assembly {
let returndata_size := mload(returndata)
revert(add(32, returndata), returndata_size)
}
} else {
revert(errorMessage);
}
}
}
文件 3 的 29:AggregatorV3Interface.sol
pragma solidity ^0.8.0;
interface AggregatorV3Interface {
function decimals() external view returns (uint8);
function description() external view returns (string memory);
function version() external view returns (uint256);
function getRoundData(uint80 _roundId)
external
view
returns (
uint80 roundId,
int256 answer,
uint256 startedAt,
uint256 updatedAt,
uint80 answeredInRound
);
function latestRoundData()
external
view
returns (
uint80 roundId,
int256 answer,
uint256 startedAt,
uint256 updatedAt,
uint80 answeredInRound
);
}
文件 4 的 29:AssetAdapter.sol
pragma solidity ^0.8.24;
import {AccessControl} from "openzeppelin-contracts/contracts/access/AccessControl.sol";
import {AggregatorV3Interface} from "chainlink/contracts/src/v0.8/interfaces/AggregatorV3Interface.sol";
import {IERC20} from "openzeppelin-contracts/contracts/token/ERC20/IERC20.sol";
import {IERC20Metadata} from "openzeppelin-contracts/contracts/token/ERC20/extensions/IERC20Metadata.sol";
import {IFund} from "src/interfaces/IFund.sol";
import {IAssetAdapter} from "src/interfaces/IAssetAdapter.sol";
import {IOracle} from "src/interfaces/IOracle.sol";
contract UnderlyingAssetPrice is IOracle {
AggregatorV3Interface aggregator;
constructor(address _aggregator) {
aggregator = AggregatorV3Interface(_aggregator);
}
function latestAnswer() external view returns (int256) {
int256 answer;
uint256 updatedAt;
(, answer, , updatedAt, ) = aggregator.latestRoundData();
return (block.timestamp > 1.1 days + updatedAt) ? int256(1e8) : answer;
}
}
contract AssetPrice is IOracle {
IFund public immutable fund;
constructor(address fundAddress) {
fund = IFund(fundAddress);
}
function latestAnswer() external view returns (int256) {
uint256 currentPrice = fund.currentPrice();
return
currentPrice > uint256(type(int256).max)
? type(int256).max
: int256(currentPrice);
}
}
contract AssetAdapter is AccessControl, IAssetAdapter {
bytes32 public constant MANAGER =
keccak256(abi.encode("asset.adapter.manager"));
bytes32 public constant CONTROLLER =
keccak256(abi.encode("asset.adapter.controller"));
uint256 public immutable duration;
IOracle public immutable underlyingPriceOracle;
IOracle public immutable fundPriceOracle;
uint256 public underlyingRiskWeight = 0e6;
uint256 public fundRiskWeight = 0e6;
IFund public immutable fund;
IERC20 public immutable underlying;
uint8 public immutable DECIMAL_FACTOR;
constructor(
address admin,
address underlyingAddr,
address fundAddr,
address underlyingPriceOracleAddr,
address fundPriceOracleAddr,
uint256 _duration
) {
_grantRole(DEFAULT_ADMIN_ROLE, admin);
duration = _duration;
underlying = IERC20(underlyingAddr);
DECIMAL_FACTOR = IERC20Metadata(address(underlying)).decimals();
fund = IFund(fundAddr);
underlyingPriceOracle = IOracle(underlyingPriceOracleAddr);
fundPriceOracle = IOracle(fundPriceOracleAddr);
}
function allocate(uint256 amount) external {
underlying.transferFrom(msg.sender, address(this), amount);
emit Allocate(msg.sender, amount, block.timestamp);
}
function withdraw(uint256 amount) external onlyRole(CONTROLLER) {
underlying.transfer(msg.sender, amount);
emit Withdraw(msg.sender, amount, block.timestamp);
}
function deposit(uint256 amount) external onlyRole(CONTROLLER) {
underlying.approve(address(fund), amount);
fund.deposit(amount);
emit Deposit(msg.sender, amount, block.timestamp);
}
function redeem(uint256 amount) external onlyRole(CONTROLLER) {
fund.approve(address(fund), amount);
fund.redeem(amount);
emit Redeem(msg.sender, amount, block.timestamp);
}
function setUnderlyingRiskWeight(
uint256 riskWeight
) external onlyRole(MANAGER) {
require(1e6 > riskWeight, "FA: Risk Weight can not be above 100%");
underlyingRiskWeight = riskWeight;
emit UnderlyingRiskWeightUpdate(riskWeight, block.timestamp);
}
function setFundRiskWeight(uint256 riskWeight) external onlyRole(MANAGER) {
require(1e6 > riskWeight, "FA: Risk Weight can not be above 100%");
fundRiskWeight = riskWeight;
emit FundRiskWeightUpdate(riskWeight, block.timestamp);
}
function totalValue() external view returns (uint256) {
uint256 total = 0;
total += _underlyingTotalValue();
total += _fundTotalValue();
return total;
}
function totalRiskValue() external view returns (uint256) {
uint256 total = 0;
total += _underlyingTotalRiskValue();
total += _fundTotalRiskValue();
return total;
}
function underlyingTotalRiskValue() external view returns (uint256) {
return _underlyingTotalRiskValue();
}
function _underlyingTotalRiskValue() private view returns (uint256) {
uint256 assets;
(, assets) = fund.userDeposits(address(this));
return _underlyingRiskValue(assets + _underlyingBalance());
}
function underlyingRiskValue(
uint256 amount
) external view returns (uint256) {
return _underlyingRiskValue(amount);
}
function _underlyingRiskValue(
uint256 amount
) private view returns (uint256) {
return (underlyingRiskWeight * _underlyingValue(amount)) / 1e6;
}
function underlyingTotalValue() external view returns (uint256) {
return _underlyingTotalValue();
}
function _underlyingTotalValue() private view returns (uint256) {
uint256 assets;
(, assets) = fund.userDeposits(address(this));
return _underlyingValue(assets + _underlyingBalance());
}
function underlyingValue(uint256 amount) external view returns (uint256) {
return _underlyingValue(amount);
}
function _underlyingValue(uint256 amount) private view returns (uint256) {
return
(_underlyingPriceOracleLatestAnswer() *
amount *
(10 ** (18 - DECIMAL_FACTOR))) / 1e8;
}
function underlyingBalance() external view returns (uint256) {
return _underlyingBalance();
}
function _underlyingBalance() private view returns (uint256) {
return underlying.balanceOf(address(this));
}
function fundTotalRiskValue() external view returns (uint256) {
return _fundTotalRiskValue();
}
function _fundTotalRiskValue() private view returns (uint256) {
uint256 shares;
(, shares) = fund.userRedemptions(address(this));
return _fundRiskValue(shares + _fundBalance());
}
function fundRiskValue(uint256 amount) external view returns (uint256) {
return _fundRiskValue(amount);
}
function _fundRiskValue(uint256 amount) private view returns (uint256) {
return (fundRiskWeight * _fundValue(amount)) / 1e6;
}
function fundTotalValue() external view returns (uint256) {
return _fundTotalValue();
}
function _fundTotalValue() private view returns (uint256) {
uint256 shares;
(, shares) = fund.userRedemptions(address(this));
return _fundValue(shares + _fundBalance());
}
function fundValue(uint256 amount) external view returns (uint256) {
return _fundValue(amount);
}
function _fundValue(uint256 amount) private view returns (uint256) {
return (_fundPriceOracleLatestAnswer() * amount) / 1e8;
}
function fundBalance() external view returns (uint256) {
return _fundBalance();
}
function _fundBalance() private view returns (uint256) {
return fund.balanceOf(address(this));
}
function _underlyingPriceOracleLatestAnswer()
private
view
returns (uint256)
{
int256 latestAnswer = underlyingPriceOracle.latestAnswer();
return latestAnswer > 0 ? uint256(latestAnswer) : 0;
}
function _fundPriceOracleLatestAnswer() private view returns (uint256) {
int256 latestAnswer = fundPriceOracle.latestAnswer();
return latestAnswer > 0 ? uint256(latestAnswer) : 0;
}
}
文件 5 的 29:Context.sol
pragma solidity ^0.8.0;
abstract contract Context {
function _msgSender() internal view virtual returns (address) {
return msg.sender;
}
function _msgData() internal view virtual returns (bytes calldata) {
return msg.data;
}
}
文件 6 的 29:CreditEnforcer.sol
pragma solidity ^0.8.24;
import {AccessControl} from "openzeppelin-contracts/contracts/access/AccessControl.sol";
import {ICreditEnforcer} from "src/interfaces/ICreditEnforcer.sol";
import {IERC20} from "openzeppelin-contracts/contracts/token/ERC20/IERC20.sol";
import {ITermIssuer} from "src/interfaces/ITermIssuer.sol";
import {IPegStabilityModule} from "src/interfaces/IPegStabilityModule.sol";
import {ISavingModule} from "src/interfaces/ISavingModule.sol";
import {IAssetAdapter} from "src/adapters/AssetAdapter.sol";
contract CreditEnforcer is AccessControl, ICreditEnforcer {
bytes32 public constant MANAGER =
keccak256(abi.encode("credit.enforcer.manager"));
bytes32 public constant SUPERVISOR =
keccak256(abi.encode("credit.enforcer.supervisor"));
struct AssetAdapter {
bool set;
uint256 index;
}
IERC20 public immutable underlying;
ITermIssuer public immutable termIssuer;
ISavingModule public immutable sm;
IPegStabilityModule public immutable psm;
uint256 public duration = 365 days;
uint256 public assetRatioMin = type(uint256).max;
uint256 public equityRatioMin = type(uint256).max;
uint256 public liquidityRatioMin = type(uint256).max;
uint256 public smDebtMax = 0;
uint256 public psmDebtMax = 0;
mapping(uint256 => uint256) public termDebtMax;
address[] public assetAdapterList;
mapping(address => AssetAdapter) public assetAdapterMap;
constructor(
address admin,
IERC20 underlying_,
ITermIssuer termIssuer_,
IPegStabilityModule psm_,
ISavingModule sm_
) {
_grantRole(DEFAULT_ADMIN_ROLE, admin);
underlying = underlying_;
termIssuer = termIssuer_;
psm = psm_;
sm = sm_;
}
function mintStablecoin(uint256 amount) external returns (uint256) {
return _mintStablecoin(msg.sender, msg.sender, amount);
}
function mintStablecoin(
address to,
uint256 amount
) external returns (uint256) {
return _mintStablecoin(msg.sender, to, amount);
}
function _mintStablecoin(
address from,
address to,
uint256 amount
) private returns (uint256) {
bool valid;
string memory message;
(valid, message) = _checkPSMDebtMax(amount);
require(valid, message);
psm.mint(from, to, amount);
(valid, message) = _checkRatios();
require(valid, message);
return amount;
}
function mintSavingcoin(uint256 amount) external returns (uint256) {
return _mintSavingcoin(msg.sender, msg.sender, amount);
}
function mintSavingcoin(
address to,
uint256 amount
) external returns (uint256) {
return _mintSavingcoin(msg.sender, to, amount);
}
function _mintSavingcoin(
address from,
address to,
uint256 amount
) private returns (uint256) {
bool valid;
string memory message;
(valid, message) = _checkSMDebtMax(amount);
require(valid, message);
sm.mint(from, to, amount);
(valid, message) = _checkRatios();
require(valid, message);
return amount;
}
function mintTerm(uint256 id, uint256 amount) external returns (uint256) {
return _mintTerm(msg.sender, msg.sender, id, amount);
}
function mintTerm(
address to,
uint256 id,
uint256 amount
) external returns (uint256) {
return _mintTerm(msg.sender, to, id, amount);
}
function _mintTerm(
address from,
address to,
uint256 id,
uint256 amount
) private returns (uint256) {
bool valid;
string memory message;
(valid, message) = _checkTermDebtMax(id, amount);
require(valid, message);
uint256 cost = termIssuer.mint(from, to, id, amount);
(valid, message) = _checkRatios();
require(valid, message);
return cost;
}
function allocate(
uint256 index,
uint256 amount
) external onlyRole(MANAGER) {
require(
_assetAdapterLength() > index,
"CE: Asset Adapter index out of bounds"
);
address assetAdapterAddress = assetAdapterList[index];
psm.withdraw(amount);
underlying.approve(assetAdapterAddress, amount);
IAssetAdapter(assetAdapterAddress).allocate(amount);
}
function withdraw(
uint256 index,
uint256 amount
) external onlyRole(MANAGER) {
require(
_assetAdapterLength() > index,
"CE: Asset Adapter index out of bounds"
);
address assetAdapterAddress = assetAdapterList[index];
IAssetAdapter(assetAdapterAddress).withdraw(amount);
underlying.approve(address(psm), amount);
psm.allocate(amount);
}
function deposit(uint256 index, uint256 amount) external onlyRole(MANAGER) {
require(
_assetAdapterLength() > index,
"CE: Asset Adapter index out of bounds"
);
bool valid;
string memory message;
address assetAdapterAddress = assetAdapterList[index];
IAssetAdapter(assetAdapterAddress).deposit(amount);
(valid, message) = _checkRatios();
require(valid, message);
}
function redeem(uint256 index, uint256 amount) external onlyRole(MANAGER) {
require(
_assetAdapterLength() > index,
"CE: Asset Adapter index out of bounds"
);
bool valid;
string memory message;
address assetAdapterAddress = assetAdapterList[index];
IAssetAdapter(assetAdapterAddress).redeem(amount);
(valid, message) = _checkRatios();
require(valid, message);
}
function checkPSMDebtMax(
uint256 amount
) external view returns (bool, string memory) {
return _checkPSMDebtMax(amount);
}
function _checkPSMDebtMax(
uint256 amount
) private view returns (bool, string memory) {
if (amount + psm.underlyingBalance() > psmDebtMax) {
return (false, "CE: amount exceeds PSM debt max");
}
return (true, "");
}
function checkSMDebtMax(
uint256 amount
) external view returns (bool, string memory) {
return _checkSMDebtMax(amount);
}
function _checkSMDebtMax(
uint256 amount
) private view returns (bool, string memory) {
if (amount + sm.totalDebt() > smDebtMax) {
return (false, "CE: amount exceeds SM debt max");
}
return (true, "");
}
function checkTermDebtMax(
uint256 id,
uint256 amount
) external view returns (bool, string memory) {
return _checkTermDebtMax(id, amount);
}
function _checkTermDebtMax(
uint256 id,
uint256 amount
) private view returns (bool, string memory) {
if (amount + termIssuer.totalSupply(id) > _getTermDebtMax(id)) {
return (false, "CE: amount exceeds term minter debt max");
}
return (true, "");
}
function checkRatios() external view returns (bool, string memory) {
return _checkRatios();
}
function _checkRatios() private view returns (bool, string memory) {
if (assetRatioMin > _assetRatio(0)) {
return (false, "CE: invalid asset ratio");
}
if (equityRatioMin > _equityRatio(0)) {
return (false, "CE: invalid equity ratio");
}
if (liquidityRatioMin > _liquidityRatio(duration)) {
return (false, "CE: invalid liquidity ratio");
}
return (true, "");
}
function assetRatio() external view returns (uint256) {
return _assetRatio(0);
}
function _assetRatio(uint256) private view returns (uint256) {
uint256 assets_ = _assets();
uint256 liabilities_ = _liabilities();
if (assets_ == 0) return 0;
if (liabilities_ == 0) return type(uint256).max;
return (assets_ * 1e6) / liabilities_;
}
function equityRatio() external view returns (uint256) {
return _equityRatio(0);
}
function _equityRatio(uint256) private view returns (uint256) {
uint256 equity_ = _equity();
uint256 riskWeightedAssets_ = _riskWeightedAssets();
if (equity_ == 0) return 0;
if (riskWeightedAssets_ == 0) return type(uint256).max;
return (equity_ * 1e6) / riskWeightedAssets_;
}
function liquidityRatio() external view returns (uint256) {
return _liquidityRatio(duration);
}
function _liquidityRatio(uint256 duration_) private view returns (uint256) {
uint256 assets_ = _shortTermAssets(duration_);
uint256 liabilities_ = _shortTermLiabilities(duration_);
if (assets_ == 0) return 0;
if (liabilities_ == 0) return type(uint256).max;
return (assets_ * 1e6) / liabilities_;
}
function shortTermAssets() external view returns (uint256) {
return _shortTermAssets(duration);
}
function _shortTermAssets(
uint256 _duration
) private view returns (uint256 stAssets) {
uint256 length = assetAdapterList.length;
for (uint256 i = 0; i < length; i++) {
IAssetAdapter assetAdapter = IAssetAdapter(assetAdapterList[i]);
if (_duration <= assetAdapter.duration()) continue;
stAssets += assetAdapter.totalValue();
}
stAssets += psm.totalValue();
}
function extendedAssets() external view returns (uint256) {
return _extendedAssets(duration);
}
function _extendedAssets(
uint256 _duration
) private view returns (uint256 eAssets) {
uint256 length = assetAdapterList.length;
for (uint256 i = 0; i < length; i++) {
IAssetAdapter assetAdapter = IAssetAdapter(assetAdapterList[i]);
if (_duration >= assetAdapter.duration()) continue;
eAssets += assetAdapter.totalValue();
}
}
function shortTermLiabilities() external view returns (uint256) {
return _shortTermLiabilities(duration);
}
function _shortTermLiabilities(
uint256 duration_
) private view returns (uint256 totalLiabilities) {
totalLiabilities = _liabilities();
totalLiabilities -= _extendedLiabilities(duration_);
}
function extendedLiabilities(
uint256 duration_
) external view returns (uint256) {
return _extendedLiabilities(duration_);
}
function _extendedLiabilities(
uint256 duration_
) private view returns (uint256) {
uint256 latestID = termIssuer.latestID();
uint256 earliestID = termIssuer.earliestID();
uint256 sum = 0;
for (uint256 i = earliestID; i <= latestID; i++) {
if (termIssuer.maturityTimestamp(i) <= block.timestamp + duration_) {
continue;
}
sum += termIssuer.totalSupply(i);
}
return sum;
}
function riskWeightedAssets() external view returns (uint256) {
return _riskWeightedAssets();
}
function _riskWeightedAssets() private view returns (uint256) {
uint256 total = 0;
uint256 length = assetAdapterList.length;
for (uint256 i = 0; i < length; i++) {
total += IAssetAdapter(assetAdapterList[i]).totalRiskValue();
}
return total + psm.totalRiskValue();
}
function equity() external view returns (uint256) {
return _equity();
}
function _equity() private view returns (uint256) {
uint256 assets_ = _assets();
uint256 liabilities_ = _liabilities();
return liabilities_ > assets_ ? 0 : assets_ - liabilities_;
}
function assets() external view returns (uint256) {
return _assets();
}
function _assets() private view returns (uint256) {
uint256 total = 0;
uint256 length = assetAdapterList.length;
for (uint256 i = 0; i < length; i++) {
total += IAssetAdapter(assetAdapterList[i]).totalValue();
}
return total + psm.totalValue();
}
function liabilities() external view returns (uint256) {
return _liabilities();
}
function _liabilities() private view returns (uint256) {
return sm.rusdTotalLiability() + termIssuer.totalDebt();
}
function setDuration(uint256 duration_) external onlyRole(MANAGER) {
duration = duration_;
}
function setAssetRatioMin(
uint256 assetRatioMin_
) external onlyRole(MANAGER) {
assetRatioMin = assetRatioMin_;
}
function setEquityRatioMin(
uint256 equityRatioMin_
) external onlyRole(MANAGER) {
equityRatioMin = equityRatioMin_;
}
function setLiquidityRatioMin(
uint256 liquidityRatioMin_
) external onlyRole(MANAGER) {
liquidityRatioMin = liquidityRatioMin_;
}
function setPSMDebtMax(uint256 psmDebtMax_) external onlyRole(MANAGER) {
psmDebtMax = psmDebtMax_;
}
function setSMDebtMax(uint256 smDebtMax_) external onlyRole(MANAGER) {
smDebtMax = smDebtMax_;
}
function setTermDebtMax(
uint256 id,
uint256 amount
) external onlyRole(MANAGER) {
_setTermDebtMax(id, amount);
}
function _setTermDebtMax(uint256 id, uint256 amount) private {
termDebtMax[id] = amount;
}
function getTermDebtMax(uint256 id) external view returns (uint256) {
return _getTermDebtMax(id);
}
function _getTermDebtMax(uint256 id) private view returns (uint256) {
return termDebtMax[id];
}
function assetAdapterLength() external view returns (uint256) {
return _assetAdapterLength();
}
function _assetAdapterLength() private view returns (uint256) {
return assetAdapterList.length;
}
function getAssetAdapterList(
uint256 startIndex,
uint256 length
) external view returns (address[] memory) {
return _getAssetAdapterList(startIndex, length);
}
function _getAssetAdapterList(
uint256 startIndex,
uint256 length
) private view returns (address[] memory) {
address[] memory list = new address[](length);
for (uint256 i = startIndex; i < startIndex + length; i++) {
list[i - startIndex] = assetAdapterList[i];
}
return list;
}
function getAssetAdapter(
address adapter
) external view returns (AssetAdapter memory) {
return _getAssetAdapter(adapter);
}
function _getAssetAdapter(
address adapter
) private view returns (AssetAdapter memory) {
return assetAdapterMap[adapter];
}
function addAssetAdapter(address adapter) external onlyRole(SUPERVISOR) {
AssetAdapter storage assetAdapter = assetAdapterMap[adapter];
require(!assetAdapter.set, "CE: adapter already set");
assetAdapterList.push(adapter);
assetAdapter.set = true;
assetAdapter.index = _assetAdapterLength() - 1;
}
function removeAssetAdapter(address adapter) external onlyRole(SUPERVISOR) {
AssetAdapter storage assetAdapter = assetAdapterMap[adapter];
require(assetAdapter.set, "CE: adapter not set");
uint256 lastIndex = _assetAdapterLength() - 1;
address key = assetAdapterList[lastIndex];
uint256 index = assetAdapter.index;
assetAdapterList[index] = assetAdapterList[lastIndex];
assetAdapterMap[key].index = index;
assetAdapterList.pop();
delete assetAdapterMap[adapter];
}
}
文件 7 的 29:ERC1155.sol
pragma solidity ^0.8.0;
import "./IERC1155.sol";
import "./IERC1155Receiver.sol";
import "./extensions/IERC1155MetadataURI.sol";
import "../../utils/Address.sol";
import "../../utils/Context.sol";
import "../../utils/introspection/ERC165.sol";
contract ERC1155 is Context, ERC165, IERC1155, IERC1155MetadataURI {
using Address for address;
mapping(uint256 => mapping(address => uint256)) private _balances;
mapping(address => mapping(address => bool)) private _operatorApprovals;
string private _uri;
constructor(string memory uri_) {
_setURI(uri_);
}
function supportsInterface(bytes4 interfaceId) public view virtual override(ERC165, IERC165) returns (bool) {
return
interfaceId == type(IERC1155).interfaceId ||
interfaceId == type(IERC1155MetadataURI).interfaceId ||
super.supportsInterface(interfaceId);
}
function uri(uint256) public view virtual override returns (string memory) {
return _uri;
}
function balanceOf(address account, uint256 id) public view virtual override returns (uint256) {
require(account != address(0), "ERC1155: address zero is not a valid owner");
return _balances[id][account];
}
function balanceOfBatch(address[] memory accounts, uint256[] memory ids)
public
view
virtual
override
returns (uint256[] memory)
{
require(accounts.length == ids.length, "ERC1155: accounts and ids length mismatch");
uint256[] memory batchBalances = new uint256[](accounts.length);
for (uint256 i = 0; i < accounts.length; ++i) {
batchBalances[i] = balanceOf(accounts[i], ids[i]);
}
return batchBalances;
}
function setApprovalForAll(address operator, bool approved) public virtual override {
_setApprovalForAll(_msgSender(), operator, approved);
}
function isApprovedForAll(address account, address operator) public view virtual override returns (bool) {
return _operatorApprovals[account][operator];
}
function safeTransferFrom(
address from,
address to,
uint256 id,
uint256 amount,
bytes memory data
) public virtual override {
require(
from == _msgSender() || isApprovedForAll(from, _msgSender()),
"ERC1155: caller is not token owner or approved"
);
_safeTransferFrom(from, to, id, amount, data);
}
function safeBatchTransferFrom(
address from,
address to,
uint256[] memory ids,
uint256[] memory amounts,
bytes memory data
) public virtual override {
require(
from == _msgSender() || isApprovedForAll(from, _msgSender()),
"ERC1155: caller is not token owner or approved"
);
_safeBatchTransferFrom(from, to, ids, amounts, data);
}
function _safeTransferFrom(
address from,
address to,
uint256 id,
uint256 amount,
bytes memory data
) internal virtual {
require(to != address(0), "ERC1155: transfer to the zero address");
address operator = _msgSender();
uint256[] memory ids = _asSingletonArray(id);
uint256[] memory amounts = _asSingletonArray(amount);
_beforeTokenTransfer(operator, from, to, ids, amounts, data);
uint256 fromBalance = _balances[id][from];
require(fromBalance >= amount, "ERC1155: insufficient balance for transfer");
unchecked {
_balances[id][from] = fromBalance - amount;
}
_balances[id][to] += amount;
emit TransferSingle(operator, from, to, id, amount);
_afterTokenTransfer(operator, from, to, ids, amounts, data);
_doSafeTransferAcceptanceCheck(operator, from, to, id, amount, data);
}
function _safeBatchTransferFrom(
address from,
address to,
uint256[] memory ids,
uint256[] memory amounts,
bytes memory data
) internal virtual {
require(ids.length == amounts.length, "ERC1155: ids and amounts length mismatch");
require(to != address(0), "ERC1155: transfer to the zero address");
address operator = _msgSender();
_beforeTokenTransfer(operator, from, to, ids, amounts, data);
for (uint256 i = 0; i < ids.length; ++i) {
uint256 id = ids[i];
uint256 amount = amounts[i];
uint256 fromBalance = _balances[id][from];
require(fromBalance >= amount, "ERC1155: insufficient balance for transfer");
unchecked {
_balances[id][from] = fromBalance - amount;
}
_balances[id][to] += amount;
}
emit TransferBatch(operator, from, to, ids, amounts);
_afterTokenTransfer(operator, from, to, ids, amounts, data);
_doSafeBatchTransferAcceptanceCheck(operator, from, to, ids, amounts, data);
}
function _setURI(string memory newuri) internal virtual {
_uri = newuri;
}
function _mint(
address to,
uint256 id,
uint256 amount,
bytes memory data
) internal virtual {
require(to != address(0), "ERC1155: mint to the zero address");
address operator = _msgSender();
uint256[] memory ids = _asSingletonArray(id);
uint256[] memory amounts = _asSingletonArray(amount);
_beforeTokenTransfer(operator, address(0), to, ids, amounts, data);
_balances[id][to] += amount;
emit TransferSingle(operator, address(0), to, id, amount);
_afterTokenTransfer(operator, address(0), to, ids, amounts, data);
_doSafeTransferAcceptanceCheck(operator, address(0), to, id, amount, data);
}
function _mintBatch(
address to,
uint256[] memory ids,
uint256[] memory amounts,
bytes memory data
) internal virtual {
require(to != address(0), "ERC1155: mint to the zero address");
require(ids.length == amounts.length, "ERC1155: ids and amounts length mismatch");
address operator = _msgSender();
_beforeTokenTransfer(operator, address(0), to, ids, amounts, data);
for (uint256 i = 0; i < ids.length; i++) {
_balances[ids[i]][to] += amounts[i];
}
emit TransferBatch(operator, address(0), to, ids, amounts);
_afterTokenTransfer(operator, address(0), to, ids, amounts, data);
_doSafeBatchTransferAcceptanceCheck(operator, address(0), to, ids, amounts, data);
}
function _burn(
address from,
uint256 id,
uint256 amount
) internal virtual {
require(from != address(0), "ERC1155: burn from the zero address");
address operator = _msgSender();
uint256[] memory ids = _asSingletonArray(id);
uint256[] memory amounts = _asSingletonArray(amount);
_beforeTokenTransfer(operator, from, address(0), ids, amounts, "");
uint256 fromBalance = _balances[id][from];
require(fromBalance >= amount, "ERC1155: burn amount exceeds balance");
unchecked {
_balances[id][from] = fromBalance - amount;
}
emit TransferSingle(operator, from, address(0), id, amount);
_afterTokenTransfer(operator, from, address(0), ids, amounts, "");
}
function _burnBatch(
address from,
uint256[] memory ids,
uint256[] memory amounts
) internal virtual {
require(from != address(0), "ERC1155: burn from the zero address");
require(ids.length == amounts.length, "ERC1155: ids and amounts length mismatch");
address operator = _msgSender();
_beforeTokenTransfer(operator, from, address(0), ids, amounts, "");
for (uint256 i = 0; i < ids.length; i++) {
uint256 id = ids[i];
uint256 amount = amounts[i];
uint256 fromBalance = _balances[id][from];
require(fromBalance >= amount, "ERC1155: burn amount exceeds balance");
unchecked {
_balances[id][from] = fromBalance - amount;
}
}
emit TransferBatch(operator, from, address(0), ids, amounts);
_afterTokenTransfer(operator, from, address(0), ids, amounts, "");
}
function _setApprovalForAll(
address owner,
address operator,
bool approved
) internal virtual {
require(owner != operator, "ERC1155: setting approval status for self");
_operatorApprovals[owner][operator] = approved;
emit ApprovalForAll(owner, operator, approved);
}
function _beforeTokenTransfer(
address operator,
address from,
address to,
uint256[] memory ids,
uint256[] memory amounts,
bytes memory data
) internal virtual {}
function _afterTokenTransfer(
address operator,
address from,
address to,
uint256[] memory ids,
uint256[] memory amounts,
bytes memory data
) internal virtual {}
function _doSafeTransferAcceptanceCheck(
address operator,
address from,
address to,
uint256 id,
uint256 amount,
bytes memory data
) private {
if (to.isContract()) {
try IERC1155Receiver(to).onERC1155Received(operator, from, id, amount, data) returns (bytes4 response) {
if (response != IERC1155Receiver.onERC1155Received.selector) {
revert("ERC1155: ERC1155Receiver rejected tokens");
}
} catch Error(string memory reason) {
revert(reason);
} catch {
revert("ERC1155: transfer to non-ERC1155Receiver implementer");
}
}
}
function _doSafeBatchTransferAcceptanceCheck(
address operator,
address from,
address to,
uint256[] memory ids,
uint256[] memory amounts,
bytes memory data
) private {
if (to.isContract()) {
try IERC1155Receiver(to).onERC1155BatchReceived(operator, from, ids, amounts, data) returns (
bytes4 response
) {
if (response != IERC1155Receiver.onERC1155BatchReceived.selector) {
revert("ERC1155: ERC1155Receiver rejected tokens");
}
} catch Error(string memory reason) {
revert(reason);
} catch {
revert("ERC1155: transfer to non-ERC1155Receiver implementer");
}
}
}
function _asSingletonArray(uint256 element) private pure returns (uint256[] memory) {
uint256[] memory array = new uint256[](1);
array[0] = element;
return array;
}
}
文件 8 的 29:ERC1155Burnable.sol
pragma solidity ^0.8.0;
import "../ERC1155.sol";
abstract contract ERC1155Burnable is ERC1155 {
function burn(
address account,
uint256 id,
uint256 value
) public virtual {
require(
account == _msgSender() || isApprovedForAll(account, _msgSender()),
"ERC1155: caller is not token owner or approved"
);
_burn(account, id, value);
}
function burnBatch(
address account,
uint256[] memory ids,
uint256[] memory values
) public virtual {
require(
account == _msgSender() || isApprovedForAll(account, _msgSender()),
"ERC1155: caller is not token owner or approved"
);
_burnBatch(account, ids, values);
}
}
文件 9 的 29:ERC1155Supply.sol
pragma solidity ^0.8.0;
import "../ERC1155.sol";
abstract contract ERC1155Supply is ERC1155 {
mapping(uint256 => uint256) private _totalSupply;
function totalSupply(uint256 id) public view virtual returns (uint256) {
return _totalSupply[id];
}
function exists(uint256 id) public view virtual returns (bool) {
return ERC1155Supply.totalSupply(id) > 0;
}
function _beforeTokenTransfer(
address operator,
address from,
address to,
uint256[] memory ids,
uint256[] memory amounts,
bytes memory data
) internal virtual override {
super._beforeTokenTransfer(operator, from, to, ids, amounts, data);
if (from == address(0)) {
for (uint256 i = 0; i < ids.length; ++i) {
_totalSupply[ids[i]] += amounts[i];
}
}
if (to == address(0)) {
for (uint256 i = 0; i < ids.length; ++i) {
uint256 id = ids[i];
uint256 amount = amounts[i];
uint256 supply = _totalSupply[id];
require(supply >= amount, "ERC1155: burn amount exceeds totalSupply");
unchecked {
_totalSupply[id] = supply - amount;
}
}
}
}
}
文件 10 的 29:ERC165.sol
pragma solidity ^0.8.0;
import "./IERC165.sol";
abstract contract ERC165 is IERC165 {
function supportsInterface(bytes4 interfaceId) public view virtual override returns (bool) {
return interfaceId == type(IERC165).interfaceId;
}
}
文件 11 的 29:IAccessControl.sol
pragma solidity ^0.8.0;
interface IAccessControl {
event RoleAdminChanged(bytes32 indexed role, bytes32 indexed previousAdminRole, bytes32 indexed newAdminRole);
event RoleGranted(bytes32 indexed role, address indexed account, address indexed sender);
event RoleRevoked(bytes32 indexed role, address indexed account, address indexed sender);
function hasRole(bytes32 role, address account) external view returns (bool);
function getRoleAdmin(bytes32 role) external view returns (bytes32);
function grantRole(bytes32 role, address account) external;
function revokeRole(bytes32 role, address account) external;
function renounceRole(bytes32 role, address account) external;
}
文件 12 的 29:IAssetAdapter.sol
pragma solidity ^0.8.24;
import {IOracle} from "src/interfaces/IOracle.sol";
import {IERC20} from "openzeppelin-contracts/contracts/token/ERC20/IERC20.sol";
interface IAssetAdapter {
event Allocate(address indexed signer, uint256 amount, uint256 timestamp);
event Withdraw(address indexed signer, uint256 amount, uint256 timestamp);
event Deposit(address indexed signer, uint256 amount, uint256 timestamp);
event Redeem(address indexed signer, uint256 amount, uint256 timestamp);
event UnderlyingRiskWeightUpdate(uint256 riskWeight, uint256 timestamp);
event FundRiskWeightUpdate(uint256 riskWeight, uint256 timestamp);
function duration() external view returns (uint256);
function underlyingPriceOracle() external view returns (IOracle);
function fundPriceOracle() external view returns (IOracle);
function underlyingRiskWeight() external view returns (uint256);
function fundRiskWeight() external view returns (uint256);
function underlying() external view returns (IERC20);
function allocate(uint256) external;
function withdraw(uint256) external;
function deposit(uint256) external;
function redeem(uint256) external;
function totalValue() external view returns (uint256);
function totalRiskValue() external view returns (uint256);
function underlyingTotalRiskValue() external view returns (uint256);
function underlyingRiskValue(uint256) external view returns (uint256);
function underlyingTotalValue() external view returns (uint256);
function underlyingValue(uint256) external view returns (uint256);
function underlyingBalance() external view returns (uint256);
function fundTotalRiskValue() external view returns (uint256);
function fundRiskValue(uint256) external view returns (uint256);
function fundTotalValue() external view returns (uint256);
function fundValue(uint256) external view returns (uint256);
function fundBalance() external view returns (uint256);
}
文件 13 的 29:ICreditEnforcer.sol
pragma solidity ^0.8.24;
import {IPegStabilityModule} from "./IPegStabilityModule.sol";
import {ITermIssuer} from "./ITermIssuer.sol";
interface ICreditEnforcer {
function mintTerm(uint256, uint256) external returns (uint256);
function psm() external view returns (IPegStabilityModule);
function termIssuer() external view returns (ITermIssuer);
}
文件 14 的 29:IERC1155.sol
pragma solidity ^0.8.0;
import "../../utils/introspection/IERC165.sol";
interface IERC1155 is IERC165 {
event TransferSingle(address indexed operator, address indexed from, address indexed to, uint256 id, uint256 value);
event TransferBatch(
address indexed operator,
address indexed from,
address indexed to,
uint256[] ids,
uint256[] values
);
event ApprovalForAll(address indexed account, address indexed operator, bool approved);
event URI(string value, uint256 indexed id);
function balanceOf(address account, uint256 id) external view returns (uint256);
function balanceOfBatch(address[] calldata accounts, uint256[] calldata ids)
external
view
returns (uint256[] memory);
function setApprovalForAll(address operator, bool approved) external;
function isApprovedForAll(address account, address operator) external view returns (bool);
function safeTransferFrom(
address from,
address to,
uint256 id,
uint256 amount,
bytes calldata data
) external;
function safeBatchTransferFrom(
address from,
address to,
uint256[] calldata ids,
uint256[] calldata amounts,
bytes calldata data
) external;
}
文件 15 的 29:IERC1155MetadataURI.sol
pragma solidity ^0.8.0;
import "../IERC1155.sol";
interface IERC1155MetadataURI is IERC1155 {
function uri(uint256 id) external view returns (string memory);
}
文件 16 的 29:IERC1155Receiver.sol
pragma solidity ^0.8.0;
import "../../utils/introspection/IERC165.sol";
interface IERC1155Receiver is IERC165 {
function onERC1155Received(
address operator,
address from,
uint256 id,
uint256 value,
bytes calldata data
) external returns (bytes4);
function onERC1155BatchReceived(
address operator,
address from,
uint256[] calldata ids,
uint256[] calldata values,
bytes calldata data
) external returns (bytes4);
}
文件 17 的 29:IERC165.sol
pragma solidity ^0.8.0;
interface IERC165 {
function supportsInterface(bytes4 interfaceId) external view returns (bool);
}
文件 18 的 29:IERC20.sol
pragma solidity ^0.8.0;
interface IERC20 {
event Transfer(address indexed from, address indexed to, uint256 value);
event Approval(address indexed owner, address indexed spender, uint256 value);
function totalSupply() external view returns (uint256);
function balanceOf(address account) external view returns (uint256);
function transfer(address to, uint256 amount) external returns (bool);
function allowance(address owner, address spender) external view returns (uint256);
function approve(address spender, uint256 amount) external returns (bool);
function transferFrom(
address from,
address to,
uint256 amount
) external returns (bool);
}
文件 19 的 29:IERC20Metadata.sol
pragma solidity ^0.8.0;
import "../IERC20.sol";
interface IERC20Metadata is IERC20 {
function name() external view returns (string memory);
function symbol() external view returns (string memory);
function decimals() external view returns (uint8);
}
文件 20 的 29:IFund.sol
import {IERC20} from "openzeppelin-contracts/contracts/token/ERC20/IERC20.sol";
pragma solidity ^0.8.24;
interface IFund is IERC20 {
function deposit(uint256) external;
function redeem(uint256) external;
function userDeposits(address) external view returns (uint256, uint256);
function userRedemptions(address) external view returns (uint256, uint256);
function currentPrice() external view returns (uint256);
}
文件 21 的 29:IOracle.sol
pragma solidity ^0.8.24;
interface IOracle {
function latestAnswer() external view returns (int256);
}
文件 22 的 29:IPegStabilityModule.sol
pragma solidity ^0.8.24;
interface IPegStabilityModule {
event Allocate(address indexed signer, uint256 amount, uint256 timestamp);
event Withdraw(address indexed signer, uint256 amount, uint256 timestamp);
event UnderlyingRiskWeightUpdate(uint256 riskWeight, uint256 timestamp);
event Mint(
address indexed from,
address indexed to,
uint256 amount,
uint256 timestamp
);
event Redeem(
address indexed from,
address indexed to,
uint256 amount,
uint256 timestamp
);
function allocate(uint256) external;
function withdraw(uint256) external;
function mint(address, address, uint256) external;
function totalValue() external view returns (uint256);
function totalRiskValue() external view returns (uint256);
function underlyingBalance() external view returns (uint256);
}
文件 23 的 29:ISavingModule.sol
pragma solidity ^0.8.24;
interface ISavingModule {
event Mint(
address indexed from,
address indexed to,
uint256 mintAmount,
uint256 burnAmount,
uint256 timestamp
);
event Redeem(
address indexed from,
address indexed to,
uint256 redeemAmount,
uint256 burnAmount,
uint256 timestamp
);
event Update(
uint256 compoundFactorAccum,
uint256 currentRate,
uint256 rate,
uint256 timestamp
);
function mint(address, address, uint256) external;
function rusdTotalLiability() external view returns (uint256);
function totalDebt() external view returns (uint256);
}
文件 24 的 29:ITerm.sol
pragma solidity ^0.8.24;
interface ITerm {
function mint(address, uint256, uint256) external;
function burn(address, uint256, uint256) external;
function totalSupply(uint256) external view returns (uint256);
}
文件 25 的 29:ITermIssuer.sol
pragma solidity ^0.8.24;
import {IToken} from "./IToken.sol";
import {ITerm} from "../Term.sol";
interface ITermIssuer {
event MintTerm(
address indexed from,
address indexed to,
uint256 indexed termId,
uint256 principle,
uint256 cost,
uint256 timestamp
);
event RedeemTerm(
address indexed from,
address indexed to,
uint256 indexed termId,
uint256 principle,
uint256 timestamp
);
function mint(
address,
address,
uint256,
uint256
) external returns (uint256);
function redeem(uint256, uint256) external;
function redeem(address, uint256, uint256) external;
function applyDiscount(
uint256,
uint256,
uint256
) external view returns (uint256);
function getDiscountRate(uint256 id) external view returns (uint256);
function latestID() external view returns (uint256);
function earliestID() external view returns (uint256);
function maturityTimestamp(uint256) external view returns (uint256);
function totalSupply(uint256) external view returns (uint256);
function totalDebt() external view returns (uint256);
function rusd() external view returns (IToken);
function term() external view returns (ITerm);
}
文件 26 的 29:IToken.sol
pragma solidity ^0.8.24;
import {IERC20} from "openzeppelin-contracts/contracts/token/ERC20/IERC20.sol";
interface IToken is IERC20 {
function mint(address, uint256) external;
function burnFrom(address, uint256) external;
}
文件 27 的 29:Math.sol
pragma solidity ^0.8.0;
library Math {
enum Rounding {
Down,
Up,
Zero
}
function max(uint256 a, uint256 b) internal pure returns (uint256) {
return a > b ? a : b;
}
function min(uint256 a, uint256 b) internal pure returns (uint256) {
return a < b ? a : b;
}
function average(uint256 a, uint256 b) internal pure returns (uint256) {
return (a & b) + (a ^ b) / 2;
}
function ceilDiv(uint256 a, uint256 b) internal pure returns (uint256) {
return a == 0 ? 0 : (a - 1) / b + 1;
}
function mulDiv(
uint256 x,
uint256 y,
uint256 denominator
) internal pure returns (uint256 result) {
unchecked {
uint256 prod0;
uint256 prod1;
assembly {
let mm := mulmod(x, y, not(0))
prod0 := mul(x, y)
prod1 := sub(sub(mm, prod0), lt(mm, prod0))
}
if (prod1 == 0) {
return prod0 / denominator;
}
require(denominator > prod1);
uint256 remainder;
assembly {
remainder := mulmod(x, y, denominator)
prod1 := sub(prod1, gt(remainder, prod0))
prod0 := sub(prod0, remainder)
}
uint256 twos = denominator & (~denominator + 1);
assembly {
denominator := div(denominator, twos)
prod0 := div(prod0, twos)
twos := add(div(sub(0, twos), twos), 1)
}
prod0 |= prod1 * twos;
uint256 inverse = (3 * denominator) ^ 2;
inverse *= 2 - denominator * inverse;
inverse *= 2 - denominator * inverse;
inverse *= 2 - denominator * inverse;
inverse *= 2 - denominator * inverse;
inverse *= 2 - denominator * inverse;
inverse *= 2 - denominator * inverse;
result = prod0 * inverse;
return result;
}
}
function mulDiv(
uint256 x,
uint256 y,
uint256 denominator,
Rounding rounding
) internal pure returns (uint256) {
uint256 result = mulDiv(x, y, denominator);
if (rounding == Rounding.Up && mulmod(x, y, denominator) > 0) {
result += 1;
}
return result;
}
function sqrt(uint256 a) internal pure returns (uint256) {
if (a == 0) {
return 0;
}
uint256 result = 1 << (log2(a) >> 1);
unchecked {
result = (result + a / result) >> 1;
result = (result + a / result) >> 1;
result = (result + a / result) >> 1;
result = (result + a / result) >> 1;
result = (result + a / result) >> 1;
result = (result + a / result) >> 1;
result = (result + a / result) >> 1;
return min(result, a / result);
}
}
function sqrt(uint256 a, Rounding rounding) internal pure returns (uint256) {
unchecked {
uint256 result = sqrt(a);
return result + (rounding == Rounding.Up && result * result < a ? 1 : 0);
}
}
function log2(uint256 value) internal pure returns (uint256) {
uint256 result = 0;
unchecked {
if (value >> 128 > 0) {
value >>= 128;
result += 128;
}
if (value >> 64 > 0) {
value >>= 64;
result += 64;
}
if (value >> 32 > 0) {
value >>= 32;
result += 32;
}
if (value >> 16 > 0) {
value >>= 16;
result += 16;
}
if (value >> 8 > 0) {
value >>= 8;
result += 8;
}
if (value >> 4 > 0) {
value >>= 4;
result += 4;
}
if (value >> 2 > 0) {
value >>= 2;
result += 2;
}
if (value >> 1 > 0) {
result += 1;
}
}
return result;
}
function log2(uint256 value, Rounding rounding) internal pure returns (uint256) {
unchecked {
uint256 result = log2(value);
return result + (rounding == Rounding.Up && 1 << result < value ? 1 : 0);
}
}
function log10(uint256 value) internal pure returns (uint256) {
uint256 result = 0;
unchecked {
if (value >= 10**64) {
value /= 10**64;
result += 64;
}
if (value >= 10**32) {
value /= 10**32;
result += 32;
}
if (value >= 10**16) {
value /= 10**16;
result += 16;
}
if (value >= 10**8) {
value /= 10**8;
result += 8;
}
if (value >= 10**4) {
value /= 10**4;
result += 4;
}
if (value >= 10**2) {
value /= 10**2;
result += 2;
}
if (value >= 10**1) {
result += 1;
}
}
return result;
}
function log10(uint256 value, Rounding rounding) internal pure returns (uint256) {
unchecked {
uint256 result = log10(value);
return result + (rounding == Rounding.Up && 10**result < value ? 1 : 0);
}
}
function log256(uint256 value) internal pure returns (uint256) {
uint256 result = 0;
unchecked {
if (value >> 128 > 0) {
value >>= 128;
result += 16;
}
if (value >> 64 > 0) {
value >>= 64;
result += 8;
}
if (value >> 32 > 0) {
value >>= 32;
result += 4;
}
if (value >> 16 > 0) {
value >>= 16;
result += 2;
}
if (value >> 8 > 0) {
result += 1;
}
}
return result;
}
function log256(uint256 value, Rounding rounding) internal pure returns (uint256) {
unchecked {
uint256 result = log256(value);
return result + (rounding == Rounding.Up && 1 << (result * 8) < value ? 1 : 0);
}
}
}
文件 28 的 29:Strings.sol
pragma solidity ^0.8.0;
import "./math/Math.sol";
library Strings {
bytes16 private constant _SYMBOLS = "0123456789abcdef";
uint8 private constant _ADDRESS_LENGTH = 20;
function toString(uint256 value) internal pure returns (string memory) {
unchecked {
uint256 length = Math.log10(value) + 1;
string memory buffer = new string(length);
uint256 ptr;
assembly {
ptr := add(buffer, add(32, length))
}
while (true) {
ptr--;
assembly {
mstore8(ptr, byte(mod(value, 10), _SYMBOLS))
}
value /= 10;
if (value == 0) break;
}
return buffer;
}
}
function toHexString(uint256 value) internal pure returns (string memory) {
unchecked {
return toHexString(value, Math.log256(value) + 1);
}
}
function toHexString(uint256 value, uint256 length) internal pure returns (string memory) {
bytes memory buffer = new bytes(2 * length + 2);
buffer[0] = "0";
buffer[1] = "x";
for (uint256 i = 2 * length + 1; i > 1; --i) {
buffer[i] = _SYMBOLS[value & 0xf];
value >>= 4;
}
require(value == 0, "Strings: hex length insufficient");
return string(buffer);
}
function toHexString(address addr) internal pure returns (string memory) {
return toHexString(uint256(uint160(addr)), _ADDRESS_LENGTH);
}
}
文件 29 的 29:Term.sol
pragma solidity ^0.8.24;
import {AccessControl} from "openzeppelin-contracts/contracts/access/AccessControl.sol";
import {ITerm} from "src/interfaces/ITerm.sol";
import {ERC1155} from "openzeppelin-contracts/contracts/token/ERC1155/ERC1155.sol";
import {ERC1155Supply} from "openzeppelin-contracts/contracts/token/ERC1155/extensions/ERC1155Supply.sol";
import {ERC1155Burnable} from "openzeppelin-contracts/contracts/token/ERC1155/extensions/ERC1155Burnable.sol";
contract Term is AccessControl, ERC1155Supply, ERC1155Burnable {
bytes32 public constant MINTER = keccak256(abi.encode("term.minter"));
constructor(address admin, string memory uri) ERC1155(uri) {
_grantRole(DEFAULT_ADMIN_ROLE, admin);
}
function mint(
address to,
uint256 id,
uint256 amount
) external onlyRole(MINTER) {
_mint(to, id, amount, "");
}
function mintBatch(
address to,
uint256[] memory ids,
uint256[] memory amounts
) external onlyRole(MINTER) {
_mintBatch(to, ids, amounts, "");
}
function supportsInterface(
bytes4 interfaceId
) public view virtual override(ERC1155, AccessControl) returns (bool) {
return super.supportsInterface(interfaceId);
}
function _beforeTokenTransfer(
address operator,
address from,
address to,
uint256[] memory ids,
uint256[] memory amounts,
bytes memory data
) internal override(ERC1155, ERC1155Supply) {
ERC1155Supply._beforeTokenTransfer(
operator,
from,
to,
ids,
amounts,
data
);
}
}
{
"compilationTarget": {
"src/CreditEnforcer.sol": "CreditEnforcer"
},
"evmVersion": "paris",
"libraries": {},
"metadata": {
"bytecodeHash": "ipfs"
},
"optimizer": {
"enabled": true,
"runs": 200
},
"remappings": [
":chainlink/=lib/chainlink/",
":ds-test/=lib/forge-std/lib/ds-test/src/",
":erc4626-tests/=lib/offchain-fund/lib/openzeppelin-contracts/lib/erc4626-tests/",
":forge-std/=lib/forge-std/src/",
":offchain-fund/=lib/offchain-fund/",
":openzeppelin-contracts/=lib/openzeppelin-contracts/"
]
}
[{"inputs":[{"internalType":"address","name":"admin","type":"address"},{"internalType":"contract IERC20","name":"underlying_","type":"address"},{"internalType":"contract ITermIssuer","name":"termIssuer_","type":"address"},{"internalType":"contract IPegStabilityModule","name":"psm_","type":"address"},{"internalType":"contract ISavingModule","name":"sm_","type":"address"}],"stateMutability":"nonpayable","type":"constructor"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"bytes32","name":"role","type":"bytes32"},{"indexed":true,"internalType":"bytes32","name":"previousAdminRole","type":"bytes32"},{"indexed":true,"internalType":"bytes32","name":"newAdminRole","type":"bytes32"}],"name":"RoleAdminChanged","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"bytes32","name":"role","type":"bytes32"},{"indexed":true,"internalType":"address","name":"account","type":"address"},{"indexed":true,"internalType":"address","name":"sender","type":"address"}],"name":"RoleGranted","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"bytes32","name":"role","type":"bytes32"},{"indexed":true,"internalType":"address","name":"account","type":"address"},{"indexed":true,"internalType":"address","name":"sender","type":"address"}],"name":"RoleRevoked","type":"event"},{"inputs":[],"name":"DEFAULT_ADMIN_ROLE","outputs":[{"internalType":"bytes32","name":"","type":"bytes32"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"MANAGER","outputs":[{"internalType":"bytes32","name":"","type":"bytes32"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"SUPERVISOR","outputs":[{"internalType":"bytes32","name":"","type":"bytes32"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"adapter","type":"address"}],"name":"addAssetAdapter","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"index","type":"uint256"},{"internalType":"uint256","name":"amount","type":"uint256"}],"name":"allocate","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"assetAdapterLength","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"}],"name":"assetAdapterList","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"assetAdapterMap","outputs":[{"internalType":"bool","name":"set","type":"bool"},{"internalType":"uint256","name":"index","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"assetRatio","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"assetRatioMin","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"assets","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"amount","type":"uint256"}],"name":"checkPSMDebtMax","outputs":[{"internalType":"bool","name":"","type":"bool"},{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"checkRatios","outputs":[{"internalType":"bool","name":"","type":"bool"},{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"amount","type":"uint256"}],"name":"checkSMDebtMax","outputs":[{"internalType":"bool","name":"","type":"bool"},{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"id","type":"uint256"},{"internalType":"uint256","name":"amount","type":"uint256"}],"name":"checkTermDebtMax","outputs":[{"internalType":"bool","name":"","type":"bool"},{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"index","type":"uint256"},{"internalType":"uint256","name":"amount","type":"uint256"}],"name":"deposit","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"duration","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"equity","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"equityRatio","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"equityRatioMin","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"extendedAssets","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"duration_","type":"uint256"}],"name":"extendedLiabilities","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"adapter","type":"address"}],"name":"getAssetAdapter","outputs":[{"components":[{"internalType":"bool","name":"set","type":"bool"},{"internalType":"uint256","name":"index","type":"uint256"}],"internalType":"struct CreditEnforcer.AssetAdapter","name":"","type":"tuple"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"startIndex","type":"uint256"},{"internalType":"uint256","name":"length","type":"uint256"}],"name":"getAssetAdapterList","outputs":[{"internalType":"address[]","name":"","type":"address[]"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"bytes32","name":"role","type":"bytes32"}],"name":"getRoleAdmin","outputs":[{"internalType":"bytes32","name":"","type":"bytes32"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"id","type":"uint256"}],"name":"getTermDebtMax","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"bytes32","name":"role","type":"bytes32"},{"internalType":"address","name":"account","type":"address"}],"name":"grantRole","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"bytes32","name":"role","type":"bytes32"},{"internalType":"address","name":"account","type":"address"}],"name":"hasRole","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"liabilities","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"liquidityRatio","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"liquidityRatioMin","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"amount","type":"uint256"}],"name":"mintSavingcoin","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"amount","type":"uint256"}],"name":"mintSavingcoin","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"amount","type":"uint256"}],"name":"mintStablecoin","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"amount","type":"uint256"}],"name":"mintStablecoin","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"id","type":"uint256"},{"internalType":"uint256","name":"amount","type":"uint256"}],"name":"mintTerm","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"id","type":"uint256"},{"internalType":"uint256","name":"amount","type":"uint256"}],"name":"mintTerm","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"psm","outputs":[{"internalType":"contract IPegStabilityModule","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"psmDebtMax","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"index","type":"uint256"},{"internalType":"uint256","name":"amount","type":"uint256"}],"name":"redeem","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"adapter","type":"address"}],"name":"removeAssetAdapter","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"bytes32","name":"role","type":"bytes32"},{"internalType":"address","name":"account","type":"address"}],"name":"renounceRole","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"bytes32","name":"role","type":"bytes32"},{"internalType":"address","name":"account","type":"address"}],"name":"revokeRole","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"riskWeightedAssets","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"assetRatioMin_","type":"uint256"}],"name":"setAssetRatioMin","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"duration_","type":"uint256"}],"name":"setDuration","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"equityRatioMin_","type":"uint256"}],"name":"setEquityRatioMin","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"liquidityRatioMin_","type":"uint256"}],"name":"setLiquidityRatioMin","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"psmDebtMax_","type":"uint256"}],"name":"setPSMDebtMax","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"smDebtMax_","type":"uint256"}],"name":"setSMDebtMax","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"id","type":"uint256"},{"internalType":"uint256","name":"amount","type":"uint256"}],"name":"setTermDebtMax","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"shortTermAssets","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"shortTermLiabilities","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"sm","outputs":[{"internalType":"contract ISavingModule","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"smDebtMax","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"bytes4","name":"interfaceId","type":"bytes4"}],"name":"supportsInterface","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"}],"name":"termDebtMax","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"termIssuer","outputs":[{"internalType":"contract ITermIssuer","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"underlying","outputs":[{"internalType":"contract IERC20","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"index","type":"uint256"},{"internalType":"uint256","name":"amount","type":"uint256"}],"name":"withdraw","outputs":[],"stateMutability":"nonpayable","type":"function"}]