More Info
Private Name Tags
ContractCreator
TokenTracker
Latest 25 from a total of 2,165 transactions
Transaction Hash |
Method
|
Block
|
From
|
To
|
|||||
---|---|---|---|---|---|---|---|---|---|
Emit Loan | 21704034 | 1 hr ago | IN | 0 ETH | 0.00137406 | ||||
Refinance Full | 21704004 | 1 hr ago | IN | 0 ETH | 0.00057465 | ||||
Refinance From L... | 21703611 | 2 hrs ago | IN | 0 ETH | 0.00095356 | ||||
Emit Loan | 21703598 | 2 hrs ago | IN | 0 ETH | 0.00114544 | ||||
Emit Loan | 21703203 | 4 hrs ago | IN | 0 ETH | 0.00157806 | ||||
Refinance Full | 21702934 | 5 hrs ago | IN | 0 ETH | 0.00101646 | ||||
Refinance Full | 21702931 | 5 hrs ago | IN | 0 ETH | 0.00101199 | ||||
Emit Loan | 21702703 | 5 hrs ago | IN | 0 ETH | 0.00217824 | ||||
Repay Loan | 21702701 | 5 hrs ago | IN | 0 ETH | 0.00172588 | ||||
Emit Loan | 21702699 | 5 hrs ago | IN | 0 ETH | 0.00205616 | ||||
Repay Loan | 21702696 | 5 hrs ago | IN | 0 ETH | 0.00144813 | ||||
Emit Loan | 21702683 | 5 hrs ago | IN | 0 ETH | 0.00228174 | ||||
Refinance Full | 21702344 | 7 hrs ago | IN | 0 ETH | 0.0009907 | ||||
Refinance Full | 21702336 | 7 hrs ago | IN | 0 ETH | 0.00110504 | ||||
Repay Loan | 21701950 | 8 hrs ago | IN | 0 ETH | 0.00065136 | ||||
Repay Loan | 21701948 | 8 hrs ago | IN | 0 ETH | 0.00071293 | ||||
Repay Loan | 21701946 | 8 hrs ago | IN | 0 ETH | 0.00070705 | ||||
Repay Loan | 21701945 | 8 hrs ago | IN | 0 ETH | 0.00069175 | ||||
Repay Loan | 21701943 | 8 hrs ago | IN | 0 ETH | 0.00065635 | ||||
Repay Loan | 21701939 | 8 hrs ago | IN | 0 ETH | 0.00098094 | ||||
Repay Loan | 21701935 | 8 hrs ago | IN | 0 ETH | 0.00063404 | ||||
Repay Loan | 21701928 | 8 hrs ago | IN | 0 ETH | 0.00064425 | ||||
Repay Loan | 21701926 | 8 hrs ago | IN | 0 ETH | 0.00065922 | ||||
Repay Loan | 21701881 | 8 hrs ago | IN | 0 ETH | 0.00071458 | ||||
Refinance Full | 21701768 | 8 hrs ago | IN | 0 ETH | 0.00062882 |
View more zero value Internal Transactions in Advanced View mode
Advanced mode:
Loading...
Loading
This contract may be a proxy contract. Click on More Options and select Is this a proxy? to confirm and enable the "Read as Proxy" & "Write as Proxy" tabs.
Similar Match Source Code This contract matches the deployed Bytecode of the Source Code for Contract 0xDFf8873f...f55859634 The constructor portion of the code might be different and could alter the actual behaviour of the contract
Contract Name:
MultiSourceLoan
Compiler Version
v0.8.21+commit.d9974bed
Contract Source Code (Solidity Standard Json-Input format)
// SPDX-License-Identifier: AGPL-3.0 pragma solidity ^0.8.21; import "@delegate/IDelegateRegistry.sol"; import "@openzeppelin/utils/cryptography/ECDSA.sol"; import "@solmate/tokens/ERC20.sol"; import "@solmate/tokens/ERC721.sol"; import "@solmate/utils/FixedPointMathLib.sol"; import "@solmate/utils/ReentrancyGuard.sol"; import "@solmate/utils/SafeTransferLib.sol"; import "../../interfaces/validators/IOfferValidator.sol"; import "../../interfaces/INFTFlashAction.sol"; import "../../interfaces/loans/ILoanManager.sol"; import "../../interfaces/loans/ILoanManagerRegistry.sol"; import "../../interfaces/loans/IMultiSourceLoan.sol"; import "../utils/Hash.sol"; import "../utils/Interest.sol"; import "../Multicall.sol"; import "./BaseLoan.sol"; /// @title MultiSourceLoan (v3) /// @author Florida St /// @notice Loan contract that allows for multiple tranches with different /// seniorities. Each loan is collateralized by an NFT. Loans have a duration, /// principal, and APR. Loans can be refinanced automatically by lenders (if terms /// are improved). Borrowers can also get renegotiation offers which they can then /// accept. If a loan is not repaid by its end time, it's considered to have defaulted. /// If it had only one lender behind it, then the lender (unless it's a pool), can claim /// the collateral. If there are multiple lenders or the sole lender is a pool, then there's /// a liquidation process (run by an instance of `ILoanLiquidator`). contract MultiSourceLoan is IMultiSourceLoan, Multicall, ReentrancyGuard, BaseLoan { using FixedPointMathLib for uint256; using Hash for ExecutionData; using Hash for Loan; using Hash for LoanOffer; using Hash for SignableRepaymentData; using Hash for RenegotiationOffer; using InputChecker for address; using Interest for uint256; using ECDSA for bytes32; using MessageHashUtils for bytes32; using SafeTransferLib for ERC20; /// @notice Loan Id to hash mapping(uint256 loanId => bytes32 loanHash) private _loans; /// This is used in _getMinTranchePrincipal. uint256 private constant _MAX_RATIO_TRANCHE_MIN_PRINCIPAL = 2; /// @notice Maximum number of tranches per loan uint256 public immutable getMaxTranches; /// @notice delegate registry address public immutable getDelegateRegistry; /// @notice Contract to execute flash actions. address public getFlashActionContract; /// @notice Loan manager registry (we currently have Gondi's pools) ILoanManagerRegistry public immutable getLoanManagerRegistry; /// @notice Min lock period for a tranche uint256 private _minLockPeriod; error InvalidParametersError(); error MismatchError(); error InvalidCollateralIdError(); error InvalidMethodError(); error InvalidAddressesError(); error InvalidCallerError(); error InvalidTrancheError(); error InvalidRenegotiationOfferError(); error TooManyTranchesError(); error LoanExpiredError(); error NFTNotReturnedError(); error TrancheCannotBeRefinancedError(uint256 minTimestamp); error LoanLockedError(); /// @param loanLiquidator Address of the liquidator contract. /// @param protocolFee Protocol fee charged on gains. /// @param currencyManager Address of the currency manager. /// @param collectionManager Address of the collection manager. /// @param maxTranches Maximum number of tranches per loan. /// @param minLockPeriod Minimum lock period for a tranche/loan. /// @param delegateRegistry Address of the delegate registry (Delegate.xyz). /// @param loanManagerRegistry Address of the loan manager registry. /// @param flashActionContract Address of the flash action contract. /// @param minWaitTime The time to wait before a new owner can be set. constructor( address loanLiquidator, ProtocolFee memory protocolFee, address currencyManager, address collectionManager, uint256 maxTranches, uint256 minLockPeriod, address delegateRegistry, address loanManagerRegistry, address flashActionContract, uint256 minWaitTime ) BaseLoan( "GONDI_MULTI_SOURCE_LOAN", currencyManager, collectionManager, protocolFee, loanLiquidator, tx.origin, minWaitTime ) { loanLiquidator.checkNotZero(); _minLockPeriod = minLockPeriod; getMaxTranches = maxTranches; getDelegateRegistry = delegateRegistry; getFlashActionContract = flashActionContract; getLoanManagerRegistry = ILoanManagerRegistry(loanManagerRegistry); } /// @inheritdoc IMultiSourceLoan function emitLoan(LoanExecutionData calldata _loanExecutionData) external nonReentrant returns (uint256, Loan memory) { address borrower = _loanExecutionData.borrower; ExecutionData calldata executionData = _loanExecutionData.executionData; (address principalAddress, address nftCollateralAddress) = _getAddressesFromExecutionData(executionData); OfferExecution[] calldata offerExecution = executionData.offerExecution; _validateExecutionData(_loanExecutionData, borrower); _checkWhitelists(principalAddress, nftCollateralAddress); (uint256 loanId, uint256[] memory offerIds, Loan memory loan, uint256 totalFee) = _processOffersFromExecutionData( borrower, executionData.principalReceiver, principalAddress, nftCollateralAddress, executionData.tokenId, executionData.duration, offerExecution ); if (_hasCallback(executionData.callbackData)) { handleAfterPrincipalTransferCallback(loan, msg.sender, executionData.callbackData, totalFee); } ERC721(nftCollateralAddress).transferFrom(borrower, address(this), executionData.tokenId); _loans[loanId] = loan.hash(); emit LoanEmitted(loanId, offerIds, loan, totalFee); return (loanId, loan); } /// @inheritdoc IMultiSourceLoan function refinanceFull( RenegotiationOffer calldata _renegotiationOffer, Loan memory _loan, bytes calldata _renegotiationOfferSignature ) external nonReentrant returns (uint256, Loan memory) { _baseLoanChecks(_renegotiationOffer.loanId, _loan); _baseRenegotiationChecks(_renegotiationOffer, _loan); if (_renegotiationOffer.trancheIndex.length != _loan.tranche.length) { revert InvalidRenegotiationOfferError(); } bool lenderInitiated = msg.sender == _renegotiationOffer.lender; uint256 netNewLender = _renegotiationOffer.principalAmount - _renegotiationOffer.fee; uint256 totalAccruedInterest; uint256 totalAnnualInterest; /// @dev If it's lender initiated, needs to be strictly better. if (lenderInitiated) { (totalAccruedInterest, totalAnnualInterest,) = _processOldTranchesFull(_renegotiationOffer, _loan, lenderInitiated, 0); if (_isLoanLocked(_loan.startTime, _loan.duration)) { revert LoanLockedError(); } _checkStrictlyBetter( _renegotiationOffer.principalAmount, _loan.principalAmount, _renegotiationOffer.duration + block.timestamp, _loan.duration + _loan.startTime, _renegotiationOffer.aprBps, totalAnnualInterest / _loan.principalAmount, _renegotiationOffer.fee ); if (_renegotiationOffer.principalAmount > _loan.principalAmount) { ERC20(_loan.principalAddress).safeTransferFrom( _renegotiationOffer.lender, _loan.borrower, _renegotiationOffer.principalAmount - _loan.principalAmount ); } } else if (msg.sender != _loan.borrower) { revert InvalidCallerError(); } else { (totalAccruedInterest, totalAnnualInterest, netNewLender) = _processOldTranchesFull(_renegotiationOffer, _loan, lenderInitiated, netNewLender); /// @notice Borrowers clears interest _checkSignature(_renegotiationOffer.lender, _renegotiationOffer.hash(), _renegotiationOfferSignature); if (netNewLender > 0) { ERC20(_loan.principalAddress).safeTransferFrom(_renegotiationOffer.lender, _loan.borrower, netNewLender); } totalAccruedInterest = 0; } uint256 newLoanId = _getAndSetNewLoanId(); Tranche[] memory newTranche = new Tranche[](1); newTranche[0] = Tranche( newLoanId, 0, _renegotiationOffer.principalAmount, _renegotiationOffer.lender, totalAccruedInterest, block.timestamp, _renegotiationOffer.aprBps ); _loan.tranche = newTranche; _loan.startTime = block.timestamp; _loan.duration = _renegotiationOffer.duration; _loan.principalAmount = _renegotiationOffer.principalAmount; _loans[newLoanId] = _loan.hash(); delete _loans[_renegotiationOffer.loanId]; emit LoanRefinanced( _renegotiationOffer.renegotiationId, _renegotiationOffer.loanId, newLoanId, _loan, _renegotiationOffer.fee ); return (newLoanId, _loan); } /// @inheritdoc IMultiSourceLoan function refinancePartial(RenegotiationOffer calldata _renegotiationOffer, Loan memory _loan) external nonReentrant returns (uint256, Loan memory) { if (msg.sender != _renegotiationOffer.lender) { revert InvalidCallerError(); } if (_isLoanLocked(_loan.startTime, _loan.duration)) { revert LoanLockedError(); } if (_renegotiationOffer.trancheIndex.length == 0) { revert InvalidRenegotiationOfferError(); } uint256 loanId = _renegotiationOffer.loanId; _baseLoanChecks(loanId, _loan); _baseRenegotiationChecks(_renegotiationOffer, _loan); uint256 newLoanId = _getAndSetNewLoanId(); uint256 totalProtocolFee; uint256 totalAnnualInterest; uint256 totalRefinanced; /// @dev bring to mem uint256 minImprovementApr = _minImprovementApr; /// @dev We iterate over all tranches to execute repayments. uint256 totalTranchesRenegotiated = _renegotiationOffer.trancheIndex.length; for (uint256 i; i < totalTranchesRenegotiated;) { uint256 index = _renegotiationOffer.trancheIndex[i]; if (index >= _loan.tranche.length) { revert InvalidRenegotiationOfferError(); } Tranche memory tranche = _loan.tranche[index]; _checkTrancheStrictly(true, tranche.aprBps, _renegotiationOffer.aprBps, minImprovementApr); (uint256 accruedInterest, uint256 thisProtocolFee,) = _processOldTranche( _renegotiationOffer.lender, _loan.borrower, _loan.principalAddress, tranche, _loan.startTime + _loan.duration, _loan.protocolFee, type(uint256).max ); unchecked { totalRefinanced += tranche.principalAmount; totalAnnualInterest += tranche.principalAmount * tranche.aprBps; totalProtocolFee += thisProtocolFee; } tranche.loanId = newLoanId; tranche.lender = _renegotiationOffer.lender; tranche.accruedInterest = accruedInterest; tranche.startTime = block.timestamp; tranche.aprBps = _renegotiationOffer.aprBps; unchecked { ++i; } } if (_renegotiationOffer.principalAmount != totalRefinanced) { revert InvalidRenegotiationOfferError(); } _handleProtocolFeeForFee( _loan.principalAddress, _renegotiationOffer.lender, totalProtocolFee, _protocolFee.recipient ); _loans[newLoanId] = _loan.hash(); delete _loans[loanId]; /// @dev Here reneg fee is always 0 emit LoanRefinanced(_renegotiationOffer.renegotiationId, loanId, newLoanId, _loan, 0); return (newLoanId, _loan); } /// @inheritdoc IMultiSourceLoan function refinanceFromLoanExecutionData( uint256 _loanId, Loan calldata _loan, LoanExecutionData calldata _loanExecutionData ) external nonReentrant returns (uint256, Loan memory) { if (msg.sender != _loan.borrower) { revert InvalidCallerError(); } _baseLoanChecks(_loanId, _loan); ExecutionData calldata executionData = _loanExecutionData.executionData; /// @dev We ignore the borrower in executionData, and used existing one. address borrower = _loan.borrower; (address principalAddress, address nftCollateralAddress) = _getAddressesFromExecutionData(executionData); OfferExecution[] calldata offerExecution = executionData.offerExecution; _validateExecutionData(_loanExecutionData, borrower); _checkWhitelists(principalAddress, nftCollateralAddress); if (_loan.principalAddress != principalAddress || _loan.nftCollateralAddress != nftCollateralAddress) { revert InvalidAddressesError(); } if (_loan.nftCollateralTokenId != executionData.tokenId) { revert InvalidCollateralIdError(); } /// @dev We first process the incoming offers so borrower gets the capital. After that, we process repayments. /// NFT doesn't need to be transfered (it was already in escrow) (uint256 newLoanId, uint256[] memory offerIds, Loan memory loan, uint256 totalFee) = _processOffersFromExecutionData( borrower, executionData.principalReceiver, principalAddress, nftCollateralAddress, executionData.tokenId, executionData.duration, offerExecution ); _processRepayments(_loan); emit LoanRefinancedFromNewOffers(_loanId, newLoanId, loan, offerIds, totalFee); _loans[newLoanId] = loan.hash(); delete _loans[_loanId]; return (newLoanId, loan); } /// @inheritdoc IMultiSourceLoan function addNewTranche( RenegotiationOffer calldata _renegotiationOffer, Loan memory _loan, bytes calldata _renegotiationOfferSignature ) external nonReentrant returns (uint256, Loan memory) { if (msg.sender != _loan.borrower) { revert InvalidCallerError(); } uint256 loanId = _renegotiationOffer.loanId; _baseLoanChecks(loanId, _loan); _baseRenegotiationChecks(_renegotiationOffer, _loan); _checkSignature(_renegotiationOffer.lender, _renegotiationOffer.hash(), _renegotiationOfferSignature); if (_renegotiationOffer.trancheIndex.length != 1 || _renegotiationOffer.trancheIndex[0] != _loan.tranche.length) { revert InvalidRenegotiationOfferError(); } if (_loan.tranche.length == getMaxTranches) { revert TooManyTranchesError(); } uint256 newLoanId = _getAndSetNewLoanId(); Loan memory loanWithTranche = _addNewTranche(newLoanId, _loan, _renegotiationOffer); _loans[newLoanId] = loanWithTranche.hash(); delete _loans[loanId]; ERC20(_loan.principalAddress).safeTransferFrom( _renegotiationOffer.lender, _loan.borrower, _renegotiationOffer.principalAmount - _renegotiationOffer.fee ); if (_renegotiationOffer.fee != 0) { /// @dev Cached ERC20(_loan.principalAddress).safeTransferFrom( _renegotiationOffer.lender, _protocolFee.recipient, _renegotiationOffer.fee.mulDivUp(_loan.protocolFee, _PRECISION) ); } emit LoanRefinanced( _renegotiationOffer.renegotiationId, loanId, newLoanId, loanWithTranche, _renegotiationOffer.fee ); return (newLoanId, loanWithTranche); } /// @inheritdoc IMultiSourceLoan function repayLoan(LoanRepaymentData calldata _repaymentData) external override nonReentrant { uint256 loanId = _repaymentData.data.loanId; Loan calldata loan = _repaymentData.loan; /// @dev If the caller is not the borrower itself, check the signature to avoid someone else forcing an unwanted repayment. if (msg.sender != loan.borrower) { _checkSignature(loan.borrower, _repaymentData.data.hash(), _repaymentData.borrowerSignature); } _baseLoanChecks(loanId, loan); /// @dev Unlikely this is used outside of the callback with a seaport sell, but leaving here in case that's not correct. if (_repaymentData.data.shouldDelegate) { IDelegateRegistry(getDelegateRegistry).delegateERC721( loan.borrower, loan.nftCollateralAddress, loan.nftCollateralTokenId, bytes32(""), true ); } ERC721(loan.nftCollateralAddress).transferFrom(address(this), loan.borrower, loan.nftCollateralTokenId); /// @dev After returning the NFT to the borrower, check if there's an action to be taken (eg: sell it to cover repayment). if (_hasCallback(_repaymentData.data.callbackData)) { handleAfterNFTTransferCallback(loan, msg.sender, _repaymentData.data.callbackData); } (uint256 totalRepayment, uint256 totalProtocolFee) = _processRepayments(loan); emit LoanRepaid(loanId, totalRepayment, totalProtocolFee); /// @dev Reclaim space. delete _loans[loanId]; } /// @inheritdoc IMultiSourceLoan function liquidateLoan(uint256 _loanId, Loan calldata _loan) external override nonReentrant returns (bytes memory) { if (_loan.hash() != _loans[_loanId]) { revert InvalidLoanError(_loanId); } (bool liquidated, bytes memory liquidation) = _liquidateLoan( _loanId, _loan, _loan.tranche.length == 1 && !getLoanManagerRegistry.isLoanManager(_loan.tranche[0].lender) ); if (liquidated) { delete _loans[_loanId]; } return liquidation; } /// @inheritdoc IMultiSourceLoan function loanLiquidated(uint256 _loanId, Loan calldata _loan) external override onlyLiquidator { if (_loan.hash() != _loans[_loanId]) { revert InvalidLoanError(_loanId); } emit LoanLiquidated(_loanId); /// @dev Reclaim space. delete _loans[_loanId]; } /// @inheritdoc IMultiSourceLoan function delegate(uint256 _loanId, Loan calldata loan, address _delegate, bytes32 _rights, bool _value) external { if (loan.hash() != _loans[_loanId]) { revert InvalidLoanError(_loanId); } if (msg.sender != loan.borrower) { revert InvalidCallerError(); } IDelegateRegistry(getDelegateRegistry).delegateERC721( _delegate, loan.nftCollateralAddress, loan.nftCollateralTokenId, _rights, _value ); emit Delegated(_loanId, _delegate, _rights, _value); } /// @inheritdoc IMultiSourceLoan function revokeDelegate(address _delegate, address _collection, uint256 _tokenId, bytes32 _rights) external { if (ERC721(_collection).ownerOf(_tokenId) == address(this)) { revert InvalidMethodError(); } IDelegateRegistry(getDelegateRegistry).delegateERC721(_delegate, _collection, _tokenId, _rights, false); emit RevokeDelegate(_delegate, _collection, _tokenId, _rights); } /// @inheritdoc IMultiSourceLoan function getMinLockPeriod() external view returns (uint256) { return _minLockPeriod; } /// @inheritdoc IMultiSourceLoan function setMinLockPeriod(uint256 __minLockPeriod) external onlyOwner { _minLockPeriod = __minLockPeriod; emit MinLockPeriodUpdated(__minLockPeriod); } /// @inheritdoc IMultiSourceLoan function getLoanHash(uint256 _loanId) external view returns (bytes32) { return _loans[_loanId]; } /// @inheritdoc IMultiSourceLoan function executeFlashAction(uint256 _loanId, Loan calldata _loan, address _target, bytes calldata _data) external nonReentrant { if (_loan.hash() != _loans[_loanId]) { revert InvalidLoanError(_loanId); } if (msg.sender != _loan.borrower) { revert InvalidCallerError(); } address flashActionContract = getFlashActionContract; ERC721(_loan.nftCollateralAddress).transferFrom(address(this), flashActionContract, _loan.nftCollateralTokenId); INFTFlashAction(flashActionContract).execute( _loan.nftCollateralAddress, _loan.nftCollateralTokenId, _target, _data ); if (ERC721(_loan.nftCollateralAddress).ownerOf(_loan.nftCollateralTokenId) != address(this)) { revert NFTNotReturnedError(); } emit FlashActionExecuted(_loanId, _target, _data); } /// @inheritdoc IMultiSourceLoan function setFlashActionContract(address _newFlashActionContract) external onlyOwner { getFlashActionContract = _newFlashActionContract; emit FlashActionContractUpdated(_newFlashActionContract); } /// @notice Process repayments for tranches upon a full renegotiation. /// @param _renegotiationOffer The renegotiation offer. /// @param _loan The loan to be processed. /// @param _isStrictlyBetter Whether the new tranche needs to be strictly better than all previous ones. /// @param _remainingNewLender Amount left for new lender to pay function _processOldTranchesFull( RenegotiationOffer calldata _renegotiationOffer, Loan memory _loan, bool _isStrictlyBetter, uint256 _remainingNewLender ) private returns (uint256 totalAccruedInterest, uint256 totalAnnualInterest, uint256 remainingNewLender) { uint256 totalProtocolFee = _renegotiationOffer.fee.mulDivUp(_loan.protocolFee, _PRECISION); unchecked { _remainingNewLender += totalProtocolFee; /// @dev bring to mem uint256 minImprovementApr = _minImprovementApr; remainingNewLender = _isStrictlyBetter ? type(uint256).max : _remainingNewLender; // We iterate first for the new lender and then for the rest. // This way if he is owed some principal, // it's discounted before transfering to the other lenders for (uint256 i = 0; i < _loan.tranche.length << 1;) { Tranche memory tranche = _loan.tranche[i % _loan.tranche.length]; bool onlyNewLenderPass = i < _loan.tranche.length; bool isNewLender = tranche.lender == _renegotiationOffer.lender; ++i; if (onlyNewLenderPass != isNewLender) continue; uint256 accruedInterest; uint256 thisProtocolFee; (accruedInterest, thisProtocolFee, remainingNewLender) = _processOldTranche( _renegotiationOffer.lender, _loan.borrower, _loan.principalAddress, tranche, _loan.startTime + _loan.duration, _loan.protocolFee, remainingNewLender ); _checkTrancheStrictly(_isStrictlyBetter, tranche.aprBps, _renegotiationOffer.aprBps, minImprovementApr); totalAnnualInterest += tranche.principalAmount * tranche.aprBps; totalAccruedInterest += accruedInterest; totalProtocolFee += thisProtocolFee; } uint256 lenderFee = remainingNewLender > totalProtocolFee ? totalProtocolFee : remainingNewLender; uint256 borrowerFee = totalProtocolFee - lenderFee; _handleProtocolFeeForFee( _loan.principalAddress, _renegotiationOffer.lender, lenderFee, _protocolFee.recipient ); _handleProtocolFeeForFee(_loan.principalAddress, _loan.borrower, borrowerFee, _protocolFee.recipient); remainingNewLender -= lenderFee; } } /// @notice Process the current source tranche during a renegotiation. /// @param _lender The new lender. /// @param _borrower The borrower of the loan. /// @param _principalAddress The principal address of the loan. /// @param _tranche The tranche to be processed. /// @param _endTime The end time of the loan. /// @param _protocolFeeFraction The protocol fee fraction. /// @param _remainingNewLender The amount left for the new lender to pay. /// @return accruedInterest The accrued interest paid. /// @return thisProtocolFee The protocol fee paid for this tranche. /// @return remainingNewLender The amount left for the new lender to pay. function _processOldTranche( address _lender, address _borrower, address _principalAddress, Tranche memory _tranche, uint256 _endTime, uint256 _protocolFeeFraction, uint256 _remainingNewLender ) private returns (uint256 accruedInterest, uint256 thisProtocolFee, uint256 remainingNewLender) { uint256 unlockTime = _getUnlockedTime(_tranche.startTime, _endTime); if (unlockTime > block.timestamp) { revert TrancheCannotBeRefinancedError(unlockTime); } unchecked { accruedInterest = _tranche.principalAmount.getInterest(_tranche.aprBps, block.timestamp - _tranche.startTime); thisProtocolFee = accruedInterest.mulDivUp(_protocolFeeFraction, _PRECISION); accruedInterest += _tranche.accruedInterest; } if (getLoanManagerRegistry.isLoanManager(_tranche.lender)) { ILoanManager(_tranche.lender).loanRepayment( _tranche.loanId, _tranche.principalAmount, _tranche.aprBps, _tranche.accruedInterest, _protocolFeeFraction, _tranche.startTime ); } uint256 oldLenderDebt; unchecked { oldLenderDebt = _tranche.principalAmount + accruedInterest - thisProtocolFee; } ERC20 asset = ERC20(_principalAddress); if (oldLenderDebt > _remainingNewLender) { /// @dev already checked in the condition asset.safeTransferFrom(_borrower, _tranche.lender, oldLenderDebt - _remainingNewLender); oldLenderDebt = _remainingNewLender; } if (oldLenderDebt > 0) { if (_lender != _tranche.lender) { asset.safeTransferFrom(_lender, _tranche.lender, oldLenderDebt); } /// @dev oldLenderDebt < _remainingNewLender because it would enter previous condition if not and set to _remainingNewLender unchecked { _remainingNewLender -= oldLenderDebt; } } remainingNewLender = _remainingNewLender; } /// @notice Basic loan checks (check if the hash is correct) + whether loan is still active. /// @param _loanId The loan ID. /// @param _loan The loan to be checked. function _baseLoanChecks(uint256 _loanId, Loan memory _loan) private view { if (_loan.hash() != _loans[_loanId]) { revert InvalidLoanError(_loanId); } if (_loan.startTime + _loan.duration <= block.timestamp) { revert LoanExpiredError(); } } /// @notice Basic renegotiation checks. Check basic parameters + expiration + whether the offer is active. function _baseRenegotiationChecks(RenegotiationOffer calldata _renegotiationOffer, Loan memory _loan) private view { if ( (_renegotiationOffer.principalAmount == 0) || (_loan.tranche.length < _renegotiationOffer.trancheIndex.length) ) { revert InvalidRenegotiationOfferError(); } if (block.timestamp > _renegotiationOffer.expirationTime) { revert ExpiredOfferError(_renegotiationOffer.expirationTime); } uint256 renegotiationId = _renegotiationOffer.renegotiationId; address lender = _renegotiationOffer.lender; if (isRenegotiationOfferCancelled[lender][renegotiationId]) { revert CancelledOrExecutedOfferError(lender, renegotiationId); } } /// @notice Protocol fee for fees charged on offers/renegotationOffers. /// @param _principalAddress The principal address of the loan. /// @param _lender The lender of the loan. /// @param _fee The fee to be charged. /// @param _feeRecipient The protocol fee recipient. function _handleProtocolFeeForFee(address _principalAddress, address _lender, uint256 _fee, address _feeRecipient) private { if (_fee != 0) { ERC20(_principalAddress).safeTransferFrom(_lender, _feeRecipient, _fee); } } /// @notice Check condition for strictly better tranches /// @param _isStrictlyBetter Whether the new tranche needs to be strictly better than the old one. /// @param _currentAprBps The current apr of the tranche. /// @param _targetAprBps The target apr of the tranche. /// @param __minImprovementApr The minimum improvement in APR. function _checkTrancheStrictly( bool _isStrictlyBetter, uint256 _currentAprBps, uint256 _targetAprBps, uint256 __minImprovementApr ) private pure { /// @dev If _isStrictlyBetter is set, and the new apr is higher, then it'll underflow. if ( _isStrictlyBetter && ((_currentAprBps - _targetAprBps).mulDivDown(_PRECISION, _currentAprBps) < __minImprovementApr) ) { revert InvalidRenegotiationOfferError(); } } /// @dev Tranches are locked from any refi after they are initiated for some time. function _getUnlockedTime(uint256 _trancheStartTime, uint256 _loanEndTime) private view returns (uint256) { uint256 delta; unchecked { delta = _loanEndTime - _trancheStartTime; } return _trancheStartTime + delta.mulDivUp(_minLockPeriod, _PRECISION); } /// @dev Loans are locked from lender initiated refis in the end. function _isLoanLocked(uint256 _loanStartTime, uint256 _loanDuration) private view returns (bool) { unchecked { /// @dev doesn't overflow because _minLockPeriod should be < 1 return block.timestamp > _loanStartTime + _loanDuration - _loanDuration.mulDivUp(_minLockPeriod, _PRECISION); } } /// @notice Base ExecutionData Checks /// @dev Note that we do not validate fee < principalAmount since this is done in the child class in this case. /// @param _offerExecution The offer execution. /// @param _tokenId The token ID. /// @param _lender The lender. /// @param _duration The duration. /// @param _lenderOfferSignature The signature of the lender of LoanOffer. /// @param _feeFraction The protocol fee fraction. /// @param _totalAmount The total amount ahead. function _validateOfferExecution( OfferExecution calldata _offerExecution, uint256 _tokenId, address _lender, uint256 _duration, bytes calldata _lenderOfferSignature, uint256 _feeFraction, uint256 _totalAmount ) private { LoanOffer calldata offer = _offerExecution.offer; address lender = offer.lender; uint256 offerId = offer.offerId; uint256 totalAmountAfterExecution = _offerExecution.amount + _totalAmount; if (lender.code.length != 0 && getLoanManagerRegistry.isLoanManager(lender)) { ILoanManager(lender).validateOffer(_tokenId, abi.encode(_offerExecution), _feeFraction); } else { _checkSignature(lender, offer.hash(), _lenderOfferSignature); } if (block.timestamp > offer.expirationTime) { revert ExpiredOfferError(offer.expirationTime); } if (isOfferCancelled[_lender][offerId] || (offerId <= minOfferId[_lender])) { revert CancelledOrExecutedOfferError(_lender, offerId); } if (totalAmountAfterExecution > offer.principalAmount) { revert InvalidAmountError(totalAmountAfterExecution, offer.principalAmount); } if (offer.duration == 0 || _duration > offer.duration) { revert InvalidDurationError(); } if (offer.aprBps == 0) { revert ZeroInterestError(); } if ((offer.capacity != 0) && (_used[_lender][offer.offerId] + _offerExecution.amount > offer.capacity)) { revert MaxCapacityExceededError(); } _checkValidators(_offerExecution.offer, _tokenId); } /// @notice Basic checks (expiration / signature if diff than borrower) for execution data. function _validateExecutionData(LoanExecutionData calldata _executionData, address _borrower) private view { if (msg.sender != _borrower) { _checkSignature(_borrower, _executionData.executionData.hash(), _executionData.borrowerOfferSignature); } if (block.timestamp > _executionData.executionData.expirationTime) { revert ExpiredOfferError(_executionData.executionData.expirationTime); } if (_executionData.executionData.offerExecution.length > getMaxTranches) { revert TooManyTranchesError(); } } /// @notice Extract addresses from first offer. Used for validations. /// @param _executionData Execution data. /// @return principalAddress Address of the principal token. /// @return nftCollateralAddress Address of the NFT collateral. function _getAddressesFromExecutionData(ExecutionData calldata _executionData) private pure returns (address, address) { LoanOffer calldata one = _executionData.offerExecution[0].offer; return (one.principalAddress, one.nftCollateralAddress); } /// @notice Check addresses are whitelisted. /// @param _principalAddress Address of the principal token. /// @param _nftCollateralAddress Address of the NFT collateral. function _checkWhitelists(address _principalAddress, address _nftCollateralAddress) private view { if (!_currencyManager.isWhitelisted(_principalAddress)) { revert CurrencyNotWhitelistedError(); } if (!_collectionManager.isWhitelisted(_nftCollateralAddress)) { revert CollectionNotWhitelistedError(); } } /// @notice Check principal/collateral addresses match. /// @param _offer The offer to check. /// @param _principalAddress Address of the principal token. /// @param _nftCollateralAddress Address of the NFT collateral. /// @param _amountWithInterestAhead Amount of more senior principal + max accrued interest ahead. function _checkOffer( LoanOffer calldata _offer, address _principalAddress, address _nftCollateralAddress, uint256 _amountWithInterestAhead ) private pure { if (_offer.principalAddress != _principalAddress || _offer.nftCollateralAddress != _nftCollateralAddress) { revert InvalidAddressesError(); } if (_amountWithInterestAhead > _offer.maxSeniorRepayment) { revert InvalidTrancheError(); } } /// @notice Check generic offer validators for a given offer or /// an exact match if no validators are given. The validators /// check is performed only if tokenId is set to 0. /// Having one empty validator is used for collection offers (all IDs match). /// @param _loanOffer The loan offer to check. /// @param _tokenId The token ID to check. function _checkValidators(LoanOffer calldata _loanOffer, uint256 _tokenId) private view { uint256 offerTokenId = _loanOffer.nftCollateralTokenId; if (_loanOffer.nftCollateralTokenId != 0) { if (offerTokenId != _tokenId) { revert InvalidCollateralIdError(); } } else { uint256 totalValidators = _loanOffer.validators.length; if (totalValidators == 0 && _tokenId != 0) { revert InvalidCollateralIdError(); } else if ((totalValidators == 1) && _loanOffer.validators[0].validator == address(0)) { return; } for (uint256 i = 0; i < totalValidators;) { IBaseLoan.OfferValidator memory thisValidator = _loanOffer.validators[i]; IOfferValidator(thisValidator.validator).validateOffer(_loanOffer, _tokenId, thisValidator.arguments); unchecked { ++i; } } } } /// @dev Check new trnches are at least this big. function _getMinTranchePrincipal(uint256 _loanPrincipal) private view returns (uint256) { return _loanPrincipal / (_MAX_RATIO_TRANCHE_MIN_PRINCIPAL * getMaxTranches); } function _hasCallback(bytes calldata _callbackData) private pure returns (bool) { return _callbackData.length != 0; } function _processRepayments(Loan calldata loan) private returns (uint256, uint256) { bool withProtocolFee = loan.protocolFee != 0; uint256 totalRepayment = 0; uint256 totalProtocolFee = 0; ERC20 asset = ERC20(loan.principalAddress); uint256 totalTranches = loan.tranche.length; for (uint256 i; i < totalTranches;) { Tranche memory tranche = loan.tranche[i]; uint256 newInterest = tranche.principalAmount.getInterest(tranche.aprBps, block.timestamp - tranche.startTime); uint256 thisProtocolFee = 0; if (withProtocolFee) { thisProtocolFee = newInterest.mulDivUp(loan.protocolFee, _PRECISION); unchecked { totalProtocolFee += thisProtocolFee; } } uint256 repayment = tranche.principalAmount + tranche.accruedInterest + newInterest - thisProtocolFee; asset.safeTransferFrom(loan.borrower, tranche.lender, repayment); unchecked { totalRepayment += repayment; } if (getLoanManagerRegistry.isLoanManager(tranche.lender)) { ILoanManager(tranche.lender).loanRepayment( tranche.loanId, tranche.principalAmount, tranche.aprBps, tranche.accruedInterest, loan.protocolFee, tranche.startTime ); } unchecked { ++i; } } if (withProtocolFee) { asset.safeTransferFrom(loan.borrower, _protocolFee.recipient, totalProtocolFee); } return (totalRepayment, totalProtocolFee); } /// @notice Process a series of offers and return the loan ID, offer IDs, loan (built from such offers) and total fee. /// @param _borrower The borrower of the loan. /// @param _principalReceiver The receiver of the principal. /// @param _principalAddress The principal address of the loan. /// @param _nftCollateralAddress The NFT collateral address of the loan. /// @param _tokenId The token ID of the loan. /// @param _duration The duration of the loan. /// @param _offerExecution The offer execution. /// @return loanId The loan ID. /// @return offerIds The offer IDs. /// @return loan The loan. /// @return totalFee The total fee. function _processOffersFromExecutionData( address _borrower, address _principalReceiver, address _principalAddress, address _nftCollateralAddress, uint256 _tokenId, uint256 _duration, OfferExecution[] calldata _offerExecution ) private returns (uint256, uint256[] memory, Loan memory, uint256) { Tranche[] memory tranche = new Tranche[](_offerExecution.length); uint256[] memory offerIds = new uint256[](_offerExecution.length); uint256 totalAmount; uint256 loanId = _getAndSetNewLoanId(); ProtocolFee memory protocolFee = _protocolFee; LoanOffer calldata offer; uint256 totalFee; uint256 totalAmountWithMaxInterest; uint256 minAmount = type(uint256).max; uint256 totalOffers = _offerExecution.length; for (uint256 i = 0; i < totalOffers;) { OfferExecution calldata thisOfferExecution = _offerExecution[i]; offer = thisOfferExecution.offer; _validateOfferExecution( thisOfferExecution, _tokenId, offer.lender, _duration, thisOfferExecution.lenderOfferSignature, protocolFee.fraction, totalAmount ); uint256 amount = thisOfferExecution.amount; if (amount < minAmount) { minAmount = amount; } address lender = offer.lender; _checkOffer(offer, _principalAddress, _nftCollateralAddress, totalAmountWithMaxInterest); /// @dev Please note that we can now have many tranches with same `loanId`. tranche[i] = Tranche(loanId, totalAmount, amount, lender, 0, block.timestamp, offer.aprBps); unchecked { totalAmount += amount; totalAmountWithMaxInterest += amount + amount.getInterest(offer.aprBps, _duration); } uint256 fee; unchecked { fee = offer.fee.mulDivUp(amount, offer.principalAmount); totalFee += fee; } _handleProtocolFeeForFee( offer.principalAddress, lender, fee.mulDivUp(protocolFee.fraction, _PRECISION), protocolFee.recipient ); ERC20(offer.principalAddress).safeTransferFrom(lender, _principalReceiver, amount - fee); if (offer.capacity != 0) { unchecked { _used[lender][offer.offerId] += amount; } } else { isOfferCancelled[lender][offer.offerId] = true; } offerIds[i] = offer.offerId; unchecked { ++i; } } if (minAmount < _getMinTranchePrincipal(totalAmount)) { revert InvalidTrancheError(); } Loan memory loan = Loan( _borrower, _tokenId, _nftCollateralAddress, _principalAddress, totalAmount, block.timestamp, _duration, tranche, protocolFee.fraction ); return (loanId, offerIds, loan, totalFee); } function _addNewTranche( uint256 _newLoanId, IMultiSourceLoan.Loan memory _loan, IMultiSourceLoan.RenegotiationOffer calldata _renegotiationOffer ) private view returns (IMultiSourceLoan.Loan memory) { if (_renegotiationOffer.principalAmount < _getMinTranchePrincipal(_loan.principalAmount)) { revert InvalidTrancheError(); } uint256 newTrancheIndex = _loan.tranche.length; IMultiSourceLoan.Tranche[] memory tranches = new IMultiSourceLoan.Tranche[](newTrancheIndex + 1); /// @dev Copy old tranches for (uint256 i = 0; i < newTrancheIndex;) { tranches[i] = _loan.tranche[i]; unchecked { ++i; } } tranches[newTrancheIndex] = IMultiSourceLoan.Tranche( _newLoanId, _loan.principalAmount, _renegotiationOffer.principalAmount, _renegotiationOffer.lender, 0, block.timestamp, _renegotiationOffer.aprBps ); _loan.tranche = tranches; unchecked { _loan.principalAmount += _renegotiationOffer.principalAmount; } return _loan; } /// @notice Check a signature is valid given a hash and signer. /// @dev Comply with IERC1271 and EIP-712. function _checkSignature(address _signer, bytes32 _hash, bytes calldata _signature) private view { bytes32 typedDataHash = DOMAIN_SEPARATOR().toTypedDataHash(_hash); if (_signer.code.length != 0) { if (IERC1271(_signer).isValidSignature(typedDataHash, _signature) != MAGICVALUE_1271) { revert InvalidSignatureError(); } } else { address recovered = typedDataHash.recover(_signature); if (_signer != recovered) { revert InvalidSignatureError(); } } } /// @dev Check whether an offer is strictly better than a tranche function _checkStrictlyBetter( uint256 _offerPrincipalAmount, uint256 _loanPrincipalAmount, uint256 _offerEndTime, uint256 _loanEndTime, uint256 _offerAprBps, uint256 _loanAprBps, uint256 _offerFee ) internal view { uint256 minImprovementApr = _minImprovementApr; /// @dev If principal is increased, then we need to check net daily interest is better. /// interestDelta = (_loanAprBps * _loanPrincipalAmount - _offerAprBps * _offerPrincipalAmount) /// We already checked that all tranches are strictly better. /// We check that the duration is not decreased or the offer charges a fee. if ( ( (_offerPrincipalAmount - _loanPrincipalAmount != 0) && ( (_loanAprBps * _loanPrincipalAmount - _offerAprBps * _offerPrincipalAmount).mulDivDown( _PRECISION, _loanAprBps * _loanPrincipalAmount ) < minImprovementApr ) ) || (_offerFee != 0) || (_offerEndTime < _loanEndTime) ) { revert NotStrictlyImprovedError(); } } }
// SPDX-License-Identifier: CC0-1.0 pragma solidity >=0.8.13; /** * @title IDelegateRegistry * @custom:version 2.0 * @custom:author foobar (0xfoobar) * @notice A standalone immutable registry storing delegated permissions from one address to another */ interface IDelegateRegistry { /// @notice Delegation type, NONE is used when a delegation does not exist or is revoked enum DelegationType { NONE, ALL, CONTRACT, ERC721, ERC20, ERC1155 } /// @notice Struct for returning delegations struct Delegation { DelegationType type_; address to; address from; bytes32 rights; address contract_; uint256 tokenId; uint256 amount; } /// @notice Emitted when an address delegates or revokes rights for their entire wallet event DelegateAll(address indexed from, address indexed to, bytes32 rights, bool enable); /// @notice Emitted when an address delegates or revokes rights for a contract address event DelegateContract(address indexed from, address indexed to, address indexed contract_, bytes32 rights, bool enable); /// @notice Emitted when an address delegates or revokes rights for an ERC721 tokenId event DelegateERC721(address indexed from, address indexed to, address indexed contract_, uint256 tokenId, bytes32 rights, bool enable); /// @notice Emitted when an address delegates or revokes rights for an amount of ERC20 tokens event DelegateERC20(address indexed from, address indexed to, address indexed contract_, bytes32 rights, uint256 amount); /// @notice Emitted when an address delegates or revokes rights for an amount of an ERC1155 tokenId event DelegateERC1155(address indexed from, address indexed to, address indexed contract_, uint256 tokenId, bytes32 rights, uint256 amount); /// @notice Thrown if multicall calldata is malformed error MulticallFailed(); /** * ----------- WRITE ----------- */ /** * @notice Call multiple functions in the current contract and return the data from all of them if they all succeed * @param data The encoded function data for each of the calls to make to this contract * @return results The results from each of the calls passed in via data */ function multicall(bytes[] calldata data) external payable returns (bytes[] memory results); /** * @notice Allow the delegate to act on behalf of `msg.sender` for all contracts * @param to The address to act as delegate * @param rights Specific subdelegation rights granted to the delegate, pass an empty bytestring to encompass all rights * @param enable Whether to enable or disable this delegation, true delegates and false revokes * @return delegationHash The unique identifier of the delegation */ function delegateAll(address to, bytes32 rights, bool enable) external payable returns (bytes32 delegationHash); /** * @notice Allow the delegate to act on behalf of `msg.sender` for a specific contract * @param to The address to act as delegate * @param contract_ The contract whose rights are being delegated * @param rights Specific subdelegation rights granted to the delegate, pass an empty bytestring to encompass all rights * @param enable Whether to enable or disable this delegation, true delegates and false revokes * @return delegationHash The unique identifier of the delegation */ function delegateContract(address to, address contract_, bytes32 rights, bool enable) external payable returns (bytes32 delegationHash); /** * @notice Allow the delegate to act on behalf of `msg.sender` for a specific ERC721 token * @param to The address to act as delegate * @param contract_ The contract whose rights are being delegated * @param tokenId The token id to delegate * @param rights Specific subdelegation rights granted to the delegate, pass an empty bytestring to encompass all rights * @param enable Whether to enable or disable this delegation, true delegates and false revokes * @return delegationHash The unique identifier of the delegation */ function delegateERC721(address to, address contract_, uint256 tokenId, bytes32 rights, bool enable) external payable returns (bytes32 delegationHash); /** * @notice Allow the delegate to act on behalf of `msg.sender` for a specific amount of ERC20 tokens * @dev The actual amount is not encoded in the hash, just the existence of a amount (since it is an upper bound) * @param to The address to act as delegate * @param contract_ The address for the fungible token contract * @param rights Specific subdelegation rights granted to the delegate, pass an empty bytestring to encompass all rights * @param amount The amount to delegate, > 0 delegates and 0 revokes * @return delegationHash The unique identifier of the delegation */ function delegateERC20(address to, address contract_, bytes32 rights, uint256 amount) external payable returns (bytes32 delegationHash); /** * @notice Allow the delegate to act on behalf of `msg.sender` for a specific amount of ERC1155 tokens * @dev The actual amount is not encoded in the hash, just the existence of a amount (since it is an upper bound) * @param to The address to act as delegate * @param contract_ The address of the contract that holds the token * @param tokenId The token id to delegate * @param rights Specific subdelegation rights granted to the delegate, pass an empty bytestring to encompass all rights * @param amount The amount of that token id to delegate, > 0 delegates and 0 revokes * @return delegationHash The unique identifier of the delegation */ function delegateERC1155(address to, address contract_, uint256 tokenId, bytes32 rights, uint256 amount) external payable returns (bytes32 delegationHash); /** * ----------- CHECKS ----------- */ /** * @notice Check if `to` is a delegate of `from` for the entire wallet * @param to The potential delegate address * @param from The potential address who delegated rights * @param rights Specific rights to check for, pass the zero value to ignore subdelegations and check full delegations only * @return valid Whether delegate is granted to act on the from's behalf */ function checkDelegateForAll(address to, address from, bytes32 rights) external view returns (bool); /** * @notice Check if `to` is a delegate of `from` for the specified `contract_` or the entire wallet * @param to The delegated address to check * @param contract_ The specific contract address being checked * @param from The cold wallet who issued the delegation * @param rights Specific rights to check for, pass the zero value to ignore subdelegations and check full delegations only * @return valid Whether delegate is granted to act on from's behalf for entire wallet or that specific contract */ function checkDelegateForContract(address to, address from, address contract_, bytes32 rights) external view returns (bool); /** * @notice Check if `to` is a delegate of `from` for the specific `contract` and `tokenId`, the entire `contract_`, or the entire wallet * @param to The delegated address to check * @param contract_ The specific contract address being checked * @param tokenId The token id for the token to delegating * @param from The wallet that issued the delegation * @param rights Specific rights to check for, pass the zero value to ignore subdelegations and check full delegations only * @return valid Whether delegate is granted to act on from's behalf for entire wallet, that contract, or that specific tokenId */ function checkDelegateForERC721(address to, address from, address contract_, uint256 tokenId, bytes32 rights) external view returns (bool); /** * @notice Returns the amount of ERC20 tokens the delegate is granted rights to act on the behalf of * @param to The delegated address to check * @param contract_ The address of the token contract * @param from The cold wallet who issued the delegation * @param rights Specific rights to check for, pass the zero value to ignore subdelegations and check full delegations only * @return balance The delegated balance, which will be 0 if the delegation does not exist */ function checkDelegateForERC20(address to, address from, address contract_, bytes32 rights) external view returns (uint256); /** * @notice Returns the amount of a ERC1155 tokens the delegate is granted rights to act on the behalf of * @param to The delegated address to check * @param contract_ The address of the token contract * @param tokenId The token id to check the delegated amount of * @param from The cold wallet who issued the delegation * @param rights Specific rights to check for, pass the zero value to ignore subdelegations and check full delegations only * @return balance The delegated balance, which will be 0 if the delegation does not exist */ function checkDelegateForERC1155(address to, address from, address contract_, uint256 tokenId, bytes32 rights) external view returns (uint256); /** * ----------- ENUMERATIONS ----------- */ /** * @notice Returns all enabled delegations a given delegate has received * @param to The address to retrieve delegations for * @return delegations Array of Delegation structs */ function getIncomingDelegations(address to) external view returns (Delegation[] memory delegations); /** * @notice Returns all enabled delegations an address has given out * @param from The address to retrieve delegations for * @return delegations Array of Delegation structs */ function getOutgoingDelegations(address from) external view returns (Delegation[] memory delegations); /** * @notice Returns all hashes associated with enabled delegations an address has received * @param to The address to retrieve incoming delegation hashes for * @return delegationHashes Array of delegation hashes */ function getIncomingDelegationHashes(address to) external view returns (bytes32[] memory delegationHashes); /** * @notice Returns all hashes associated with enabled delegations an address has given out * @param from The address to retrieve outgoing delegation hashes for * @return delegationHashes Array of delegation hashes */ function getOutgoingDelegationHashes(address from) external view returns (bytes32[] memory delegationHashes); /** * @notice Returns the delegations for a given array of delegation hashes * @param delegationHashes is an array of hashes that correspond to delegations * @return delegations Array of Delegation structs, return empty structs for nonexistent or revoked delegations */ function getDelegationsFromHashes(bytes32[] calldata delegationHashes) external view returns (Delegation[] memory delegations); /** * ----------- STORAGE ACCESS ----------- */ /** * @notice Allows external contracts to read arbitrary storage slots */ function readSlot(bytes32 location) external view returns (bytes32); /** * @notice Allows external contracts to read an arbitrary array of storage slots */ function readSlots(bytes32[] calldata locations) external view returns (bytes32[] memory); }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v5.0.0) (utils/cryptography/ECDSA.sol) pragma solidity ^0.8.20; /** * @dev Elliptic Curve Digital Signature Algorithm (ECDSA) operations. * * These functions can be used to verify that a message was signed by the holder * of the private keys of a given address. */ library ECDSA { enum RecoverError { NoError, InvalidSignature, InvalidSignatureLength, InvalidSignatureS } /** * @dev The signature derives the `address(0)`. */ error ECDSAInvalidSignature(); /** * @dev The signature has an invalid length. */ error ECDSAInvalidSignatureLength(uint256 length); /** * @dev The signature has an S value that is in the upper half order. */ error ECDSAInvalidSignatureS(bytes32 s); /** * @dev Returns the address that signed a hashed message (`hash`) with `signature` or an error. This will not * return address(0) without also returning an error description. Errors are documented using an enum (error type) * and a bytes32 providing additional information about the error. * * If no error is returned, then the address can be used for verification purposes. * * The `ecrecover` EVM precompile allows for malleable (non-unique) signatures: * this function rejects them by requiring the `s` value to be in the lower * half order, and the `v` value to be either 27 or 28. * * IMPORTANT: `hash` _must_ be the result of a hash operation for the * verification to be secure: it is possible to craft signatures that * recover to arbitrary addresses for non-hashed data. A safe way to ensure * this is by receiving a hash of the original message (which may otherwise * be too long), and then calling {MessageHashUtils-toEthSignedMessageHash} on it. * * Documentation for signature generation: * - with https://web3js.readthedocs.io/en/v1.3.4/web3-eth-accounts.html#sign[Web3.js] * - with https://docs.ethers.io/v5/api/signer/#Signer-signMessage[ethers] */ function tryRecover(bytes32 hash, bytes memory signature) internal pure returns (address, RecoverError, bytes32) { if (signature.length == 65) { bytes32 r; bytes32 s; uint8 v; // ecrecover takes the signature parameters, and the only way to get them // currently is to use assembly. /// @solidity memory-safe-assembly assembly { r := mload(add(signature, 0x20)) s := mload(add(signature, 0x40)) v := byte(0, mload(add(signature, 0x60))) } return tryRecover(hash, v, r, s); } else { return (address(0), RecoverError.InvalidSignatureLength, bytes32(signature.length)); } } /** * @dev Returns the address that signed a hashed message (`hash`) with * `signature`. This address can then be used for verification purposes. * * The `ecrecover` EVM precompile allows for malleable (non-unique) signatures: * this function rejects them by requiring the `s` value to be in the lower * half order, and the `v` value to be either 27 or 28. * * IMPORTANT: `hash` _must_ be the result of a hash operation for the * verification to be secure: it is possible to craft signatures that * recover to arbitrary addresses for non-hashed data. A safe way to ensure * this is by receiving a hash of the original message (which may otherwise * be too long), and then calling {MessageHashUtils-toEthSignedMessageHash} on it. */ function recover(bytes32 hash, bytes memory signature) internal pure returns (address) { (address recovered, RecoverError error, bytes32 errorArg) = tryRecover(hash, signature); _throwError(error, errorArg); return recovered; } /** * @dev Overload of {ECDSA-tryRecover} that receives the `r` and `vs` short-signature fields separately. * * See https://eips.ethereum.org/EIPS/eip-2098[ERC-2098 short signatures] */ function tryRecover(bytes32 hash, bytes32 r, bytes32 vs) internal pure returns (address, RecoverError, bytes32) { unchecked { bytes32 s = vs & bytes32(0x7fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff); // We do not check for an overflow here since the shift operation results in 0 or 1. uint8 v = uint8((uint256(vs) >> 255) + 27); return tryRecover(hash, v, r, s); } } /** * @dev Overload of {ECDSA-recover} that receives the `r and `vs` short-signature fields separately. */ function recover(bytes32 hash, bytes32 r, bytes32 vs) internal pure returns (address) { (address recovered, RecoverError error, bytes32 errorArg) = tryRecover(hash, r, vs); _throwError(error, errorArg); return recovered; } /** * @dev Overload of {ECDSA-tryRecover} that receives the `v`, * `r` and `s` signature fields separately. */ function tryRecover( bytes32 hash, uint8 v, bytes32 r, bytes32 s ) internal pure returns (address, RecoverError, bytes32) { // EIP-2 still allows signature malleability for ecrecover(). Remove this possibility and make the signature // unique. Appendix F in the Ethereum Yellow paper (https://ethereum.github.io/yellowpaper/paper.pdf), defines // the valid range for s in (301): 0 < s < secp256k1n ÷ 2 + 1, and for v in (302): v ∈ {27, 28}. Most // signatures from current libraries generate a unique signature with an s-value in the lower half order. // // If your library generates malleable signatures, such as s-values in the upper range, calculate a new s-value // with 0xFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFEBAAEDCE6AF48A03BBFD25E8CD0364141 - s1 and flip v from 27 to 28 or // vice versa. If your library also generates signatures with 0/1 for v instead 27/28, add 27 to v to accept // these malleable signatures as well. if (uint256(s) > 0x7FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF5D576E7357A4501DDFE92F46681B20A0) { return (address(0), RecoverError.InvalidSignatureS, s); } // If the signature is valid (and not malleable), return the signer address address signer = ecrecover(hash, v, r, s); if (signer == address(0)) { return (address(0), RecoverError.InvalidSignature, bytes32(0)); } return (signer, RecoverError.NoError, bytes32(0)); } /** * @dev Overload of {ECDSA-recover} that receives the `v`, * `r` and `s` signature fields separately. */ function recover(bytes32 hash, uint8 v, bytes32 r, bytes32 s) internal pure returns (address) { (address recovered, RecoverError error, bytes32 errorArg) = tryRecover(hash, v, r, s); _throwError(error, errorArg); return recovered; } /** * @dev Optionally reverts with the corresponding custom error according to the `error` argument provided. */ function _throwError(RecoverError error, bytes32 errorArg) private pure { if (error == RecoverError.NoError) { return; // no error: do nothing } else if (error == RecoverError.InvalidSignature) { revert ECDSAInvalidSignature(); } else if (error == RecoverError.InvalidSignatureLength) { revert ECDSAInvalidSignatureLength(uint256(errorArg)); } else if (error == RecoverError.InvalidSignatureS) { revert ECDSAInvalidSignatureS(errorArg); } } }
// SPDX-License-Identifier: AGPL-3.0-only pragma solidity >=0.8.0; /// @notice Modern and gas efficient ERC20 + EIP-2612 implementation. /// @author Solmate (https://github.com/transmissions11/solmate/blob/main/src/tokens/ERC20.sol) /// @author Modified from Uniswap (https://github.com/Uniswap/uniswap-v2-core/blob/master/contracts/UniswapV2ERC20.sol) /// @dev Do not manually set balances without updating totalSupply, as the sum of all user balances must not exceed it. abstract contract ERC20 { /*////////////////////////////////////////////////////////////// EVENTS //////////////////////////////////////////////////////////////*/ event Transfer(address indexed from, address indexed to, uint256 amount); event Approval(address indexed owner, address indexed spender, uint256 amount); /*////////////////////////////////////////////////////////////// METADATA STORAGE //////////////////////////////////////////////////////////////*/ string public name; string public symbol; uint8 public immutable decimals; /*////////////////////////////////////////////////////////////// ERC20 STORAGE //////////////////////////////////////////////////////////////*/ uint256 public totalSupply; mapping(address => uint256) public balanceOf; mapping(address => mapping(address => uint256)) public allowance; /*////////////////////////////////////////////////////////////// EIP-2612 STORAGE //////////////////////////////////////////////////////////////*/ uint256 internal immutable INITIAL_CHAIN_ID; bytes32 internal immutable INITIAL_DOMAIN_SEPARATOR; mapping(address => uint256) public nonces; /*////////////////////////////////////////////////////////////// CONSTRUCTOR //////////////////////////////////////////////////////////////*/ constructor( string memory _name, string memory _symbol, uint8 _decimals ) { name = _name; symbol = _symbol; decimals = _decimals; INITIAL_CHAIN_ID = block.chainid; INITIAL_DOMAIN_SEPARATOR = computeDomainSeparator(); } /*////////////////////////////////////////////////////////////// ERC20 LOGIC //////////////////////////////////////////////////////////////*/ function approve(address spender, uint256 amount) public virtual returns (bool) { allowance[msg.sender][spender] = amount; emit Approval(msg.sender, spender, amount); return true; } function transfer(address to, uint256 amount) public virtual returns (bool) { balanceOf[msg.sender] -= amount; // Cannot overflow because the sum of all user // balances can't exceed the max uint256 value. unchecked { balanceOf[to] += amount; } emit Transfer(msg.sender, to, amount); return true; } function transferFrom( address from, address to, uint256 amount ) public virtual returns (bool) { uint256 allowed = allowance[from][msg.sender]; // Saves gas for limited approvals. if (allowed != type(uint256).max) allowance[from][msg.sender] = allowed - amount; balanceOf[from] -= amount; // Cannot overflow because the sum of all user // balances can't exceed the max uint256 value. unchecked { balanceOf[to] += amount; } emit Transfer(from, to, amount); return true; } /*////////////////////////////////////////////////////////////// EIP-2612 LOGIC //////////////////////////////////////////////////////////////*/ function permit( address owner, address spender, uint256 value, uint256 deadline, uint8 v, bytes32 r, bytes32 s ) public virtual { require(deadline >= block.timestamp, "PERMIT_DEADLINE_EXPIRED"); // Unchecked because the only math done is incrementing // the owner's nonce which cannot realistically overflow. unchecked { address recoveredAddress = ecrecover( keccak256( abi.encodePacked( "\x19\x01", DOMAIN_SEPARATOR(), keccak256( abi.encode( keccak256( "Permit(address owner,address spender,uint256 value,uint256 nonce,uint256 deadline)" ), owner, spender, value, nonces[owner]++, deadline ) ) ) ), v, r, s ); require(recoveredAddress != address(0) && recoveredAddress == owner, "INVALID_SIGNER"); allowance[recoveredAddress][spender] = value; } emit Approval(owner, spender, value); } function DOMAIN_SEPARATOR() public view virtual returns (bytes32) { return block.chainid == INITIAL_CHAIN_ID ? INITIAL_DOMAIN_SEPARATOR : computeDomainSeparator(); } function computeDomainSeparator() internal view virtual returns (bytes32) { return keccak256( abi.encode( keccak256("EIP712Domain(string name,string version,uint256 chainId,address verifyingContract)"), keccak256(bytes(name)), keccak256("1"), block.chainid, address(this) ) ); } /*////////////////////////////////////////////////////////////// INTERNAL MINT/BURN LOGIC //////////////////////////////////////////////////////////////*/ function _mint(address to, uint256 amount) internal virtual { totalSupply += amount; // Cannot overflow because the sum of all user // balances can't exceed the max uint256 value. unchecked { balanceOf[to] += amount; } emit Transfer(address(0), to, amount); } function _burn(address from, uint256 amount) internal virtual { balanceOf[from] -= amount; // Cannot underflow because a user's balance // will never be larger than the total supply. unchecked { totalSupply -= amount; } emit Transfer(from, address(0), amount); } }
// SPDX-License-Identifier: AGPL-3.0-only pragma solidity >=0.8.0; /// @notice Modern, minimalist, and gas efficient ERC-721 implementation. /// @author Solmate (https://github.com/transmissions11/solmate/blob/main/src/tokens/ERC721.sol) abstract contract ERC721 { /*////////////////////////////////////////////////////////////// EVENTS //////////////////////////////////////////////////////////////*/ event Transfer(address indexed from, address indexed to, uint256 indexed id); event Approval(address indexed owner, address indexed spender, uint256 indexed id); event ApprovalForAll(address indexed owner, address indexed operator, bool approved); /*////////////////////////////////////////////////////////////// METADATA STORAGE/LOGIC //////////////////////////////////////////////////////////////*/ string public name; string public symbol; function tokenURI(uint256 id) public view virtual returns (string memory); /*////////////////////////////////////////////////////////////// ERC721 BALANCE/OWNER STORAGE //////////////////////////////////////////////////////////////*/ mapping(uint256 => address) internal _ownerOf; mapping(address => uint256) internal _balanceOf; function ownerOf(uint256 id) public view virtual returns (address owner) { require((owner = _ownerOf[id]) != address(0), "NOT_MINTED"); } function balanceOf(address owner) public view virtual returns (uint256) { require(owner != address(0), "ZERO_ADDRESS"); return _balanceOf[owner]; } /*////////////////////////////////////////////////////////////// ERC721 APPROVAL STORAGE //////////////////////////////////////////////////////////////*/ mapping(uint256 => address) public getApproved; mapping(address => mapping(address => bool)) public isApprovedForAll; /*////////////////////////////////////////////////////////////// CONSTRUCTOR //////////////////////////////////////////////////////////////*/ constructor(string memory _name, string memory _symbol) { name = _name; symbol = _symbol; } /*////////////////////////////////////////////////////////////// ERC721 LOGIC //////////////////////////////////////////////////////////////*/ function approve(address spender, uint256 id) public virtual { address owner = _ownerOf[id]; require(msg.sender == owner || isApprovedForAll[owner][msg.sender], "NOT_AUTHORIZED"); getApproved[id] = spender; emit Approval(owner, spender, id); } function setApprovalForAll(address operator, bool approved) public virtual { isApprovedForAll[msg.sender][operator] = approved; emit ApprovalForAll(msg.sender, operator, approved); } function transferFrom( address from, address to, uint256 id ) public virtual { require(from == _ownerOf[id], "WRONG_FROM"); require(to != address(0), "INVALID_RECIPIENT"); require( msg.sender == from || isApprovedForAll[from][msg.sender] || msg.sender == getApproved[id], "NOT_AUTHORIZED" ); // Underflow of the sender's balance is impossible because we check for // ownership above and the recipient's balance can't realistically overflow. unchecked { _balanceOf[from]--; _balanceOf[to]++; } _ownerOf[id] = to; delete getApproved[id]; emit Transfer(from, to, id); } function safeTransferFrom( address from, address to, uint256 id ) public virtual { transferFrom(from, to, id); require( to.code.length == 0 || ERC721TokenReceiver(to).onERC721Received(msg.sender, from, id, "") == ERC721TokenReceiver.onERC721Received.selector, "UNSAFE_RECIPIENT" ); } function safeTransferFrom( address from, address to, uint256 id, bytes calldata data ) public virtual { transferFrom(from, to, id); require( to.code.length == 0 || ERC721TokenReceiver(to).onERC721Received(msg.sender, from, id, data) == ERC721TokenReceiver.onERC721Received.selector, "UNSAFE_RECIPIENT" ); } /*////////////////////////////////////////////////////////////// ERC165 LOGIC //////////////////////////////////////////////////////////////*/ function supportsInterface(bytes4 interfaceId) public view virtual returns (bool) { return interfaceId == 0x01ffc9a7 || // ERC165 Interface ID for ERC165 interfaceId == 0x80ac58cd || // ERC165 Interface ID for ERC721 interfaceId == 0x5b5e139f; // ERC165 Interface ID for ERC721Metadata } /*////////////////////////////////////////////////////////////// INTERNAL MINT/BURN LOGIC //////////////////////////////////////////////////////////////*/ function _mint(address to, uint256 id) internal virtual { require(to != address(0), "INVALID_RECIPIENT"); require(_ownerOf[id] == address(0), "ALREADY_MINTED"); // Counter overflow is incredibly unrealistic. unchecked { _balanceOf[to]++; } _ownerOf[id] = to; emit Transfer(address(0), to, id); } function _burn(uint256 id) internal virtual { address owner = _ownerOf[id]; require(owner != address(0), "NOT_MINTED"); // Ownership check above ensures no underflow. unchecked { _balanceOf[owner]--; } delete _ownerOf[id]; delete getApproved[id]; emit Transfer(owner, address(0), id); } /*////////////////////////////////////////////////////////////// INTERNAL SAFE MINT LOGIC //////////////////////////////////////////////////////////////*/ function _safeMint(address to, uint256 id) internal virtual { _mint(to, id); require( to.code.length == 0 || ERC721TokenReceiver(to).onERC721Received(msg.sender, address(0), id, "") == ERC721TokenReceiver.onERC721Received.selector, "UNSAFE_RECIPIENT" ); } function _safeMint( address to, uint256 id, bytes memory data ) internal virtual { _mint(to, id); require( to.code.length == 0 || ERC721TokenReceiver(to).onERC721Received(msg.sender, address(0), id, data) == ERC721TokenReceiver.onERC721Received.selector, "UNSAFE_RECIPIENT" ); } } /// @notice A generic interface for a contract which properly accepts ERC721 tokens. /// @author Solmate (https://github.com/transmissions11/solmate/blob/main/src/tokens/ERC721.sol) abstract contract ERC721TokenReceiver { function onERC721Received( address, address, uint256, bytes calldata ) external virtual returns (bytes4) { return ERC721TokenReceiver.onERC721Received.selector; } }
// SPDX-License-Identifier: AGPL-3.0-only pragma solidity >=0.8.0; /// @notice Arithmetic library with operations for fixed-point numbers. /// @author Solmate (https://github.com/transmissions11/solmate/blob/main/src/utils/FixedPointMathLib.sol) /// @author Inspired by USM (https://github.com/usmfum/USM/blob/master/contracts/WadMath.sol) library FixedPointMathLib { /*////////////////////////////////////////////////////////////// SIMPLIFIED FIXED POINT OPERATIONS //////////////////////////////////////////////////////////////*/ uint256 internal constant MAX_UINT256 = 2**256 - 1; uint256 internal constant WAD = 1e18; // The scalar of ETH and most ERC20s. function mulWadDown(uint256 x, uint256 y) internal pure returns (uint256) { return mulDivDown(x, y, WAD); // Equivalent to (x * y) / WAD rounded down. } function mulWadUp(uint256 x, uint256 y) internal pure returns (uint256) { return mulDivUp(x, y, WAD); // Equivalent to (x * y) / WAD rounded up. } function divWadDown(uint256 x, uint256 y) internal pure returns (uint256) { return mulDivDown(x, WAD, y); // Equivalent to (x * WAD) / y rounded down. } function divWadUp(uint256 x, uint256 y) internal pure returns (uint256) { return mulDivUp(x, WAD, y); // Equivalent to (x * WAD) / y rounded up. } /*////////////////////////////////////////////////////////////// LOW LEVEL FIXED POINT OPERATIONS //////////////////////////////////////////////////////////////*/ function mulDivDown( uint256 x, uint256 y, uint256 denominator ) internal pure returns (uint256 z) { /// @solidity memory-safe-assembly assembly { // Equivalent to require(denominator != 0 && (y == 0 || x <= type(uint256).max / y)) if iszero(mul(denominator, iszero(mul(y, gt(x, div(MAX_UINT256, y)))))) { revert(0, 0) } // Divide x * y by the denominator. z := div(mul(x, y), denominator) } } function mulDivUp( uint256 x, uint256 y, uint256 denominator ) internal pure returns (uint256 z) { /// @solidity memory-safe-assembly assembly { // Equivalent to require(denominator != 0 && (y == 0 || x <= type(uint256).max / y)) if iszero(mul(denominator, iszero(mul(y, gt(x, div(MAX_UINT256, y)))))) { revert(0, 0) } // If x * y modulo the denominator is strictly greater than 0, // 1 is added to round up the division of x * y by the denominator. z := add(gt(mod(mul(x, y), denominator), 0), div(mul(x, y), denominator)) } } function rpow( uint256 x, uint256 n, uint256 scalar ) internal pure returns (uint256 z) { /// @solidity memory-safe-assembly assembly { switch x case 0 { switch n case 0 { // 0 ** 0 = 1 z := scalar } default { // 0 ** n = 0 z := 0 } } default { switch mod(n, 2) case 0 { // If n is even, store scalar in z for now. z := scalar } default { // If n is odd, store x in z for now. z := x } // Shifting right by 1 is like dividing by 2. let half := shr(1, scalar) for { // Shift n right by 1 before looping to halve it. n := shr(1, n) } n { // Shift n right by 1 each iteration to halve it. n := shr(1, n) } { // Revert immediately if x ** 2 would overflow. // Equivalent to iszero(eq(div(xx, x), x)) here. if shr(128, x) { revert(0, 0) } // Store x squared. let xx := mul(x, x) // Round to the nearest number. let xxRound := add(xx, half) // Revert if xx + half overflowed. if lt(xxRound, xx) { revert(0, 0) } // Set x to scaled xxRound. x := div(xxRound, scalar) // If n is even: if mod(n, 2) { // Compute z * x. let zx := mul(z, x) // If z * x overflowed: if iszero(eq(div(zx, x), z)) { // Revert if x is non-zero. if iszero(iszero(x)) { revert(0, 0) } } // Round to the nearest number. let zxRound := add(zx, half) // Revert if zx + half overflowed. if lt(zxRound, zx) { revert(0, 0) } // Return properly scaled zxRound. z := div(zxRound, scalar) } } } } } /*////////////////////////////////////////////////////////////// GENERAL NUMBER UTILITIES //////////////////////////////////////////////////////////////*/ function sqrt(uint256 x) internal pure returns (uint256 z) { /// @solidity memory-safe-assembly assembly { let y := x // We start y at x, which will help us make our initial estimate. z := 181 // The "correct" value is 1, but this saves a multiplication later. // This segment is to get a reasonable initial estimate for the Babylonian method. With a bad // start, the correct # of bits increases ~linearly each iteration instead of ~quadratically. // We check y >= 2^(k + 8) but shift right by k bits // each branch to ensure that if x >= 256, then y >= 256. if iszero(lt(y, 0x10000000000000000000000000000000000)) { y := shr(128, y) z := shl(64, z) } if iszero(lt(y, 0x1000000000000000000)) { y := shr(64, y) z := shl(32, z) } if iszero(lt(y, 0x10000000000)) { y := shr(32, y) z := shl(16, z) } if iszero(lt(y, 0x1000000)) { y := shr(16, y) z := shl(8, z) } // Goal was to get z*z*y within a small factor of x. More iterations could // get y in a tighter range. Currently, we will have y in [256, 256*2^16). // We ensured y >= 256 so that the relative difference between y and y+1 is small. // That's not possible if x < 256 but we can just verify those cases exhaustively. // Now, z*z*y <= x < z*z*(y+1), and y <= 2^(16+8), and either y >= 256, or x < 256. // Correctness can be checked exhaustively for x < 256, so we assume y >= 256. // Then z*sqrt(y) is within sqrt(257)/sqrt(256) of sqrt(x), or about 20bps. // For s in the range [1/256, 256], the estimate f(s) = (181/1024) * (s+1) is in the range // (1/2.84 * sqrt(s), 2.84 * sqrt(s)), with largest error when s = 1 and when s = 256 or 1/256. // Since y is in [256, 256*2^16), let a = y/65536, so that a is in [1/256, 256). Then we can estimate // sqrt(y) using sqrt(65536) * 181/1024 * (a + 1) = 181/4 * (y + 65536)/65536 = 181 * (y + 65536)/2^18. // There is no overflow risk here since y < 2^136 after the first branch above. z := shr(18, mul(z, add(y, 65536))) // A mul() is saved from starting z at 181. // Given the worst case multiplicative error of 2.84 above, 7 iterations should be enough. z := shr(1, add(z, div(x, z))) z := shr(1, add(z, div(x, z))) z := shr(1, add(z, div(x, z))) z := shr(1, add(z, div(x, z))) z := shr(1, add(z, div(x, z))) z := shr(1, add(z, div(x, z))) z := shr(1, add(z, div(x, z))) // If x+1 is a perfect square, the Babylonian method cycles between // floor(sqrt(x)) and ceil(sqrt(x)). This statement ensures we return floor. // See: https://en.wikipedia.org/wiki/Integer_square_root#Using_only_integer_division // Since the ceil is rare, we save gas on the assignment and repeat division in the rare case. // If you don't care whether the floor or ceil square root is returned, you can remove this statement. z := sub(z, lt(div(x, z), z)) } } function unsafeMod(uint256 x, uint256 y) internal pure returns (uint256 z) { /// @solidity memory-safe-assembly assembly { // Mod x by y. Note this will return // 0 instead of reverting if y is zero. z := mod(x, y) } } function unsafeDiv(uint256 x, uint256 y) internal pure returns (uint256 r) { /// @solidity memory-safe-assembly assembly { // Divide x by y. Note this will return // 0 instead of reverting if y is zero. r := div(x, y) } } function unsafeDivUp(uint256 x, uint256 y) internal pure returns (uint256 z) { /// @solidity memory-safe-assembly assembly { // Add 1 to x * y if x % y > 0. Note this will // return 0 instead of reverting if y is zero. z := add(gt(mod(x, y), 0), div(x, y)) } } }
// SPDX-License-Identifier: AGPL-3.0-only pragma solidity >=0.8.0; /// @notice Gas optimized reentrancy protection for smart contracts. /// @author Solmate (https://github.com/transmissions11/solmate/blob/main/src/utils/ReentrancyGuard.sol) /// @author Modified from OpenZeppelin (https://github.com/OpenZeppelin/openzeppelin-contracts/blob/master/contracts/security/ReentrancyGuard.sol) abstract contract ReentrancyGuard { uint256 private locked = 1; modifier nonReentrant() virtual { require(locked == 1, "REENTRANCY"); locked = 2; _; locked = 1; } }
// SPDX-License-Identifier: AGPL-3.0-only pragma solidity >=0.8.0; import {ERC20} from "../tokens/ERC20.sol"; /// @notice Safe ETH and ERC20 transfer library that gracefully handles missing return values. /// @author Solmate (https://github.com/transmissions11/solmate/blob/main/src/utils/SafeTransferLib.sol) /// @dev Use with caution! Some functions in this library knowingly create dirty bits at the destination of the free memory pointer. /// @dev Note that none of the functions in this library check that a token has code at all! That responsibility is delegated to the caller. library SafeTransferLib { /*////////////////////////////////////////////////////////////// ETH OPERATIONS //////////////////////////////////////////////////////////////*/ function safeTransferETH(address to, uint256 amount) internal { bool success; /// @solidity memory-safe-assembly assembly { // Transfer the ETH and store if it succeeded or not. success := call(gas(), to, amount, 0, 0, 0, 0) } require(success, "ETH_TRANSFER_FAILED"); } /*////////////////////////////////////////////////////////////// ERC20 OPERATIONS //////////////////////////////////////////////////////////////*/ function safeTransferFrom( ERC20 token, address from, address to, uint256 amount ) internal { bool success; /// @solidity memory-safe-assembly assembly { // Get a pointer to some free memory. let freeMemoryPointer := mload(0x40) // Write the abi-encoded calldata into memory, beginning with the function selector. mstore(freeMemoryPointer, 0x23b872dd00000000000000000000000000000000000000000000000000000000) mstore(add(freeMemoryPointer, 4), and(from, 0xffffffffffffffffffffffffffffffffffffffff)) // Append and mask the "from" argument. mstore(add(freeMemoryPointer, 36), and(to, 0xffffffffffffffffffffffffffffffffffffffff)) // Append and mask the "to" argument. mstore(add(freeMemoryPointer, 68), amount) // Append the "amount" argument. Masking not required as it's a full 32 byte type. success := and( // Set success to whether the call reverted, if not we check it either // returned exactly 1 (can't just be non-zero data), or had no return data. or(and(eq(mload(0), 1), gt(returndatasize(), 31)), iszero(returndatasize())), // We use 100 because the length of our calldata totals up like so: 4 + 32 * 3. // We use 0 and 32 to copy up to 32 bytes of return data into the scratch space. // Counterintuitively, this call must be positioned second to the or() call in the // surrounding and() call or else returndatasize() will be zero during the computation. call(gas(), token, 0, freeMemoryPointer, 100, 0, 32) ) } require(success, "TRANSFER_FROM_FAILED"); } function safeTransfer( ERC20 token, address to, uint256 amount ) internal { bool success; /// @solidity memory-safe-assembly assembly { // Get a pointer to some free memory. let freeMemoryPointer := mload(0x40) // Write the abi-encoded calldata into memory, beginning with the function selector. mstore(freeMemoryPointer, 0xa9059cbb00000000000000000000000000000000000000000000000000000000) mstore(add(freeMemoryPointer, 4), and(to, 0xffffffffffffffffffffffffffffffffffffffff)) // Append and mask the "to" argument. mstore(add(freeMemoryPointer, 36), amount) // Append the "amount" argument. Masking not required as it's a full 32 byte type. success := and( // Set success to whether the call reverted, if not we check it either // returned exactly 1 (can't just be non-zero data), or had no return data. or(and(eq(mload(0), 1), gt(returndatasize(), 31)), iszero(returndatasize())), // We use 68 because the length of our calldata totals up like so: 4 + 32 * 2. // We use 0 and 32 to copy up to 32 bytes of return data into the scratch space. // Counterintuitively, this call must be positioned second to the or() call in the // surrounding and() call or else returndatasize() will be zero during the computation. call(gas(), token, 0, freeMemoryPointer, 68, 0, 32) ) } require(success, "TRANSFER_FAILED"); } function safeApprove( ERC20 token, address to, uint256 amount ) internal { bool success; /// @solidity memory-safe-assembly assembly { // Get a pointer to some free memory. let freeMemoryPointer := mload(0x40) // Write the abi-encoded calldata into memory, beginning with the function selector. mstore(freeMemoryPointer, 0x095ea7b300000000000000000000000000000000000000000000000000000000) mstore(add(freeMemoryPointer, 4), and(to, 0xffffffffffffffffffffffffffffffffffffffff)) // Append and mask the "to" argument. mstore(add(freeMemoryPointer, 36), amount) // Append the "amount" argument. Masking not required as it's a full 32 byte type. success := and( // Set success to whether the call reverted, if not we check it either // returned exactly 1 (can't just be non-zero data), or had no return data. or(and(eq(mload(0), 1), gt(returndatasize(), 31)), iszero(returndatasize())), // We use 68 because the length of our calldata totals up like so: 4 + 32 * 2. // We use 0 and 32 to copy up to 32 bytes of return data into the scratch space. // Counterintuitively, this call must be positioned second to the or() call in the // surrounding and() call or else returndatasize() will be zero during the computation. call(gas(), token, 0, freeMemoryPointer, 68, 0, 32) ) } require(success, "APPROVE_FAILED"); } }
// SPDX-License-Identifier: AGPL-3.0 pragma solidity ^0.8.20; import "../loans/IMultiSourceLoan.sol"; /// @title Interface for Loan Offer Validators. /// @author Florida St /// @notice Verify the given `_offer` is valid for `_tokenId` and `_validatorData`. interface IOfferValidator { /// @notice Validate a loan offer. function validateOffer(IMultiSourceLoan.LoanOffer calldata _offer, uint256 _tokenId, bytes calldata _validatorData) external view; }
// SPDX-License-Identifier: AGPL-3.0 pragma solidity ^0.8.20; /// @title NFT Flash Action Interface /// @author Florida St /// @notice Interface for Flash Actions on NFTs in outstanding loans. interface INFTFlashAction { error InvalidOwnerError(); /// @notice Execute an arbitrary flash action on a given NFT. This contract owns it and must return it. /// @param _collection The NFT collection. /// @param _tokenId The NFT token ID. /// @param _target The target contract. /// @param _data The data to send to the target. function execute(address _collection, uint256 _tokenId, address _target, bytes calldata _data) external; }
// SPDX-License-Identifier: AGPL-3.0 pragma solidity ^0.8.21; /// @title Multi Source Loan Interface /// @author Florida St /// @notice A multi source loan is one with multiple tranches. interface ILoanManager { struct ProposedCaller { address caller; bool isLoanContract; } /// @notice Validate an offer. Can only be called by an accepted caller. /// @param _tokenId The token id. /// @param _offer The offer to validate. /// @param _protocolFee The protocol fee. function validateOffer(uint256 _tokenId, bytes calldata _offer, uint256 _protocolFee) external; /// @notice Update the offer handler. /// @param _offerHandler The new offer handler. function updateOfferHandler(address _offerHandler) external; /// @notice Get the offer handler setter. /// @dev Had to take this out from the contract because of size issues. /// @return The offer handler setter. function getParameterSetter() external view returns (address); /// @notice Add allowed callers. /// @param _callers The callers to add. function addCallers(ProposedCaller[] calldata _callers) external; /// @notice Called on loan repayment. /// @param _loanId The loan id. /// @param _principalAmount The principal amount. /// @param _apr The APR. /// @param _accruedInterest The accrued interest. /// @param _protocolFee The protocol fee. /// @param _startTime The start time. function loanRepayment( uint256 _loanId, uint256 _principalAmount, uint256 _apr, uint256 _accruedInterest, uint256 _protocolFee, uint256 _startTime ) external; /// @notice Called on loan liquidation. /// @param _loanAddress The address of the loan contract since this might be called by a liquidator. /// @param _loanId The loan id. /// @param _principalAmount The principal amount. /// @param _apr The APR. /// @param _accruedInterest The accrued interest. /// @param _protocolFee The protocol fee. /// @param _received The received amount (from liquidation proceeds) /// @param _startTime The start time. function loanLiquidation( address _loanAddress, uint256 _loanId, uint256 _principalAmount, uint256 _apr, uint256 _accruedInterest, uint256 _protocolFee, uint256 _received, uint256 _startTime ) external; }
// SPDX-License-Identifier: AGPL-3.0 pragma solidity ^0.8.20; /// @title Interface Loan Manager Registry /// @author Florida St /// @notice Interface for a Loan Manager Registry. interface ILoanManagerRegistry { /// @notice Add a loan manager to the registry /// @param _loanManager Address of the loan manager function addLoanManager(address _loanManager) external; /// @notice Remove a loan manager from the registry /// @param _loanManager Address of the loan manager function removeLoanManager(address _loanManager) external; /// @notice Check if a loan manager is registered /// @param _loanManager Address of the loan manager /// @return True if the loan manager is registered function isLoanManager(address _loanManager) external view returns (bool); }
// SPDX-License-Identifier: AGPL-3.0 pragma solidity ^0.8.21; import "./IBaseLoan.sol"; /// @title Multi Source Loan Interface /// @author Florida St /// @notice A multi source loan is one with multiple tranches. interface IMultiSourceLoan { /// @notice Borrowers receive offers that are then validated. /// @dev Setting the nftCollateralTokenId to 0 triggers validation through `validators`. /// @param offerId Offer ID. Used for canceling/setting as executed. /// @param lender Lender of the offer. /// @param fee Origination fee. /// @param capacity Capacity of the offer. /// @param nftCollateralAddress Address of the NFT collateral. /// @param nftCollateralTokenId NFT collateral token ID. /// @param principalAddress Address of the principal. /// @param principalAmount Principal amount of the loan. /// @param aprBps APR in BPS. /// @param expirationTime Expiration time of the offer. /// @param duration Duration of the loan in seconds. /// @param maxSeniorRepayment Max amount of senior capital ahead (principal + interest). /// @param validators Arbitrary contract to validate offers implementing `IBaseOfferValidator`. struct LoanOffer { uint256 offerId; address lender; uint256 fee; uint256 capacity; address nftCollateralAddress; uint256 nftCollateralTokenId; address principalAddress; uint256 principalAmount; uint256 aprBps; uint256 expirationTime; uint256 duration; uint256 maxSeniorRepayment; IBaseLoan.OfferValidator[] validators; } /// @notice Offer + how much will be filled (always <= principalAmount). /// @param offer Offer. /// @param amount Amount to be filled. struct OfferExecution { LoanOffer offer; uint256 amount; bytes lenderOfferSignature; } /// @notice Offer + necessary fields to execute a specific loan. This has a separate expirationTime to avoid /// someone holding an offer and executing much later, without the borrower's awareness. /// @dev It's advised that borrowers only set an expirationTime close to the actual time they will execute the loan /// to avoid replays. /// @param offerExecution List of offers to be filled and amount for each. /// @param tokenId NFT collateral token ID. /// @param amount The amount the borrower is willing to take (must be <= _loanOffer principalAmount) /// @param expirationTime Expiration time of the signed offer by the borrower. /// @param callbackData Data to pass to the callback. struct ExecutionData { OfferExecution[] offerExecution; uint256 tokenId; uint256 duration; uint256 expirationTime; address principalReceiver; bytes callbackData; } /// @param executionData Execution data. /// @param borrower Address that owns the NFT and will take over the loan. /// @param borrowerOfferSignature Signature of the offer (signed by borrower). /// @param callbackData Whether to call the afterPrincipalTransfer callback struct LoanExecutionData { ExecutionData executionData; address borrower; bytes borrowerOfferSignature; } /// @param loanId Loan ID. /// @param callbackData Whether to call the afterNFTTransfer callback /// @param shouldDelegate Whether to delegate ownership of the NFT (avoid seaport flags). struct SignableRepaymentData { uint256 loanId; bytes callbackData; bool shouldDelegate; } /// @param loan Loan. /// @param borrowerLoanSignature Signature of the loan (signed by borrower). struct LoanRepaymentData { SignableRepaymentData data; Loan loan; bytes borrowerSignature; } /// @notice Tranches have different seniority levels. /// @param loanId Loan ID. /// @param floor Amount of principal more senior to this tranche. /// @param principalAmount Total principal in this tranche. /// @param lender Lender for this given tranche. /// @param accruedInterest Accrued Interest. /// @param startTime Start Time. Either the time at which the loan initiated / was refinanced. /// @param aprBps APR in basis points. struct Tranche { uint256 loanId; uint256 floor; uint256 principalAmount; address lender; uint256 accruedInterest; uint256 startTime; uint256 aprBps; } /// @dev Principal Amount is equal to the sum of all tranches principalAmount. /// We keep it for caching purposes. Since we are not saving this on chain but the hash, /// it does not have a huge impact on gas. /// @param borrower Borrower. /// @param nftCollateralTokenId NFT Collateral Token ID. /// @param nftCollateralAddress NFT Collateral Address. /// @param principalAddress Principal Address. /// @param principalAmount Principal Amount. /// @param startTime Start Time. /// @param duration Duration. /// @param tranche Tranches. /// @param protocolFee Protocol Fee. struct Loan { address borrower; uint256 nftCollateralTokenId; address nftCollateralAddress; address principalAddress; uint256 principalAmount; uint256 startTime; uint256 duration; Tranche[] tranche; uint256 protocolFee; } /// @notice Renegotiation offer. /// @param renegotiationId Renegotiation ID. /// @param loanId Loan ID. /// @param lender Lender. /// @param fee Fee. /// @param trancheIndex Tranche Indexes to be refinanced. /// @param principalAmount Principal Amount. If more than one tranche, it must be the sum. /// @param aprBps APR in basis points. /// @param expirationTime Expiration Time. /// @param duration Duration. struct RenegotiationOffer { uint256 renegotiationId; uint256 loanId; address lender; uint256 fee; uint256[] trancheIndex; uint256 principalAmount; uint256 aprBps; uint256 expirationTime; uint256 duration; } event LoanLiquidated(uint256 loanId); event LoanEmitted(uint256 loanId, uint256[] offerId, Loan loan, uint256 fee); event LoanRefinanced(uint256 renegotiationId, uint256 oldLoanId, uint256 newLoanId, Loan loan, uint256 fee); event LoanRepaid(uint256 loanId, uint256 totalRepayment, uint256 fee); event LoanRefinancedFromNewOffers( uint256 loanId, uint256 newLoanId, Loan loan, uint256[] offerIds, uint256 totalFee ); event Delegated(uint256 loanId, address delegate, bytes32 _rights, bool value); event FlashActionContractUpdated(address newFlashActionContract); event FlashActionExecuted(uint256 loanId, address target, bytes data); event RevokeDelegate(address delegate, address collection, uint256 tokenId, bytes32 _rights); event MinLockPeriodUpdated(uint256 minLockPeriod); /// @notice Call by the borrower when emiting a new loan. /// @param _loanExecutionData Loan execution data. /// @return loanId Loan ID. /// @return loan Loan. function emitLoan(LoanExecutionData calldata _loanExecutionData) external returns (uint256, Loan memory); /// @notice Refinance whole loan (leaving just one tranche). /// @param _renegotiationOffer Offer to refinance a loan. /// @param _loan Current loan. /// @param _renegotiationOfferSignature Signature of the offer. /// @return loanId New Loan Id, New Loan. function refinanceFull( RenegotiationOffer calldata _renegotiationOffer, Loan memory _loan, bytes calldata _renegotiationOfferSignature ) external returns (uint256, Loan memory); /// @notice Add a new tranche to a loan. /// @param _renegotiationOffer Offer for new tranche. /// @param _loan Current loan. /// @param _renegotiationOfferSignature Signature of the offer. /// @return loanId New Loan Id /// @return loan New Loan. function addNewTranche( RenegotiationOffer calldata _renegotiationOffer, Loan memory _loan, bytes calldata _renegotiationOfferSignature ) external returns (uint256, Loan memory); /// @notice Refinance a loan partially. It can only be called by the new lender /// (they are always a strict improvement on apr). /// @param _renegotiationOffer Offer to refinance a loan partially. /// @param _loan Current loan. /// @return loanId New Loan Id, New Loan. /// @return loan New Loan. function refinancePartial(RenegotiationOffer calldata _renegotiationOffer, Loan memory _loan) external returns (uint256, Loan memory); /// @notice Refinance a loan from LoanExecutionData. We let borrowers use outstanding offers for new loans /// to refinance their current loan. /// @param _loanId Loan ID. /// @param _loan Current loan. /// @param _loanExecutionData Loan Execution Data. /// @return loanId New Loan Id. /// @return loan New Loan. function refinanceFromLoanExecutionData( uint256 _loanId, Loan calldata _loan, LoanExecutionData calldata _loanExecutionData ) external returns (uint256, Loan memory); /// @notice Repay loan. Interest is calculated pro-rata based on time. Lender is defined by nft ownership. /// @param _repaymentData Repayment data. function repayLoan(LoanRepaymentData calldata _repaymentData) external; /// @notice Call when a loan is past its due date. /// @param _loanId Loan ID. /// @param _loan Loan. /// @return Liquidation Struct of the liquidation. function liquidateLoan(uint256 _loanId, Loan calldata _loan) external returns (bytes memory); /// @return getMaxTranches Maximum number of tranches per loan. function getMaxTranches() external view returns (uint256); /// @notice Set min lock period (in BPS). /// @param _minLockPeriod Min lock period. function setMinLockPeriod(uint256 _minLockPeriod) external; /// @notice Get min lock period (in BPS). /// @return minLockPeriod Min lock period. function getMinLockPeriod() external view returns (uint256); /// @notice Get delegation registry. /// @return delegateRegistry Delegate registry. function getDelegateRegistry() external view returns (address); /// @notice Delegate ownership. /// @param _loanId Loan ID. /// @param _loan Loan. /// @param _rights Delegation Rights. Empty for all. /// @param _delegate Delegate address. /// @param _value True if delegate, false if undelegate. function delegate(uint256 _loanId, Loan calldata _loan, address _delegate, bytes32 _rights, bool _value) external; /// @notice Anyone can reveke a delegation on an NFT that's no longer in escrow. /// @param _delegate Delegate address. /// @param _collection Collection address. /// @param _tokenId Token ID. /// @param _rights Delegation Rights. Empty for all. function revokeDelegate(address _delegate, address _collection, uint256 _tokenId, bytes32 _rights) external; /// @notice Get Flash Action Contract. /// @return flashActionContract Flash Action Contract. function getFlashActionContract() external view returns (address); /// @notice Update Flash Action Contract. /// @param _newFlashActionContract Flash Action Contract. function setFlashActionContract(address _newFlashActionContract) external; /// @notice Get Loan Hash. /// @param _loanId Loan ID. /// @return loanHash Loan Hash. function getLoanHash(uint256 _loanId) external view returns (bytes32); /// @notice Transfer NFT to the flash action contract (expected use cases here are for airdrops and similar scenarios). /// The flash action contract would implement specific interactions with given contracts. /// Only the the borrower can call this function for a given loan. By the end of the transaction, the NFT must have /// been returned to escrow. /// @param _loanId Loan ID. /// @param _loan Loan. /// @param _target Target address for the flash action contract to interact with. /// @param _data Data to be passed to be passed to the ultimate contract. function executeFlashAction(uint256 _loanId, Loan calldata _loan, address _target, bytes calldata _data) external; /// @notice Called by the liquidator for accounting purposes. /// @param _loanId The id of the loan. /// @param _loan The loan object. function loanLiquidated(uint256 _loanId, Loan calldata _loan) external; }
// SPDX-License-Identifier: AGPL-3.0 pragma solidity ^0.8.21; import "../../interfaces/loans/IMultiSourceLoan.sol"; import "../../interfaces/loans/IBaseLoan.sol"; import "../../interfaces/IAuctionLoanLiquidator.sol"; library Hash { // keccak256("OfferValidator(address validator,bytes arguments)") bytes32 private constant _VALIDATOR_HASH = 0x4def3e04bd42194484d5f8a5b268ec0df03b9d9d0402606fe3100023c5d79ac4; // keccak256("LoanOffer(uint256 offerId,address lender,uint256 fee,uint256 capacity,address nftCollateralAddress,uint256 nftCollateralTokenId,address principalAddress,uint256 principalAmount,uint256 aprBps,uint256 expirationTime,uint256 duration,uint256 maxSeniorRepayment,OfferValidator[] validators)OfferValidator(address validator,bytes arguments)") bytes32 private constant _LOAN_OFFER_HASH = 0xa87df46e2d2684eb0bbc7abfb05483167cdccac6d7302078a9eaad540c119958; // keccak256("OfferExecution(LoanOffer offer,uint256 amount,bytes lenderOfferSignature)LoanOffer(uint256 offerId,address lender,uint256 fee,uint256 capacity,address nftCollateralAddress,uint256 nftCollateralTokenId,address principalAddress,uint256 principalAmount,uint256 aprBps,uint256 expirationTime,uint256 duration,uint256 maxSeniorRepayment,OfferValidator[] validators)OfferValidator(address validator,bytes arguments)") bytes32 private constant _OFFER_EXECUTION_HASH = 0x00c14ad24a24ef957b8af9ebdfbc5d353bba0d3b20bbd97fb243c9f5fb361282; /// keccak256("ExecutionData(OfferExecution[] offerExecution,uint256 tokenId,uint256 duration,uint256 expirationTime,address principalReceiver,bytes callbackData)OfferExecution(LoanOffer offer,uint256 amount,bytes lenderOfferSignature)LoanOffer(uint256 offerId,address lender,uint256 fee,uint256 capacity,address nftCollateralAddress,uint256 nftCollateralTokenId,address principalAddress,uint256 principalAmount,uint256 aprBps,uint256 expirationTime,uint256 duration,uint256 maxSeniorRepayment,OfferValidator[] validators)OfferValidator(address validator,bytes arguments)") bytes32 private constant _EXECUTION_DATA_HASH = 0xa5cb06a0c5f03000a6afa6b0d5080d0f863338257beb253058bc2c184ad7d4e1; /// keccak256("SignableRepaymentData(uint256 loanId,bytes callbackData,bool shouldDelegate)") bytes32 private constant _SIGNABLE_REPAYMENT_DATA_HASH = 0x41277b3c1cbe08ea7bbdd10a13f24dc956f3936bf46526f904c73697d9958e0c; // keccak256("Loan(address borrower,uint256 nftCollateralTokenId,address nftCollateralAddress,address principalAddress,uint256 principalAmount,uint256 startTime,uint256 duration,Tranche[] tranche,uint256 protocolFee)Tranche(uint256 floor,uint256 principalAmount,Source[] source)Source(uint256 loanId,address lender,uint256 principalAmount,uint256 accruedInterest,uint256 startTime,uint256 aprBps)") bytes32 private constant _MULTI_SOURCE_LOAN_HASH = 0x47dba7e6940f0063b21c2ef8f7b0beaf1a2f4c2f84144c36b274ceec12e99b57; /// keccak256("Tranche(uint256 loanId,uint256 floor,uint256 principalAmount,address lender,uint256 accruedInterest,uint256 startTime,uint256 aprBps)") bytes32 private constant _TRANCHE_HASH = 0x6ac594952a72f2e6b24efaf9744b05c23b1b92ce25aa97d18a4338f484c41b95; /// keccak256("RenegotiationOffer(uint256 renegotiationId,uint256 loanId,address lender,uint256 fee,uint256[] trancheIndex,uint256 principalAmount,uint256 aprBps,uint256 expirationTime,uint256 duration)") bytes32 private constant _MULTI_RENEGOTIATION_OFFER_HASH = 0x986a160abc209a64a5b0786817ff0aa7a5f5737a4ee6a95197f86290598cd03d; /// keccak256("Auction(address loanAddress,uint256 loanId,uint256 highestBid,uint256 triggerFee,uint256 minBid,address highestBidder,uint96 duration,address asset,uint96 startTime,address originator,uint96 lastBidTime)") bytes32 private constant _AUCTION_HASH = 0x091bb2c766793330514b24dc458b085f596716d69fcb631d53788558ff148646; function hash(IMultiSourceLoan.LoanOffer memory _loanOffer) internal pure returns (bytes32) { bytes memory encodedValidators; uint256 totalValidators = _loanOffer.validators.length; for (uint256 i = 0; i < totalValidators;) { encodedValidators = abi.encodePacked(encodedValidators, _hashValidator(_loanOffer.validators[i])); unchecked { ++i; } } return keccak256( abi.encode( _LOAN_OFFER_HASH, _loanOffer.offerId, _loanOffer.lender, _loanOffer.fee, _loanOffer.capacity, _loanOffer.nftCollateralAddress, _loanOffer.nftCollateralTokenId, _loanOffer.principalAddress, _loanOffer.principalAmount, _loanOffer.aprBps, _loanOffer.expirationTime, _loanOffer.duration, _loanOffer.maxSeniorRepayment, keccak256(encodedValidators) ) ); } function hash(IMultiSourceLoan.ExecutionData memory _executionData) internal pure returns (bytes32) { bytes memory encodedOfferExecution; uint256 totalOfferExecution = _executionData.offerExecution.length; for (uint256 i = 0; i < totalOfferExecution;) { encodedOfferExecution = abi.encodePacked(encodedOfferExecution, _hashOfferExecution(_executionData.offerExecution[i])); unchecked { ++i; } } return keccak256( abi.encode( _EXECUTION_DATA_HASH, keccak256(encodedOfferExecution), _executionData.tokenId, _executionData.duration, _executionData.expirationTime, _executionData.principalReceiver, keccak256(_executionData.callbackData) ) ); } function hash(IMultiSourceLoan.SignableRepaymentData memory _repaymentData) internal pure returns (bytes32) { return keccak256( abi.encode( _SIGNABLE_REPAYMENT_DATA_HASH, _repaymentData.loanId, keccak256(_repaymentData.callbackData), _repaymentData.shouldDelegate ) ); } function hash(IMultiSourceLoan.Loan memory _loan) internal pure returns (bytes32) { bytes memory trancheHashes; uint256 totalTranches = _loan.tranche.length; for (uint256 i; i < totalTranches;) { trancheHashes = abi.encodePacked(trancheHashes, _hashTranche(_loan.tranche[i])); unchecked { ++i; } } return keccak256( abi.encode( _MULTI_SOURCE_LOAN_HASH, _loan.borrower, _loan.nftCollateralTokenId, _loan.nftCollateralAddress, _loan.principalAddress, _loan.principalAmount, _loan.startTime, _loan.duration, keccak256(trancheHashes), _loan.protocolFee ) ); } function hash(IMultiSourceLoan.RenegotiationOffer memory _refinanceOffer) internal pure returns (bytes32) { bytes memory encodedIndexes; uint256 totalIndexes = _refinanceOffer.trancheIndex.length; for (uint256 i = 0; i < totalIndexes;) { encodedIndexes = abi.encodePacked(encodedIndexes, _refinanceOffer.trancheIndex[i]); unchecked { ++i; } } return keccak256( abi.encode( _MULTI_RENEGOTIATION_OFFER_HASH, _refinanceOffer.renegotiationId, _refinanceOffer.loanId, _refinanceOffer.lender, _refinanceOffer.fee, keccak256(encodedIndexes), _refinanceOffer.principalAmount, _refinanceOffer.aprBps, _refinanceOffer.expirationTime, _refinanceOffer.duration ) ); } function hash(IAuctionLoanLiquidator.Auction memory _auction) internal pure returns (bytes32) { return keccak256( abi.encode( _AUCTION_HASH, _auction.loanAddress, _auction.loanId, _auction.highestBid, _auction.triggerFee, _auction.minBid, _auction.highestBidder, _auction.duration, _auction.asset, _auction.startTime, _auction.originator, _auction.lastBidTime ) ); } function _hashTranche(IMultiSourceLoan.Tranche memory _tranche) private pure returns (bytes32) { return keccak256( abi.encode( _TRANCHE_HASH, _tranche.loanId, _tranche.floor, _tranche.principalAmount, _tranche.lender, _tranche.accruedInterest, _tranche.startTime, _tranche.aprBps ) ); } function _hashValidator(IBaseLoan.OfferValidator memory _validator) private pure returns (bytes32) { return keccak256(abi.encode(_VALIDATOR_HASH, _validator.validator, keccak256(_validator.arguments))); } function _hashOfferExecution(IMultiSourceLoan.OfferExecution memory _offerExecution) private pure returns (bytes32) { return keccak256( abi.encode( _OFFER_EXECUTION_HASH, hash(_offerExecution.offer), _offerExecution.amount, keccak256(_offerExecution.lenderOfferSignature) ) ); } }
// SPDX-License-Identifier: AGPL-3.0 pragma solidity ^0.8.21; import "@solmate/utils/FixedPointMathLib.sol"; import "../../interfaces/loans/IMultiSourceLoan.sol"; import "../../interfaces/loans/IBaseLoan.sol"; library Interest { using FixedPointMathLib for uint256; uint256 private constant _PRECISION = 10000; uint256 private constant _SECONDS_PER_YEAR = 31536000; function getInterest(IMultiSourceLoan.LoanOffer memory _loanOffer) internal pure returns (uint256) { return _getInterest(_loanOffer.principalAmount, _loanOffer.aprBps, _loanOffer.duration); } function getInterest(uint256 _amount, uint256 _aprBps, uint256 _duration) internal pure returns (uint256) { return _getInterest(_amount, _aprBps, _duration); } function getTotalOwed(IMultiSourceLoan.Loan memory _loan, uint256 _timestamp) internal pure returns (uint256) { uint256 owed = 0; for (uint256 i = 0; i < _loan.tranche.length;) { IMultiSourceLoan.Tranche memory tranche = _loan.tranche[i]; owed += tranche.principalAmount + tranche.accruedInterest + _getInterest(tranche.principalAmount, tranche.aprBps, _timestamp - tranche.startTime); unchecked { ++i; } } return owed; } function _getInterest(uint256 _amount, uint256 _aprBps, uint256 _duration) private pure returns (uint256) { return _amount.mulDivUp(_aprBps * _duration, _PRECISION * _SECONDS_PER_YEAR); } }
// SPDX-License-Identifier: AGPL-3.0 pragma solidity ^0.8.21; import "../interfaces/IMulticall.sol"; /// @title Multicall /// @author Florida St /// @notice Base implementation for multicall. abstract contract Multicall is IMulticall { function multicall(bytes[] calldata data) external payable override returns (bytes[] memory results) { results = new bytes[](data.length); bool success; uint256 totalCalls = data.length; for (uint256 i = 0; i < totalCalls;) { //slither-disable-next-line calls-loop,delegatecall-loop (success, results[i]) = address(this).delegatecall(data[i]); if (!success) { revert MulticallFailed(i, results[i]); } unchecked { ++i; } } } }
// SPDX-License-Identifier: AGPL-3.0 pragma solidity ^0.8.21; import "@openzeppelin/utils/cryptography/MessageHashUtils.sol"; import "@openzeppelin/interfaces/IERC1271.sol"; import "@solmate/auth/Owned.sol"; import "@solmate/tokens/ERC721.sol"; import "@solmate/utils/FixedPointMathLib.sol"; import "../../interfaces/loans/IBaseLoan.sol"; import "../utils/Hash.sol"; import "../AddressManager.sol"; import "../LiquidationHandler.sol"; /// @title BaseLoan /// @author Florida St /// @notice Base implementation that we expect all loans to share. Offers can either be /// for new loans or renegotiating existing ones. /// Offers are signed off-chain. /// Offers have a nonce associated that is used for cancelling and /// marking as executed. abstract contract BaseLoan is ERC721TokenReceiver, IBaseLoan, LiquidationHandler { using FixedPointMathLib for uint256; using InputChecker for address; using MessageHashUtils for bytes32; /// @notice Used in compliance with EIP712 uint256 internal immutable INITIAL_CHAIN_ID; bytes32 public immutable INITIAL_DOMAIN_SEPARATOR; bytes4 internal constant MAGICVALUE_1271 = 0x1626ba7e; /// @notice Precision used for calculating interests. uint256 internal constant _PRECISION = 10000; bytes public constant VERSION = "3"; /// @notice Minimum improvement (in BPS) required for a strict improvement. uint256 internal _minImprovementApr = 1000; string public name; /// @notice Total number of loans issued. Given it's a serial value, we use it /// as loan id. uint256 public override getTotalLoansIssued; /// @notice Offer capacity mapping(address user => mapping(uint256 offerId => uint256 used)) internal _used; /// @notice Used for validate off chain maker offers / canceling one mapping(address user => mapping(uint256 offerId => bool notActive)) public isOfferCancelled; /// @notice Used for validating off chain maker offers / canceling all mapping(address user => uint256 minOfferId) public minOfferId; /// @notice Used in a similar way as `isOfferCancelled` to handle renegotiations. mapping(address user => mapping(uint256 renegotiationIf => bool notActive)) public isRenegotiationOfferCancelled; /// @notice Loans are only denominated in whitelisted addresses. Within each struct, /// we save those as their `uint` representation. AddressManager internal immutable _currencyManager; /// @notice Only whilteslited collections are accepted as collateral. Within each struct, /// we save those as their `uint` representation. AddressManager internal immutable _collectionManager; event OfferCancelled(address lender, uint256 offerId); event AllOffersCancelled(address lender, uint256 minOfferId); event RenegotiationOfferCancelled(address lender, uint256 renegotiationId); event MinAprImprovementUpdated(uint256 _minimum); error CancelledOrExecutedOfferError(address _lender, uint256 _offerId); error ExpiredOfferError(uint256 _expirationTime); error LowOfferIdError(address _lender, uint256 _newMinOfferId, uint256 _minOfferId); error LowRenegotiationOfferIdError(address _lender, uint256 _newMinRenegotiationOfferId, uint256 _minOfferId); error ZeroInterestError(); error InvalidSignatureError(); error CurrencyNotWhitelistedError(); error CollectionNotWhitelistedError(); error MaxCapacityExceededError(); error InvalidLoanError(uint256 _loanId); error NotStrictlyImprovedError(); error InvalidAmountError(uint256 _amount, uint256 _principalAmount); /// @notice Constructor /// @param _name The name of the loan contract /// @param currencyManager The address of the currency manager /// @param collectionManager The address of the collection manager /// @param protocolFee The protocol fee /// @param loanLiquidator The liquidator contract /// @param owner The owner of the contract /// @param minWaitTime The time to wait before a new owner can be set constructor( string memory _name, address currencyManager, address collectionManager, ProtocolFee memory protocolFee, address loanLiquidator, address owner, uint256 minWaitTime ) LiquidationHandler(owner, minWaitTime, loanLiquidator, protocolFee) { name = _name; currencyManager.checkNotZero(); collectionManager.checkNotZero(); _currencyManager = AddressManager(currencyManager); _collectionManager = AddressManager(collectionManager); INITIAL_CHAIN_ID = block.chainid; INITIAL_DOMAIN_SEPARATOR = _computeDomainSeparator(); } /// @return The minimum improvement for a loan to be considered strictly better. function getMinImprovementApr() external view returns (uint256) { return _minImprovementApr; } /// @notice Updates the minimum improvement for a loan to be considered strictly better. /// Only the owner can call this function. /// @param _newMinimum The new minimum improvement. function updateMinImprovementApr(uint256 _newMinimum) external onlyOwner { _minImprovementApr = _newMinimum; emit MinAprImprovementUpdated(_minImprovementApr); } /// @return Address of the currency manager. function getCurrencyManager() external view returns (address) { return address(_currencyManager); } /// @return Address of the collection manager. function getCollectionManager() external view returns (address) { return address(_collectionManager); } /// @inheritdoc IBaseLoan function cancelOffer(uint256 _offerId) external { address user = msg.sender; isOfferCancelled[user][_offerId] = true; emit OfferCancelled(user, _offerId); } /// @inheritdoc IBaseLoan function cancelAllOffers(uint256 _minOfferId) external virtual { address user = msg.sender; uint256 currentMinOfferId = minOfferId[user]; if (currentMinOfferId >= _minOfferId) { revert LowOfferIdError(user, _minOfferId, currentMinOfferId); } minOfferId[user] = _minOfferId; emit AllOffersCancelled(user, _minOfferId); } /// @inheritdoc IBaseLoan function cancelRenegotiationOffer(uint256 _renegotiationId) external virtual { address lender = msg.sender; isRenegotiationOfferCancelled[lender][_renegotiationId] = true; emit RenegotiationOfferCancelled(lender, _renegotiationId); } /// @notice Returns the remaining capacity for a given loan offer. /// @param _lender The address of the lender. /// @param _offerId The id of the offer. /// @return The amount lent out. function getUsedCapacity(address _lender, uint256 _offerId) external view returns (uint256) { return _used[_lender][_offerId]; } /// @notice Get the domain separator requried to comply with EIP-712. function DOMAIN_SEPARATOR() public view returns (bytes32) { return block.chainid == INITIAL_CHAIN_ID ? INITIAL_DOMAIN_SEPARATOR : _computeDomainSeparator(); } /// @notice Call when issuing a new loan to get/set a unique serial id. /// @dev This id should never be 0. /// @return The new loan id. function _getAndSetNewLoanId() internal returns (uint256) { unchecked { return ++getTotalLoansIssued; } } /// @notice Compute domain separator for EIP-712. /// @return The domain separator. function _computeDomainSeparator() private view returns (bytes32) { return keccak256( abi.encode( keccak256("EIP712Domain(string name,string version,uint256 chainId,address verifyingContract)"), keccak256(bytes(name)), keccak256(VERSION), block.chainid, address(this) ) ); } }
// SPDX-License-Identifier: AGPL-3.0 pragma solidity ^0.8.20; import "../../interfaces/ILoanLiquidator.sol"; /// @title Interface for Loans. /// @author Florida St /// @notice Basic Loan interface IBaseLoan { /// @notice Minimum improvement (in BPS) required for a strict improvement. /// @param principalAmount Minimum delta of principal amount. /// @param interest Minimum delta of interest. /// @param duration Minimum delta of duration. struct ImprovementMinimum { uint256 principalAmount; uint256 interest; uint256 duration; } /// @notice Arbitrary contract to validate offers implementing `IBaseOfferValidator`. /// @param validator Address of the validator contract. /// @param arguments Arguments to pass to the validator. struct OfferValidator { address validator; bytes arguments; } /// @notice Total number of loans issued by this contract. function getTotalLoansIssued() external view returns (uint256); /// @notice Cancel offer for `msg.sender`. Each lender has unique offerIds. /// @param _offerId Offer ID. function cancelOffer(uint256 _offerId) external; /// @notice Cancell all offers with offerId < _minOfferId /// @param _minOfferId Minimum offer ID. function cancelAllOffers(uint256 _minOfferId) external; /// @notice Cancel renegotiation offer. Similar to offers. /// @param _renegotiationId Renegotiation offer ID. function cancelRenegotiationOffer(uint256 _renegotiationId) external; }
// SPDX-License-Identifier: AGPL-3.0 pragma solidity ^0.8.20; import "./loans/IMultiSourceLoan.sol"; /// @title Liquidates Collateral for Defaulted Loans using English Auctions. /// @author Florida St /// @notice It liquidates collateral corresponding to defaulted loans /// and sends back the proceeds to the loan contract for distribution. interface IAuctionLoanLiquidator { /// @notice The auction struct. /// @param loanAddress The loan contract address. /// @param loanId The loan id. /// @param highestBid The highest bid. /// @param triggerFee The trigger fee. /// @param minBid The minimum bid. /// @param highestBidder The highest bidder. /// @param duration The auction duration. /// @param asset The asset address. /// @param startTime The auction start time. /// @param originator The address that triggered the liquidation. /// @param lastBidTime The last bid time. struct Auction { address loanAddress; uint256 loanId; uint256 highestBid; uint256 triggerFee; uint256 minBid; address highestBidder; uint96 duration; address asset; uint96 startTime; address originator; uint96 lastBidTime; } /// @notice Add a loan contract to the list of accepted contracts. /// @param _loanContract The loan contract to be added. function addLoanContract(address _loanContract) external; /// @notice Remove a loan contract from the list of accepted contracts. /// @param _loanContract The loan contract to be removed. function removeLoanContract(address _loanContract) external; /// @return The loan contracts that are accepted by this liquidator. function getValidLoanContracts() external view returns (address[] memory); /// @notice Update liquidation distributor. /// @param _liquidationDistributor The new liquidation distributor. function updateLiquidationDistributor(address _liquidationDistributor) external; /// @return liquidationDistributor The liquidation distributor address. function getLiquidationDistributor() external view returns (address); /// @notice Called by the owner to update the trigger fee. /// @param triggerFee The new trigger fee. function updateTriggerFee(uint256 triggerFee) external; /// @return triggerFee The trigger fee. function getTriggerFee() external view returns (uint256); /// @notice When a bid is placed, the contract takes possesion of the bid, and /// if there was a previous bid, it returns that capital to the original /// bidder. /// @param _contract The nft contract address. /// @param _tokenId The nft id. /// @param _auction The auction struct. /// @param _bid The bid amount. /// @return auction The updated auction struct. function placeBid(address _contract, uint256 _tokenId, Auction memory _auction, uint256 _bid) external returns (Auction memory); /// @notice On settlement, the NFT is sent to the highest bidder. /// Calls loan liquidated for accounting purposes. /// @param _auction The auction struct. /// @param _loan The loan struct. function settleAuction(Auction calldata _auction, IMultiSourceLoan.Loan calldata _loan) external; /// @notice The contract has hashes of all auctions to save space (not the actual struct) /// @param _contract The nft contract address. /// @param _tokenId The nft id. /// @return auctionHash The auction hash. function getAuctionHash(address _contract, uint256 _tokenId) external view returns (bytes32); }
// SPDX-License-Identifier: AGPL-3.0 pragma solidity ^0.8.20; interface IMulticall { error MulticallFailed(uint256 i, bytes returndata); /// @notice Call multiple functions in the contract. Revert if one of them fails, return results otherwise. /// @param data Encoded function calls. /// @return results The results of the function calls. function multicall(bytes[] calldata data) external payable returns (bytes[] memory results); }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v5.0.0) (utils/cryptography/MessageHashUtils.sol) pragma solidity ^0.8.20; import {Strings} from "../Strings.sol"; /** * @dev Signature message hash utilities for producing digests to be consumed by {ECDSA} recovery or signing. * * The library provides methods for generating a hash of a message that conforms to the * https://eips.ethereum.org/EIPS/eip-191[ERC-191] and https://eips.ethereum.org/EIPS/eip-712[EIP 712] * specifications. */ library MessageHashUtils { /** * @dev Returns the keccak256 digest of an ERC-191 signed data with version * `0x45` (`personal_sign` messages). * * The digest is calculated by prefixing a bytes32 `messageHash` with * `"\x19Ethereum Signed Message:\n32"` and hashing the result. It corresponds with the * hash signed when using the https://eth.wiki/json-rpc/API#eth_sign[`eth_sign`] JSON-RPC method. * * NOTE: The `messageHash` parameter is intended to be the result of hashing a raw message with * keccak256, although any bytes32 value can be safely used because the final digest will * be re-hashed. * * See {ECDSA-recover}. */ function toEthSignedMessageHash(bytes32 messageHash) internal pure returns (bytes32 digest) { /// @solidity memory-safe-assembly assembly { mstore(0x00, "\x19Ethereum Signed Message:\n32") // 32 is the bytes-length of messageHash mstore(0x1c, messageHash) // 0x1c (28) is the length of the prefix digest := keccak256(0x00, 0x3c) // 0x3c is the length of the prefix (0x1c) + messageHash (0x20) } } /** * @dev Returns the keccak256 digest of an ERC-191 signed data with version * `0x45` (`personal_sign` messages). * * The digest is calculated by prefixing an arbitrary `message` with * `"\x19Ethereum Signed Message:\n" + len(message)` and hashing the result. It corresponds with the * hash signed when using the https://eth.wiki/json-rpc/API#eth_sign[`eth_sign`] JSON-RPC method. * * See {ECDSA-recover}. */ function toEthSignedMessageHash(bytes memory message) internal pure returns (bytes32) { return keccak256(bytes.concat("\x19Ethereum Signed Message:\n", bytes(Strings.toString(message.length)), message)); } /** * @dev Returns the keccak256 digest of an ERC-191 signed data with version * `0x00` (data with intended validator). * * The digest is calculated by prefixing an arbitrary `data` with `"\x19\x00"` and the intended * `validator` address. Then hashing the result. * * See {ECDSA-recover}. */ function toDataWithIntendedValidatorHash(address validator, bytes memory data) internal pure returns (bytes32) { return keccak256(abi.encodePacked(hex"19_00", validator, data)); } /** * @dev Returns the keccak256 digest of an EIP-712 typed data (ERC-191 version `0x01`). * * The digest is calculated from a `domainSeparator` and a `structHash`, by prefixing them with * `\x19\x01` and hashing the result. It corresponds to the hash signed by the * https://eips.ethereum.org/EIPS/eip-712[`eth_signTypedData`] JSON-RPC method as part of EIP-712. * * See {ECDSA-recover}. */ function toTypedDataHash(bytes32 domainSeparator, bytes32 structHash) internal pure returns (bytes32 digest) { /// @solidity memory-safe-assembly assembly { let ptr := mload(0x40) mstore(ptr, hex"19_01") mstore(add(ptr, 0x02), domainSeparator) mstore(add(ptr, 0x22), structHash) digest := keccak256(ptr, 0x42) } } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v5.0.0) (interfaces/IERC1271.sol) pragma solidity ^0.8.20; /** * @dev Interface of the ERC-1271 standard signature validation method for * contracts as defined in https://eips.ethereum.org/EIPS/eip-1271[ERC-1271]. */ interface IERC1271 { /** * @dev Should return whether the signature provided is valid for the provided data * @param hash Hash of the data to be signed * @param signature Signature byte array associated with _data */ function isValidSignature(bytes32 hash, bytes memory signature) external view returns (bytes4 magicValue); }
// SPDX-License-Identifier: AGPL-3.0-only pragma solidity >=0.8.0; /// @notice Simple single owner authorization mixin. /// @author Solmate (https://github.com/transmissions11/solmate/blob/main/src/auth/Owned.sol) abstract contract Owned { /*////////////////////////////////////////////////////////////// EVENTS //////////////////////////////////////////////////////////////*/ event OwnershipTransferred(address indexed user, address indexed newOwner); /*////////////////////////////////////////////////////////////// OWNERSHIP STORAGE //////////////////////////////////////////////////////////////*/ address public owner; modifier onlyOwner() virtual { require(msg.sender == owner, "UNAUTHORIZED"); _; } /*////////////////////////////////////////////////////////////// CONSTRUCTOR //////////////////////////////////////////////////////////////*/ constructor(address _owner) { owner = _owner; emit OwnershipTransferred(address(0), _owner); } /*////////////////////////////////////////////////////////////// OWNERSHIP LOGIC //////////////////////////////////////////////////////////////*/ function transferOwnership(address newOwner) public virtual onlyOwner { owner = newOwner; emit OwnershipTransferred(msg.sender, newOwner); } }
// SPDX-License-Identifier: AGPL-3.0 pragma solidity ^0.8.21; import "@solmate/auth/Owned.sol"; import "@solmate/utils/ReentrancyGuard.sol"; import "./InputChecker.sol"; /// @title AddressManager /// @notice A contract that handles a whitelist of addresses and their indexes. /// @dev We assume no more than 65535 addresses will be added to the directory. contract AddressManager is Owned, ReentrancyGuard { using InputChecker for address; event AddressAdded(address address_added); event AddressRemovedFromWhitelist(address address_removed); event AddressWhitelisted(address address_whitelisted); error AddressAlreadyAddedError(address _address); error AddressNotAddedError(address _address); mapping(address => uint16) private _directory; mapping(uint16 => address) private _inverseDirectory; mapping(address => bool) private _whitelist; uint16 private _lastAdded; constructor(address[] memory _original) Owned(tx.origin) { uint256 total = _original.length; for (uint256 i; i < total;) { _add(_original[i]); unchecked { ++i; } } } /// @notice Adds an address to the directory. If it already exists, /// reverts. It assumes it's whitelisted. /// @param _entry The address to add. /// @return The index of the address in the directory. function add(address _entry) external payable onlyOwner returns (uint16) { return _add(_entry); } /// @notice Whitelist an address that's already part of the directory. /// @param _entry The address to whitelist. function addToWhitelist(address _entry) external payable onlyOwner { if (_directory[_entry] == 0) { revert AddressNotAddedError(_entry); } _whitelist[_entry] = true; emit AddressWhitelisted(_entry); } /// @notice Removes an address from the whitelist. We still keep it /// in the directory since this mapping is relevant across time. /// @param _entry The address to remove from the whitelist. function removeFromWhitelist(address _entry) external payable onlyOwner { _whitelist[_entry] = false; emit AddressRemovedFromWhitelist(_entry); } /// @param _address The address to get the index for. /// @return The index for a given address. function addressToIndex(address _address) external view returns (uint16) { return _directory[_address]; } /// @param _index The index to get the address for. /// @return The address for a given index. function indexToAddress(uint16 _index) external view returns (address) { return _inverseDirectory[_index]; } /// @param _entry The address to check if it's whitelisted. /// @return Whether the address is whitelisted or not. function isWhitelisted(address _entry) external view returns (bool) { return _whitelist[_entry]; } function _add(address _entry) private returns (uint16) { _entry.checkNotZero(); if (_directory[_entry] != 0) { revert AddressAlreadyAddedError(_entry); } unchecked { ++_lastAdded; } _directory[_entry] = _lastAdded; _inverseDirectory[_lastAdded] = _entry; _whitelist[_entry] = true; emit AddressAdded(_entry); return _lastAdded; } }
// SPDX-License-Identifier: AGPL-3.0 pragma solidity ^0.8.21; import "@solmate/auth/Owned.sol"; import "@solmate/tokens/ERC721.sol"; import "@solmate/utils/FixedPointMathLib.sol"; import "@solmate/utils/ReentrancyGuard.sol"; import "../interfaces/ILiquidationHandler.sol"; import "../interfaces/loans/IMultiSourceLoan.sol"; import "./callbacks/CallbackHandler.sol"; import "./InputChecker.sol"; /// @title Liquidation Handler /// @author Florida St /// @notice Liquidation Handler for defaulted loans abstract contract LiquidationHandler is ILiquidationHandler, ReentrancyGuard, CallbackHandler { using InputChecker for address; using FixedPointMathLib for uint256; uint48 public constant MIN_AUCTION_DURATION = 3 days; uint48 public constant MAX_AUCTION_DURATION = 7 days; uint256 public constant MIN_BID_LIQUIDATION = 50; uint256 private constant _BPS = 10000; /// @notice Duration of the auction when a loan defaults requires a liquidation. uint48 internal _liquidationAuctionDuration = 3 days; /// @notice Liquidator used defaulted loans that requires liquidation. address internal _loanLiquidator; event MinBidLiquidationUpdated(uint256 newMinBid); event LoanSentToLiquidator(uint256 loanId, address liquidator); event LoanForeclosed(uint256 loanId); event LiquidationContractUpdated(address liquidator); event LiquidationAuctionDurationUpdated(uint256 newDuration); error LiquidatorOnlyError(address _liquidator); error LoanNotDueError(uint256 _expirationTime); error InvalidDurationError(); /// @notice Constructor /// @param __owner The owner of the contract /// @param _updateWaitTime The time to wait before a new owner can be set /// @param __loanLiquidator The liquidator contract /// @param __protocolFee The protocol fee constructor(address __owner, uint256 _updateWaitTime, address __loanLiquidator, ProtocolFee memory __protocolFee) CallbackHandler(__owner, _updateWaitTime, __protocolFee) { __loanLiquidator.checkNotZero(); _loanLiquidator = __loanLiquidator; } modifier onlyLiquidator() { if (msg.sender != address(_loanLiquidator)) { revert LiquidatorOnlyError(address(_loanLiquidator)); } _; } /// @inheritdoc ILiquidationHandler function getLiquidator() external view override returns (address) { return _loanLiquidator; } /// @inheritdoc ILiquidationHandler function updateLiquidationContract(address __loanLiquidator) external override onlyOwner { __loanLiquidator.checkNotZero(); _loanLiquidator = __loanLiquidator; emit LiquidationContractUpdated(__loanLiquidator); } /// @inheritdoc ILiquidationHandler function updateLiquidationAuctionDuration(uint48 _newDuration) external override onlyOwner { if (_newDuration < MIN_AUCTION_DURATION || _newDuration > MAX_AUCTION_DURATION) { revert InvalidDurationError(); } _liquidationAuctionDuration = _newDuration; emit LiquidationAuctionDurationUpdated(_newDuration); } /// @inheritdoc ILiquidationHandler function getLiquidationAuctionDuration() external view override returns (uint48) { return _liquidationAuctionDuration; } function _liquidateLoan(uint256 _loanId, IMultiSourceLoan.Loan calldata _loan, bool _canClaim) internal returns (bool liquidated, bytes memory liquidation) { uint256 expirationTime; unchecked { expirationTime = _loan.startTime + _loan.duration; } if (expirationTime > block.timestamp) { revert LoanNotDueError(expirationTime); } if (_canClaim) { ERC721(_loan.nftCollateralAddress).transferFrom( address(this), _loan.tranche[0].lender, _loan.nftCollateralTokenId ); emit LoanForeclosed(_loanId); liquidated = true; } else { address liquidator = _loanLiquidator; ERC721(_loan.nftCollateralAddress).transferFrom(address(this), liquidator, _loan.nftCollateralTokenId); liquidation = ILoanLiquidator(liquidator).liquidateLoan( _loanId, _loan.nftCollateralAddress, _loan.nftCollateralTokenId, _loan.principalAddress, _liquidationAuctionDuration, _loan.principalAmount.mulDivDown(MIN_BID_LIQUIDATION, _BPS), msg.sender ); emit LoanSentToLiquidator(_loanId, liquidator); } } }
// SPDX-License-Identifier: AGPL-3.0 pragma solidity ^0.8.20; import "../interfaces/loans/IMultiSourceLoan.sol"; /// @title Liquidates Collateral for Defaulted Loans /// @author Florida St /// @notice It liquidates collateral corresponding to defaulted loans /// and sends back the proceeds to the loan contract for distribution. interface ILoanLiquidator { /// @notice Given a loan, it takes posession of the NFT and liquidates it. /// @param _loanId The loan id. /// @param _contract The loan contract address. /// @param _tokenId The NFT id. /// @param _asset The asset address. /// @param _duration The liquidation duration. /// @param _minBid The minimum bid. /// @param _originator The address that trigger the liquidation. /// @return encodedAuction Encoded struct. function liquidateLoan( uint256 _loanId, address _contract, uint256 _tokenId, address _asset, uint96 _duration, uint256 _minBid, address _originator ) external returns (bytes memory); }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v5.0.0) (utils/Strings.sol) pragma solidity ^0.8.20; import {Math} from "./math/Math.sol"; import {SignedMath} from "./math/SignedMath.sol"; /** * @dev String operations. */ library Strings { bytes16 private constant HEX_DIGITS = "0123456789abcdef"; uint8 private constant ADDRESS_LENGTH = 20; /** * @dev The `value` string doesn't fit in the specified `length`. */ error StringsInsufficientHexLength(uint256 value, uint256 length); /** * @dev Converts a `uint256` to its ASCII `string` decimal representation. */ function toString(uint256 value) internal pure returns (string memory) { unchecked { uint256 length = Math.log10(value) + 1; string memory buffer = new string(length); uint256 ptr; /// @solidity memory-safe-assembly assembly { ptr := add(buffer, add(32, length)) } while (true) { ptr--; /// @solidity memory-safe-assembly assembly { mstore8(ptr, byte(mod(value, 10), HEX_DIGITS)) } value /= 10; if (value == 0) break; } return buffer; } } /** * @dev Converts a `int256` to its ASCII `string` decimal representation. */ function toStringSigned(int256 value) internal pure returns (string memory) { return string.concat(value < 0 ? "-" : "", toString(SignedMath.abs(value))); } /** * @dev Converts a `uint256` to its ASCII `string` hexadecimal representation. */ function toHexString(uint256 value) internal pure returns (string memory) { unchecked { return toHexString(value, Math.log256(value) + 1); } } /** * @dev Converts a `uint256` to its ASCII `string` hexadecimal representation with fixed length. */ function toHexString(uint256 value, uint256 length) internal pure returns (string memory) { uint256 localValue = value; 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] = HEX_DIGITS[localValue & 0xf]; localValue >>= 4; } if (localValue != 0) { revert StringsInsufficientHexLength(value, length); } return string(buffer); } /** * @dev Converts an `address` with fixed length of 20 bytes to its not checksummed ASCII `string` hexadecimal * representation. */ function toHexString(address addr) internal pure returns (string memory) { return toHexString(uint256(uint160(addr)), ADDRESS_LENGTH); } /** * @dev Returns true if the two strings are equal. */ function equal(string memory a, string memory b) internal pure returns (bool) { return bytes(a).length == bytes(b).length && keccak256(bytes(a)) == keccak256(bytes(b)); } }
// SPDX-License-Identifier: AGPL-3.0 pragma solidity ^0.8.21; /// @title InputChecker /// @author Florida St /// @notice Some basic input checks. library InputChecker { error AddressZeroError(); function checkNotZero(address _address) internal pure { if (_address == address(0)) { revert AddressZeroError(); } } }
// SPDX-License-Identifier: AGPL-3.0 pragma solidity ^0.8.21; import "./loans/IMultiSourceLoan.sol"; /// @title Interface for liquidation handlers. /// @author Florida St /// @notice Liquidation Handler interface ILiquidationHandler { /// @return Liquidator contract address function getLiquidator() external returns (address); /// @notice Updates the liquidation contract. /// @param loanLiquidator New liquidation contract. function updateLiquidationContract(address loanLiquidator) external; /// @notice Updates the auction duration for liquidations. /// @param _newDuration New auction duration. function updateLiquidationAuctionDuration(uint48 _newDuration) external; /// @return auctionDuration Returns the auction's duration for liquidations. function getLiquidationAuctionDuration() external returns (uint48); }
// SPDX-License-Identifier: AGPL-3.0 pragma solidity ^0.8.21; import "../utils/TwoStepOwned.sol"; import "../InputChecker.sol"; import "../utils/WithProtocolFee.sol"; import "../../interfaces/callbacks/ILoanCallback.sol"; /// @title CallbackHandler /// @author Florida St /// @notice Handle callbacks from the MultiSourceLoan contract. abstract contract CallbackHandler is WithProtocolFee { using InputChecker for address; /// @notice For security reasons we only allow a whitelisted set of callback contracts. mapping(address callbackContract => bool isWhitelisted) internal _isWhitelistedCallbackContract; address private immutable _multiSourceLoan; event WhitelistedCallbackContractAdded(address contractAdded); event WhitelistedCallbackContractRemoved(address contractRemoved); constructor(address __owner, uint256 _minWaitTime, ProtocolFee memory __protocolFee) WithProtocolFee(__owner, _minWaitTime, __protocolFee) {} /// @notice Add a whitelisted callback contract. /// @param _contract Address of the contract. function addWhitelistedCallbackContract(address _contract) external onlyOwner { _contract.checkNotZero(); _isWhitelistedCallbackContract[_contract] = true; emit WhitelistedCallbackContractAdded(_contract); } /// @notice Remove a whitelisted callback contract. /// @param _contract Address of the contract. function removeWhitelistedCallbackContract(address _contract) external onlyOwner { _isWhitelistedCallbackContract[_contract] = false; emit WhitelistedCallbackContractRemoved(_contract); } /// @return Whether a callback contract is whitelisted function isWhitelistedCallbackContract(address _contract) external view returns (bool) { return _isWhitelistedCallbackContract[_contract]; } /// @notice Handle the afterPrincipalTransfer callback. /// @param _loan Loan. /// @param _callbackAddress Callback address. /// @param _callbackData Callback data. /// @param _fee Fee. function handleAfterPrincipalTransferCallback( IMultiSourceLoan.Loan memory _loan, address _callbackAddress, bytes memory _callbackData, uint256 _fee ) internal { if ( !_isWhitelistedCallbackContract[_callbackAddress] || ILoanCallback(_callbackAddress).afterPrincipalTransfer(_loan, _fee, _callbackData) != ILoanCallback.afterPrincipalTransfer.selector ) { revert ILoanCallback.InvalidCallbackError(); } } /// @notice Handle the afterNFTTransfer callback. /// @param _loan Loan. /// @param _callbackAddress Callback address. /// @param _callbackData Callback data. function handleAfterNFTTransferCallback( IMultiSourceLoan.Loan memory _loan, address _callbackAddress, bytes calldata _callbackData ) internal { if ( !_isWhitelistedCallbackContract[_callbackAddress] || ILoanCallback(_callbackAddress).afterNFTTransfer(_loan, _callbackData) != ILoanCallback.afterNFTTransfer.selector ) { revert ILoanCallback.InvalidCallbackError(); } } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v5.0.0) (utils/math/Math.sol) pragma solidity ^0.8.20; /** * @dev Standard math utilities missing in the Solidity language. */ library Math { /** * @dev Muldiv operation overflow. */ error MathOverflowedMulDiv(); enum Rounding { Floor, // Toward negative infinity Ceil, // Toward positive infinity Trunc, // Toward zero Expand // Away from zero } /** * @dev Returns the addition of two unsigned integers, with an success flag (no overflow). */ function tryAdd(uint256 a, uint256 b) internal pure returns (bool success, uint256 result) { unchecked { uint256 c = a + b; if (c < a) return (false, 0); return (true, c); } } /** * @dev Returns the subtraction of two unsigned integers, with an success flag (no overflow). */ function trySub(uint256 a, uint256 b) internal pure returns (bool success, uint256 result) { unchecked { if (b > a) return (false, 0); return (true, a - b); } } /** * @dev Returns the multiplication of two unsigned integers, with an success flag (no overflow). */ function tryMul(uint256 a, uint256 b) internal pure returns (bool success, uint256 result) { unchecked { // Gas optimization: this is cheaper than requiring 'a' not being zero, but the // benefit is lost if 'b' is also tested. // See: https://github.com/OpenZeppelin/openzeppelin-contracts/pull/522 if (a == 0) return (true, 0); uint256 c = a * b; if (c / a != b) return (false, 0); return (true, c); } } /** * @dev Returns the division of two unsigned integers, with a success flag (no division by zero). */ function tryDiv(uint256 a, uint256 b) internal pure returns (bool success, uint256 result) { unchecked { if (b == 0) return (false, 0); return (true, a / b); } } /** * @dev Returns the remainder of dividing two unsigned integers, with a success flag (no division by zero). */ function tryMod(uint256 a, uint256 b) internal pure returns (bool success, uint256 result) { unchecked { if (b == 0) return (false, 0); return (true, a % b); } } /** * @dev Returns the largest of two numbers. */ function max(uint256 a, uint256 b) internal pure returns (uint256) { return a > b ? a : b; } /** * @dev Returns the smallest of two numbers. */ function min(uint256 a, uint256 b) internal pure returns (uint256) { return a < b ? a : b; } /** * @dev Returns the average of two numbers. The result is rounded towards * zero. */ function average(uint256 a, uint256 b) internal pure returns (uint256) { // (a + b) / 2 can overflow. return (a & b) + (a ^ b) / 2; } /** * @dev Returns the ceiling of the division of two numbers. * * This differs from standard division with `/` in that it rounds towards infinity instead * of rounding towards zero. */ function ceilDiv(uint256 a, uint256 b) internal pure returns (uint256) { if (b == 0) { // Guarantee the same behavior as in a regular Solidity division. return a / b; } // The following calculation ensures accurate ceiling division without overflow. // Since a is non-zero, (a - 1) / b will not overflow. // The largest possible result occurs when (a - 1) / b is type(uint256).max, // but the largest value we can obtain is type(uint256).max - 1, which happens // when a = type(uint256).max and b = 1. unchecked { return a == 0 ? 0 : (a - 1) / b + 1; } } /** * @notice Calculates floor(x * y / denominator) with full precision. Throws if result overflows a uint256 or * denominator == 0. * @dev Original credit to Remco Bloemen under MIT license (https://xn--2-umb.com/21/muldiv) with further edits by * Uniswap Labs also under MIT license. */ function mulDiv(uint256 x, uint256 y, uint256 denominator) internal pure returns (uint256 result) { unchecked { // 512-bit multiply [prod1 prod0] = x * y. Compute the product mod 2^256 and mod 2^256 - 1, then use // use the Chinese Remainder Theorem to reconstruct the 512 bit result. The result is stored in two 256 // variables such that product = prod1 * 2^256 + prod0. uint256 prod0 = x * y; // Least significant 256 bits of the product uint256 prod1; // Most significant 256 bits of the product assembly { let mm := mulmod(x, y, not(0)) prod1 := sub(sub(mm, prod0), lt(mm, prod0)) } // Handle non-overflow cases, 256 by 256 division. if (prod1 == 0) { // Solidity will revert if denominator == 0, unlike the div opcode on its own. // The surrounding unchecked block does not change this fact. // See https://docs.soliditylang.org/en/latest/control-structures.html#checked-or-unchecked-arithmetic. return prod0 / denominator; } // Make sure the result is less than 2^256. Also prevents denominator == 0. if (denominator <= prod1) { revert MathOverflowedMulDiv(); } /////////////////////////////////////////////// // 512 by 256 division. /////////////////////////////////////////////// // Make division exact by subtracting the remainder from [prod1 prod0]. uint256 remainder; assembly { // Compute remainder using mulmod. remainder := mulmod(x, y, denominator) // Subtract 256 bit number from 512 bit number. prod1 := sub(prod1, gt(remainder, prod0)) prod0 := sub(prod0, remainder) } // Factor powers of two out of denominator and compute largest power of two divisor of denominator. // Always >= 1. See https://cs.stackexchange.com/q/138556/92363. uint256 twos = denominator & (0 - denominator); assembly { // Divide denominator by twos. denominator := div(denominator, twos) // Divide [prod1 prod0] by twos. prod0 := div(prod0, twos) // Flip twos such that it is 2^256 / twos. If twos is zero, then it becomes one. twos := add(div(sub(0, twos), twos), 1) } // Shift in bits from prod1 into prod0. prod0 |= prod1 * twos; // Invert denominator mod 2^256. Now that denominator is an odd number, it has an inverse modulo 2^256 such // that denominator * inv = 1 mod 2^256. Compute the inverse by starting with a seed that is correct for // four bits. That is, denominator * inv = 1 mod 2^4. uint256 inverse = (3 * denominator) ^ 2; // Use the Newton-Raphson iteration to improve the precision. Thanks to Hensel's lifting lemma, this also // works in modular arithmetic, doubling the correct bits in each step. inverse *= 2 - denominator * inverse; // inverse mod 2^8 inverse *= 2 - denominator * inverse; // inverse mod 2^16 inverse *= 2 - denominator * inverse; // inverse mod 2^32 inverse *= 2 - denominator * inverse; // inverse mod 2^64 inverse *= 2 - denominator * inverse; // inverse mod 2^128 inverse *= 2 - denominator * inverse; // inverse mod 2^256 // Because the division is now exact we can divide by multiplying with the modular inverse of denominator. // This will give us the correct result modulo 2^256. Since the preconditions guarantee that the outcome is // less than 2^256, this is the final result. We don't need to compute the high bits of the result and prod1 // is no longer required. result = prod0 * inverse; return result; } } /** * @notice Calculates x * y / denominator with full precision, following the selected rounding direction. */ function mulDiv(uint256 x, uint256 y, uint256 denominator, Rounding rounding) internal pure returns (uint256) { uint256 result = mulDiv(x, y, denominator); if (unsignedRoundsUp(rounding) && mulmod(x, y, denominator) > 0) { result += 1; } return result; } /** * @dev Returns the square root of a number. If the number is not a perfect square, the value is rounded * towards zero. * * Inspired by Henry S. Warren, Jr.'s "Hacker's Delight" (Chapter 11). */ function sqrt(uint256 a) internal pure returns (uint256) { if (a == 0) { return 0; } // For our first guess, we get the biggest power of 2 which is smaller than the square root of the target. // // We know that the "msb" (most significant bit) of our target number `a` is a power of 2 such that we have // `msb(a) <= a < 2*msb(a)`. This value can be written `msb(a)=2**k` with `k=log2(a)`. // // This can be rewritten `2**log2(a) <= a < 2**(log2(a) + 1)` // → `sqrt(2**k) <= sqrt(a) < sqrt(2**(k+1))` // → `2**(k/2) <= sqrt(a) < 2**((k+1)/2) <= 2**(k/2 + 1)` // // Consequently, `2**(log2(a) / 2)` is a good first approximation of `sqrt(a)` with at least 1 correct bit. uint256 result = 1 << (log2(a) >> 1); // At this point `result` is an estimation with one bit of precision. We know the true value is a uint128, // since it is the square root of a uint256. Newton's method converges quadratically (precision doubles at // every iteration). We thus need at most 7 iteration to turn our partial result with one bit of precision // into the expected uint128 result. 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); } } /** * @notice Calculates sqrt(a), following the selected rounding direction. */ function sqrt(uint256 a, Rounding rounding) internal pure returns (uint256) { unchecked { uint256 result = sqrt(a); return result + (unsignedRoundsUp(rounding) && result * result < a ? 1 : 0); } } /** * @dev Return the log in base 2 of a positive value rounded towards zero. * Returns 0 if given 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; } /** * @dev Return the log in base 2, following the selected rounding direction, of a positive value. * Returns 0 if given 0. */ function log2(uint256 value, Rounding rounding) internal pure returns (uint256) { unchecked { uint256 result = log2(value); return result + (unsignedRoundsUp(rounding) && 1 << result < value ? 1 : 0); } } /** * @dev Return the log in base 10 of a positive value rounded towards zero. * Returns 0 if given 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; } /** * @dev Return the log in base 10, following the selected rounding direction, of a positive value. * Returns 0 if given 0. */ function log10(uint256 value, Rounding rounding) internal pure returns (uint256) { unchecked { uint256 result = log10(value); return result + (unsignedRoundsUp(rounding) && 10 ** result < value ? 1 : 0); } } /** * @dev Return the log in base 256 of a positive value rounded towards zero. * Returns 0 if given 0. * * Adding one to the result gives the number of pairs of hex symbols needed to represent `value` as a hex string. */ 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; } /** * @dev Return the log in base 256, following the selected rounding direction, of a positive value. * Returns 0 if given 0. */ function log256(uint256 value, Rounding rounding) internal pure returns (uint256) { unchecked { uint256 result = log256(value); return result + (unsignedRoundsUp(rounding) && 1 << (result << 3) < value ? 1 : 0); } } /** * @dev Returns whether a provided rounding mode is considered rounding up for unsigned integers. */ function unsignedRoundsUp(Rounding rounding) internal pure returns (bool) { return uint8(rounding) % 2 == 1; } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v5.0.0) (utils/math/SignedMath.sol) pragma solidity ^0.8.20; /** * @dev Standard signed math utilities missing in the Solidity language. */ library SignedMath { /** * @dev Returns the largest of two signed numbers. */ function max(int256 a, int256 b) internal pure returns (int256) { return a > b ? a : b; } /** * @dev Returns the smallest of two signed numbers. */ function min(int256 a, int256 b) internal pure returns (int256) { return a < b ? a : b; } /** * @dev Returns the average of two signed numbers without overflow. * The result is rounded towards zero. */ function average(int256 a, int256 b) internal pure returns (int256) { // Formula from the book "Hacker's Delight" int256 x = (a & b) + ((a ^ b) >> 1); return x + (int256(uint256(x) >> 255) & (a ^ b)); } /** * @dev Returns the absolute unsigned value of a signed value. */ function abs(int256 n) internal pure returns (uint256) { unchecked { // Formula from the "Bit Twiddling Hacks" by Sean Eron Anderson. // Since `n` is a signed integer, the generated bytecode will use the SAR opcode to perform the right shift, // taking advantage of the most significant (or "sign" bit) in two's complement representation. // This opcode adds new most significant bits set to the value of the previous most significant bit. As a result, // the mask will either be `bytes(0)` (if n is positive) or `~bytes32(0)` (if n is negative). int256 mask = n >> 255; // A `bytes(0)` mask leaves the input unchanged, while a `~bytes32(0)` mask complements it. return uint256((n + mask) ^ mask); } } }
// SPDX-License-Identifier: AGPL-3.0 pragma solidity ^0.8.21; import "@solmate/auth/Owned.sol"; /// @title TwoStepOwned /// @author Florida St /// @notice This contract is used to transfer ownership of a contract in two steps. abstract contract TwoStepOwned is Owned { event TransferOwnerRequested(address newOwner); error TooSoonError(); error InvalidInputError(); uint256 public immutable MIN_WAIT_TIME; address public pendingOwner; uint256 public pendingOwnerTime; constructor(address _owner, uint256 _minWaitTime) Owned(_owner) { pendingOwnerTime = type(uint256).max; MIN_WAIT_TIME = _minWaitTime; } /// @notice First step transferring ownership to the new owner. /// @param _newOwner The address of the new owner. function requestTransferOwner(address _newOwner) external onlyOwner { pendingOwner = _newOwner; pendingOwnerTime = block.timestamp; emit TransferOwnerRequested(_newOwner); } /// @notice Second step transferring ownership to the new owner. function transferOwnership() public { address newOwner = msg.sender; if (pendingOwnerTime + MIN_WAIT_TIME > block.timestamp) { revert TooSoonError(); } if (newOwner != pendingOwner) { revert InvalidInputError(); } owner = newOwner; pendingOwner = address(0); pendingOwnerTime = type(uint256).max; emit OwnershipTransferred(owner, newOwner); } }
// SPDX-License-Identifier: AGPL-3.0 pragma solidity ^0.8.21; import "./TwoStepOwned.sol"; import "../InputChecker.sol"; abstract contract WithProtocolFee is TwoStepOwned { using InputChecker for address; /// @notice Recipient address and fraction of gains charged by the protocol. struct ProtocolFee { address recipient; uint256 fraction; } uint256 public constant FEE_UPDATE_NOTICE = 30 days; /// @notice Protocol fee charged on gains. ProtocolFee internal _protocolFee; /// @notice Set as the target new protocol fee. ProtocolFee internal _pendingProtocolFee; /// @notice Set when the protocol fee updating mechanisms starts. uint256 internal _pendingProtocolFeeSetTime; event ProtocolFeeUpdated(ProtocolFee fee); event ProtocolFeePendingUpdate(ProtocolFee fee); error TooEarlyError(uint256 _pendingProtocolFeeSetTime); /// @notice Constructor /// @param _owner The owner of the contract /// @param _minWaitTime The time to wait before a new owner can be set /// @param __protocolFee The protocol fee constructor(address _owner, uint256 _minWaitTime, ProtocolFee memory __protocolFee) TwoStepOwned(_owner, _minWaitTime) { _protocolFee = __protocolFee; _pendingProtocolFeeSetTime = type(uint256).max; } /// @return protocolFee The Protocol fee. function getProtocolFee() external view returns (ProtocolFee memory) { return _protocolFee; } /// @return pendingProtocolFee The pending protocol fee. function getPendingProtocolFee() external view returns (ProtocolFee memory) { return _pendingProtocolFee; } /// @return protocolFeeSetTime Time when the protocol fee was set to be changed. function getPendingProtocolFeeSetTime() external view returns (uint256) { return _pendingProtocolFeeSetTime; } /// @notice Kicks off the process to update the protocol fee. /// @param _newProtocolFee New protocol fee. function updateProtocolFee(ProtocolFee calldata _newProtocolFee) external onlyOwner { _newProtocolFee.recipient.checkNotZero(); _pendingProtocolFee = _newProtocolFee; _pendingProtocolFeeSetTime = block.timestamp; emit ProtocolFeePendingUpdate(_pendingProtocolFee); } /// @notice Set the protocol fee if enough notice has been given. function setProtocolFee() external virtual { if (block.timestamp < _pendingProtocolFeeSetTime + FEE_UPDATE_NOTICE) { revert TooSoonError(); } ProtocolFee memory protocolFee = _pendingProtocolFee; _protocolFee = protocolFee; emit ProtocolFeeUpdated(protocolFee); } }
// SPDX-License-Identifier: AGPL-3.0 pragma solidity ^0.8.20; import "../loans/IMultiSourceLoan.sol"; interface ILoanCallback { error InvalidCallbackError(); /// @notice Called by the MSL contract after the principal of loan has been tranfered (when a loan is initiated) /// but before it tries to transfer the NFT into escrow. /// @param _loan The loan. /// @param _fee The origination fee. /// @param _executionData Execution data for purchase. /// @return The bytes4 magic value. function afterPrincipalTransfer(IMultiSourceLoan.Loan memory _loan, uint256 _fee, bytes calldata _executionData) external returns (bytes4); /// @notice Call by the MSL contract after the NFT has been transfered to the borrower repaying the loan, but before /// transfering the principal to the lender. /// @param _loan The loan. /// @param _executionData Execution data for the offer. /// @return The bytes4 magic value. function afterNFTTransfer(IMultiSourceLoan.Loan memory _loan, bytes calldata _executionData) external returns (bytes4); }
{ "remappings": [ "@forge-std/=lib/forge-std/src/", "@openzeppelin/=lib/openzeppelin-contracts/contracts/", "@solady/=lib/solady/src/", "@solmate/=lib/solmate/src/", "@zora/=lib/v3/contracts/", "@chainlink/=lib/chainlink/contracts/src/v0.8/", "@delegate/=lib/delegate-registry/src/", "@seaport/=lib/seaport/lib/", "test/=test/", "@const/=src/const/default/", "@manifoldxyz/=lib/v3/node_modules/@manifoldxyz/", "@openzeppelin/contracts/=lib/openzeppelin-contracts/contracts/", "@rari-capital/=lib/v3/node_modules/@rari-capital/", "@rari-capital/solmate/=lib/seaport/lib/solmate/", "chainlink/=lib/chainlink/contracts/", "delegate-registry/=lib/delegate-registry/", "ds-test/=lib/seaport/lib/ds-test/src/", "erc4626-tests/=lib/openzeppelin-contracts/lib/erc4626-tests/", "forge-std/=lib/forge-std/src/", "murky/=lib/delegate-registry/lib/murky/", "openzeppelin-contracts/=lib/openzeppelin-contracts/", "openzeppelin/=lib/delegate-registry/lib/openzeppelin-contracts/contracts/", "seaport-core/=lib/seaport/lib/seaport-core/", "seaport-sol/=lib/seaport/lib/seaport-sol/", "seaport-types/=lib/seaport/lib/seaport-types/", "seaport/=lib/seaport/", "solady/=lib/solady/src/", "solarray/=lib/seaport/lib/solarray/src/", "solmate/=lib/solmate/src/", "v3/=lib/v3/contracts/" ], "optimizer": { "enabled": true, "runs": 20 }, "metadata": { "useLiteralContent": false, "bytecodeHash": "ipfs", "appendCBOR": true }, "outputSelection": { "*": { "*": [ "evm.bytecode", "evm.deployedBytecode", "abi" ] } }, "evmVersion": "paris", "viaIR": true, "libraries": {} }
Contract Security Audit
- No Contract Security Audit Submitted- Submit Audit Here
[{"inputs":[{"internalType":"address","name":"loanLiquidator","type":"address"},{"components":[{"internalType":"address","name":"recipient","type":"address"},{"internalType":"uint256","name":"fraction","type":"uint256"}],"internalType":"struct WithProtocolFee.ProtocolFee","name":"protocolFee","type":"tuple"},{"internalType":"address","name":"currencyManager","type":"address"},{"internalType":"address","name":"collectionManager","type":"address"},{"internalType":"uint256","name":"maxTranches","type":"uint256"},{"internalType":"uint256","name":"minLockPeriod","type":"uint256"},{"internalType":"address","name":"delegateRegistry","type":"address"},{"internalType":"address","name":"loanManagerRegistry","type":"address"},{"internalType":"address","name":"flashActionContract","type":"address"},{"internalType":"uint256","name":"minWaitTime","type":"uint256"}],"stateMutability":"nonpayable","type":"constructor"},{"inputs":[],"name":"AddressZeroError","type":"error"},{"inputs":[{"internalType":"address","name":"_lender","type":"address"},{"internalType":"uint256","name":"_offerId","type":"uint256"}],"name":"CancelledOrExecutedOfferError","type":"error"},{"inputs":[],"name":"CollectionNotWhitelistedError","type":"error"},{"inputs":[],"name":"CurrencyNotWhitelistedError","type":"error"},{"inputs":[],"name":"ECDSAInvalidSignature","type":"error"},{"inputs":[{"internalType":"uint256","name":"length","type":"uint256"}],"name":"ECDSAInvalidSignatureLength","type":"error"},{"inputs":[{"internalType":"bytes32","name":"s","type":"bytes32"}],"name":"ECDSAInvalidSignatureS","type":"error"},{"inputs":[{"internalType":"uint256","name":"_expirationTime","type":"uint256"}],"name":"ExpiredOfferError","type":"error"},{"inputs":[],"name":"InvalidAddressesError","type":"error"},{"inputs":[{"internalType":"uint256","name":"_amount","type":"uint256"},{"internalType":"uint256","name":"_principalAmount","type":"uint256"}],"name":"InvalidAmountError","type":"error"},{"inputs":[],"name":"InvalidCallbackError","type":"error"},{"inputs":[],"name":"InvalidCallerError","type":"error"},{"inputs":[],"name":"InvalidCollateralIdError","type":"error"},{"inputs":[],"name":"InvalidDurationError","type":"error"},{"inputs":[],"name":"InvalidInputError","type":"error"},{"inputs":[{"internalType":"uint256","name":"_loanId","type":"uint256"}],"name":"InvalidLoanError","type":"error"},{"inputs":[],"name":"InvalidMethodError","type":"error"},{"inputs":[],"name":"InvalidParametersError","type":"error"},{"inputs":[],"name":"InvalidRenegotiationOfferError","type":"error"},{"inputs":[],"name":"InvalidSignatureError","type":"error"},{"inputs":[],"name":"InvalidTrancheError","type":"error"},{"inputs":[{"internalType":"address","name":"_liquidator","type":"address"}],"name":"LiquidatorOnlyError","type":"error"},{"inputs":[],"name":"LoanExpiredError","type":"error"},{"inputs":[],"name":"LoanLockedError","type":"error"},{"inputs":[{"internalType":"uint256","name":"_expirationTime","type":"uint256"}],"name":"LoanNotDueError","type":"error"},{"inputs":[{"internalType":"address","name":"_lender","type":"address"},{"internalType":"uint256","name":"_newMinOfferId","type":"uint256"},{"internalType":"uint256","name":"_minOfferId","type":"uint256"}],"name":"LowOfferIdError","type":"error"},{"inputs":[{"internalType":"address","name":"_lender","type":"address"},{"internalType":"uint256","name":"_newMinRenegotiationOfferId","type":"uint256"},{"internalType":"uint256","name":"_minOfferId","type":"uint256"}],"name":"LowRenegotiationOfferIdError","type":"error"},{"inputs":[],"name":"MaxCapacityExceededError","type":"error"},{"inputs":[],"name":"MismatchError","type":"error"},{"inputs":[{"internalType":"uint256","name":"i","type":"uint256"},{"internalType":"bytes","name":"returndata","type":"bytes"}],"name":"MulticallFailed","type":"error"},{"inputs":[],"name":"NFTNotReturnedError","type":"error"},{"inputs":[],"name":"NotStrictlyImprovedError","type":"error"},{"inputs":[{"internalType":"uint256","name":"_pendingProtocolFeeSetTime","type":"uint256"}],"name":"TooEarlyError","type":"error"},{"inputs":[],"name":"TooManyTranchesError","type":"error"},{"inputs":[],"name":"TooSoonError","type":"error"},{"inputs":[{"internalType":"uint256","name":"minTimestamp","type":"uint256"}],"name":"TrancheCannotBeRefinancedError","type":"error"},{"inputs":[],"name":"ZeroInterestError","type":"error"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"lender","type":"address"},{"indexed":false,"internalType":"uint256","name":"minOfferId","type":"uint256"}],"name":"AllOffersCancelled","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"loanId","type":"uint256"},{"indexed":false,"internalType":"address","name":"delegate","type":"address"},{"indexed":false,"internalType":"bytes32","name":"_rights","type":"bytes32"},{"indexed":false,"internalType":"bool","name":"value","type":"bool"}],"name":"Delegated","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"newFlashActionContract","type":"address"}],"name":"FlashActionContractUpdated","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"loanId","type":"uint256"},{"indexed":false,"internalType":"address","name":"target","type":"address"},{"indexed":false,"internalType":"bytes","name":"data","type":"bytes"}],"name":"FlashActionExecuted","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"newDuration","type":"uint256"}],"name":"LiquidationAuctionDurationUpdated","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"liquidator","type":"address"}],"name":"LiquidationContractUpdated","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"loanId","type":"uint256"},{"indexed":false,"internalType":"uint256[]","name":"offerId","type":"uint256[]"},{"components":[{"internalType":"address","name":"borrower","type":"address"},{"internalType":"uint256","name":"nftCollateralTokenId","type":"uint256"},{"internalType":"address","name":"nftCollateralAddress","type":"address"},{"internalType":"address","name":"principalAddress","type":"address"},{"internalType":"uint256","name":"principalAmount","type":"uint256"},{"internalType":"uint256","name":"startTime","type":"uint256"},{"internalType":"uint256","name":"duration","type":"uint256"},{"components":[{"internalType":"uint256","name":"loanId","type":"uint256"},{"internalType":"uint256","name":"floor","type":"uint256"},{"internalType":"uint256","name":"principalAmount","type":"uint256"},{"internalType":"address","name":"lender","type":"address"},{"internalType":"uint256","name":"accruedInterest","type":"uint256"},{"internalType":"uint256","name":"startTime","type":"uint256"},{"internalType":"uint256","name":"aprBps","type":"uint256"}],"internalType":"struct IMultiSourceLoan.Tranche[]","name":"tranche","type":"tuple[]"},{"internalType":"uint256","name":"protocolFee","type":"uint256"}],"indexed":false,"internalType":"struct IMultiSourceLoan.Loan","name":"loan","type":"tuple"},{"indexed":false,"internalType":"uint256","name":"fee","type":"uint256"}],"name":"LoanEmitted","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"loanId","type":"uint256"}],"name":"LoanForeclosed","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"loanId","type":"uint256"}],"name":"LoanLiquidated","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"renegotiationId","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"oldLoanId","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"newLoanId","type":"uint256"},{"components":[{"internalType":"address","name":"borrower","type":"address"},{"internalType":"uint256","name":"nftCollateralTokenId","type":"uint256"},{"internalType":"address","name":"nftCollateralAddress","type":"address"},{"internalType":"address","name":"principalAddress","type":"address"},{"internalType":"uint256","name":"principalAmount","type":"uint256"},{"internalType":"uint256","name":"startTime","type":"uint256"},{"internalType":"uint256","name":"duration","type":"uint256"},{"components":[{"internalType":"uint256","name":"loanId","type":"uint256"},{"internalType":"uint256","name":"floor","type":"uint256"},{"internalType":"uint256","name":"principalAmount","type":"uint256"},{"internalType":"address","name":"lender","type":"address"},{"internalType":"uint256","name":"accruedInterest","type":"uint256"},{"internalType":"uint256","name":"startTime","type":"uint256"},{"internalType":"uint256","name":"aprBps","type":"uint256"}],"internalType":"struct IMultiSourceLoan.Tranche[]","name":"tranche","type":"tuple[]"},{"internalType":"uint256","name":"protocolFee","type":"uint256"}],"indexed":false,"internalType":"struct IMultiSourceLoan.Loan","name":"loan","type":"tuple"},{"indexed":false,"internalType":"uint256","name":"fee","type":"uint256"}],"name":"LoanRefinanced","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"loanId","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"newLoanId","type":"uint256"},{"components":[{"internalType":"address","name":"borrower","type":"address"},{"internalType":"uint256","name":"nftCollateralTokenId","type":"uint256"},{"internalType":"address","name":"nftCollateralAddress","type":"address"},{"internalType":"address","name":"principalAddress","type":"address"},{"internalType":"uint256","name":"principalAmount","type":"uint256"},{"internalType":"uint256","name":"startTime","type":"uint256"},{"internalType":"uint256","name":"duration","type":"uint256"},{"components":[{"internalType":"uint256","name":"loanId","type":"uint256"},{"internalType":"uint256","name":"floor","type":"uint256"},{"internalType":"uint256","name":"principalAmount","type":"uint256"},{"internalType":"address","name":"lender","type":"address"},{"internalType":"uint256","name":"accruedInterest","type":"uint256"},{"internalType":"uint256","name":"startTime","type":"uint256"},{"internalType":"uint256","name":"aprBps","type":"uint256"}],"internalType":"struct IMultiSourceLoan.Tranche[]","name":"tranche","type":"tuple[]"},{"internalType":"uint256","name":"protocolFee","type":"uint256"}],"indexed":false,"internalType":"struct IMultiSourceLoan.Loan","name":"loan","type":"tuple"},{"indexed":false,"internalType":"uint256[]","name":"offerIds","type":"uint256[]"},{"indexed":false,"internalType":"uint256","name":"totalFee","type":"uint256"}],"name":"LoanRefinancedFromNewOffers","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"loanId","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"totalRepayment","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"fee","type":"uint256"}],"name":"LoanRepaid","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"loanId","type":"uint256"},{"indexed":false,"internalType":"address","name":"liquidator","type":"address"}],"name":"LoanSentToLiquidator","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"_minimum","type":"uint256"}],"name":"MinAprImprovementUpdated","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"newMinBid","type":"uint256"}],"name":"MinBidLiquidationUpdated","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"minLockPeriod","type":"uint256"}],"name":"MinLockPeriodUpdated","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"lender","type":"address"},{"indexed":false,"internalType":"uint256","name":"offerId","type":"uint256"}],"name":"OfferCancelled","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"user","type":"address"},{"indexed":true,"internalType":"address","name":"newOwner","type":"address"}],"name":"OwnershipTransferred","type":"event"},{"anonymous":false,"inputs":[{"components":[{"internalType":"address","name":"recipient","type":"address"},{"internalType":"uint256","name":"fraction","type":"uint256"}],"indexed":false,"internalType":"struct WithProtocolFee.ProtocolFee","name":"fee","type":"tuple"}],"name":"ProtocolFeePendingUpdate","type":"event"},{"anonymous":false,"inputs":[{"components":[{"internalType":"address","name":"recipient","type":"address"},{"internalType":"uint256","name":"fraction","type":"uint256"}],"indexed":false,"internalType":"struct WithProtocolFee.ProtocolFee","name":"fee","type":"tuple"}],"name":"ProtocolFeeUpdated","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"lender","type":"address"},{"indexed":false,"internalType":"uint256","name":"renegotiationId","type":"uint256"}],"name":"RenegotiationOfferCancelled","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"delegate","type":"address"},{"indexed":false,"internalType":"address","name":"collection","type":"address"},{"indexed":false,"internalType":"uint256","name":"tokenId","type":"uint256"},{"indexed":false,"internalType":"bytes32","name":"_rights","type":"bytes32"}],"name":"RevokeDelegate","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"newOwner","type":"address"}],"name":"TransferOwnerRequested","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"contractAdded","type":"address"}],"name":"WhitelistedCallbackContractAdded","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"contractRemoved","type":"address"}],"name":"WhitelistedCallbackContractRemoved","type":"event"},{"inputs":[],"name":"DOMAIN_SEPARATOR","outputs":[{"internalType":"bytes32","name":"","type":"bytes32"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"FEE_UPDATE_NOTICE","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"INITIAL_DOMAIN_SEPARATOR","outputs":[{"internalType":"bytes32","name":"","type":"bytes32"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"MAX_AUCTION_DURATION","outputs":[{"internalType":"uint48","name":"","type":"uint48"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"MIN_AUCTION_DURATION","outputs":[{"internalType":"uint48","name":"","type":"uint48"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"MIN_BID_LIQUIDATION","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"MIN_WAIT_TIME","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"VERSION","outputs":[{"internalType":"bytes","name":"","type":"bytes"}],"stateMutability":"view","type":"function"},{"inputs":[{"components":[{"internalType":"uint256","name":"renegotiationId","type":"uint256"},{"internalType":"uint256","name":"loanId","type":"uint256"},{"internalType":"address","name":"lender","type":"address"},{"internalType":"uint256","name":"fee","type":"uint256"},{"internalType":"uint256[]","name":"trancheIndex","type":"uint256[]"},{"internalType":"uint256","name":"principalAmount","type":"uint256"},{"internalType":"uint256","name":"aprBps","type":"uint256"},{"internalType":"uint256","name":"expirationTime","type":"uint256"},{"internalType":"uint256","name":"duration","type":"uint256"}],"internalType":"struct IMultiSourceLoan.RenegotiationOffer","name":"_renegotiationOffer","type":"tuple"},{"components":[{"internalType":"address","name":"borrower","type":"address"},{"internalType":"uint256","name":"nftCollateralTokenId","type":"uint256"},{"internalType":"address","name":"nftCollateralAddress","type":"address"},{"internalType":"address","name":"principalAddress","type":"address"},{"internalType":"uint256","name":"principalAmount","type":"uint256"},{"internalType":"uint256","name":"startTime","type":"uint256"},{"internalType":"uint256","name":"duration","type":"uint256"},{"components":[{"internalType":"uint256","name":"loanId","type":"uint256"},{"internalType":"uint256","name":"floor","type":"uint256"},{"internalType":"uint256","name":"principalAmount","type":"uint256"},{"internalType":"address","name":"lender","type":"address"},{"internalType":"uint256","name":"accruedInterest","type":"uint256"},{"internalType":"uint256","name":"startTime","type":"uint256"},{"internalType":"uint256","name":"aprBps","type":"uint256"}],"internalType":"struct IMultiSourceLoan.Tranche[]","name":"tranche","type":"tuple[]"},{"internalType":"uint256","name":"protocolFee","type":"uint256"}],"internalType":"struct IMultiSourceLoan.Loan","name":"_loan","type":"tuple"},{"internalType":"bytes","name":"_renegotiationOfferSignature","type":"bytes"}],"name":"addNewTranche","outputs":[{"internalType":"uint256","name":"","type":"uint256"},{"components":[{"internalType":"address","name":"borrower","type":"address"},{"internalType":"uint256","name":"nftCollateralTokenId","type":"uint256"},{"internalType":"address","name":"nftCollateralAddress","type":"address"},{"internalType":"address","name":"principalAddress","type":"address"},{"internalType":"uint256","name":"principalAmount","type":"uint256"},{"internalType":"uint256","name":"startTime","type":"uint256"},{"internalType":"uint256","name":"duration","type":"uint256"},{"components":[{"internalType":"uint256","name":"loanId","type":"uint256"},{"internalType":"uint256","name":"floor","type":"uint256"},{"internalType":"uint256","name":"principalAmount","type":"uint256"},{"internalType":"address","name":"lender","type":"address"},{"internalType":"uint256","name":"accruedInterest","type":"uint256"},{"internalType":"uint256","name":"startTime","type":"uint256"},{"internalType":"uint256","name":"aprBps","type":"uint256"}],"internalType":"struct IMultiSourceLoan.Tranche[]","name":"tranche","type":"tuple[]"},{"internalType":"uint256","name":"protocolFee","type":"uint256"}],"internalType":"struct IMultiSourceLoan.Loan","name":"","type":"tuple"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_contract","type":"address"}],"name":"addWhitelistedCallbackContract","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_minOfferId","type":"uint256"}],"name":"cancelAllOffers","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_offerId","type":"uint256"}],"name":"cancelOffer","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_renegotiationId","type":"uint256"}],"name":"cancelRenegotiationOffer","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_loanId","type":"uint256"},{"components":[{"internalType":"address","name":"borrower","type":"address"},{"internalType":"uint256","name":"nftCollateralTokenId","type":"uint256"},{"internalType":"address","name":"nftCollateralAddress","type":"address"},{"internalType":"address","name":"principalAddress","type":"address"},{"internalType":"uint256","name":"principalAmount","type":"uint256"},{"internalType":"uint256","name":"startTime","type":"uint256"},{"internalType":"uint256","name":"duration","type":"uint256"},{"components":[{"internalType":"uint256","name":"loanId","type":"uint256"},{"internalType":"uint256","name":"floor","type":"uint256"},{"internalType":"uint256","name":"principalAmount","type":"uint256"},{"internalType":"address","name":"lender","type":"address"},{"internalType":"uint256","name":"accruedInterest","type":"uint256"},{"internalType":"uint256","name":"startTime","type":"uint256"},{"internalType":"uint256","name":"aprBps","type":"uint256"}],"internalType":"struct IMultiSourceLoan.Tranche[]","name":"tranche","type":"tuple[]"},{"internalType":"uint256","name":"protocolFee","type":"uint256"}],"internalType":"struct IMultiSourceLoan.Loan","name":"loan","type":"tuple"},{"internalType":"address","name":"_delegate","type":"address"},{"internalType":"bytes32","name":"_rights","type":"bytes32"},{"internalType":"bool","name":"_value","type":"bool"}],"name":"delegate","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"components":[{"components":[{"components":[{"components":[{"internalType":"uint256","name":"offerId","type":"uint256"},{"internalType":"address","name":"lender","type":"address"},{"internalType":"uint256","name":"fee","type":"uint256"},{"internalType":"uint256","name":"capacity","type":"uint256"},{"internalType":"address","name":"nftCollateralAddress","type":"address"},{"internalType":"uint256","name":"nftCollateralTokenId","type":"uint256"},{"internalType":"address","name":"principalAddress","type":"address"},{"internalType":"uint256","name":"principalAmount","type":"uint256"},{"internalType":"uint256","name":"aprBps","type":"uint256"},{"internalType":"uint256","name":"expirationTime","type":"uint256"},{"internalType":"uint256","name":"duration","type":"uint256"},{"internalType":"uint256","name":"maxSeniorRepayment","type":"uint256"},{"components":[{"internalType":"address","name":"validator","type":"address"},{"internalType":"bytes","name":"arguments","type":"bytes"}],"internalType":"struct IBaseLoan.OfferValidator[]","name":"validators","type":"tuple[]"}],"internalType":"struct IMultiSourceLoan.LoanOffer","name":"offer","type":"tuple"},{"internalType":"uint256","name":"amount","type":"uint256"},{"internalType":"bytes","name":"lenderOfferSignature","type":"bytes"}],"internalType":"struct IMultiSourceLoan.OfferExecution[]","name":"offerExecution","type":"tuple[]"},{"internalType":"uint256","name":"tokenId","type":"uint256"},{"internalType":"uint256","name":"duration","type":"uint256"},{"internalType":"uint256","name":"expirationTime","type":"uint256"},{"internalType":"address","name":"principalReceiver","type":"address"},{"internalType":"bytes","name":"callbackData","type":"bytes"}],"internalType":"struct IMultiSourceLoan.ExecutionData","name":"executionData","type":"tuple"},{"internalType":"address","name":"borrower","type":"address"},{"internalType":"bytes","name":"borrowerOfferSignature","type":"bytes"}],"internalType":"struct IMultiSourceLoan.LoanExecutionData","name":"_loanExecutionData","type":"tuple"}],"name":"emitLoan","outputs":[{"internalType":"uint256","name":"","type":"uint256"},{"components":[{"internalType":"address","name":"borrower","type":"address"},{"internalType":"uint256","name":"nftCollateralTokenId","type":"uint256"},{"internalType":"address","name":"nftCollateralAddress","type":"address"},{"internalType":"address","name":"principalAddress","type":"address"},{"internalType":"uint256","name":"principalAmount","type":"uint256"},{"internalType":"uint256","name":"startTime","type":"uint256"},{"internalType":"uint256","name":"duration","type":"uint256"},{"components":[{"internalType":"uint256","name":"loanId","type":"uint256"},{"internalType":"uint256","name":"floor","type":"uint256"},{"internalType":"uint256","name":"principalAmount","type":"uint256"},{"internalType":"address","name":"lender","type":"address"},{"internalType":"uint256","name":"accruedInterest","type":"uint256"},{"internalType":"uint256","name":"startTime","type":"uint256"},{"internalType":"uint256","name":"aprBps","type":"uint256"}],"internalType":"struct IMultiSourceLoan.Tranche[]","name":"tranche","type":"tuple[]"},{"internalType":"uint256","name":"protocolFee","type":"uint256"}],"internalType":"struct IMultiSourceLoan.Loan","name":"","type":"tuple"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_loanId","type":"uint256"},{"components":[{"internalType":"address","name":"borrower","type":"address"},{"internalType":"uint256","name":"nftCollateralTokenId","type":"uint256"},{"internalType":"address","name":"nftCollateralAddress","type":"address"},{"internalType":"address","name":"principalAddress","type":"address"},{"internalType":"uint256","name":"principalAmount","type":"uint256"},{"internalType":"uint256","name":"startTime","type":"uint256"},{"internalType":"uint256","name":"duration","type":"uint256"},{"components":[{"internalType":"uint256","name":"loanId","type":"uint256"},{"internalType":"uint256","name":"floor","type":"uint256"},{"internalType":"uint256","name":"principalAmount","type":"uint256"},{"internalType":"address","name":"lender","type":"address"},{"internalType":"uint256","name":"accruedInterest","type":"uint256"},{"internalType":"uint256","name":"startTime","type":"uint256"},{"internalType":"uint256","name":"aprBps","type":"uint256"}],"internalType":"struct IMultiSourceLoan.Tranche[]","name":"tranche","type":"tuple[]"},{"internalType":"uint256","name":"protocolFee","type":"uint256"}],"internalType":"struct IMultiSourceLoan.Loan","name":"_loan","type":"tuple"},{"internalType":"address","name":"_target","type":"address"},{"internalType":"bytes","name":"_data","type":"bytes"}],"name":"executeFlashAction","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"getCollectionManager","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"getCurrencyManager","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"getDelegateRegistry","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"getFlashActionContract","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"getLiquidationAuctionDuration","outputs":[{"internalType":"uint48","name":"","type":"uint48"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"getLiquidator","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"_loanId","type":"uint256"}],"name":"getLoanHash","outputs":[{"internalType":"bytes32","name":"","type":"bytes32"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"getLoanManagerRegistry","outputs":[{"internalType":"contract ILoanManagerRegistry","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"getMaxTranches","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"getMinImprovementApr","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"getMinLockPeriod","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"getPendingProtocolFee","outputs":[{"components":[{"internalType":"address","name":"recipient","type":"address"},{"internalType":"uint256","name":"fraction","type":"uint256"}],"internalType":"struct WithProtocolFee.ProtocolFee","name":"","type":"tuple"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"getPendingProtocolFeeSetTime","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"getProtocolFee","outputs":[{"components":[{"internalType":"address","name":"recipient","type":"address"},{"internalType":"uint256","name":"fraction","type":"uint256"}],"internalType":"struct WithProtocolFee.ProtocolFee","name":"","type":"tuple"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"getTotalLoansIssued","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_lender","type":"address"},{"internalType":"uint256","name":"_offerId","type":"uint256"}],"name":"getUsedCapacity","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"user","type":"address"},{"internalType":"uint256","name":"offerId","type":"uint256"}],"name":"isOfferCancelled","outputs":[{"internalType":"bool","name":"notActive","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"user","type":"address"},{"internalType":"uint256","name":"renegotiationIf","type":"uint256"}],"name":"isRenegotiationOfferCancelled","outputs":[{"internalType":"bool","name":"notActive","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_contract","type":"address"}],"name":"isWhitelistedCallbackContract","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"_loanId","type":"uint256"},{"components":[{"internalType":"address","name":"borrower","type":"address"},{"internalType":"uint256","name":"nftCollateralTokenId","type":"uint256"},{"internalType":"address","name":"nftCollateralAddress","type":"address"},{"internalType":"address","name":"principalAddress","type":"address"},{"internalType":"uint256","name":"principalAmount","type":"uint256"},{"internalType":"uint256","name":"startTime","type":"uint256"},{"internalType":"uint256","name":"duration","type":"uint256"},{"components":[{"internalType":"uint256","name":"loanId","type":"uint256"},{"internalType":"uint256","name":"floor","type":"uint256"},{"internalType":"uint256","name":"principalAmount","type":"uint256"},{"internalType":"address","name":"lender","type":"address"},{"internalType":"uint256","name":"accruedInterest","type":"uint256"},{"internalType":"uint256","name":"startTime","type":"uint256"},{"internalType":"uint256","name":"aprBps","type":"uint256"}],"internalType":"struct IMultiSourceLoan.Tranche[]","name":"tranche","type":"tuple[]"},{"internalType":"uint256","name":"protocolFee","type":"uint256"}],"internalType":"struct IMultiSourceLoan.Loan","name":"_loan","type":"tuple"}],"name":"liquidateLoan","outputs":[{"internalType":"bytes","name":"","type":"bytes"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_loanId","type":"uint256"},{"components":[{"internalType":"address","name":"borrower","type":"address"},{"internalType":"uint256","name":"nftCollateralTokenId","type":"uint256"},{"internalType":"address","name":"nftCollateralAddress","type":"address"},{"internalType":"address","name":"principalAddress","type":"address"},{"internalType":"uint256","name":"principalAmount","type":"uint256"},{"internalType":"uint256","name":"startTime","type":"uint256"},{"internalType":"uint256","name":"duration","type":"uint256"},{"components":[{"internalType":"uint256","name":"loanId","type":"uint256"},{"internalType":"uint256","name":"floor","type":"uint256"},{"internalType":"uint256","name":"principalAmount","type":"uint256"},{"internalType":"address","name":"lender","type":"address"},{"internalType":"uint256","name":"accruedInterest","type":"uint256"},{"internalType":"uint256","name":"startTime","type":"uint256"},{"internalType":"uint256","name":"aprBps","type":"uint256"}],"internalType":"struct IMultiSourceLoan.Tranche[]","name":"tranche","type":"tuple[]"},{"internalType":"uint256","name":"protocolFee","type":"uint256"}],"internalType":"struct IMultiSourceLoan.Loan","name":"_loan","type":"tuple"}],"name":"loanLiquidated","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"user","type":"address"}],"name":"minOfferId","outputs":[{"internalType":"uint256","name":"minOfferId","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"bytes[]","name":"data","type":"bytes[]"}],"name":"multicall","outputs":[{"internalType":"bytes[]","name":"results","type":"bytes[]"}],"stateMutability":"payable","type":"function"},{"inputs":[],"name":"name","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"},{"internalType":"address","name":"","type":"address"},{"internalType":"uint256","name":"","type":"uint256"},{"internalType":"bytes","name":"","type":"bytes"}],"name":"onERC721Received","outputs":[{"internalType":"bytes4","name":"","type":"bytes4"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"owner","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"pendingOwner","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"pendingOwnerTime","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"_loanId","type":"uint256"},{"components":[{"internalType":"address","name":"borrower","type":"address"},{"internalType":"uint256","name":"nftCollateralTokenId","type":"uint256"},{"internalType":"address","name":"nftCollateralAddress","type":"address"},{"internalType":"address","name":"principalAddress","type":"address"},{"internalType":"uint256","name":"principalAmount","type":"uint256"},{"internalType":"uint256","name":"startTime","type":"uint256"},{"internalType":"uint256","name":"duration","type":"uint256"},{"components":[{"internalType":"uint256","name":"loanId","type":"uint256"},{"internalType":"uint256","name":"floor","type":"uint256"},{"internalType":"uint256","name":"principalAmount","type":"uint256"},{"internalType":"address","name":"lender","type":"address"},{"internalType":"uint256","name":"accruedInterest","type":"uint256"},{"internalType":"uint256","name":"startTime","type":"uint256"},{"internalType":"uint256","name":"aprBps","type":"uint256"}],"internalType":"struct IMultiSourceLoan.Tranche[]","name":"tranche","type":"tuple[]"},{"internalType":"uint256","name":"protocolFee","type":"uint256"}],"internalType":"struct IMultiSourceLoan.Loan","name":"_loan","type":"tuple"},{"components":[{"components":[{"components":[{"components":[{"internalType":"uint256","name":"offerId","type":"uint256"},{"internalType":"address","name":"lender","type":"address"},{"internalType":"uint256","name":"fee","type":"uint256"},{"internalType":"uint256","name":"capacity","type":"uint256"},{"internalType":"address","name":"nftCollateralAddress","type":"address"},{"internalType":"uint256","name":"nftCollateralTokenId","type":"uint256"},{"internalType":"address","name":"principalAddress","type":"address"},{"internalType":"uint256","name":"principalAmount","type":"uint256"},{"internalType":"uint256","name":"aprBps","type":"uint256"},{"internalType":"uint256","name":"expirationTime","type":"uint256"},{"internalType":"uint256","name":"duration","type":"uint256"},{"internalType":"uint256","name":"maxSeniorRepayment","type":"uint256"},{"components":[{"internalType":"address","name":"validator","type":"address"},{"internalType":"bytes","name":"arguments","type":"bytes"}],"internalType":"struct IBaseLoan.OfferValidator[]","name":"validators","type":"tuple[]"}],"internalType":"struct IMultiSourceLoan.LoanOffer","name":"offer","type":"tuple"},{"internalType":"uint256","name":"amount","type":"uint256"},{"internalType":"bytes","name":"lenderOfferSignature","type":"bytes"}],"internalType":"struct IMultiSourceLoan.OfferExecution[]","name":"offerExecution","type":"tuple[]"},{"internalType":"uint256","name":"tokenId","type":"uint256"},{"internalType":"uint256","name":"duration","type":"uint256"},{"internalType":"uint256","name":"expirationTime","type":"uint256"},{"internalType":"address","name":"principalReceiver","type":"address"},{"internalType":"bytes","name":"callbackData","type":"bytes"}],"internalType":"struct IMultiSourceLoan.ExecutionData","name":"executionData","type":"tuple"},{"internalType":"address","name":"borrower","type":"address"},{"internalType":"bytes","name":"borrowerOfferSignature","type":"bytes"}],"internalType":"struct IMultiSourceLoan.LoanExecutionData","name":"_loanExecutionData","type":"tuple"}],"name":"refinanceFromLoanExecutionData","outputs":[{"internalType":"uint256","name":"","type":"uint256"},{"components":[{"internalType":"address","name":"borrower","type":"address"},{"internalType":"uint256","name":"nftCollateralTokenId","type":"uint256"},{"internalType":"address","name":"nftCollateralAddress","type":"address"},{"internalType":"address","name":"principalAddress","type":"address"},{"internalType":"uint256","name":"principalAmount","type":"uint256"},{"internalType":"uint256","name":"startTime","type":"uint256"},{"internalType":"uint256","name":"duration","type":"uint256"},{"components":[{"internalType":"uint256","name":"loanId","type":"uint256"},{"internalType":"uint256","name":"floor","type":"uint256"},{"internalType":"uint256","name":"principalAmount","type":"uint256"},{"internalType":"address","name":"lender","type":"address"},{"internalType":"uint256","name":"accruedInterest","type":"uint256"},{"internalType":"uint256","name":"startTime","type":"uint256"},{"internalType":"uint256","name":"aprBps","type":"uint256"}],"internalType":"struct IMultiSourceLoan.Tranche[]","name":"tranche","type":"tuple[]"},{"internalType":"uint256","name":"protocolFee","type":"uint256"}],"internalType":"struct IMultiSourceLoan.Loan","name":"","type":"tuple"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"components":[{"internalType":"uint256","name":"renegotiationId","type":"uint256"},{"internalType":"uint256","name":"loanId","type":"uint256"},{"internalType":"address","name":"lender","type":"address"},{"internalType":"uint256","name":"fee","type":"uint256"},{"internalType":"uint256[]","name":"trancheIndex","type":"uint256[]"},{"internalType":"uint256","name":"principalAmount","type":"uint256"},{"internalType":"uint256","name":"aprBps","type":"uint256"},{"internalType":"uint256","name":"expirationTime","type":"uint256"},{"internalType":"uint256","name":"duration","type":"uint256"}],"internalType":"struct IMultiSourceLoan.RenegotiationOffer","name":"_renegotiationOffer","type":"tuple"},{"components":[{"internalType":"address","name":"borrower","type":"address"},{"internalType":"uint256","name":"nftCollateralTokenId","type":"uint256"},{"internalType":"address","name":"nftCollateralAddress","type":"address"},{"internalType":"address","name":"principalAddress","type":"address"},{"internalType":"uint256","name":"principalAmount","type":"uint256"},{"internalType":"uint256","name":"startTime","type":"uint256"},{"internalType":"uint256","name":"duration","type":"uint256"},{"components":[{"internalType":"uint256","name":"loanId","type":"uint256"},{"internalType":"uint256","name":"floor","type":"uint256"},{"internalType":"uint256","name":"principalAmount","type":"uint256"},{"internalType":"address","name":"lender","type":"address"},{"internalType":"uint256","name":"accruedInterest","type":"uint256"},{"internalType":"uint256","name":"startTime","type":"uint256"},{"internalType":"uint256","name":"aprBps","type":"uint256"}],"internalType":"struct IMultiSourceLoan.Tranche[]","name":"tranche","type":"tuple[]"},{"internalType":"uint256","name":"protocolFee","type":"uint256"}],"internalType":"struct IMultiSourceLoan.Loan","name":"_loan","type":"tuple"},{"internalType":"bytes","name":"_renegotiationOfferSignature","type":"bytes"}],"name":"refinanceFull","outputs":[{"internalType":"uint256","name":"","type":"uint256"},{"components":[{"internalType":"address","name":"borrower","type":"address"},{"internalType":"uint256","name":"nftCollateralTokenId","type":"uint256"},{"internalType":"address","name":"nftCollateralAddress","type":"address"},{"internalType":"address","name":"principalAddress","type":"address"},{"internalType":"uint256","name":"principalAmount","type":"uint256"},{"internalType":"uint256","name":"startTime","type":"uint256"},{"internalType":"uint256","name":"duration","type":"uint256"},{"components":[{"internalType":"uint256","name":"loanId","type":"uint256"},{"internalType":"uint256","name":"floor","type":"uint256"},{"internalType":"uint256","name":"principalAmount","type":"uint256"},{"internalType":"address","name":"lender","type":"address"},{"internalType":"uint256","name":"accruedInterest","type":"uint256"},{"internalType":"uint256","name":"startTime","type":"uint256"},{"internalType":"uint256","name":"aprBps","type":"uint256"}],"internalType":"struct IMultiSourceLoan.Tranche[]","name":"tranche","type":"tuple[]"},{"internalType":"uint256","name":"protocolFee","type":"uint256"}],"internalType":"struct IMultiSourceLoan.Loan","name":"","type":"tuple"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"components":[{"internalType":"uint256","name":"renegotiationId","type":"uint256"},{"internalType":"uint256","name":"loanId","type":"uint256"},{"internalType":"address","name":"lender","type":"address"},{"internalType":"uint256","name":"fee","type":"uint256"},{"internalType":"uint256[]","name":"trancheIndex","type":"uint256[]"},{"internalType":"uint256","name":"principalAmount","type":"uint256"},{"internalType":"uint256","name":"aprBps","type":"uint256"},{"internalType":"uint256","name":"expirationTime","type":"uint256"},{"internalType":"uint256","name":"duration","type":"uint256"}],"internalType":"struct IMultiSourceLoan.RenegotiationOffer","name":"_renegotiationOffer","type":"tuple"},{"components":[{"internalType":"address","name":"borrower","type":"address"},{"internalType":"uint256","name":"nftCollateralTokenId","type":"uint256"},{"internalType":"address","name":"nftCollateralAddress","type":"address"},{"internalType":"address","name":"principalAddress","type":"address"},{"internalType":"uint256","name":"principalAmount","type":"uint256"},{"internalType":"uint256","name":"startTime","type":"uint256"},{"internalType":"uint256","name":"duration","type":"uint256"},{"components":[{"internalType":"uint256","name":"loanId","type":"uint256"},{"internalType":"uint256","name":"floor","type":"uint256"},{"internalType":"uint256","name":"principalAmount","type":"uint256"},{"internalType":"address","name":"lender","type":"address"},{"internalType":"uint256","name":"accruedInterest","type":"uint256"},{"internalType":"uint256","name":"startTime","type":"uint256"},{"internalType":"uint256","name":"aprBps","type":"uint256"}],"internalType":"struct IMultiSourceLoan.Tranche[]","name":"tranche","type":"tuple[]"},{"internalType":"uint256","name":"protocolFee","type":"uint256"}],"internalType":"struct IMultiSourceLoan.Loan","name":"_loan","type":"tuple"}],"name":"refinancePartial","outputs":[{"internalType":"uint256","name":"","type":"uint256"},{"components":[{"internalType":"address","name":"borrower","type":"address"},{"internalType":"uint256","name":"nftCollateralTokenId","type":"uint256"},{"internalType":"address","name":"nftCollateralAddress","type":"address"},{"internalType":"address","name":"principalAddress","type":"address"},{"internalType":"uint256","name":"principalAmount","type":"uint256"},{"internalType":"uint256","name":"startTime","type":"uint256"},{"internalType":"uint256","name":"duration","type":"uint256"},{"components":[{"internalType":"uint256","name":"loanId","type":"uint256"},{"internalType":"uint256","name":"floor","type":"uint256"},{"internalType":"uint256","name":"principalAmount","type":"uint256"},{"internalType":"address","name":"lender","type":"address"},{"internalType":"uint256","name":"accruedInterest","type":"uint256"},{"internalType":"uint256","name":"startTime","type":"uint256"},{"internalType":"uint256","name":"aprBps","type":"uint256"}],"internalType":"struct IMultiSourceLoan.Tranche[]","name":"tranche","type":"tuple[]"},{"internalType":"uint256","name":"protocolFee","type":"uint256"}],"internalType":"struct IMultiSourceLoan.Loan","name":"","type":"tuple"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_contract","type":"address"}],"name":"removeWhitelistedCallbackContract","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"components":[{"components":[{"internalType":"uint256","name":"loanId","type":"uint256"},{"internalType":"bytes","name":"callbackData","type":"bytes"},{"internalType":"bool","name":"shouldDelegate","type":"bool"}],"internalType":"struct IMultiSourceLoan.SignableRepaymentData","name":"data","type":"tuple"},{"components":[{"internalType":"address","name":"borrower","type":"address"},{"internalType":"uint256","name":"nftCollateralTokenId","type":"uint256"},{"internalType":"address","name":"nftCollateralAddress","type":"address"},{"internalType":"address","name":"principalAddress","type":"address"},{"internalType":"uint256","name":"principalAmount","type":"uint256"},{"internalType":"uint256","name":"startTime","type":"uint256"},{"internalType":"uint256","name":"duration","type":"uint256"},{"components":[{"internalType":"uint256","name":"loanId","type":"uint256"},{"internalType":"uint256","name":"floor","type":"uint256"},{"internalType":"uint256","name":"principalAmount","type":"uint256"},{"internalType":"address","name":"lender","type":"address"},{"internalType":"uint256","name":"accruedInterest","type":"uint256"},{"internalType":"uint256","name":"startTime","type":"uint256"},{"internalType":"uint256","name":"aprBps","type":"uint256"}],"internalType":"struct IMultiSourceLoan.Tranche[]","name":"tranche","type":"tuple[]"},{"internalType":"uint256","name":"protocolFee","type":"uint256"}],"internalType":"struct IMultiSourceLoan.Loan","name":"loan","type":"tuple"},{"internalType":"bytes","name":"borrowerSignature","type":"bytes"}],"internalType":"struct IMultiSourceLoan.LoanRepaymentData","name":"_repaymentData","type":"tuple"}],"name":"repayLoan","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_newOwner","type":"address"}],"name":"requestTransferOwner","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_delegate","type":"address"},{"internalType":"address","name":"_collection","type":"address"},{"internalType":"uint256","name":"_tokenId","type":"uint256"},{"internalType":"bytes32","name":"_rights","type":"bytes32"}],"name":"revokeDelegate","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_newFlashActionContract","type":"address"}],"name":"setFlashActionContract","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"__minLockPeriod","type":"uint256"}],"name":"setMinLockPeriod","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"setProtocolFee","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"transferOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"newOwner","type":"address"}],"name":"transferOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint48","name":"_newDuration","type":"uint48"}],"name":"updateLiquidationAuctionDuration","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"__loanLiquidator","type":"address"}],"name":"updateLiquidationContract","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_newMinimum","type":"uint256"}],"name":"updateMinImprovementApr","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"components":[{"internalType":"address","name":"recipient","type":"address"},{"internalType":"uint256","name":"fraction","type":"uint256"}],"internalType":"struct WithProtocolFee.ProtocolFee","name":"_newProtocolFee","type":"tuple"}],"name":"updateProtocolFee","outputs":[],"stateMutability":"nonpayable","type":"function"}]
Deployed Bytecode
0x6102a080604052600436101561001457600080fd5b600090813560e01c90816306fdde031461351b5750806308910dce146131515780630a90bd9b14613133578063150b7a02146130dc57806324fd49e414612f6857806326f86c2614612a215780632891433d146129e85780632fd8abab146129715780633644e5151461294e57806336cb351b146128f35780633fb35178146128d557806344204ee31461289a57806344af681c1461287c5780634fd930ba146128085780635122afb11461277a578063584d158f146127515780635a3298941461270c57806365e03b9c14611f96578063665458a714611f4d57806367e2240414611f125780636ccc9dde14611e835780637320ca2614611e2857806373b99f1014611d8b57806373c2390c14611d655780637795960b14611cd65780637b8d1a0b14611ca95780638617780b14611c8b578063880ad0af14611bc25780638cc09aea14611b295780638da5cb5b14611b00578063912d819e14611ab7578063931c56e914611a99578063986a7ace14611a32578063995fe720146119ed578063a009ca9a14611773578063a258451714611557578063a5a4103114611526578063aa29dad1146114fc578063ac9650d81461135f578063acb1dfdb1461131c578063b97e527a14611274578063bca67b9314611239578063c2f50a7a1461121b578063c4ca53ad146111fd578063c9f7383a14611065578063cc37ef4f14611026578063cf23797d14611008578063da8d76b514610f88578063db540a8a14610f15578063dd7d978214610ed0578063e30c397814610ea7578063e570267b14610bac578063e69d8c4714610b6f578063ef706adf14610b02578063f18de94f14610abd578063f2fde38b14610a4e578063f59421f91461079d578063f6fa901f14610308578063f838d945146102ec5763ffa1ad74146102b657600080fd5b346102e957806003193601126102e9576102e56102d1613b2c565b604051918291602083526020830190613726565b0390f35b80fd5b50346102e957806003193601126102e957602060405160328152f35b50346102e95761031736613abb565b916103256001825414613dd1565b6002815561033b6103363685613877565b615d2a565b91808252602092601284526040832054036107855760e084019360016103618683614205565b90501494856106d2575b8360609660c084013560a0850135014281116106ba57501561049157506103a76001600160a01b0361039f60408501613e20565b169183614205565b1561047d5760606103b89101613e20565b91813b156104795791849186836103ea95604051968795869485936323b872dd60e01b85520135903060048501613eb5565b03925af1801561046e57610456575b50906001917f5985c1ea46d5ed2be4837f88c0563e18ed022c3bfe10dee15f6ce74daf897d6384604051838152a1825b610444575b50556102e5604051928284938452830190613726565b8152601283528060408120553861042e565b6104608391613646565b61046a57386103f9565b5080fd5b6040513d85823e3d90fd5b8480fd5b634e487b7160e01b85526032600452602485fd5b600a546001600160a01b039760309190911c881696959493919250906040810190886104bc83613e20565b16918782013592803b156106b65787846104ef928c836040518096819582946323b872dd60e01b84523060048501613eb5565b03925af180156106ab57908891610693575b505061050f61051791613e20565b938201613e20565b608065ffffffffffff600a541692013592612710947f051eb851eb851eb851eb851eb851eb851eb851eb851eb851eb851eb851eb851e85116032021586021561068f5760405163378685eb60e21b815260048101899052908c1660248201526044810191909152991660648a015260848901526032020460a48701523360c4870152828660e48183895af195861561046e5783966105e8575b507f0c0f58e3bcc7b4b6f90dac7e83ad948f2dfe00aff9393c51aa2181fa08fb94e160406001959681519085825288820152a1610429565b95503d8084883e6105f981886136e2565b860195848188031261068b578051906001600160401b038211610479570186601f8201121561068b5780519061062e82613b11565b9761063c604051998a6136e2565b82895286838301011161047957876106817f0c0f58e3bcc7b4b6f90dac7e83ad948f2dfe00aff9393c51aa2181fa08fb94e193604093898060019b9c9d019101613703565b97969550506105b0565b8380fd5b8880fd5b61069c90613646565b6106a7578638610501565b8680fd5b6040513d8a823e3d90fd5b8780fd5b6024906040519063b73a6a1360e01b82526004820152fd5b94506001600160a01b036106e68683614205565b1561047d578160246106fb6060899401613e20565b604051948593849263f4070e2d60e01b84521660048301527f0000000000000000000000001e09403be51029fe676e684743b6aad2dab7c9f5165afa90811561077a57849161074d575b50159461036b565b61076d9150853d8711610773575b61076581836136e2565b81019061423a565b38610745565b503d61075b565b6040513d86823e3d90fd5b6024906040519063047e261360e21b82526004820152fd5b50346102e95760031960803682011261046a576001600160401b036004356024358281116104795761012081600401948236030112610479576107de6137a8565b92606435908111610a4a576107f79036906004016137d2565b9190946108076001885414613dd1565b600287556108186103363683613877565b8488526012602052604088205403610a31576001600160a01b0390819061083e90613e20565b163303610a1f57806013541660448301908860248461085c85613e20565b1695013594803b1561046a578160405180926323b872dd60e01b825281838161088a8c8a3060048501613eb5565b03925af18015610a1457610a00575b506108a383613e20565b823b1561046a5789988593848480946108ef6040519889968795869463ee534a3f60e01b86521660048501528d6024850152169e8f6044840152608060648401528d60848401916141e4565b03925af19182156109f35784926109dc575b505061090e602092613e20565b16926024604051809581936331a9108f60e11b835260048301525afa9182156109d15787926109a1575b503091160361098f577f13ab7606a034e1d4320c1129442ffa929fe2789f8cd0d03d147ff4f4707567a59361098591604051948594855260208501526060604085015260608401916141e4565b0390a16001815580f35b60405163807a2b8f60e01b8152600490fd5b6109c391925060203d81116109ca575b6109bb81836136e2565b810190614252565b9038610938565b503d6109b1565b6040513d89823e3d90fd5b6109e7919250613646565b6106b657818838610901565b50604051903d90823e3d90fd5b610a0990613646565b61068f578838610899565b6040513d84823e3d90fd5b6040516325d3d62760e21b8152600490fd5b60405163047e261360e21b815260048101859052602490fd5b8580fd5b50346102e95760203660031901126102e957610a6861377c565b600154906001600160a01b0390610a823383851614613b49565b166001600160a01b0319919091168117600155337f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e08380a380f35b50346102e957806003193601126102e9576040517f0000000000000000000000001e09403be51029fe676e684743b6aad2dab7c9f56001600160a01b03168152602090f35b50346102e95760203660031901126102e9577f899cab278284ae4a91172caa0943607a0bcb19766254c3ebe1139be00650b102610b69600435338452600f6020526040842081855260205260408420600160ff198254161790556040519182913383613bf9565b0390a180f35b50346102e957806003193601126102e95760006020604051610b9081613690565b82815201526102e5610ba0614e32565b60405191829182613af1565b50346102e957610bbb36613a57565b9093610bc8939293613d8b565b50600191610bd883835414613dd1565b6002825584516001600160a01b03949085163303610a1f57610c33602082013592610c038885614493565b610c0d88846144e0565b6040830198610c1b8a613e20565b610c2d610c283687613f38565b615e94565b90615925565b6080810184610c428284613e49565b90501490811591610e73575b50610e615760e08601968751517f000000000000000000000000000000000000000000000000000000000000000a14610e4f5784600d54019586600d55610c93613d8b565b5060a08301359160808901610ca881516158e6565b8410610e3d578a5151888101808211610e2957610cc490614031565b90885b818110610dfe575091600080516020615f8b83398151915297969593918d610d4287958e610db1998651610cfa89613e20565b9060405193610d0885613659565b845260208401528a60408401521660608201528d60808201524260a082015260c08b013560c0820152610d3b8286613be5565b5283613be5565b5052838151019052610d538b615d2a565b8a8952601260205260408920558588528760408120558a6060810192610d9981855116610d7f85613e20565b8385511690610d9360608c0135809a613f2b565b926140e1565b84610dd1575b509150508860405195869535866140b3565b0390a1556102e56040519283928352604060208401526040830190613975565b610d93610100610de783610df597511695613e20565b926004541693015186614168565b89808a81610d9f565b80610e0c8f928d9351613be5565b51610e178286613be5565b52610e228185613be5565b5001610cc7565b634e487b7160e01b89526011600452602489fd5b6040516391de6ae760e01b8152600490fd5b604051638efb932960e01b8152600490fd5b604051632544458760e01b8152600490fd5b610e7e915082613e49565b15610e935760e0870151519035141538610c4e565b634e487b7160e01b84526032600452602484fd5b50346102e957806003193601126102e9576002546040516001600160a01b039091168152602090f35b50346102e957806003193601126102e9576040517f00000000000000000000000052ac424ef7b283aa5badb8c6254832e3280d73986001600160a01b03168152602090f35b50346102e95760203660031901126102e9577faed358a2bd4ca37fb72362cad5ae66b0b49a2a2031f9474a656e47b1f4afc4196020610f5261377c565b6001546001600160a01b039190610f6c9083163314613b49565b168060018060a01b03196013541617601355604051908152a180f35b50346102e95760203660031901126102e9577f6a6b10ec403319c89f7cc46cb63debaa4dd4b26822f357ca8f43e7a05059a69b6020610fc561377c565b6001546001600160a01b039190610fdf9083163314613b49565b610fe881613b84565b16808452600982526040808520805460ff1916600117905551908152a180f35b50346102e957806003193601126102e9576020601454604051908152f35b50346102e95760203660031901126102e95760209060ff906040906001600160a01b0361105161377c565b168152600984522054166040519015158152f35b50346102e95760031960a03682011261046a57600435602435916001600160401b03831161068b576101208360040191843603011261068b576110a66137a8565b9060643590608435928315158094036111d8576110c66103363684613877565b91858852602092601284526040892054036111e4576001600160a01b03919082906110f090613e20565b163303610a1f578160a484928a83602461110c60448e01613e20565b9c604051988997889663b18e2bbb60e01b8852169e8f60048801521682860152013560448401528860648401528960848401527f00000000000000000000000000000000000000447e69651d841bd8d104bed493165af180156109d1576111ae575b50907fda896c547bd922b72f626042731e13ee6399eebfb99cbe5b5f6f172c539a9a3394608094939260405194855284015260408301526060820152a180f35b908094939291813d83116111dd575b6111c781836136e2565b810103126111d8579091923861116e565b600080fd5b503d6111bd565b60405163047e261360e21b815260048101879052602490fd5b50346102e957806003193601126102e9576020600354604051908152f35b50346102e957806003193601126102e95760206040516203f4808152f35b50346102e957806003193601126102e95760206040517f000000000000000000000000000000000000000000000000000000000002a3008152f35b50346102e95760203660031901126102e95760043565ffffffffffff811680910361046a576112ae60018060a01b03600154163314613b49565b6203f48081108015611310575b6112fe576020817f9b0306e96c09148e30f9acd9a1ebc2c7cb1bc0348adbf320530ead875f78292c9265ffffffffffff19600a541617600a55604051908152a180f35b604051632a48be4b60e21b8152600490fd5b5062093a8081116112bb565b50346102e95760403660031901126102e9576020906040906001600160a01b0361134461377c565b168152600e8352818120602435825283522054604051908152f35b5060208060031936011261046a576004356001600160401b03928382116102e957366023830112156102e95781600401359384116102e957602480830194600595368383891b8701011161068b576113b6826137ff565b946113c460405196876136e2565b8286526113d0836137ff565b601f190187865b8281106114ec57505050845b8381106114495750505050506040519280840190808552835180925280604083818801981b870101940192955b82871061141d5785850386f35b909192938280611439600193603f198a82030186528851613726565b9601920196019592919092611410565b858061145b87848d1b86010186613ba6565b9081604051928392833781018381520390305af43d156114e5573d61147f81613b11565b9061148d60405192836136e2565b81523d888b83013e5b6114a0838a613be5565b52156114ae576001016113e3565b806114e16114bd87938a613be5565b5160408051948594631b3dcf4560e21b865260048601528401526044830190613726565b0390fd5b6060611496565b606082828b0101520188906113d7565b50346102e95760203660031901126102e95760406020916004358152601283522054604051908152f35b50346102e957806003193601126102e9576000602060405161154781613690565b82815201526102e5610ba0614e57565b50346102e957600319906060368301126102e9576001600160401b03906024359060043583831161046a57826004019461012081853603011261176f5760443594851161176f5760609085600401953603011261046a576115b6613d8b565b506115c46001835414613dd1565b600282556001600160a01b0393846115db87613e20565b163303610a1f576115f56115ef3688613877565b83614493565b6115ff8180613e34565b9461160987613e20565b958661161482614a72565b94909361162c6116248580613e49565b94909361471c565b6116368686614aad565b61164260648a01613e20565b81168582161480159190611754575b506117425760208301359760240135889003611730576116b9986001987f2fee6b2365f3376bd2edcf343488ffb5fd987d4fbe034770ba7ed3a448f8edde966116ec968660406116a660806116af9a01613e20565b91013594614e7c565b9192989099614bd6565b505060405192839286845289602085015260a060408501526116de60a085018c613975565b908482036060860152613ed7565b9060808301520390a16116fe85615d2a565b848352601260205260408320558152806040812055556102e56040519283928352604060208401526040830190613975565b60405163734d5b5f60e11b8152600490fd5b604051636902417b60e11b8152600490fd5b905061176260448a01613e20565b8116908616141538611651565b8280fd5b50346102e9576117823661374b565b9061178b613d8b565b506117996001825414613dd1565b600281556020916117ab838201613e20565b916117b68280613e34565b91836117fe6117c485614a72565b938493916117d56116248980613e49565b6117df8582614aad565b6117eb60808901613e20565b888b01359560408a01359287928c614e7c565b9392979194909660a08101906118148282613ba6565b9050611904575b50506001600160a01b031690813b15610a4a5791859161185493836040518096819582946323b872dd60e01b8452309060048501613eb5565b03925af1801561077a576118f0575b50917fcb618933e86a4dbdce8a8ba3cbf5c6544be6e1984c53a465ad18a9b472c63bae6118c2600193869561189a6102e598615d2a565b89865260128b5260408620556118d06040519384938b855260808d8601526080850190613ed7565b838103604085015288613975565b9060608301520390a1556040805194859485528401526040830190613975565b6118fa8491613646565b61176f5738611863565b6119189161191191613ba6565b3691613e7e565b33875260098a52878a60ff60408a20541615928315611952575b50505061194057388061181b565b60405163275d4ca360e11b8152600490fd5b6119979293506040518093819261197f634ba6e49d60e11b97888552606060048601526064850190613975565b908b6024850152600319848303016044850152613726565b03818b335af19081156106ab5788916119c0575b506001600160e01b0319161415878a38611932565b6119e091508b3d8d116119e6575b6119d881836136e2565b810190613f0b565b386119ab565b503d6119ce565b50346102e957806003193601126102e9576040517f00000000000000000000000000000000000000447e69651d841bd8d104bed4936001600160a01b03168152602090f35b50346102e95760203660031901126102e9577fa39a039982552bc613a549c5a31d7c9a911b4e57c7da274435fca0e156c9c340610b6960043533845260116020526040842081855260205260408420600160ff198254161790556040519182913383613bf9565b50346102e957806003193601126102e957602060405162278d008152f35b50346102e95760403660031901126102e95760209060ff906040906001600160a01b03611ae261377c565b16815260118452818120602435825284522054166040519015158152f35b50346102e957806003193601126102e9576001546040516001600160a01b039091168152602090f35b50346102e957611b3836613abb565b600a5460301c6001600160a01b031633819003611baa5750610336611b5e913690613877565b8183526012602052604083205403610785577faf91ffe368225045d42f7c2b4b90feb438133ac1e375e73bc9066652b70a1d776020604051838152a18152601260205280604081205580f35b60249060405190638afaa9b560e01b82526004820152fd5b50346102e957806003193601126102e957611c006003547f000000000000000000000000000000000000000000000000000000000002a3009061401a565b4210611c79576002546001600160a01b0381163303611c675760018054336001600160a01b0319808316821790935592909116600255600019600355821681177f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e08380a380f35b6040516319e9109760e11b8152600490fd5b604051632a28ea7760e21b8152600490fd5b50346102e957806003193601126102e9576020600d54604051908152f35b50346102e957806003193601126102e957600a5460405160309190911c6001600160a01b03168152602090f35b50346102e95760203660031901126102e9576004353382526010602052604082205481811015611d415750610b697feaacd06746bfd850a83bbe929f94f4e1413f9aac82dc76e4331418aaf8bba4689133845260106020528060408520556040519182913383613bf9565b6064925060405191630ae1eb6f60e31b835233600484015260248301526044820152fd5b50346102e957806003193601126102e957602065ffffffffffff600a5416604051908152f35b50346102e957806003193601126102e95760085462278d008101809111611e14574210611c79577fb3c1d38dbdc9199d0ce01f386d70e29014ed4af7af1d321ca6641a91f4b4dc0c610b69611dde614e32565b8051600480546001600160a01b0319166001600160a01b0392909216919091179055602081015160055560405191829182613af1565b634e487b7160e01b82526011600452602482fd5b50346102e95760203660031901126102e9577fe818ff6972bf8970ca14a893539e57452588b197b76e6afe29c40a31b3363c616020600435611e7560018060a01b03600154163314613b49565b80601455604051908152a180f35b50346102e95760203660031901126102e9577f0656dc0fa3e59bca53331c14b87a4778ffac8c184915152fdd0f103838a4f3786020611ec061377c565b6001546001600160a01b0390611ed99082163314613b49565b611ee282613b84565b600a8054600160301b600160d01b031916603084901b600160301b600160d01b031617905560405191168152a180f35b50346102e957806003193601126102e95760206040517ff3171f78f373ce7d69f28649c2b033ea27b739d1cbbb4f85c857bc06d20142188152f35b50346102e95760403660031901126102e95760209060ff906040906001600160a01b03611f7861377c565b168152600f8452818120602435825284522054166040519015158152f35b50346102e957611fa536613a57565b611fb3949194939293613d8b565b50611fc16001835414613dd1565b60028255611fd3846020850135614493565b611fdd84846144e0565b611fea6080840184613e49565b905060e08501515103610e61576001600160a01b0361200b60408501613e20565b16331494612021606085013560a0860135613f2b565b9186156124ee575050506000906000906120446101008601516060860135614168565b600b546080529486156124e557919492916000195b865b8760e08401518051908160011b83101561218357811561216d5761208191830690613be5565b5160e08501515160608201519192916001906001600160a01b03908116906120ab60408e01613e20565b161492019a1003612164576120fb6120c560408a01613e20565b8551606087015160a088015160c0808a01516101008b0151919995919492019287926001600160a01b0391821692911690614299565b9391929093948c8183015181612129575b50610e61576040820151910151020195019501965b96949361205b565b9050612138828d013582613f2b565b90612710600080516020615fab83398151915283118102158202156111d857608051920204103861210c565b50939496612121565b634e487b7160e01b600052601260045260246000fd5b5050509650909650939290938181116000146124e05750805b60608701516001600160a01b0316816121b760408701613e20565b6004546001600160a01b031690826124ce575b50505050606087015187516004546001600160a01b039283169390831692909116908085036124b9575b50505050509260a085015160c086015161221060145482614168565b91010342116124a75760808501519061222e4261010085013561401a565b61225061224460c089015160a08a01519061401a565b92608089015190614027565b600b54906122628560a0880135613f2b565b15159485612448575b505050821561243a575b8215612430575b505061241e5760808401518060a0830135116123dd575b505b6001600d54019283600d55604051906122ad82613690565b6001825260005b60208110612395575082600080516020615f8b83398151915292610db1926122e160406001989701613e20565b90604051916122ef83613659565b89835287602084015260a08501356040840152888060a01b0316606083015260808201524260a082015260c083013560c082015261232c82613bd8565b5261233681613bd8565b5060e08801524260a088015261010081013560c088015260a0810135608088015261236087615d2a565b8685526012602052604085205560208101358452836040812055604051918188886060869501359260208101359035866140b3565b6020906040516123a481613659565b60008152600083820152600060408201526000606082015260006080820152600060a0820152600060c0820152828286010152016122b4565b6060850151612418916001600160a01b03909116906123fe60408501613e20565b87516001600160a01b031691610d939060a0870135613f2b565b38612293565b604051633240c6ed60e21b8152600490fd5b109050388061227c565b606084013515159250612275565b819293955061247561245d8261247b946158d3565b61246f60a08a013560c08b01356158d3565b90613f2b565b926158d3565b9061271090600080516020615fab83398151915281118202158302156106b6570204109138808061226b565b6040516341ba0a1760e11b8152600490fd5b6124c49403926140e1565b38808080806121f4565b6124d7936140e1565b388181806121ca565b61219c565b85939295612059565b855193965091949390926001600160a01b03163314612519576040516325d3d62760e21b8152600490fd5b6000949291949160006125356101008701516060850135614168565b80950191815b8260e08901518051908160011b83101561260157811561216d5761256191830690613be5565b5160e08a01515160608201519192916001906001600160a01b039081169061258b60408c01613e20565b161492019510036125f8576125da6125a560408801613e20565b9560018060a01b038b5116968b8460018060a01b0360608301511660a08301519a61010060c09c8d8601510194015194614299565b9391929093966040820151910151020196019601975b97959461253b565b509495976125f0565b505050509050612692929691935096949684841160001461270557845b60608901516001600160a01b03168661263960408b01613e20565b6004546001600160a01b031690826126f3575b50505050606089015189516004546001600160a01b03928316929081169116888085036126de575b505050505061268560408801613e20565b610c2d610c28368a613f38565b8181036126a3575b50508091612295565b60608501516126d7926001600160a01b03909116906126c460408701613e20565b92039160018060a01b03875116916140e1565b388061269a565b6126e99403926140e1565b3880808088612674565b6126fc936140e1565b3886818061264c565b839461261e565b50346102e957806003193601126102e9576040517f0000000000000000000000004150ded32a6d3bfecae76e7558af4801903449276001600160a01b03168152602090f35b50346102e957806003193601126102e9576013546040516001600160a01b039091168152602090f35b50346102e95760403660031901126102e9577f160fef22fef07b45037a807beef2c89408a81168d1055cf34024c85396b882af604060018060a01b036127c581600154163314613b49565b6127d56127d0613e0a565b613b84565b6127dd613e0a565b168060018060a01b03196006541617600655602435806007554260085582519182526020820152a180f35b50346102e95760203660031901126102e9577f924773930b2679ab8bf328330b211bedfb8f917551fd856f536bec008d17f9a6602061284561377c565b6001546001600160a01b03919061285f9083163314613b49565b16808452600982526040808520805460ff1916905551908152a180f35b50346102e957806003193601126102e9576020600854604051908152f35b50346102e957806003193601126102e95760206040517f000000000000000000000000000000000000000000000000000000000000000a8152f35b50346102e957806003193601126102e957602060405162093a808152f35b50346102e95760203660031901126102e9577f257d95d6b6cd5a1a1e56bddff8487147cadd06fce4618b2197b174ccc9634201602060043561294060018060a01b03600154163314613b49565b80600b55604051908152a180f35b50346102e957806003193601126102e9576020612969613c14565b604051908152f35b50346102e95760203660031901126102e9577f73541446fa014a1018bb483540e823dcc1bb0c55215d112b403bf614ff86672f60206129ae61377c565b6001546001600160a01b0391906129c89083163314613b49565b168060018060a01b0319600254161760025542600355604051908152a180f35b50346102e95760203660031901126102e9576020906040906001600160a01b03612a1061377c565b168152601083522054604051908152f35b50346102e95760031960403682011261046a576001600160401b0390816004351161176f57610120906004353603011261046a5760243590811161046a57612a6d903690600401613877565b90612a76613d8b565b50612a846001825414613dd1565b600281556001600160a01b03612a9e600435604401613e20565b163303610a1f5760a082015160c0830151612abb60145482614168565b91010342116124a757612ad8608460043501600435600401613e49565b905015610e6157612aef8260246004350135614493565b612afe826004356004016144e0565b6001600d54019081600d5580808190600b54612b24608460043501600435600401613e49565b91905084925b828410612c17575050505060a4600435013503610e615760608401516001926001600160a01b0390911690612b63604460043501613e20565b848060a01b03600454169082612c05575b50505050612b8184615d2a565b83825260126020526040822055602460043501358152806040812055600080516020615f8b83398151915260405160043560040135815260246004350135602082015284604082015260a0606082015280612bdf60a0820188613975565b8460808301520390a1556102e56040519283928352604060208401526040830190613975565b612c0e936140e1565b38808080612b74565b612c306084600498939495983501600435600401613e49565b881015612f525760e08a0151518860051b8201351015610e6157612c5e908860051b013560e08b0151613be5565b519660c0880151612c7560c4600435013582613f2b565b90612710600080516020615fab83398151915283118102158202156111d8578692020410610e6157612cab604460043501613e20565b9160018060a01b038b51169260018060a01b0360608d015116908c8b612cfe60a0610100612ce18286015160c08701519061401a565b940151920151612cf8600094826014549103614168565b9061401a565b428111612f3a5750908c94939291612d2f6040870151612d2960c089015160a08a01514203906158d3565b9061418b565b6080612d3b8383614168565b9701510160a05260608e015160405163f4070e2d60e01b81526001600160a01b03918216600482015290602090829060249082907f0000000000000000000000001e09403be51029fe676e684743b6aad2dab7c9f5165afa90811561046e57908f92918491612f1b575b50612ea0575b505090848d604060a051910151010391612e55575b6001965081612e20575b50505060408a01518098019760c08b01510201930197898152828060a01b03612df7604460043501613e20565b16606082015260a05160808201524260a082015260c060c4600435013591015201929190612b2a565b8c6060888060a01b03910151169081888060a01b03821603612e43575b50612dca565b612e4c936140e1565b38808080612e3d565b60608d01516001600160a01b031696909190600182018210612e8a5760019788612e81930191856140e1565b60001990612dc0565b634e487b7160e01b600052601160045260246000fd5b60018060a01b03606083015116825160408401519060c08501519360a0608087015196015190843b156106b65791612ef68896959493928793604051998a988997889663faf7b8b160e01b885260048801614271565b03925af18015610a1457612f0c575b8d90612dab565b612f1590613646565b38612f05565b612f34915060203d6020116107735761076581836136e2565b38612da5565b60249060405190630d9ad65160e31b82526004820152fd5b634e487b7160e01b600052603260045260246000fd5b50346102e95760803660031901126102e957612f8261377c565b612f8a613792565b6040516331a9108f60e11b8152604435600482018190526001600160a01b0392831693606435926020919082816024818a5afa9081156106ab5788916130bf575b50851630146130ad57818560a48993604051948593849263b18e2bbb60e01b845216998a60048401528b6024840152886044840152896064840152600060848401527f00000000000000000000000000000000000000447e69651d841bd8d104bed493165af180156109d15761307c575b50907f222cb479d85029b2315fc9f22998b9d5616cac50c80dc1a1126c0804ab3126fd94608094939260405194855284015260408301526060820152a180f35b908094939291813d83116130a6575b61309581836136e2565b810103126111d8579091923861303c565b503d61308b565b604051636443654760e01b8152600490fd5b6130d69150833d85116109ca576109bb81836136e2565b38612fcb565b50346102e95760803660031901126102e9576130f661377c565b506130ff613792565b506064356001600160401b03811161046a5761311f9036906004016137d2565b5050604051630a85bd0160e11b8152602090f35b50346102e957806003193601126102e9576020600b54604051908152f35b50346102e9576131603661374b565b61316d6001835414613dd1565b6002825561317b81806141b1565b356020908183013561011e198436030181121561047957830192846001600160a01b03806131a887613e20565b163303613440575b6131c36131bd3688613877565b85614493565b60406131cf84806141b1565b0135801515810361176f5761338f575b6131eb60408701613e20565b166131f586613e20565b813b1561176f578291613224916040519485809481936323b872dd60e01b83528b8d0135903060048501613eb5565b03925af180156133715761337c575b5061324a61324182806141b1565b84810190613ba6565b90506132a1575b507fd9cfbcfb9ae5d65b8d34072de4575f602ee25cf3715bc05c291701fe9e258f9a6060613280601295614bd6565b60405191858352868301526040820152a18352528060408120556001815580f35b6132ae816132b7926141b1565b83810190613ba6565b90836132c33687613877565b913388526009825260ff604089205416159384156132ea575b505050506119405738613251565b61332a939450604051938492839261331863188e78a960e11b98898652604060048701526044860190613975565b848103600319016024860152916141e4565b038189335af1908115613371578691613354575b506001600160e01b0319161415388381806132dc565b61336b9150843d86116119e6576119d881836136e2565b3861333e565b6040513d88823e3d90fd5b61338890959195613646565b9338613233565b61339886613e20565b816133a560408901613e20565b60405163b18e2bbb60e01b815292821660048401521660248201528686013560448201526064810183905260016084820152858160a481867f00000000000000000000000000000000000000447e69651d841bd8d104bed49387165af190811561046e578691613417575b50506131df565b813d8311613439575b61342a81836136e2565b8101031261046a578438613410565b503d613420565b61344986613e20565b61345384806141b1565b9060608236031261068b576040519161346b83613615565b80358352878101356001600160401b038111610a4a57613490604091369084016141c6565b91828a860152013592831515908185036106a757613516946040820152519189815191012090604051918a8301937f41277b3c1cbe08ea7bbdd10a13f24dc956f3936bf46526f904c73697d9958e0c8552604084015260608301526080820152608081526134fd816136ab565b5190209061350e6040870187613ba6565b929091615925565b6131b0565b82346102e957806003193601126102e957600c54828261353a836135db565b928383526020946001918683821691826000146135b957505060011461357d575b5050613569925003836136e2565b6102e5604051928284938452830190613726565b90859250600c82528282205b8583106135a15750506135699350820101858061355b565b80548389018501528794508693909201918101613589565b925093505061356994915060ff191682840152151560051b820101858061355b565b90600182811c9216801561360b575b60208310146135f557565b634e487b7160e01b600052602260045260246000fd5b91607f16916135ea565b606081019081106001600160401b0382111761363057604052565b634e487b7160e01b600052604160045260246000fd5b6001600160401b03811161363057604052565b60e081019081106001600160401b0382111761363057604052565b61012081019081106001600160401b0382111761363057604052565b604081019081106001600160401b0382111761363057604052565b60a081019081106001600160401b0382111761363057604052565b61016081019081106001600160401b0382111761363057604052565b90601f801991011681019081106001600160401b0382111761363057604052565b60005b8381106137165750506000910152565b8181015183820152602001613706565b9060209161373f81518092818552858086019101613703565b601f01601f1916010190565b600319906020818301126111d857600435916001600160401b0383116111d857826060920301126111d85760040190565b600435906001600160a01b03821682036111d857565b602435906001600160a01b03821682036111d857565b604435906001600160a01b03821682036111d857565b35906001600160a01b03821682036111d857565b9181601f840112156111d8578235916001600160401b0383116111d857602083818601950101116111d857565b6001600160401b0381116136305760051b60200190565b91908260e09103126111d85760405161382e81613659565b60c0808294803584526020810135602085015260408101356040850152613857606082016137be565b60608501526080810135608085015260a081013560a08501520135910152565b919091610120818403126111d8576040519061389282613674565b819361389d826137be565b8352602080830135818501526138b5604084016137be565b60408501526138c6606084016137be565b60608501526080830135608085015260a083013560a085015260c083013560c085015260e090818401356001600160401b0381116111d85784019083601f830112156111d8578135613917816137ff565b9461392560405196876136e2565b81865282858188019302850101938185116111d85783869101925b85841061395c5750505050505083015261010080910135910152565b84906139688486613816565b8152019201918590613940565b906101209182820160018060a01b0390818351168452602080840151818601526040928084860151168487015260609381858701511685880152608090818701518289015260a09081880151828a015260c09182890151838b015260e093848a01519b858c01528b51809852866101408c019c01986000965b898810613a0d5750505050505050505050506101008091015191015290565b8a5180518f52808a01518f8b0152808601518f8701528083015182168f840152808401518f850152808501518f8601528601518e8701529c86019c998801996001909701966139ee565b600319916060828401126111d8576004356001600160401b03938482116111d8576101209082850301126111d857600401926024358181116111d85783613aa091600401613877565b926044359182116111d857613ab7916004016137d2565b9091565b906003196040818401126111d85760043592602435916001600160401b0383116111d85782610120920301126111d85760040190565b81516001600160a01b031681526020918201519181019190915260400190565b6001600160401b03811161363057601f01601f191660200190565b60405190613b3982613690565b60018252603360f81b6020830152565b15613b5057565b60405162461bcd60e51b815260206004820152600c60248201526b15539055551213d49256915160a21b6044820152606490fd5b6001600160a01b031615613b9457565b6040516349431df560e11b8152600490fd5b903590601e19813603018212156111d857018035906001600160401b0382116111d8576020019181360383136111d857565b805115612f525760200190565b8051821015612f525760209160051b010190565b6001600160a01b039091168152602081019190915260400190565b6000467f000000000000000000000000000000000000000000000000000000000000000103613c6257507ff3171f78f373ce7d69f28649c2b033ea27b739d1cbbb4f85c857bc06d201421890565b604051600c548291613c73826135db565b8082528160209485820194600190878282169182600014613d6d575050600114613d30575b50613ca5925003826136e2565b51902091613cb1613b2c565b828151910120604051928301937f8b73c3c69bb8fe3d512ecc4cf759cc79239f7b179b0ffacaa9a75d522b39400f8552604084015260608301524660808301523060a083015260a0825260c08201908282106001600160401b03831117613d1c575060405251902090565b634e487b7160e01b81526041600452602490fd5b869150600c88528188209088915b858310613d55575050613ca5935082010138613c98565b80548388018501528694508893909201918101613d3e565b60ff19168852613ca595151560051b8501019250389150613c989050565b60405190613d9882613674565b816101006000918281528260208201528260408201528260608201528260808201528260a08201528260c0820152606060e08201520152565b15613dd857565b60405162461bcd60e51b815260206004820152600a6024820152695245454e5452414e435960b01b6044820152606490fd5b6004356001600160a01b03811681036111d85790565b356001600160a01b03811681036111d85790565b90359060be19813603018212156111d8570190565b903590601e19813603018212156111d857018035906001600160401b0382116111d857602001918160051b360383136111d857565b929192613e8a82613b11565b91613e9860405193846136e2565b8294818452818301116111d8578281602093846000960137010152565b6001600160a01b03918216815291166020820152604081019190915260600190565b90815180825260208080930193019160005b828110613ef7575050505090565b835185529381019392810192600101613ee9565b908160209103126111d857516001600160e01b0319811681036111d85790565b91908203918211612e8a57565b919091610120818403126111d85760405190613f5382613674565b81938135835260208083013581850152613f6f604084016137be565b60408501526060830135606085015260808301356001600160401b0381116111d85783019180601f840112156111d8578235613faa816137ff565b93613fb860405195866136e2565b818552838086019260051b8201019283116111d8578301905b82821061400b5750505050608083015260a081013560a083015260c081013560c083015260e081013560e083015261010080910135910152565b81358152908301908301613fd1565b91908201809211612e8a57565b811561216d570490565b9061403b826137ff565b60409061404a825191826136e2565b838152809361405b601f19916137ff565b019160005b83811061406d5750505050565b602090825161407b81613659565b60008152826000818301526000858301526000606083015260006080830152600060a0830152600060c0830152828601015201614060565b95949390926140dc9260809488526020880152604087015260a0606087015260a0860190613975565b930152565b9160008093602095606494604051946323b872dd60e01b865260018060a01b03809216600487015216602485015260448401525af13d15601f3d116001600051141617161561412c57565b60405162461bcd60e51b81526020600482015260146024820152731514905394d1915497d19493d357d1905253115160621b6044820152606490fd5b9061271091816000190481118202158302156111d8570290808204910615150190565b9064496cebb80091816000190481118202158302156111d8570290808204910615150190565b903590605e19813603018212156111d8570190565b9080601f830112156111d8578160206141e193359101613e7e565b90565b908060209392818452848401376000828201840152601f01601f1916010190565b903590601e19813603018212156111d857018035906001600160401b0382116111d8576020019160e08202360383136111d857565b908160209103126111d8575180151581036111d85790565b908160209103126111d857516001600160a01b03811681036111d85790565b9260a094919796959260c0850198855260208501526040840152606083015260808201520152565b909193959296949660a08101966142bc8851612cf8600093826014549103614168565b428111612f3a575060408201906142e08251612d2960c08601518c514203906158d3565b986142eb8b8b614168565b6080850151606086015160405163f4070e2d60e01b81526001600160a01b03918216600482015291909c019c919b602090829060249082907f0000000000000000000000001e09403be51029fe676e684743b6aad2dab7c9f5165afa90811561077a57918c95939185938f9691614475575b506143f5575b5050505101039460018060a01b0316928486116143c6575b508461438a575b505050905090565b6060015184916001600160a01b039182169181168290036143b4575b505050500380388080614382565b6143bd936140e1565b388281806143a6565b606082015191956143ed926001600160a01b0316906143e6908790613f2b565b91856140e1565b82933861437b565b91935091935060018060a01b03606086015116855185519060c0880151936080890151955190843b156106b6579161444b8896959493928793604051998a988997889663faf7b8b160e01b885260048801614271565b03925af18015610a1457918a93918c93614466575b80614363565b61446f90613646565b38614460565b61448d915060203d81116107735761076581836136e2565b3861435d565b61449c82615d2a565b8160005260126020526040600020540361078557508060c060a06144c59301519101519061401a565b4210156144ce57565b6040516301b5090760e61b8152600490fd5b9060a082013515908115614571575b50610e615760e0810135804211614559575061450f604082359201613e20565b60018060a01b038116600052601160205260406000208260005260205260ff6040600020541661453d575050565b6114e160405192839263fc4a5c2560e01b845260048401613bf9565b60249060405190635929d5bb60e01b82526004820152fd5b60e091500151516145856080830183613e49565b91905010386144ef565b91906040838203126111d857604051906145a882613690565b81936145b3816137be565b83526020810135916001600160401b0383116111d8576020926145d692016141c6565b910152565b9190916101a09081818503126111d8576040516001600160401b03928101838111828210176136305760405280948235825260209061461b8285016137be565b82840152604084013560408401526060840135606084015261463f608085016137be565b608084015260a084013560a084015261465a60c085016137be565b60c084015260e084013560e084015261010080850135908401526101208085013590840152610140808501359084015261016080850135908401526101809384810135908682116111d8570181601f820112156111d85780356146bc816137ff565b966146ca60405198896136e2565b818852848089019260051b840101938085116111d857858401925b8584106146f757505050505050500152565b83358381116111d8578791614711848480948a010161458f565b8152019301926146e5565b906001600160a01b0380821633036147a7575b5050606061473d8280613e34565b01354211614782576147528161475892613e34565b80613e49565b90507f000000000000000000000000000000000000000000000000000000000000000a10610e4f57565b606061479082602493613e34565b604051635929d5bb60e01b81529101356004820152fd5b6147b18380613e34565b9160c0833603126111d85760409182519360c08501916001600160401b03928681108482111761363057855281358381116111d85782019536601f880112156111d8578635602097614802826137ff565b9161480f895193846136e2565b808352898084019160051b830101913683116111d8578a8101915b8383106149e05750505050815286810192878101358452868201878201358152606093848401918584013583526080916148658386016137be565b9483870195865260a0810135908a82116111d857614885913691016141c6565b9560a081019687528c8c8992805151916000935b8c84861061493257509350505050815191012097519151935194511694518b8151910120958a519b8c01977fa5cb06a0c5f03000a6afa6b0d5080d0f863338257beb253058bc2c184ad7d4e189528b8d01528b015289015260a088015260c087015260e086015260e08552610100850191858310908311176136305761492b9461350e92855251902092850185613ba6565b388061472f565b916149d49186979361494986979860019751613be5565b51906149558251615b65565b91848482015191015184815191012090855192858401947ec14ad24a24ef957b8af9ebdfbc5d353bba0d3b20bbd97fb243c9f5fb3612828652878501528301528d8201528c81526149a5816136ab565b51902093825194816149c08793518092868087019101613703565b8201908382015203908101845201826136e2565b930191908f8f90614899565b82358981116111d85782019060609081601f1984360301126111d8578c51918e614a0984613615565b8401358c81116111d8578f908f90614a26908336918901016145db565b855285013590840152830135918b83116111d8578d8f94614a4d90868096369201016141c6565b9082015281520192019161482a565b90359061019e19813603018212156111d8570190565b614a7c9080613e49565b15612f5257614a8e81614a94926141b1565b80614a5c565b906141e16080614aa660c08501613e20565b9301613e20565b60408051633af32abf60e01b8082526001600160a01b039384166004830152919360209392909184816024817f0000000000000000000000004150ded32a6d3bfecae76e7558af48019034492788165afa908115614bcb57600091614bae575b5015614b9d579060248492848751958694859384521660048301527f00000000000000000000000052ac424ef7b283aa5badb8c6254832e3280d7398165afa918215614b9257600092614b75575b505015614b655750565b516378d31c7360e11b8152600490fd5b614b8b9250803d106107735761076581836136e2565b3880614b5b565b83513d6000823e3d90fd5b845163a212b5d360e01b8152600490fd5b614bc59150853d87116107735761076581836136e2565b38614b0d565b86513d6000823e3d90fd5b600091908290816001600160a01b03614bf160608401613e20565b16614bff60e0840184614205565b8591505b808210614c44575050610100830135614c1e575b5050509190565b614c2a614c3c93613e20565b6004546001600160a01b0316916140e1565b388381614c17565b91925093614c5560e0850185614205565b861015612f5257614c6c90369060e0880201613816565b90614c916040830151612d2960c0850151614c8b60a087015142613f2b565b906158d3565b600090610100870135614e0d575b90614cbf614cc492614cba604087015160808801519061401a565b61401a565b613f2b565b90614ce782614cd288613e20565b60608601516001600160a01b031690886140e1565b606083015160405163f4070e2d60e01b81526001600160a01b039182166004820152919092019691602090829060249082907f0000000000000000000000001e09403be51029fe676e684743b6aad2dab7c9f5165afa908115614de257600091614dee575b50614d5f575b6001915001869291614c03565b60018060a01b0360608301511682519060408401519060c08501519460a06080820151910151823b156111d85789614dba600096928793610100956040519b8c998a98899763faf7b8b160e01b895201359360048801614271565b03925af1918215614de257600192614dd3575b50614d52565b614ddc90613646565b38614dcd565b6040513d6000823e3d90fd5b614e07915060203d6020116107735761076581836136e2565b38614d4c565b98614cc49150614cbf614e256101008901358c614168565b8092019a91925050614c9f565b60405190614e3f82613690565b6006546001600160a01b031682526007546020830152565b60405190614e6482613690565b6004546001600160a01b031682526005546020830152565b9695949392919096610200526101405261028052610180526101c052614ea0613d8b565b50614eaa82614031565b6101e052614eca614eba836137ff565b60405161026052610260516136e2565b816102605152614ed9826137ff565b601f190136602061026051013760006101a0526001600d54016102205261022051600d55614f05614e57565b610240526101a05160001960c052928391825b848110614fb4575050505050614f306101a0516158e6565b60c05110610e3d5760206102405101519060405191614f4e83613674565b610200516001600160a01b0390811684526101805160208501526102805181166040850152610140511660608401526101a05160808401524260a08401526101c05160c08401526101e05160e08401526101008301526102205161026051909390929190565b614fc38160051b8301836141b1565b90614fce8280614a5c565b94614fdb60208701613e20565b614fe86040850185613ba6565b90610160526020610240510151614fff8680614a5c565b9161500c60208401613e20565b9061501e6101a05160208a013561401a565b61012052813b151580615526575b156155045750604051602080820152908735906001600160a01b03163689900361019e19018212156111d8576150ad8361509f6150776060958660408501528d60a0850191016155d7565b60208d01358387015261508d60408e018e6155a6565b848303603f19016080860152906141e4565b03601f1981018552846136e2565b803b156111d857600092836150e8936040519687958694859363fc1c5a7960e01b855261018051600486015260248501526064840190613726565b90604483015203925af18015614de2576154f5575b505b610120810135804211614559575060018060a01b038216600052600f6020526040600020813560005260205260ff6040600020541680156154d2575b6154b45760e081013561010052610100516101205111615490576101408101358015908115615483575b506112fe5761010081013515615471576060810135908115159283615434575b505050615422576151a26101805161519d8580614a5c565b615759565b60c051602084013510615415575b6151bc60208701613e20565b60e0526151cb60c08701613e20565b610140516001600160a01b039182169116148015906153f0575b611742576101608601358111610e3d5760405161520181613659565b6102205181526101a05160208201526020840135604082015260018060a01b0360e051166060820152600060808201524260a082015261010087013560c082015261524f836101e051613be5565b5261525d826101e051613be5565b5060208301356101a051016101a05261528b6152816101c0516101008901356158d3565b602085013561418b565b60208401350101946020830135600019046040820135116020840135021560e082013502156111d85782615352602060019501356040840135029960e08401358b0460e08501358c0615150101996152e560c08501613e20565b61024051602001516153049060e0870135808506151590850401614168565b878060a01b03610240515116816153d4575b50505087615348878060a01b0361532f60c08801613e20565b169260e087013580820615159104016020860135613f2b565b9160e051906140e1565b6060820135156153a057838060a01b0360e05116600052600e602052604060002082356000526020526020604060002091013581540190555b356153998261026051613be5565b5201614f18565b50828060a01b0360e05116600052600f6020526040600020813560005260205260406000208360ff1982541617905561538b565b6153e89260e051908a8060a01b03166140e1565b388080615316565b506153fd60808701613e20565b610280516001600160a01b03918216911614156151e5565b602083013560c0526151b0565b604051636d5b4eaf60e11b8152600490fd5b9091925060018060a01b0316600052600e60205260406000209035600052602052615468602085013560406000205461401a565b11388080615185565b6040516320dac71b60e01b8152600490fd5b90506101c0511138615165565b60405163111c5f2960e01b8152610120516004820152610100516024820152604490fd5b6114e160405192839263fc4a5c2560e01b8452359060048401613bf9565b506001600160a01b0382166000908152601060205260409020548135111561513b565b6154fe90613646565b386150fd565b9061552192506101605190610c2d61551c36876145db565b615b65565b6150ff565b5060405163f4070e2d60e01b81526001600160a01b038381166004830152602090829060249082907f0000000000000000000000001e09403be51029fe676e684743b6aad2dab7c9f5165afa908115614de257600091615587575b5061502c565b6155a0915060203d6020116107735761076581836136e2565b38615581565b9035601e19823603018112156111d85701602081359101916001600160401b0382116111d85781360383136111d857565b6101a09182810192823582526020906155f18285016137be565b6001600160a01b039081168484015260408581013581860152606080870135908601529080615622608088016137be565b16608086015260a086013560a08601528061563f60c088016137be565b1660c086015260e086013560e086015261010080870135908601526101208087013590860152610140808701359086015261016080870135908601526101809586810135601e19823603018112156111d85701803596858201986001600160401b0389116111d8578860051b958636038b136111d857899796989594939992860152526101c080840196840101979660009384605e1983360301915b8887106156f057505050505050505050505090565b909192939495969798999a6101bf19828203018b528b358481121561068b57896157466001938a6157378b828c8798018781019261572d846137be565b16865201906155a6565b919092818682015201916141e4565b9d019b01999897019594939291906156db565b60a081013590811561576d57500361173057565b916101808301915061577f8284613e49565b8093915015806158ca575b156157a15760405163734d5b5f60e11b8152600490fd5b6001831480615890575b61588a5760005b8381106157c0575050505050565b6157ca8286613e49565b821015612f52578160051b81013590603e19813603018212156111d8576157f39136910161458f565b8051602090910151906001600160a01b0316803b156111d85760006158549260409261583c8a8551968794859384936202044360ed1b85526060600486015260648501906155d7565b908c6024850152600319848303016044850152613726565b03915afa90811561588057509060019291615871575b50016157b2565b61587a90613646565b3861586a565b513d6000823e3d90fd5b50505050565b5061589b8185613e49565b15612f52578035603e19823603018112156111d8576001600160a01b03916158c39101613e20565b16156157ab565b5081151561578a565b81810292918115918404141715612e8a57565b7f000000000000000000000000000000000000000000000000000000000000000a906001600160ff1b0382168203612e8a576141e19160011b90614027565b929190926042615933613c14565b60409586519161190160f01b8352600283015260228201522092813b15156000146159e65791602091615989938651809581948293630b135d3f60e11b998a855260048501528a602485015260448401916141e4565b03916001600160a01b03165afa908115614b92576000916159c8575b506001600160e01b031916036159b85750565b51636cb40d4360e11b8152600490fd5b6159e0915060203d81116119e6576119d881836136e2565b386159a5565b926159fa615a0993615a0093953691613e7e565b90615a1e565b90929192615ae0565b6001600160a01b039081169116036159b85750565b8151919060418303615a4f57615a4892506020820151906060604084015193015160001a90615a5a565b9192909190565b505060009160029190565b91906fa2a8918ca85bafe22016d0b997e4df60600160ff1b038411615ad457926020929160ff608095604051948552168484015260408301526060820152600092839182805260015afa15615ac85780516001600160a01b03811615615abf57918190565b50809160019190565b604051903d90823e3d90fd5b50505060009160039190565b6004811015615b4f5780615af2575050565b60018103615b0c5760405163f645eedf60e01b8152600490fd5b60028103615b2d5760405163fce698f760e01b815260048101839052602490fd5b600314615b375750565b602490604051906335e2f38360e21b82526004820152fd5b634e487b7160e01b600052602160045260246000fd5b60609061018081018051516000915b818310615c745750505080519160018060a01b039182602082015116926040820151906060830151908060808501511660a08501519160c0860151169060e086015192610100918288015195610120890151976101606101408b01519a01519a602081519101209b60606040519e8f907fa87df46e2d2684eb0bbc7abfb05483167cdccac6d7302078a9eaad540c11995860208301526040820152015260808d015260a08c015260c08b015260e08a01528801526101208701526101408601526101608501526101808401526101a08301526101c0908183015281526101e081018181106001600160401b03821117613630576040526020815191012090565b909193615c82858351613be5565b5160018060a01b0381511660208092015182815191012091604090815193818501937f4def3e04bd42194484d5f8a5b268ec0df03b9d9d0402606fe3100023c5d79ac485528386015260608501526060845260808401938085106001600160401b0386111761363057600195615d21948685528251902060a091615d0e81518092878688019101613703565b83019182015203908101845201826136e2565b94019190615b74565b60609060e08082018051516000915b818310615df15750505060018060a01b0390818351169360208401519383604082015116936060820151169260808201519060a08301519160c084015194602081519101209561010080950151976040519960208b019b7f47dba7e6940f0063b21c2ef8f7b0beaf1a2f4c2f84144c36b274ceec12e99b578d5260408c015260608b015260808a015260a089015260c088015286015284015261012083015261014090818301528152615deb816136c6565b51902090565b909194600190615e8b615e05888551613be5565b5180516020908183015160409384810151888060a01b03606083015116608092838101519160a0808301519460c080940151968b51988b8a019a7f6ac594952a72f2e6b24efaf9744b05c23b1b92ce25aa97d18a4338f484c41b958c528d8b015260608a01528801528601528401528b830152610100908183015281526149a581613674565b95019190615d39565b6060608082018051516000915b818310615f4f5750505081519160208101519160018060a01b0360408301511691606081015191602081519101209060a081015160c08201519260e08301519461010080940151966040519860208a019a7f986a160abc209a64a5b0786817ff0aa7a5f5737a4ee6a95197f86290598cd03d8c5260408b015260608a0152608089015260a088015260c087015260e086015284015261012083015261014090818301528152615deb816136c6565b909192600190615f81615f63868551613be5565b5191604090815180946020926149c081518092868087019101613703565b93019190615ea156fe05ff99c6f7e97b6a8a71cfc68f38acaad22981d6e3b79164c1c3c569cff19d7f00068db8bac710cb295e9e1b089a027525460aa64c2f837b4a2339c0ebedfa43a26469706673582212201b566ddca7c19c6233fca6e856d14023e942c3f3c85e55eade5f3fe3f7a4dce464736f6c63430008150033
Deployed Bytecode Sourcemap
1519:45198:30:-:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;844:2:26;1519:45198:30;;;;;;;;;;;:::i;:::-;;504:34:11;522:1;1519:45198:30;;512:11:11;504:34;:::i;:::-;558:1;1519:45198:30;;18278:12;1519:45198;;;;:::i;:::-;18278:12;:::i;:::-;1519:45198;;;;;;18294:6;1519:45198;;;;;;18278:31;18274:94;;1519:45198;18467:13;;;522:1:11;18467:13:30;;;;:::i;:::-;:25;;;:91;;;;1519:45198;3455:15:26;1519:45198:30;3603:14:26;;;;1519:45198:30;3585:15:26;;;1519:45198:30;;3658:15:26;3641:32;;3637:101;;-1:-1:-1;3747:889:26;;;-1:-1:-1;3856:13:26;-1:-1:-1;;;;;3783:26:26;1519:45198:30;3783:26:26;;;:::i;:::-;1519:45198:30;3856:13:26;;;:::i;:::-;1519:45198:30;;;;3856:23:26;;;;:::i;:::-;3776:145;;;;;;1519:45198:30;;;;;3776:145:26;1519:45198:30;;;;;;;;;;;;;3776:145:26;;3881:26;1519:45198:30;3849:4:26;;1519:45198:30;3776:145:26;;;:::i;:::-;;;;;;;;;;;3747:889;1519:45198:30;;522:1:11;1519:45198:30;3940:23:26;1519:45198:30;;;;;;3940:23:26;3978:17;3747:889;18578:63:30;;3747:889:26;1519:45198:30;;;;;;;;;;;;;;;:::i;18578:63::-;1519:45198;;18294:6;1519:45198;;;;;;;18578:63;;;3776:145:26;;;;;:::i;:::-;1519:45198:30;;3776:145:26;;;1519:45198:30;;;;3776:145:26;1519:45198:30;;;;;;;;;3776:145:26;1519:45198:30;;;;-1:-1:-1;;;1519:45198:30;;;;;;;;3747:889:26;4047:15;1519:45198:30;-1:-1:-1;;;;;1519:45198:30;;;;;;;;;;;;;;-1:-1:-1;1519:45198:30;;4083:26:26;;;1519:45198:30;4083:26:26;;;:::i;:::-;1519:45198:30;4151:26:26;;;;1519:45198:30;4076:102:26;;;;;;1519:45198:30;;4076:102:26;1519:45198:30;;;;;;;;;;;;;;4076:102:26;;4132:4;1519:45198:30;4076:102:26;;;:::i;:::-;;;;;;;;;;;;;;3747:889;4290:26;;;4378:22;4290:26;;:::i;:::-;4378:22;;;;:::i;:::-;4463:21;1519:45198:30;4047:15:26;1519:45198:30;;4463:21:26;;1519:45198:30;1291:5:29;;1744:340:10;;;;844:2:26;1744:340:10;;;;;;;1519:45198:30;;-1:-1:-1;;;4206:358:26;;1519:45198:30;4206:358:26;;1519:45198:30;;;;;;884:5:26;;;1519:45198:30;884:5:26;;;1519:45198:30;;;;;;884:5:26;;;1519:45198:30;884:5:26;;;;844:2;1744:340:10;;884:5:26;;;1519:45198:30;4540:10:26;884:5;;;1519:45198:30;4206:358:26;1519:45198:30;884:5:26;1519:45198:30;4206:358:26;;;;;;;;;;;;;3747:889;4192:372;4584:41;1519:45198:30;522:1:11;4192:372:26;;1519:45198:30;;;;;;884:5:26;;;1519:45198:30;4584:41:26;3747:889;;4206:358;;;;;;;;;;;;:::i;:::-;;;884:5;;;;;;;;;;;-1:-1:-1;;;;;884:5:26;;;;;;;;;;;;;;;;;;;:::i;:::-;1519:45198:30;;;;;;;:::i;:::-;884:5:26;;;;;;;;;;;;;4584:41;884:5;1519:45198:30;884:5:26;;;522:1:11;884:5:26;;;;;;;:::i;:::-;4206:358;;;;;;;884:5;1519:45198:30;;;1744:340:10;;;;4076:102:26;;;;:::i;:::-;1519:45198:30;;4076:102:26;;;;1519:45198:30;;;;4076:102:26;1519:45198:30;;;;;;;;;4076:102:26;1519:45198:30;;;3637:101:26;1519:45198:30;;;;3696:31:26;;;;;;1519:45198:30;3696:31:26;;1519:45198:30;3696:31:26;18467:91:30;1519:45198;-1:-1:-1;;;;;;18534:13:30;1519:45198;18534:13;;:::i;:::-;1519:45198;;;18534:23;1519:45198;18534:23;1519:45198;18534:23;;;;:::i;:::-;1519:45198;;;;;;;;;;18497:61;;1519:45198;;18497:61;;1519:45198;18497:22;1519:45198;18497:61;;;;;;;;;;;18467:91;18496:62;;18467:91;;;18497:61;;;;;;;;;;;;;;;:::i;:::-;;;;;:::i;:::-;;;;;;;;;;1519:45198;;;;;;;;;18274:94;1519:45198;;;;18881:25;;;;18332;;1519:45198;18332:25;;1519:45198;18332:25;1519:45198;;;;;-1:-1:-1;;1519:45198:30;;;;;;;-1:-1:-1;;;;;1519:45198:30;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;:::i;:::-;;;;504:34:11;1519:45198:30;;;512:11:11;504:34;:::i;:::-;558:1;1519:45198:30;;20793:12;1519:45198;;;;:::i;20793:12::-;1519:45198;;;20809:6;1519:45198;;;;;;20793:31;20789:94;;-1:-1:-1;;;;;1519:45198:30;;;20910:14;;;:::i;:::-;1519:45198;20896:10;:28;20892:86;;1519:45198;21017:22;1519:45198;;;21056:26;;;;1519:45198;21056:26;;;;:::i;:::-;1519:45198;21133:26;;1519:45198;21049:111;;;;;;1519:45198;;;;;;;;21049:111;;21105:4;;;21049:111;21105:4;;;1519:45198;21049:111;;;:::i;:::-;;;;;;;;;;;1519:45198;21228:26;;;;:::i;:::-;21170:138;;;;;1519:45198;;;;;;;;;;;;;;;;;;;;;21170:138;;1519:45198;;21170:138;;1519:45198;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;21170:138;;;;;;;;;;;;;1519:45198;21330:26;;;1519:45198;21330:26;;:::i;:::-;1519:45198;;;;;;;;;;;;21323:70;;1519:45198;21323:70;;1519:45198;21323:70;;;;;;;;;;;1519:45198;21105:4;;1519:45198;;21323:87;21319:146;;21480:44;1519:45198;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;21480:44;;;1519:45198;;;;;21319:146;1519:45198;;-1:-1:-1;;;21433:21:30;;1519:45198;;21433:21;21323:70;;;;;1519:45198;21323:70;;;;;;;;;;:::i;:::-;;;;;:::i;:::-;;;;;;;;;;;1519:45198;;;;;;;;;21170:138;;;;;;:::i;:::-;1519:45198;;21170:138;;;;;;1519:45198;;;;;;;;;;;21049:111;;;;:::i;:::-;1519:45198;;21049:111;;;;;1519:45198;;;;;;;;;20892:86;1519:45198;;-1:-1:-1;;;20947:20:30;;1519:45198;;20947:20;20789:94;1519:45198;;-1:-1:-1;;;20847:25:30;;1519:45198;20847:25;;1519:45198;;;;;20847:25;1519:45198;;;;;;;;;;;-1:-1:-1;;1519:45198:30;;;;;;:::i;:::-;778:5:7;1519:45198:30;;-1:-1:-1;;;;;1519:45198:30;756:44:7;764:10;1519:45198:30;;;764:19:7;756:44;:::i;:::-;1519:45198:30;-1:-1:-1;;;;;;1519:45198:30;;;;;;;;764:10:7;1424:42;;;;1519:45198:30;;;;;;;;;;;;;;;;;2558:60;-1:-1:-1;;;;;1519:45198:30;;;;;;;;;;;;;-1:-1:-1;;1519:45198:30;;;;5867:30:29;;1519:45198:30;;5792:10:29;1519:45198:30;;5812:16:29;1519:45198:30;;;;;;;;;;;;;;;;;;;;;;;;5792:10:29;;;;5867:30;;:::i;:::-;;;;1519:45198:30;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;:::i;:::-;;;;;;;;:::i;:::-;;;;;;;;:::i;:::-;;;;;;;;:::i;:::-;;522:1:11;1519:45198:30;504:34:11;1519:45198:30;;;512:11:11;504:34;:::i;:::-;558:1;1519:45198:30;;;;-1:-1:-1;;;;;1519:45198:30;;;;14896:10;:28;14892:86;;15215:28;15004:26;;;1519:45198;15065:5;;;;;:::i;:::-;15127;;;;:::i;:::-;1519:45198;15159:26;;;;;;:::i;:::-;15187;1519:45198;;;;:::i;:::-;15187:26;:::i;:::-;15215:28;;:::i;:::-;15259:32;;;;;;;;:::i;:::-;:44;;;;;;:107;;;1519:45198;15255:185;;;1519:45198;15454:13;;;;;1519:45198;15478:14;15454:38;15450:98;;1519:45198;7475:21:29;1519:45198:30;;;;7475:21:29;1519:45198:30;;;:::i;:::-;;43763:35;;;1519:45198;43825:21;15259:32;43825:21;;43801:46;1519:45198;;43801:46;:::i;:::-;43763:84;;43759:143;;43937:13;;1519:45198;;;;;;;;;44012:51;;;:::i;:::-;44114:13;;44129:19;;;;;;1519:45198;;-1:-1:-1;;;;;;;;;;;1519:45198:30;;;;;;44275:294;1519:45198;;;16298:134;1519:45198;;;44449:26;;;:::i;:::-;1519:45198;;;;;;;:::i;:::-;;;15004:26;44303:266;;1519:45198;44303:266;1519:45198;44303:266;;1519:45198;;44303:266;;;1519:45198;44303:266;15259:32;44303:266;;1519:45198;44504:15;43763:35;44303:266;;1519:45198;44533:26;;;1519:45198;44533:26;44303:266;;1519:45198;44275:294;;;;:::i;:::-;;;;:::i;:::-;;44579:24;1519:45198;;;;;;15722:22;;;:::i;:::-;1519:45198;;;15702:6;15004:26;1519:45198;;;;;;;;;;;;;15792:22;44303:266;15792:22;;1519:45198;15890:61;1519:45198;;;;15846:26;;;:::i;:::-;1519:45198;;;;15928:23;15890:61;44303:266;15928:23;;1519:45198;15890:61;;;:::i;:::-;;;:::i;:::-;15975:28;15971:312;;44109:156;1519:45198;;;;;;;;;;;16298:134;;:::i;:::-;;;;1519:45198;;;;;;;;;;15004:26;1519:45198;;;;;;;;:::i;15971:312::-;16195:63;16228:17;16111:26;1519:45198;16195:63;1519:45198;;;16111:26;;:::i;:::-;1519:45198;;;;16228:17;;1519:45198;16195:63;;:::i;:::-;15971:312;;;;;;44114:13;44179;:16;:13;;;;;:16;:::i;:::-;;44165:30;;;;:::i;:::-;;;;;;:::i;:::-;;1519:45198;44114:13;;1519:45198;-1:-1:-1;;;1519:45198:30;;;;;;;;43759:143;1519:45198;;-1:-1:-1;;;43870:21:30;;1519:45198;;43870:21;15450:98;1519:45198;;-1:-1:-1;;;15515:22:30;;1519:45198;;15515:22;15255:185;1519:45198;;-1:-1:-1;;;15397:32:30;;1519:45198;;15397:32;15259:107;15307:32;;;;;:::i;:::-;1519:45198;;;;15346:13;;;1519:45198;;;15307:59;;15259:107;;;1519:45198;-1:-1:-1;;;1519:45198:30;;;;;;;;;;;;;;;;;;;;;431:27:33;1519:45198:30;;;-1:-1:-1;;;;;1519:45198:30;;;;;;;;;;;;;;;;;;;;;;;5657:18:29;-1:-1:-1;;;;;1519:45198:30;;;;;;;;;;;;;-1:-1:-1;;1519:45198:30;;;;21732:51;1519:45198;;;:::i;:::-;;;-1:-1:-1;;;;;1519:45198:30;;756:44:7;;1519:45198:30;;764:10:7;:19;756:44;:::i;:::-;1519:45198:30;;;;;;;;21668:48;1519:45198;;;21668:48;1519:45198;;;;;;21732:51;1519:45198;;;;;;;;;-1:-1:-1;;1519:45198:30;;;;1264:43:28;1519:45198:30;;;:::i;:::-;;;-1:-1:-1;;;;;1519:45198:30;;756:44:7;;1519:45198:30;;764:10:7;:19;756:44;:::i;:::-;1166:22:28;;;:::i;:::-;1519:45198:30;;;;1200:30:28;1519:45198:30;;;;;;;;-1:-1:-1;;1519:45198:30;778:5:7;1519:45198:30;;;;;;;1264:43:28;1519:45198:30;;;;;;;;;;;;;;;;20201:14;1519:45198;;;;;;;;;;;;;;-1:-1:-1;;1519:45198:30;;;;;;;;;;-1:-1:-1;;;;;1519:45198:30;;:::i;:::-;;;;1803:30:28;1519:45198:30;;;;;;;;;;;;;;;;;;-1:-1:-1;;1519:45198:30;;;;;;;;;;;;-1:-1:-1;;;;;1519:45198:30;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;19197:11;1519:45198;;;;:::i;19197:11::-;1519:45198;;;;;;19212:6;1519:45198;;;;;;19197:30;19193:93;;-1:-1:-1;;;;;1519:45198:30;;;;19313:13;;;:::i;:::-;1519:45198;19299:10;:27;19295:85;;19467:25;1519:45198;19467:25;;;;1519:45198;19467:25;1519:45198;19467:25;;;:::i;:::-;1519:45198;;;;;;;;;;;19389:157;;1519:45198;19389:157;;1519:45198;19389:157;;1519:45198;;;;;;19494:25;1519:45198;;;;;;;;;;;;;;;19407:19;1519:45198;19389:157;;;;;;;;1519:45198;;;19562:46;1519:45198;;;;;;;;;;;;;;;;;;;;;19562:46;1519:45198;;19389:157;;;;;;;;;;;;;;;;;;:::i;:::-;;;1519:45198;;;;19389:157;;;;;;1519:45198;;;;19389:157;;;;;19193:93;1519:45198;;-1:-1:-1;;;19250:25:30;;1519:45198;19250:25;;1519:45198;;;;;19250:25;1519:45198;;;;;;;;;;;;;;464:31:33;1519:45198:30;;;;;;;;;;;;;;;;;;;;;;;728:6:26;1519:45198:30;;;;;;;;;;;;;;;;;;;386:38:33;1519:45198:30;;;;;;;;;;-1:-1:-1;;1519:45198:30;;;;;;;;;;;;;;756:44:7;1519:45198:30;;;;;778:5:7;1519:45198:30;;764:10:7;:19;756:44;:::i;:::-;728:6:26;2891:35;;:74;;;;1519:45198:30;2887:134:26;;1519:45198:30;;3088:47:26;1519:45198:30;;;3030:42:26;1519:45198:30;;;3030:42:26;1519:45198:30;;;;;;3088:47:26;1519:45198:30;;2887:134:26;1519:45198:30;;-1:-1:-1;;;2988:22:26;;1519:45198:30;;2988:22:26;2891:74;2930:35;786:6;2930:35;;2891:74;;1519:45198:30;;;;;;;-1:-1:-1;;1519:45198:30;;;;;;;;-1:-1:-1;;;;;1519:45198:30;;:::i;:::-;;;;6940:5:29;1519:45198:30;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;;;;;1519:45198:30;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;:::i;:::-;;;;;;;:::i;:::-;-1:-1:-1;;1519:45198:30;;;;;;;;;466:13:27;;;;481:14;;;;;;1519:45198:30;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;466:13:27;1519:45198:30;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;605:35:27;613:4;;605:35;;1519:45198:30;;;;;;;;:::i;:::-;;;;;;;;:::i;:::-;;;;;;;;;;581:59:27;;;;:::i;:::-;;658:8;654:84;;1519:45198:30;;466:13:27;;654:84;712:10;1519:45198:30;712:10:27;;;;;:::i;:::-;;1519:45198:30;;;693:30:27;;;;;;;;1519:45198:30;693:30:27;;1519:45198:30;;;;;;;;;:::i;:::-;693:30:27;;;1519:45198:30;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;;1519:45198:30;;;;;;;;;;;20567:6;1519:45198;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;:::i;:::-;;;;;-1:-1:-1;;1519:45198:30;;;;;;;;-1:-1:-1;;;;;1519:45198:30;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;504:34:11;1519:45198:30;;;512:11:11;504:34;:::i;:::-;558:1;1519:45198:30;;-1:-1:-1;;;;;1519:45198:30;;12878:14;;;:::i;:::-;1519:45198;12864:10;:28;12860:86;;12955:31;1519:45198;;;;:::i;:::-;12955:31;;:::i;:::-;13036:32;;;;:::i;:::-;13178:14;;;;:::i;:::-;13261:45;;;;;:::i;:::-;13360:28;;;13442:8;13360:28;;;;:::i;:::-;13442:8;;;;:::i;:::-;13496:20;;;;:::i;:::-;13532:22;;;;;:::i;:::-;1519:45198;;;;;13532:42;;;;1519:45198;13532:96;;1519:45198;13528:157;;;1519:45198;13728:21;;1519:45198;;;13698:26;1519:45198;13698:51;;;13694:115;;14390:25;14176:31;1519:45198;14176:31;14431:73;14176:31;1519:45198;14176:31;;1519:45198;14176:31;;14109:271;14176:31;;;:::i;:::-;14320:22;;1519:45198;14109:271;;:::i;:::-;14390:25;;;;;;:::i;:::-;1519:45198;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;:::i;:::-;;14176:31;1519:45198;;;14431:73;;;14535:11;;;:::i;:::-;1519:45198;;;14515:6;1519:45198;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;13694:115::-;1519:45198;;-1:-1:-1;;;13772:26:30;;1519:45198;;13772:26;13528:157;1519:45198;;-1:-1:-1;;;13651:23:30;;1519:45198;;13651:23;13532:96;13578:26;;;1519:45198;13578:26;;;:::i;:::-;1519:45198;;;;;13578:50;;13532:96;;;1519:45198;;;;;;;;;;;;:::i;:::-;;;;:::i;:::-;;504:34:11;522:1;1519:45198:30;;512:11:11;504:34;:::i;:::-;558:1;1519:45198:30;;4974:27;;;;;;;:::i;:::-;5050:32;;;;;:::i;:::-;5151:45;;5508:271;5151:45;;;:::i;:::-;5250:28;;;;5332:8;5250:28;;;;:::i;5332:8::-;5386:20;;;;:::i;:::-;5575:31;;;;;:::i;:::-;5684:21;;;1519:45198;;;5719:22;;1519:45198;;;;5508:271;;:::i;:::-;5807:26;;;;;;;;;;;;;;;:::i;:::-;37785:25;;5790:163;;1519:45198;-1:-1:-1;;;;;;;1519:45198:30;;5963:89;;;;;1519:45198;;;5963:89;1519:45198;;;;;;;;;;;;;5963:89;;6023:4;5963:89;1519:45198;5963:89;;;:::i;:::-;;;;;;;;;;;1519:45198;6080:11;;6106:45;1519:45198;522:1:11;6080:11:30;;;;1519:45198;6080:11;;:::i;:::-;1519:45198;;;6063:6;1519:45198;;;;;;;;;;;;;;;5575:31;1519:45198;;;;5575:31;1519:45198;;;;:::i;:::-;;;;;;;;;;:::i;:::-;;;;;;6106:45;;;1519:45198;;;;;;;;;;;;;;;;;:::i;5963:89::-;;;;;:::i;:::-;1519:45198;;5963:89;;;5790:163;1519:45198;5905:26;;;;:::i;:::-;1519:45198;;;:::i;:::-;5893:10;1519:45198;;2288:30:28;1519:45198:30;;;;;;;;;;2287:49:28;1519:45198:30;;;2287:220:28;;5790:163:30;2270:316:28;;;;;5790:163:30;;;;2270:316:28;1519:45198:30;;-1:-1:-1;;;2539:36:28;;1519:45198:30;;2539:36:28;2287:220;1519:45198:30;;;;;;;;;;;;;;2356:82:28;;;;1519:45198:30;;2356:82:28;;1519:45198:30;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;:::i;:::-;2356:82:28;5893:10:30;;;2356:82:28;;;;;;;;;;;2287:220;-1:-1:-1;;;;;;;1519:45198:30;2356:151:28;;2287:220;;;;;2356:82;;;;;;;;;;;;;;;:::i;:::-;;;;;:::i;:::-;;;;;;;;;1519:45198:30;;;;;;;;;;;;;;;2340:44;-1:-1:-1;;;;;1519:45198:30;;;;;;;;;;;;;-1:-1:-1;;1519:45198:30;;;;6562:53:29;;1519:45198:30;;6464:10:29;1519:45198:30;;6484:29:29;1519:45198:30;;;;;;;;;;;;;;;;;;;;;;;;6464:10:29;;;;6562:53;;:::i;1519:45198:30:-;;;;;;;;;;;;;;;;427:7:34;1519:45198:30;;;;;;;;;;-1:-1:-1;;1519:45198:30;;;;;;;;;;-1:-1:-1;;;;;1519:45198:30;;:::i;:::-;;;;2177:112:29;1519:45198:30;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;;;;;1519:45198:30;;;;;;;;;;;;;;;;:::i;:::-;2190:15:26;1519:45198:30;;;-1:-1:-1;;;;;1519:45198:30;2168:10:26;:38;;;2164:121;;1519:45198:30;;18827:12;1519:45198;;;;:::i;18827:12::-;1519:45198;;;18843:6;1519:45198;;;;;;18827:31;18823:94;;18932:23;1519:45198;;;;;;18932:23;1519:45198;;18843:6;1519:45198;;;;;;;;;2164:121:26;1519:45198:30;;;;2229:45:26;;;;;;1519:45198:30;2229:45:26;;1519:45198:30;2229:45:26;1519:45198:30;;;;;;;;;;;;;1152:32:33;:16;1519:45198:30;1171:13:33;1152:32;;:::i;:::-;1187:15;-1:-1:-1;1148:102:33;;1275:12;1519:45198:30;-1:-1:-1;;;;;1519:45198:30;;1128:10:33;1263:24;1259:81;;1349:16;1519:45198:30;;1128:10:33;-1:-1:-1;;;;;;1519:45198:30;;;;;;;;;;;;1275:12:33;1519:45198:30;-1:-1:-1;;1152:16:33;1519:45198:30;;;;;1462:37:33;1519:45198:30;;1462:37:33;1519:45198:30;;1259:81:33;1519:45198:30;;-1:-1:-1;;;1310:19:33;;1519:45198:30;;1310:19:33;1148:102;1519:45198:30;;-1:-1:-1;;;1225:14:33;;1519:45198:30;;1225:14:33;1519:45198:30;;;;;;;;;;;;;;1610:43:29;1519:45198:30;;;;;;;;;;;;;;;;;;;;2431:15:26;1519:45198:30;;;;;;;;-1:-1:-1;;;;;1519:45198:30;;;;;;;;;;;;;-1:-1:-1;;1519:45198:30;;;;;;6028:10:29;1519:45198:30;;6076:10:29;1519:45198:30;;;;;;6106:32:29;;;;6102:123;;6028:10;6280:37;;6028:10;;1519:45198:30;;6076:10:29;1519:45198:30;;;;;;;;;6028:10:29;;;;6280:37;;:::i;6102:123::-;1519:45198:30;;;;;6161:53:29;;;;;;6028:10;1519:45198:30;6161:53:29;;1519:45198:30;;;;;;;;;6161:53:29;1519:45198:30;;;;;;;;;;;;;;;3286:27:26;1519:45198:30;;;;;;;;;;;;;;;;;;;;;2469:26:34;1519:45198:30;427:7:34;1519:45198:30;;;;;;;2451:15:34;:64;2447:116;;2676:31;;1519:45198:30;;:::i;:::-;;;;;;-1:-1:-1;;;;;;1519:45198:30;-1:-1:-1;;;;;1519:45198:30;;;;;;;;;;;;;;;;;;;;;;2676:31:34;:::i;1519:45198:30:-;-1:-1:-1;;;1519:45198:30;;;;;;;;;;;;;;;-1:-1:-1;;1519:45198:30;;;;20393:37;1519:45198;;;756:44:7;1519:45198:30;;;;;;;;764:10:7;:19;756:44;:::i;:::-;1519:45198:30;20345:32;1519:45198;;;;;;20393:37;1519:45198;;;;;;;;;-1:-1:-1;;1519:45198:30;;;;2689:44:26;1519:45198:30;;;:::i;:::-;;;-1:-1:-1;;;;;1519:45198:30;756:44:7;;1519:45198:30;;764:10:7;:19;756:44;:::i;:::-;2598:29:26;;;:::i;:::-;2639:34;1519:45198:30;;-1:-1:-1;;;;;;;;1519:45198:30;;;;;-1:-1:-1;;;;;;;1519:45198:30;;;;;;;;;;2689:44:26;1519:45198:30;;;;;;;;;;;;;;;;;;1078:49:29;1519:45198:30;;;;;;;;;;-1:-1:-1;;1519:45198:30;;;;;;;;;;-1:-1:-1;;;;;1519:45198:30;;:::i;:::-;;;;1851:91:29;1519:45198:30;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;:::i;:::-;;504:34:11;522:1;1519:45198:30;;512:11:11;504:34;:::i;:::-;558:1;1519:45198:30;;6506:5;6478:26;;;;1519:45198;6506:5;:::i;:::-;6568;;;;:::i;:::-;6589:32;;;;;;:::i;:::-;6632:13;;1519:45198;6632:13;;;1519:45198;6589:63;6585:133;;-1:-1:-1;;;;;6765:26:30;1519:45198;6765:26;;;:::i;:::-;1519:45198;6751:10;:40;6862:23;6824:61;6862:23;;;1519:45198;6824:35;;;1519:45198;6824:61;:::i;:::-;6895:28;7043:1698;;;;7141:71;;;1519:45198;22395:27;1519:45198;22522:17;22489:63;22522:17;;;1519:45198;6862:23;;;1519:45198;22489:63;:::i;:::-;22702:18;1519:45198;;;;22755:59;;;;;;;;-1:-1:-1;;22755:59:30;23032:13;;23051;1519:45198;6632:13;;23051;1519:45198;;;;522:1:11;1519:45198:30;23047:29;;;;;1519:45198;;;;23122:39;1519:45198;;;23122:39;;:::i;:::-;;1519:45198;6632:13;;23208;1519:45198;6862:23;23265:14;;1519:45198;23122:39;;6632:13;522:1:11;;-1:-1:-1;;;;;1519:45198:30;;;;23283:26;1519:45198;6765:26;;23283;:::i;:::-;1519:45198;23265:44;1519:45198;;;;23352:32;23348:46;;23551:326;23591:26;1519:45198;6765:26;;23591;:::i;:::-;1519:45198;;6862:23;23675:22;;1519:45198;;23748:15;;1519:45198;23766:14;;;;1519:45198;22522:17;;;1519:45198;23766:14;;1519:45198;;;;;;23748:15;;-1:-1:-1;;;;;1519:45198:30;;;;;;;23551:326;:::i;:::-;23494:383;;;;;23936:14;;;;;1519:45198;30196:132;;;23032:13;30179:224;;;1519:45198;24040:23;;1519:45198;23936:14;;1519:45198;;;;;;;23032:13;;;;;;;30196:132;23952:26;;30235:30;23952:26;;;1519:45198;30235:30;;:::i;:::-;1291:5:29;;-1:-1:-1;;;;;;;;;;;1744:340:10;;;;;;;;;;30234:93:30;;1744:340:10;;;30234:93:30;30196:132;;;23348:46;23386:8;;;;;;1519:45198;;;;;;;;;;;;23047:29;;;;;;;;;;;;;24238:37;;;:77;:37;;;:77;;;6862:23;24435:22;;1519:45198;-1:-1:-1;;;;;1519:45198:30;6765:26;24459;1519:45198;6765:26;;24459;:::i;:::-;1519:45198;;-1:-1:-1;;;;;1519:45198:30;;29420:9;29416:111;;24238:77;-1:-1:-1;;;;6862:23:30;24435:22;;1519:45198;;;;;-1:-1:-1;;;;;1519:45198:30;;;;;;;;;;;;29420:9;;;29416:111;;24238:77;7078:134;;;;;7244:15;6824:35;7244:15;;1519:45198;23766:14;7261;;1519:45198;31142:50;31165:14;1519:45198;31142:50;;:::i;:::-;1519:45198;;;31091:15;:101;7226:109;;6589:32;7439:21;;1519:45198;31091:15;7478:46;31091:15;22522:17;7478:28;;1519:45198;7478:46;:::i;:::-;7636:43;7542:32;23766:14;7261;;1519:45198;6824:35;7244:15;;1519:45198;7542:32;;:::i;:::-;7439:21;6589:32;7439:21;;1519:45198;7636:43;;:::i;:::-;22702:18;1519:45198;6824:35;46242:44;6824:35;;;;1519:45198;46242:44;:::i;:::-;:49;;46241:331;;;;24238:77;46223:383;;;;;;;24238:77;46223:417;;;;24238:77;46206:503;;;;6589:32;7439:21;;1519:45198;6824:35;;;;1519:45198;7752:59;7748:327;;24238:77;7043:1698;;522:1:11;7475:21:29;1519:45198:30;;;;7475:21:29;1519:45198:30;;;;;;;:::i;:::-;522:1:11;1519:45198:30;;;;6478:26;1519:45198;;;;6765:26;;-1:-1:-1;;;;;;;;;;;6765:26:30;9433:144;6765:26;8981;1519:45198;522:1:11;6765:26:30;;;8981;:::i;:::-;1519:45198;;;;;;;:::i;:::-;;;;8873:247;6478:26;8873:247;;1519:45198;6824:35;;;1519:45198;;8873:247;;1519:45198;;;;;;;6862:23;8873:247;;1519:45198;6589:32;8873:247;;1519:45198;9055:15;6824:35;8873:247;;1519:45198;;9084:26;;1519:45198;;8873:247;;1519:45198;8857:263;;;:::i;:::-;;;;;:::i;:::-;;1519:45198;6632:13;;9130:26;9055:15;6824:35;9166:15;;1519:45198;9226:28;;;1519:45198;;9209:14;;1519:45198;6824:35;;;1519:45198;6589:32;9264:21;;1519:45198;9354:12;;;:::i;:::-;1519:45198;;;9334:6;6478:26;1519:45198;;;;;6478:26;;;1519:45198;;;;;;;;;;6862:23;;;;;;;;1519:45198;6478:26;;;;1519:45198;;;9433:144;;:::i;1519:45198::-;6478:26;1519:45198;;;;;;:::i;:::-;;;;;;;;;;;;;;;6862:23;1519:45198;;;;6589:32;1519:45198;;;;6824:35;1519:45198;;;;;;;;;;;;;;;;;7748:327;6862:23;24435:22;;1519:45198;7983:59;;-1:-1:-1;;;;;1519:45198:30;;;;7899:26;1519:45198;6765:26;;7899;:::i;:::-;1519:45198;;-1:-1:-1;;;;;1519:45198:30;;7983:59;;1519:45198;6824:35;;1519:45198;7983:59;:::i;:::-;7748:327;;;46206:503;1519:45198;;-1:-1:-1;;;46672:26:30;;1519:45198;;46672:26;46223:417;46611:28;;-1:-1:-1;46223:417:30;;;;:383;6862:23;;;1519:45198;46591:14;;;-1:-1:-1;46223:383:30;;46241:331;46343:34;;;;;:73;:34;;46470;46343;;:::i;:::-;46380:36;6824:35;;;1519:45198;23766:14;7592:26;;1519:45198;46380:36;:::i;:::-;46343:73;;:::i;:::-;46470:34;;:::i;:::-;1291:5:29;;1744:340:10;-1:-1:-1;;;;;;;;;;;1744:340:10;;;;;;;;;;;;46342:208:30;46241:331;;;;;;7226:109;1519:45198;;-1:-1:-1;;;7303:17:30;;1519:45198;;7303:17;29416:111;29511:4;1519:45198;;29511:4;;:::i;:::-;29416:111;;;;;;;;29511:4;;;:::i;:::-;29416:111;;;;;;24238:77;;;22755:59;;;;;;;7043:1698;1519:45198;;;;-1:-1:-1;1519:45198:30;;;;;-1:-1:-1;;;;;1519:45198:30;6751:10;8095:28;1519:45198;;;;-1:-1:-1;;;8146:20:30;;1519:45198;;8146:20;8091:650;1519:45198;8273:82;;;;22395:27;1519:45198;22489:63;22522:17;;;1519:45198;6862:23;;;1519:45198;22489:63;:::i;:::-;1519:45198;;;23032:13;;;23051;1519:45198;6632:13;;23051;1519:45198;;;;522:1:11;1519:45198:30;23047:29;;;;;1519:45198;;;;23122:39;1519:45198;;;23122:39;;:::i;:::-;;1519:45198;6632:13;;23208;1519:45198;6862:23;23265:14;;1519:45198;23122:39;;6632:13;522:1:11;;-1:-1:-1;;;;;1519:45198:30;;;;23283:26;1519:45198;6765:26;;23283;:::i;:::-;1519:45198;23265:44;1519:45198;;;;23352:32;23348:46;;23551:326;23591:26;1519:45198;6765:26;;23591;:::i;:::-;1519:45198;;;;;;;;;;;;;;;;;6862:23;23675:22;;1519:45198;;6824:35;23748:15;;1519:45198;23766:14;22522:17;23766:14;;;;;1519:45198;;22522:17;;1519:45198;23551:326;;:::i;:::-;23494:383;;;;;24040:23;1519:45198;24040:23;;1519:45198;23936:14;;1519:45198;;;;;;;23032:13;;;;;;;23348:46;23386:8;;;;;;23047:29;;;;;;;8491:28;23047:29;;;;;;;;24238:37;;;:77;:37;;;:77;;6862:23;24435:22;;1519:45198;-1:-1:-1;;;;;1519:45198:30;6765:26;24459;1519:45198;6765:26;;24459;:::i;:::-;1519:45198;;-1:-1:-1;;;;;1519:45198:30;;29420:9;29416:111;;24238:77;-1:-1:-1;;;;6862:23:30;24435:22;;1519:45198;;;;;-1:-1:-1;;;;;1519:45198:30;;;;;;;;;29420:9;;;;29416:111;;24238:77;6765:26;;;;;8435;1519:45198;6765:26;;8435;:::i;:::-;8463;1519:45198;;;;:::i;8491:28::-;8538:16;;;8534:159;;24238:77;8706:24;;;8091:650;7043:1698;;8534:159;6862:23;24435:22;;1519:45198;8665:12;;-1:-1:-1;;;;;1519:45198:30;;;;8621:26;1519:45198;6765:26;;8621;:::i;:::-;1519:45198;;;;;;;;;;;8665:12;;:::i;:::-;8534:159;;;;29416:111;29511:4;1519:45198;;29511:4;;:::i;:::-;29416:111;;;;;;;;29511:4;;;:::i;:::-;29416:111;;;;;;24238:77;;;;;1519:45198;;;;;;;;;;;;;;;5487:16:29;-1:-1:-1;;;;;1519:45198:30;;;;;;;;;;;;;;;;;;;2442:37;1519:45198;;;-1:-1:-1;;;;;1519:45198:30;;;;;;;;;;;;;;;-1:-1:-1;;1519:45198:30;;;;2266:45:34;1519:45198:30;;;;;;756:44:7;1519:45198:30;778:5:7;1519:45198:30;;764:10:7;:19;756:44;:::i;:::-;2108:38:34;:25;;:::i;:::-;:38;:::i;:::-;1519:45198:30;;:::i;:::-;;;;;;;;;2159:37:34;1519:45198:30;;;2159:37:34;1519:45198:30;;;;;;2235:15:34;2206:44;1519:45198:30;;;;;;;;;;2266:45:34;1519:45198:30;;;;;;;;;-1:-1:-1;;1519:45198:30;;;;1582:45:28;1519:45198:30;;;:::i;:::-;;;-1:-1:-1;;;;;1519:45198:30;;756:44:7;;1519:45198:30;;764:10:7;:19;756:44;:::i;:::-;1519:45198:30;;;;1517:30:28;1519:45198:30;;;;;;;;-1:-1:-1;;1519:45198:30;;;;;;;1582:45:28;1519:45198:30;;;;;;;;;;;;;;;;1860:26:34;1519:45198:30;;;;;;;;;;;;;;;;;;;;;;;2260:39;1519:45198;;;;;;;;;;;;;;;;;;;786:6:26;1519:45198:30;;;;;;;;;;-1:-1:-1;;1519:45198:30;;;;5294:44:29;1519:45198:30;;;756:44:7;1519:45198:30;;;;;;;;764:10:7;:19;756:44;:::i;:::-;1519:45198:30;5246:32:29;1519:45198:30;;;;;;5294:44:29;1519:45198:30;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;-1:-1:-1;;1519:45198:30;;;;948:33:33;1519:45198:30;;;:::i;:::-;;;-1:-1:-1;;;;;1519:45198:30;;756:44:7;;1519:45198:30;;764:10:7;:19;756:44;:::i;:::-;1519:45198:30;;;;;;;;864:24:33;1519:45198:30;;;864:24:33;1519:45198:30;917:15:33;898:34;1519:45198:30;;;;;;948:33:33;1519:45198:30;;;;;;;;;-1:-1:-1;;1519:45198:30;;;;;;;;-1:-1:-1;;;;;1519:45198:30;;:::i;:::-;;;;2023:61:29;1519:45198:30;;;;;;;;;;;;;;;-1:-1:-1;;1519:45198:30;;;;;;;-1:-1:-1;;;;;1519:45198:30;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;:::i;:::-;;504:34:11;522:1;1519:45198:30;;512:11:11;504:34;:::i;:::-;558:1;1519:45198:30;;-1:-1:-1;;;;;9866:26:30;1519:45198;;9866:26;;;:::i;:::-;1519:45198;9852:10;:40;9848:98;;9974:15;;;1519:45198;9991:14;;;1519:45198;31142:50;31165:14;1519:45198;31142:50;;:::i;:::-;1519:45198;;;31091:15;:101;9956;;10070:32;;1519:45198;;10070:32;1519:45198;;;;10070:32;:::i;:::-;:44;;;10066:114;;10267:5;1519:45198;;;;10207:26;1519:45198;10267:5;:::i;:::-;10329;1519:45198;;;;;10329:5;:::i;:::-;522:1:11;7475:21:29;1519:45198:30;;;;7475:21:29;1519:45198:30;10397:24;10431:27;10468:23;1519:45198;10559:18;1519:45198;10692:32;10070;1519:45198;;10070:32;1519:45198;;;;10692:32;:::i;:::-;10746:9;;;;10741:1318;10757:29;;;;;;1519:45198;;;;12073:35;1519:45198;;12073:35;1519:45198;12073:54;12069:124;;11317:22;;;1519:45198;522:1:11;;-1:-1:-1;;;;;1519:45198:30;;;;12264:26;9866;1519:45198;;9866:26;12264;:::i;:::-;1519:45198;;;;;;;;29420:9;;29416:111;;10741:1318;12373:12;;;;;;;:::i;:::-;1519:45198;;;12353:6;26278:53;1519:45198;;;;;;;;10207:26;1519:45198;;;;;;;;-1:-1:-1;;;;;;;;;;;1519:45198:30;;;;;;;;;;;;10207:26;1519:45198;26278:53;1519:45198;;;;;;;;9974:15;11317:22;1519:45198;;;;;9974:15;1519:45198;;;;:::i;:::-;;26229:24;1519:45198;;;12476:80;;;1519:45198;;;;;;;;;;26278:53;1519:45198;;;;;;;;:::i;29416:111::-;29511:4;;;:::i;:::-;29416:111;;;;;;10746:9;10819:32;10070;1519:45198;;;;;;;10070:32;1519:45198;;;;10819:32;:::i;:::-;1519:45198;;;;;;10881:13;;;1519:45198;;;;;;;10872:29;;10868:107;;11013:20;1519:45198;;;;;;;10881:13;;11013;:20;:::i;:::-;;11075:14;9991;11075;;1519:45198;30235:30;11091:26;1519:45198;;11091:26;1519:45198;30235:30;;:::i;:::-;1291:5:29;;-1:-1:-1;;;;;;;;;;;1744:340:10;;;;;;;;;;;;;;30234:93:30;30179:224;;11241:26;9866;1519:45198;;9866:26;11241;:::i;:::-;1519:45198;;;;;;;;;;;;;;;11317:22;;;1519:45198;;9974:15;;;30732:62;9974:15;11432:17;11382:32;9974:15;;;1519:45198;9991:14;;;1519:45198;11382:32;;:::i;:::-;11432:17;;1519:45198;25801:18;;1519:45198;30752:42;1519:45198;;;31165:14;1519:45198;;;30752:42;:::i;:::-;30732:62;;:::i;:::-;31091:15;25844:28;;25840:108;;26015:24;;;;;;;1439:69:32;1519:45198:30;26015:24;;1519:45198;1456:19:32;9991:14:30;11075;;1519:45198;9974:15;25801:18;;1519:45198;31091:15;1519:45198;1456:19:32;;:::i;:::-;1439:69;;:::i;:::-;26229:24:30;26138:58;;;;:::i;:::-;26229:24;;1519:45198;;26210:43;;11317:22;26315:15;;1519:45198;;;-1:-1:-1;;;26278:53:30;;-1:-1:-1;;;;;1519:45198:30;;;;26278:53;;1519:45198;;26278:53;;1519:45198;;;;;;26278:22;1519:45198;26278:53;;;;;;;;;;;;;;;10746:9;26274:365;;;10746:9;26649:21;;;1519:45198;;;;;26015:24;;1519:45198;;;26848:256;;;10746:9;522:1:11;27117:17:30;;;27113:404;;10746:9;26015:24;;;1519:45198;26015:24;;1519:45198;;;;11075:14;9991;11075;;1519:45198;;;;;;;;;;;;;;11802:26;9866;1519:45198;;9866:26;11802;:::i;:::-;1519:45198;11317:22;26315:15;;1519:45198;;;26229:24;;;1519:45198;31091:15;9974;25801:18;;1519:45198;9991:14;11091:26;1519:45198;;11091:26;1519:45198;11075:14;;1519:45198;;10746:9;;;;;27113:404;1519:45198;11317:22;1519:45198;;;;;26315:15;;1519:45198;;;;;;;;;;;27154:26;27150:128;;27113:404;;;;27150:128;27249:13;;;:::i;:::-;27150:128;;;;;;26848:256;11317:22;26315:15;;1519:45198;-1:-1:-1;;;;;1519:45198:30;;;;;;;;;-1:-1:-1;1519:45198:30;;522:1:11;1519:45198:30;;27008:35;1519:45198;;27008:35;;;:::i;:::-;-1:-1:-1;;11467:17:30;26848:256;;1519:45198;;;;;;;;;;;;26274:365;1519:45198;;;;;11317:22;26315:15;;1519:45198;;;;;26015:24;;1519:45198;11075:14;9991;11075;;1519:45198;26229:24;9974:15;26229:24;;;1519:45198;25801:18;;1519:45198;26347:281;;;;;;1519:45198;26347:281;1519:45198;;;;;;;;;;;;;;;;;;;;26347:281;;1519:45198;26347:281;;;:::i;:::-;;;;;;;;;;;26274:365;;;;;26347:281;;;;:::i;:::-;;;;26278:53;;;;;;;;;;;;;;:::i;:::-;;;;25840:108;1519:45198;;;;25895:42;;;;;;1519:45198;25895:42;;1519:45198;25895:42;1519:45198;;;;;;;;;;;;;;;;;;;-1:-1:-1;;1519:45198:30;;;;;;:::i;:::-;;;:::i;:::-;;;-1:-1:-1;;;19780:37:30;;1519:45198;;;19780:37;;1519:45198;;;-1:-1:-1;;;;;1519:45198:30;;;;;;;;;;;;;;;19780:37;;;;;;;;;;;1519:45198;-1:-1:-1;1519:45198:30;;19829:4;19780:54;19776:112;;1519:45198;;;;;;;;;;;;;;;19898:103;;1519:45198;19898:103;;1519:45198;19898:103;;1519:45198;;;;;;;;;;;;;;;;;;;;;19916:19;1519:45198;19898:103;;;;;;;;1519:45198;;;20017:57;1519:45198;;;;;;;;;;;;;;;;;;;;;20017:57;1519:45198;;19898:103;;;;;;;;;;;;;;;;;;:::i;:::-;;;1519:45198;;;;19898:103;;;;;;;;;;;19776:112;1519:45198;;-1:-1:-1;;;19857:20:30;;1519:45198;;19857:20;19780:37;;;;;;;;;;;;;;:::i;:::-;;;;1519:45198;;;;;;;-1:-1:-1;;1519:45198:30;;;;;;:::i;:::-;;;;:::i;:::-;;;;-1:-1:-1;;;;;1519:45198:30;;;;;;;;;;;:::i;:::-;-1:-1:-1;;1519:45198:30;;-1:-1:-1;;;1519:45198:30;;;;;;;;;;;;;;;;;;;4928:18:29;1519:45198:30;;;;;;;;;;;;;;;:::i;:::-;504:34:11;522:1;1519:45198:30;;512:11:11;504:34;:::i;:::-;558:1;1519:45198:30;;16648:19;;;;:::i;:::-;1519:45198;16705:19;;;;;1519:45198;;;;;;;;;;;;;;;;-1:-1:-1;;;;;1519:45198:30;16884:13;1519:45198;16884:13;:::i;:::-;1519:45198;16870:10;:27;16866:150;;1519:45198;17026:29;1519:45198;;;;:::i;:::-;17026:29;;:::i;:::-;1519:45198;17199:19;;;;:::i;:::-;:34;1519:45198;;;;;;;;17195:236;;1519:45198;17448:25;1519:45198;17448:25;;;:::i;:::-;1519:45198;17503:13;;;:::i;:::-;17441:103;;;;;1519:45198;;17441:103;1519:45198;;;;;;;;;;;;17441:103;;17518:25;;;1519:45198;17496:4;;1519:45198;17441:103;;;:::i;:::-;;;;;;;;;;;1519:45198;17702:19;:32;:19;;;;:::i;:::-;:32;;;;;:::i;:::-;37785:25;;17685:159;;1519:45198;17907:24;17947:52;1519:45198;17907:24;18049:6;17907:24;;:::i;:::-;1519:45198;;;;;;;;;;;;;;17947:52;1519:45198;;;;;;;;522:1:11;1519:45198:30;;;;17685:159;17800:19;;:32;:19;;:::i;:::-;:32;;;;;:::i;:::-;1519:45198;;;;;;:::i;:::-;16870:10;;1519:45198;;2972:30:28;1519:45198:30;;;;;;;;2971:49:28;1519:45198:30;;;2971:202:28;;17685:159:30;2954:298:28;;;;;;17685:159:30;;;2971:202:28;1519:45198:30;;;;;;;;;;;;;;;3040:70:28;;;;1519:45198:30;;3040:70:28;;1519:45198:30;;;;;;:::i;:::-;;;;-1:-1:-1;;1519:45198:30;;;;;;;:::i;:::-;3040:70:28;16870:10:30;;;3040:70:28;;;;;;;;;;;2971:202;-1:-1:-1;;;;;;;1519:45198:30;3040:133:28;;2971:202;;;;;;3040:70;;;;;;;;;;;;;;:::i;:::-;;;;;1519:45198:30;;;;;;;;;17441:103;;;;;;;:::i;:::-;;;;;17195:236;17320:13;;;:::i;:::-;17335:25;;1519:45198;17335:25;;;:::i;:::-;1519:45198;;-1:-1:-1;;;17249:171:30;;1519:45198;;;;17249:171;;1519:45198;;;;;;17362:25;;;1519:45198;;;;;;;;;;;522:1:11;1519:45198:30;;;;17362:25;1519:45198;;;;17267:19;1519:45198;;17249:171;;;;;;;;;;;17195:236;;;;;17249:171;;;;;;;;;;;;:::i;:::-;;;1519:45198;;;;17249:171;;;;;;;;;16866:150;16929:13;;;:::i;:::-;16944:19;;;;:::i;:::-;1519:45198;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;-1:-1:-1;;;;;1519:45198:30;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;16972:32;1519:45198;;;;;;;;;;;;6127:38:31;1519:45198:30;;;6013:213:31;;;;1519:45198:30;2312:66:31;1519:45198:30;;;2312:66:31;;1519:45198:30;;2312:66:31;;1519:45198:30;2312:66:31;;;1519:45198:30;2312:66:31;6013:213;;;;;:::i;:::-;1519:45198:30;5990:246:31;;16972:32:30;;1519:45198;16972:32;;;;:::i;:::-;;;;;:::i;:::-;16866:150;;1519:45198;;;;;;;;;;;;;1474:18:29;1519:45198:30;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;:::i;:::-;;;;;1474:18:29;1519:45198:30;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;1519:45198:30;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::o;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;;;;;1519:45198:30;;;;;;;:::o;:::-;;;;;;;;;;;;;-1:-1:-1;;;;;1519:45198:30;;;;;;:::o;:::-;;;;;;;-1:-1:-1;;;;;1519:45198:30;;;;;;;:::o;:::-;;;;;;;-1:-1:-1;;;;;1519:45198:30;;;;;;;:::o;:::-;;;;;;;-1:-1:-1;;;;;1519:45198:30;;;;;;;:::o;:::-;2312:66:31;1519:45198:30;;;;;-1:-1:-1;;;;;1519:45198:30;;;;;;;:::o;:::-;2847:66:31;1519:45198:30;;;;;-1:-1:-1;;;;;1519:45198:30;;;;;;;:::o;:::-;;;;;;;;;;;;;-1:-1:-1;;;;;1519:45198:30;;;;;;;:::o;:::-;;;;;;;;-1:-1:-1;;1519:45198:30;;;;:::o;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;-1:-1:-1;;1519:45198:30;;;;:::o;:::-;-1:-1:-1;;1519:45198:30;;;;;;;;;;;-1:-1:-1;;;;;1519:45198:30;;;;;;;;;;;;;;;:::o;:::-;;;;-1:-1:-1;;;;;1519:45198:30;;;;;;:::o;:::-;;;;-1:-1:-1;;;;;1519:45198:30;;;;;;:::o;:::-;;;;-1:-1:-1;;;;;1519:45198:30;;;;;;:::o;:::-;;;-1:-1:-1;;;;;1519:45198:30;;;;;;:::o;:::-;;;;;;;;;;;;;-1:-1:-1;;;;;1519:45198:30;;;;;;;;;;;;;;;:::o;:::-;-1:-1:-1;;;;;1519:45198:30;;;;;;;;;:::o;:::-;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;:::o;:::-;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;;;;;1519:45198:30;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::o;:::-;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::o;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;;1519:45198:30;;;;;;;;;;-1:-1:-1;;;;;1519:45198:30;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;:::i;:::-;;;:::o;:::-;;-1:-1:-1;;1519:45198:30;;;;;;;;;;;;;-1:-1:-1;;;;;1519:45198:30;;;;;;;;;;;;;;;:::o;:::-;;;-1:-1:-1;;;;;1519:45198:30;;;;;;;;;;;;;;;;;;:::o;:::-;-1:-1:-1;;;;;1519:45198:30;;;;;;-1:-1:-1;;1519:45198:30;;;;:::o;:::-;;;;;;;:::i;:::-;;;;-1:-1:-1;;;1519:45198:30;;;;:::o;:::-;;;;:::o;:::-;;;-1:-1:-1;;;1519:45198:30;;;;;;;;;;;;-1:-1:-1;;;1519:45198:30;;;;;;;205:148:25;-1:-1:-1;;;;;1519:45198:30;273:22:25;269:78;;205:148::o;269:78::-;1519:45198:30;;-1:-1:-1;;;318:18:25;;;;;1519:45198:30;;;;;;;;;;;;;;;;;;;-1:-1:-1;;;;;1519:45198:30;;;;;;;;;;;;;;:::o;:::-;;;;;;;;;:::o;:::-;;;;;;;;;;;;;;;:::o;:::-;-1:-1:-1;;;;;1519:45198:30;;;;;;;;;;;;;;;:::o;7051:170:29:-;-1:-1:-1;7126:13:29;7143:16;7126:33;7143:16;;7162:24;;7051:170;:::o;7126:88::-;1519:45198:30;;7874:4:29;1519:45198:30;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;7858:22:29;;1519:45198:30;;;:::i;:::-;;;;;;7898:18:29;1519:45198:30;;7717:275:29;;;1519:45198:30;7745:95:29;1519:45198:30;;;;;;;;;;7126:13:29;1519:45198:30;;;;7973:4:29;1519:45198:30;;;;;7717:275:29;;1519:45198:30;;;;;;;-1:-1:-1;;;;;1519:45198:30;;;;;;;;;7694:308:29;;7051:170;:::o;1519:45198:30:-;-1:-1:-1;;;1519:45198:30;;;;;;;;;;;;7874:4:29;1519:45198:30;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;1519:45198:30;;;;;;;;;;;-1:-1:-1;;1519:45198:30;;;;;;;;;;;;;-1:-1:-1;1519:45198:30;;-1:-1:-1;1519:45198:30;;-1:-1:-1;1519:45198:30;;;;;;;;:::i;:::-;;;-1:-1:-1;1519:45198:30;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::o;:::-;;;;:::o;:::-;;;-1:-1:-1;;;1519:45198:30;;;;;;;;;;;;-1:-1:-1;;;1519:45198:30;;;;;;;;;;-1:-1:-1;;;;;1519:45198:30;;;;;;;:::o;:::-;;-1:-1:-1;;;;;1519:45198:30;;;;;;;:::o;:::-;;;;;;;;;;;;;;;;;:::o;:::-;;;;;;;;;;;;;;;;;;;-1:-1:-1;;;;;1519:45198:30;;;;;;;;;;;;;;;;:::o;:::-;;;;;;;:::i;:::-;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;-1:-1:-1;1519:45198:30;;;;;;:::o;:::-;-1:-1:-1;;;;;1519:45198:30;;;;;;;;;;;;;;;;;;;;;:::o;:::-;;;;;;;;;;;;;;;-1:-1:-1;1519:45198:30;;;;;;;;;;;:::o;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;;;;;;1519:45198:30;;;;;;;:::o;:::-;;;;;;;;;;:::o;:::-;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;-1:-1:-1;;;;;1519:45198:30;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::o;:::-;;;;;;;;;;;;;;;;;;;;;;;:::o;:::-;;;;;;;:::o;:::-;;;;;:::i;:::-;;;;;;;;;:::i;:::-;;;;;;;;;;;:::i;:::-;;;-1:-1:-1;1519:45198:30;;;;;;;;;;:::o;:::-;;;;;;;;:::i;:::-;-1:-1:-1;1519:45198:30;;;-1:-1:-1;1519:45198:30;;;;-1:-1:-1;1519:45198:30;;;;-1:-1:-1;1519:45198:30;;;;-1:-1:-1;1519:45198:30;;;;-1:-1:-1;1519:45198:30;;;;-1:-1:-1;1519:45198:30;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;:::o;1328:1782:12:-;;-1:-1:-1;1328:1782:12;;1532:1521;1328:1782;1532:1521;1328:1782;1532:1521;;1519:45198:30;;;;1532:1521:12;;1519:45198:30;;;;;1532:1521:12;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;1532:1521:12;;;;;1519:45198:30;;;1328:1782:12:o;1519:45198:30:-;1532:1521:12;1519:45198:30;-1:-1:-1;;;1519:45198:30;;1532:1521:12;;1519:45198:30;;;;1532:1521:12;1519:45198:30;;;-1:-1:-1;;;1532:1521:12;1519:45198:30;;;1532:1521:12;;1519:45198:30;2096:672:10;;1291:5:29;11467:17:30;;;;2274:488:10;;;;;;;;;;;;;;;;;;;;;2096:672;:::o;:::-;;2200:1:30;11467:17;;;;2274:488:10;;;;;;;;;;;;;;;;;;;;;2096:672;:::o;1519:45198:30:-;;;;;;;;;;;;;;;;;:::o;:::-;;;;;;;;;;;;;;;;;;:::i;:::-;;:::o;:::-;;;;;;;;;;;;;-1:-1:-1;1519:45198:30;;;;;;;;-1:-1:-1;;1519:45198:30;;;;:::o;:::-;;;;;;;;;;;;;;;;;;;-1:-1:-1;;;;;1519:45198:30;;;;;;;;;;;;;;;;:::o;:::-;;;;;;;;;;;;;;;;;;:::o;:::-;;;;;;;;;;-1:-1:-1;;;;;1519:45198:30;;;;;;;:::o;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::o;25403:2170::-;;;;;;;;;25801:18;;;1519:45198;30732:62;1519:45198;;30752:42;-1:-1:-1;1519:45198:30;;30767:14;1519:45198;;;30752:42;:::i;30732:62::-;25857:15;25844:28;;25840:108;;26015:24;;;;1519:45198;1439:69:32;1519:45198:30;;1456:19:32;26052:15:30;;;1519:45198;;;25857:15;1519:45198;1456:19:32;;:::i;1439:69::-;26138:58:30;;;;;:::i;:::-;26229:24;;;1519:45198;26315:15;;;1519:45198;26015:24;1519:45198;-1:-1:-1;;;26278:53:30;;-1:-1:-1;;;;;1519:45198:30;;;26278:53;;;1519:45198;;;;;;26229:24;;26278:53;;1519:45198;;;;;;26278:22;1519:45198;26278:53;;;;;;;;;;;;;;;;;;;25403:2170;26274:365;;;25403:2170;26649:21;;1519:45198;;;;;;;;;;;26852:35;;;;26848:256;;25403:2170;27117:17;;27113:404;;25403:2170;27526:40;;;;;25403:2170;:::o;27113:404::-;26315:15;;1519:45198;;;-1:-1:-1;;;;;1519:45198:30;;;;;;27154:26;;;27150:128;;27113:404;1519:45198;;;;;27113:404;;;;;;27150:128;27249:13;;;:::i;:::-;27150:128;;;;;;26848:256;26315:15;;;1519:45198;;;27008:35;;-1:-1:-1;;;;;1519:45198:30;;27008:35;;1519:45198;;27008:35;:::i;:::-;;;;:::i;:::-;27058;26848:256;;;;26274:365;1519:45198;;;;;;;;;;;26315:15;;;1519:45198;;;;;;26052:15;;;;1519:45198;26229:24;;;;1519:45198;;;26347:281;;;;;;1519:45198;26347:281;1519:45198;;;;;;;;26015:24;1519:45198;;;;;;;;;;;26347:281;;26278:53;26347:281;;;:::i;:::-;;;;;;;;;;;;;;;;;26274:365;;;;26347:281;;;;:::i;:::-;;;;26278:53;;;;;;;;;;;;;;:::i;:::-;;;;27757:300;27845:12;;;:::i;:::-;1519:45198;-1:-1:-1;1519:45198:30;27861:6;1519:45198;;;-1:-1:-1;1519:45198:30;;27845:31;27841:94;;27948:15;;27966:14;27948:15;:32;:15;;1519:45198;27966:14;;1519:45198;27948:32;;:::i;:::-;27984:15;-1:-1:-1;27948:51:30;27944:107;;27757:300::o;27944:107::-;1519:45198;;-1:-1:-1;;;28022:18:30;;;;;28174:800;;28337:35;;;1519:45198;28337:40;28336:126;;;;;28174:800;28319:218;;;28568:34;;;1519:45198;28550:15;;:52;28546:143;;1519:45198;28786:26;;1519:45198;;28786:26;;;:::i;:::-;1519:45198;;;;;;;-1:-1:-1;1519:45198:30;28826:29;1519:45198;;28786:26;-1:-1:-1;1519:45198:30;;-1:-1:-1;1519:45198:30;;;;28786:26;-1:-1:-1;1519:45198:30;;;28822:146;;28174:800;;:::o;28822:146::-;28903:54;28786:26;1519:45198;28903:54;;;;;;;;;;;;:::i;28546:143::-;1519:45198;;;;28625:53;;;;;;;;;1519:45198;28625:53;28336:126;28399:13;;;;;1519:45198;28422:32;;;;;;:::i;:::-;28399:62;;;;28336:126;;;1519:45198;;;;;;;;;;;;;;;;:::i;:::-;;;;;;:::i;:::-;;;;;;;;-1:-1:-1;;;;;1519:45198:30;;;;;;;;;;:::i;:::-;;;;:::o;:::-;;;;;;;;;;;;;;;-1:-1:-1;;;;;1519:45198:30;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::o;:::-;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;33511:585;;-1:-1:-1;;;;;1519:45198:30;;;33632:10;:23;33628:156;;33511:585;33815:28;;:43;:28;;;;:::i;:::-;:43;1519:45198;33797:15;:61;33793:161;;33967:28;;:43;:28;;:::i;:::-;:43;;:::i;:::-;34020:14;;;-1:-1:-1;33963:127:30;;33511:585::o;33793:161::-;33815:43;33899:28;;1519:45198;33899:28;;:::i;:::-;1519:45198;;-1:-1:-1;;;33881:62:30;;33899:43;;1519:45198;33881:62;;;1519:45198;33881:62;33628:156;33698:28;;;;:::i;:::-;1519:45198;;;;;;;;;;;;;;;;;-1:-1:-1;;;;;1519:45198:30;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;5072:34:31;;;5146:29;;;1519:45198:30;5197:13:31;-1:-1:-1;5192:265:31;5212:23;;;;;;;1519:45198:30;;;;;;;;;;5562:32:31;1519:45198:30;;;;;;;;;5800:27:31;;1519:45198:30;;;;;5790:38:31;1519:45198:30;;;5496:346:31;;;1519:45198:30;2076:66:31;1519:45198:30;;2076:66:31;;;1519:45198:30;2076:66:31;;1519:45198:30;2076:66:31;;1519:45198:30;;2076:66:31;;1519:45198:30;;2076:66:31;;1519:45198:30;2076:66:31;;;1519:45198:30;2076:66:31;5496:346;;2076:66;1519:45198:30;;;;;;;;;;;;33735:37;1519:45198;33735:37;1519:45198;;;;5473:379:31;;33735:37:30;;;;;:::i;:::-;33628:156;;;;5197:13:31;5352:29;5292:94;5352:29;;;;:32;:29;;;1519:45198:30;5352:29:31;;:32;:::i;:::-;;9597:21;9592:27;9597:21;;9592:27;:::i;:::-;9637:22;;;;;1519:45198:30;9687:36:31;;;1519:45198:30;;;;;9677:47:31;1519:45198:30;;;9525:213:31;;;;1519:45198:30;1377:66:31;1519:45198:30;;1377:66:31;;;1519:45198:30;1377:66:31;;1519:45198:30;1377:66:31;;;1519:45198:30;9525:213:31;;;;;;:::i;:::-;1519:45198:30;9502:246:31;;1519:45198:30;;;;;;;;;5292:94:31;;;;;;1519:45198:30;;;:::i;:::-;;;;;;;;5292:94:31;;;;;;;;;:::i;:::-;1519:45198:30;;5197:13:31;;;;;;;1519:45198:30;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::o;34355:292::-;34537:29;34355:292;34537:29;;:::i;:::-;1519:45198;;;;;34537:38;1519:45198;;:::i;:::-;34537:38;;:::i;:::-;34593:20;34615:24;;34593:20;;;;;:::i;:::-;34615:24;;;:::i;34835:364::-;1519:45198;;;-1:-1:-1;;;34947:49:30;;;-1:-1:-1;;;;;1519:45198:30;;;34947:49;;;1519:45198;;;34947:49;;1519:45198;;;34947:49;1519:45198;;;34947:16;1519:45198;;34947:49;;;;;;;-1:-1:-1;34947:49:30;;;34835:364;34946:50;;34942:117;;1519:45198;;;;;;;35073:55;;;;;;;1519:45198;34947:49;35073:55;;1519:45198;35073:18;1519:45198;35073:55;;;;;;;-1:-1:-1;35073:55:30;;;34835:364;35072:56;;;35068:125;;34835:364;:::o;35068:125::-;1519:45198;-1:-1:-1;;;35151:31:30;;34947:49;;35151:31;35073:55;;;;;;-1:-1:-1;35073:55:30;;;;;;:::i;:::-;;;;;;1519:45198;;;-1:-1:-1;1519:45198:30;;;;;34942:117;1519:45198;;-1:-1:-1;;;35019:29:30;;34947:49;;35019:29;34947:49;;;;;;;;;;;;;;:::i;:::-;;;;;1519:45198;;;-1:-1:-1;1519:45198:30;;;;;37823:1777;1519:45198;;37823:1777;1519:45198;;;-1:-1:-1;;;;;38065:21:30;;;;;:::i;:::-;1519:45198;38121:12;;;;;;:::i;:::-;38155:9;;-1:-1:-1;38166:17:30;;;;;;37939:16;;;;;1519:45198;39418:125;;38150:1258;39552:41;;;;37823:1777;:::o;39418:125::-;39476:13;39515:16;39476:13;;:::i;:::-;38953:52;1519:45198;-1:-1:-1;;;;;1519:45198:30;;39515:16;:::i;:::-;39418:125;;;;;38155:9;38121:12;;;;38225;38121;;;38225;;:::i;:::-;1519:45198;;;;;;;;;38121:12;1519:45198;;;;:::i;:::-;38292:23;1439:69:32;38292:23:30;;;1519:45198;1456:19:32;38328:14:30;;;1519:45198;38344:35;38362:17;;;1519:45198;38344:15;:35;:::i;:::-;1456:19:32;;:::i;1439:69::-;1519:45198:30;37939:16;;;;1519:45198;38435:225;;38155:9;38292:23;38693:63;:81;38292:23;38693:49;38292:23;;;1519:45198;38719:23;;;1519:45198;38693:49;;:::i;:::-;:63;:::i;:::-;:81;:::i;:::-;38811:13;38842:9;38811:13;;;;:::i;:::-;38065:21;38826:14;;1519:45198;-1:-1:-1;;;;;1519:45198:30;;38842:9;;:::i;:::-;38065:21;38826:14;;1519:45198;38292:23;1519:45198;-1:-1:-1;;;38953:52:30;;-1:-1:-1;;;;;1519:45198:30;;;38953:52;;;1519:45198;;;;;;;38953:52;;1519:45198;;;;;;38953:22;1519:45198;38953:52;;;;;;;1519:45198;38953:52;;;38155:9;38949:390;;;38155:9;1519:45198;;;;38155:9;;;;;38949:390;1519:45198;;;;;38065:21;38826:14;;1519:45198;;;;38292:23;;;;1519:45198;38328:14;;;;1519:45198;38719:23;38362:17;38719:23;;;1519:45198;38362:17;;1519:45198;39025:299;;;;;1519:45198;39025:299;1519:45198;;;;;37939:16;1519:45198;38292:23;1519:45198;;;;;;;;;;;39025:299;;37939:16;1519:45198;39025:299;38953:52;39025:299;;;:::i;:::-;;;;;;;;;;1519:45198;39025:299;;;38949:390;;;;39025:299;;;;:::i;:::-;;;;;38292:23;1519:45198;;;;;;;;38953:52;;;;;;;;;;;;;;:::i;:::-;;;;38435:225;37939:16;38693:81;37939:16;;38693:63;38492:50;37939:16;;;1519:45198;38492:50;;:::i;:::-;1519:45198;;;38435:225;;;;;;;1519:45198;;;;;;;:::i;:::-;2605:19:34;1519:45198:30;-1:-1:-1;;;;;1519:45198:30;;;;;;;;;:::o;:::-;;;;;;;:::i;:::-;;;-1:-1:-1;;;;;1519:45198:30;;;;;;;;;:::o;40290:3225::-;;;;;;;;;;;;;;;;;;;1519:45198;;:::i;:::-;;40681:37;;;:::i;:::-;;;1519:45198;;;;:::i;:::-;;;;;;;;:::i;:::-;;;;;;;;:::i;:::-;;;;;;;;;;;40803:19;;1519:45198;7475:21:29;1519:45198:30;;;;;;7475:21:29;1519:45198:30;;;:::i;:::-;40881:45;;1519:45198;;-1:-1:-1;;41040:37:30;;1519:45198;;;;41161:15;;;;;;43080:36;;;;;;;;;:::i;:::-;43068:48;;;43064:107;;1519:45198;41533:20;;;1519:45198;;;;;;;;:::i;:::-;;;-1:-1:-1;;;;;1519:45198:30;;;;;;;;43199:257;;1519:45198;;;;;;43199:257;;1519:45198;;;;33186:14;43199:257;;1519:45198;;;42000:78;43199:257;;1519:45198;42048:15;1519:45198;43199:257;;1519:45198;;;35747:23;43199:257;;1519:45198;;;32847:21;43199:257;;1519:45198;33102:12;43199:257;;1519:45198;43467:41;;;;;;;;1519:45198;43467:41;40290:3225::o;41146:13::-;1519:45198;;;;;;;;:::i;:::-;41278:24;;;;;:::i;:::-;41419:12;;1519:45198;41419:12;;;:::i;:::-;41476:39;1519:45198;41476:39;;;;:::i;:::-;;;;1519:45198;41533:20;;;1519:45198;32040:21;;;;:::i;:::-;32088:12;;1519:45198;32088:12;;;:::i;:::-;32187:37;;;;1519:45198;32187:22;;1519:45198;32187:37;:::i;:::-;;;32239:18;;:23;;:71;;;41146:13;32235:280;;;-1:-1:-1;1519:45198:30;;;32371:27;;;1519:45198;;;;;-1:-1:-1;;;;;1519:45198:30;;;;;-1:-1:-1;;1519:45198:30;;;;;;32371:27;33186:14;1519:45198;;33186:14;1519:45198;;;;;;;42000:78;1519:45198;;;;;:::i;:::-;;32187:22;;1519:45198;;;;;;;41476:39;;32187:22;1519:45198;:::i;:::-;;;;-1:-1:-1;;1519:45198:30;42000:78;1519:45198;;;;;:::i;:::-;32371:27;1519:45198;;32371:27;;;;;;:::i;:::-;32326:87;;;;;1519:45198;;;;;;;;;;;;;;;;;32326:87;;1519:45198;;40914:12;32326:87;;1519:45198;;;;;;;;;;:::i;:::-;;;;;;32326:87;;;;;;;;;;32235:280;;;32547:20;;;1519:45198;42048:15;;32529:38;32525:115;;1519:45198;;;;;;;;;;32654:16;1519:45198;;;;;;;;;;;;;;;;;32654:70;;;;32235:280;32650:155;;32847:21;;;1519:45198;;;32819:49;;;;;32815:155;;32984:14;;;1519:45198;32984:19;;:49;;;;;32235:280;32980:109;;;33102:12;;;1519:45198;33102:17;33098:74;;33186:14;;;1519:45198;33186:19;;;;33185:98;;;;32235:280;33181:162;;;;;33393:8;;;33370:21;;;;:::i;:::-;33393:8;:::i;:::-;41670:18;;1519:45198;32187:22;;1519:45198;41670:18;41666:75;;32235:280;41771:12;1519:45198;41419:12;;41771;:::i;:::-;;;35747:23;;;;;:::i;:::-;:44;;-1:-1:-1;;;;;1519:45198:30;;;;;35747:44;;;;:100;;32235:280;35743:161;;35944:25;;;1519:45198;35917:52;;35913:111;;1519:45198;;;;;:::i;:::-;;;;;;;;42000:78;;1519:45198;;32187:22;;1519:45198;;42000:78;;1519:45198;;;;;;;;;33186:14;42000:78;;1519:45198;;42000:78;;;1519:45198;42048:15;42000:78;;;1519:45198;33102:12;42065;;1519:45198;35747:23;42000:78;;1519:45198;41987:91;;;;;:::i;:::-;;;;;;;:::i;:::-;;1519:45198;32187:22;;1519:45198;42120:21;;1519:45198;42120:21;;1439:69:32;1456:19;;;33102:12:30;42065;;1519:45198;1456:19:32;:::i;:::-;1519:45198:30;32187:22;;1519:45198;1439:69:32;:::i;:::-;1519:45198:30;32187:22;;1519:45198;;;32187:22;1519:45198;32187:22;;1519:45198;11467:17;;2274:488:10;1519:45198:30;42329:9;;1519:45198;2274:488:10;1519:45198:30;32187:22;;1519:45198;2274:488:10;;32847:21:30;42356;;1519:45198;2274:488:10;;;;32187:22:30;42686:12;1519:45198;;32187:22;;1519:45198;;42329:9;;1519:45198;2274:488:10;42356:21:30;32847;42356;;1519:45198;2274:488:10;;32847:21:30;42356;;1519:45198;2274:488:10;;;;;1519:45198:30;35747:23;42481:22;35747:23;;;42481:22;:::i;:::-;41533:20;;1519:45198;41533:20;1519:45198;42513:46;;32847:21;42356;;1519:45198;2274:488:10;;;;;;;;;42513:46:30;:::i;:::-;1519:45198;;;;;42561:21;;1519:45198;;29420:9;29416:111;;32235:280;1519:45198;;;;42686:12;1519:45198;;;;;42617:22;35747:23;;;42617:22;:::i;:::-;1519:45198;;32847:21;42356;;1519:45198;2274:488:10;;;;;;;;1519:45198:30;32187:22;;1519:45198;42686:12;:::i;:::-;;;;;;:::i;:::-;33186:14;42717;;1519:45198;42717:19;33186:14;;1519:45198;;;;;;;;;;42788:5;1519:45198;;;;;;;;;;;;;;;32187:22;;1519:45198;;;;;;42713:231;1519:45198;42958:27;;;;;:::i;:::-;1519:45198;;41146:13;;42713:231;1519:45198;;;;;;;;;;;32654:16;1519:45198;;;;;;;;;;;;;;;;;;;;;;;42713:231;;29416:111;29511:4;;;;1519:45198;;;;;;;29511:4;:::i;:::-;29416:111;;;;;35747:100;35795:27;;42000:78;35795:27;;;:::i;:::-;:52;;-1:-1:-1;;;;;1519:45198:30;;;;;35795:52;;35747:100;;41666:75;1519:45198;32187:22;;1519:45198;41708:18;;41666:75;;33181:162;1519:45198;;-1:-1:-1;;;33306:26:30;;40914:12;;33306:26;33185:98;1519:45198;;;;;;;;;;;;33211:5;1519:45198;;;;;;;;;;;33211:54;1519:45198;32187:22;;1519:45198;;;;;33211:54;:::i;:::-;:71;33185:98;;;;;33098:74;1519:45198;;-1:-1:-1;;;33142:19:30;;40914:12;;33142:19;32984:49;33007:26;;;;;32984:49;;;32815:155;1519:45198;;-1:-1:-1;;;32891:68:30;;1519:45198;;40914:12;32891:68;;1519:45198;;;;;;;;;32891:68;32650:155;32747:47;1519:45198;;28903:54;;;;;;32747:47;;1519:45198;32747:47;40914:12;32747:47;;;:::i;32654:70::-;-1:-1:-1;;;;;;1519:45198:30;;;;;;32704:10;1519:45198;;;;;;;;32693:30;;32654:70;;32326:87;;;;:::i;:::-;;;;32235:280;32482:21;;;;;;1519:45198;32468:12;1519:45198;;;;:::i;:::-;32468:12;:::i;32482:21::-;32235:280;;32239:71;-1:-1:-1;1519:45198:30;;-1:-1:-1;;;32266:44:30;;-1:-1:-1;;;;;1519:45198:30;;;40914:12;32266:44;;1519:45198;;;;;;;;;32266:22;1519:45198;32266:44;;;;;;;1519:45198;32266:44;;;32239:71;;;;32266:44;;;;1519:45198;32266:44;1519:45198;32266:44;;;;;;;:::i;:::-;;;;1519:45198;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;;;;;1519:45198:30;;;;;;;;;;;:::o;:::-;;;;;;;;;;;;;;;;;;:::i;:::-;-1:-1:-1;;;;;1519:45198:30;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;;;;;1519:45198:30;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;1519:45198:30;;;;;;;;;;;;;;;;;;;;;;;;;;;:::o;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;36430:1012;36551:31;;;1519:45198;;36596:36;;;;36652:24;;36648:96;;36430:1012::o;36592:844::-;36800:21;;;;;-1:-1:-1;36800:21:30;;;;:::i;:::-;36846:20;;;;;:37;;;36592:844;36842:234;;;1519:45198;;-1:-1:-1;;;36910:26:30;;;;;36842:234;1519:45198;36962:20;;36961:74;;;36842:234;36957:119;;-1:-1:-1;37109:19:30;;;;;;36592:844;;;;;36430:1012::o;37094:13::-;37197:21;;;;:::i;:::-;1519:45198;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;37316:23;;;;;;-1:-1:-1;;;;;1519:45198:30;37239:101;;;;;-1:-1:-1;1519:45198:30;;;;;;;;;;;;;;;;;;37239:101;;1519:45198;37239:101;;;1519:45198;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;:::i;:::-;37239:101;;;;;;;;;;;1519:45198;37239:101;;;;37094:13;1519:45198;;37094:13;;37239:101;;;;:::i;:::-;;;;;1519:45198;;-1:-1:-1;1519:45198:30;;;;;36957:119;37055:7;;;;:::o;36961:74::-;36987:21;;;;;:::i;:::-;1519:45198;;;;;;;;;;;;;;;;-1:-1:-1;;;;;1519:45198:30;36987:34;;1519:45198;36987:34;:::i;:::-;1519:45198;36987:48;36961:74;;36846:37;36870:13;;;;36846:37;;2200:1;;;;;;;;;;;;;;;;:::o;37502:180::-;37660:14;;-1:-1:-1;;;;;2200:1:30;;;;;;37607:68;2200:1;;;37607:68;;:::i;44857:578::-;;;;;3515:233:4;44988:18:30;;:::i;:::-;3515:233:4;;;;;-1:-1:-1;;;3515:233:4;;;;;;;;;;;45044:19:30;;;:24;;45040:389;45044:19;;;1519:45198;45088:61;1519:45198;;;;;;;;;;;;;;45088:61;;;;;;;1519:45198;;;;;;;;;;;:::i;:::-;45088:61;;-1:-1:-1;;;;;1519:45198:30;45088:61;;;;;;;-1:-1:-1;45088:61:30;;;45040:389;-1:-1:-1;;;;;;;1519:45198:30;45088:80;45084:149;;45040:389;44857:578::o;45084:149::-;1519:45198;-1:-1:-1;;;45195:23:30;;45088:61;;45195:23;45088:61;;;;;;;;;;;;;;:::i;:::-;;;;45040:389;1519:45198;;3915:8:3;1519:45198:30;3859:27:3;1519:45198:30;;;;;:::i;:::-;3859:27:3;;:::i;:::-;3915:8;;;;;:::i;:::-;-1:-1:-1;;;;;1519:45198:30;;;;;45334:20;45330:89;;45040:389;44857:578::o;2129:766:3:-;1519:45198:30;;;2129:766:3;2276:2;2256:22;;2276:2;;2739:25;2539:180;;;;;;;;;;;;;;;-1:-1:-1;2539:180:3;2739:25;;:::i;:::-;2732:32;;;;;:::o;2252:637::-;2795:83;;2811:1;2795:83;2815:35;2795:83;;:::o;5140:1530::-;;;-1:-1:-1;;;;;6186:79:3;;6182:164;;1519:45198:30;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;6457:24:3;;;;;;;;;;;;;;-1:-1:-1;;;;;1519:45198:30;;6495:20:3;6491:113;;6614:49;;5140:1530;:::o;6491:113::-;6531:62;;;6457:24;6531:62;;:::o;6457:24::-;1519:45198:30;;;;;;;;;;6182:164:3;6281:54;;;6297:1;6281:54;6301:30;6281:54;;:::o;7196:532::-;1519:45198:30;;;;;;7282:29:3;;;7327:7;;:::o;7278:444::-;1519:45198:30;7378:38:3;;1519:45198:30;;;;-1:-1:-1;;;7439:23:3;;1519:45198:30;;7439:23:3;7374:348;7492:35;7483:44;;7492:35;;1519:45198:30;;-1:-1:-1;;;7550:46:3;;1519:45198:30;7550:46:3;;1519:45198:30;;;;;7550:46:3;7479:243;7626:30;7617:39;7613:109;;7479:243;7196:532::o;7613:109::-;1519:45198:30;;;;7679:32:3;;;;;;1519:45198:30;7679:32:3;;1519:45198:30;7679:32:3;1519:45198:30;;;;7291:20:3;1519:45198:30;;;;;7291:20:3;1519:45198:30;3877:1079:31;1519:45198:30;4045:21:31;;;;;;1519:45198:30;-1:-1:-1;4083:224:31;4103:19;;;;;;1519:45198:30;;;;;;;;;;;4444:17:31;;9305:20;4444:17;;1519:45198:30;;4479:14:31;1519:45198:30;4479:14:31;;1519:45198:30;4511:19:31;1519:45198:30;4511:19:31;;1519:45198:30;4548:31:31;;351:66;4548:31;;1519:45198:30;;4159:77:31;4597:31;;1519:45198:30;4646:27:31;;;;1519:45198:30;;4691:26:31;;;;1519:45198:30;4735:17:31;;;;;;1519:45198:30;4770:25:31;;;;1519:45198:30;4813:19:31;4850:29;4813:19;;;1519:45198:30;4850:29:31;;1519:45198:30;;9305:20:31;1519:45198:30;;;;4897:28:31;1519:45198:30;;;;4346:593:31;;;825:66;9305:20;4346:593;;1519:45198:30;;825:66:31;;1519:45198:30;825:66:31;1519:45198:30;351:66:31;825;;1519:45198:30;4159:77:31;825:66;;1519:45198:30;4646:27:31;825:66;;1519:45198:30;4691:26:31;825:66;;1519:45198:30;825:66:31;;1519:45198:30;4770:25:31;825:66;;1519:45198:30;4813:19:31;825:66;;1519:45198:30;4850:29:31;825:66;;1519:45198:30;4045:21:31;825:66;;1519:45198:30;825:66:31;;;1519:45198:30;825:66:31;;;;;1519:45198:30;4346:593:31;;825:66;1519:45198:30;;;;;-1:-1:-1;;;;;1519:45198:30;;;;;;;9305:20:31;1519:45198:30;;4346:593:31;;4323:626;3877:1079;:::o;4088:13::-;4210:21;;;:24;:21;;;:24;:::i;:::-;;1519:45198:30;;;;;;;;9305:20:31;;;;;1519:45198:30;;;;;9295:31:31;1519:45198:30;;;;;9245:82:31;;;;1519:45198:30;351:66:31;1519:45198:30;;351:66:31;;;1519:45198:30;;351:66:31;;1519:45198:30;;9245:82:31;;351:66;1519:45198:30;;;;;;-1:-1:-1;;;;;1519:45198:30;;;;;;;4159:77:31;1519:45198:30;;;;;;9235:93:31;;4159:77;1519:45198:30;;;;4159:77:31;;;;;;1519:45198:30;;;:::i;:::-;;;;;;;4159:77:31;;;;;;;;;:::i;:::-;1519:45198:30;;4088:13:31;;;;6249:843;1519:45198:30;6401:13:31;;;;;;;1519:45198:30;;6431:199:31;6447:17;;;;;;1519:45198:30;;;;;;;;;;;;;6770:26:31;8882:14;6770:26;;1519:45198:30;6814:26:31;;8914:24;6814:26;;1519:45198:30;;6858:22:31;1519:45198:30;6858:22:31;;1519:45198:30;;6898:21:31;8989:24;6898:21;;1519:45198:30;6937:15:31;9031:18;6937:15;;1519:45198:30;6970:14:31;9067:15;6970:14;;1519:45198:30;;8882:14:31;1519:45198:30;;;;7002:24:31;3116:66;;7044:17;;;1519:45198:30;;8914:24:31;1519:45198:30;6669:406:31;8882:14;6669:406;;1519:45198:30;2847:66:31;1519:45198:30;;8914:24:31;2847:66;;1519:45198:30;;2847:66:31;;1519:45198:30;8989:24:31;2847:66;;1519:45198:30;9031:18:31;2847:66;;1519:45198:30;9067:15:31;2847:66;;1519:45198:30;2847:66:31;;1519:45198:30;2847:66:31;;1519:45198:30;3116:66:31;2847;;1519:45198:30;2847:66:31;;;;;1519:45198:30;6669:406:31;;;;;:::i;:::-;1519:45198:30;6646:439:31;;6249:843;:::o;6436:9::-;6542:13;;;1519:45198:30;6542:13:31;6497:63;6542:16;:13;;;:16;:::i;:::-;;1519:45198:30;;8882:14:31;;;;;1519:45198:30;8914:24:31;;;;;1519:45198:30;;;;;;;8956:15:31;;1519:45198:30;;8989:24:31;;;;;1519:45198:30;9031:18:31;;;;;1519:45198:30;9067:15:31;;;;;1519:45198:30;;;;8790:306:31;;;;1519:45198:30;3116:66:31;1519:45198:30;;3116:66:31;;;1519:45198:30;;3116:66:31;;1519:45198:30;3116:66:31;;1519:45198:30;3116:66:31;;1519:45198:30;3116:66:31;;1519:45198:30;3116:66:31;;;1519:45198:30;3116:66:31;;;;;1519:45198:30;8790:306:31;;;;;:::i;6497:63::-;1519:45198:30;;6436:9:31;;;;7098:943;1519:45198:30;7274:28:31;;;;;1519:45198:30;-1:-1:-1;7319:205:31;7339:16;;;;;;1519:45198:30;;;;;7689:22:31;7389:65;7689:22;;1519:45198:30;;;;;;;;7729:22:31;;1519:45198:30;;7769:19:31;1519:45198:30;7769:19:31;;1519:45198:30;;7389:65:31;1519:45198:30;;;;7806:25:31;7849:31;;;;1519:45198:30;7898:22:31;;;1519:45198:30;7938:30:31;;;;1519:45198:30;7986:24:31;;;;;1519:45198:30;;;;7563:461:31;7389:65;7563:461;;1519:45198:30;3465:66:31;1519:45198:30;;;3465:66:31;;1519:45198:30;;3465:66:31;;1519:45198:30;7274:28:31;3465:66;;1519:45198:30;7849:31:31;3465:66;;1519:45198:30;7898:22:31;3465:66;;1519:45198:30;7938:30:31;3465:66;;1519:45198:30;3465:66:31;;1519:45198:30;3465:66:31;;;1519:45198:30;3465:66:31;;;;;1519:45198:30;7563:461:31;;;;;:::i;7324:13::-;7422:28;;;1519:45198:30;7422:28:31;7389:65;7422:31;:28;;;:31;:::i;:::-;1519:45198:30;;;;;;7389:65:31;;;1519:45198:30;;;;7389:65:31;;;;;;1519:45198:30;;;:::i;7389:65:31:-;1519:45198:30;;7324:13:31;;;
Swarm Source
ipfs://1b566ddca7c19c6233fca6e856d14023e942c3f3c85e55eade5f3fe3f7a4dce4
Loading...
Loading
Loading...
Loading
Multichain Portfolio | 30 Chains
Chain | Token | Portfolio % | Price | Amount | Value |
---|---|---|---|---|---|
BASE | 100.00% | $0.009129 | 210 | $1.92 |
Loading...
Loading
[ Download: CSV Export ]
A contract address hosts a smart contract, which is a set of code stored on the blockchain that runs when predetermined conditions are met. Learn more about addresses in our Knowledge Base.