More Info
Private Name Tags
ContractCreator
Latest 25 from a total of 30 transactions
Transaction Hash |
Method
|
Block
|
From
|
To
|
|||||
---|---|---|---|---|---|---|---|---|---|
Withdraw All | 18783089 | 477 days ago | IN | 0 ETH | 0.00246011 | ||||
Buyback Repo | 18776022 | 478 days ago | IN | 0 ETH | 0.0028369 | ||||
Sell Repo | 18761611 | 480 days ago | IN | 0 ETH | 0.00453437 | ||||
Buyback Repo | 18761585 | 480 days ago | IN | 0 ETH | 0.0020963 | ||||
Sell Repo | 18747287 | 482 days ago | IN | 0 ETH | 0.00595267 | ||||
Buyback Repo | 18747249 | 482 days ago | IN | 0 ETH | 0.00265049 | ||||
Sell Repo | 18732986 | 484 days ago | IN | 0 ETH | 0.00590844 | ||||
Buyback Repo | 18732861 | 484 days ago | IN | 0 ETH | 0.0037827 | ||||
Sell Repo | 18718540 | 486 days ago | IN | 0 ETH | 0.00796786 | ||||
Buyback Repo | 18717956 | 486 days ago | IN | 0 ETH | 0.0041005 | ||||
Sell Repo | 18703674 | 488 days ago | IN | 0 ETH | 0.00546365 | ||||
Buyback Repo | 18703586 | 488 days ago | IN | 0 ETH | 0.00233736 | ||||
Sell Repo | 18689278 | 490 days ago | IN | 0 ETH | 0.00626046 | ||||
Buyback Repo | 18688963 | 490 days ago | IN | 0 ETH | 0.00348037 | ||||
Sell Repo | 18674606 | 492 days ago | IN | 0 ETH | 0.00591715 | ||||
Buyback Repo | 18674562 | 492 days ago | IN | 0 ETH | 0.00393573 | ||||
Sell Repo | 18657624 | 495 days ago | IN | 0 ETH | 0.00554733 | ||||
Buyback Repo | 18654198 | 495 days ago | IN | 0 ETH | 0.00189992 | ||||
Sell Repo | 18639834 | 497 days ago | IN | 0 ETH | 0.00358438 | ||||
Edit Repo Time L... | 18639813 | 497 days ago | IN | 0 ETH | 0.00064595 | ||||
Edit Repo Buybac... | 18639688 | 497 days ago | IN | 0 ETH | 0.00057156 | ||||
Edit Repo Sell P... | 18639683 | 497 days ago | IN | 0 ETH | 0.00053901 | ||||
Edit Repo Time L... | 18639659 | 497 days ago | IN | 0 ETH | 0.00061071 | ||||
Buyback Repo | 18618057 | 500 days ago | IN | 0 ETH | 0.00273315 | ||||
Edit Pending Tim... | 18618045 | 500 days ago | IN | 0 ETH | 0.0007181 |
View more zero value Internal Transactions in Advanced View mode
Advanced mode:
Loading...
Loading
Contract Name:
BananaRepoProject
Compiler Version
v0.8.21+commit.d9974bed
Optimization Enabled:
Yes with 200 runs
Other Settings:
default evmVersion
Contract Source Code (Solidity Standard Json-Input format)
// SPDX-License-Identifier: MIT pragma solidity 0.8.21; import "@openzeppelin/contracts/access/Ownable.sol"; import "https://github.com/transmissions11/solmate/blob/main/src/utils/FixedPointMathLib.sol"; import "./RepoErrors.sol"; import "@openzeppelin/contracts/token/ERC20/IERC20.sol"; // interface /// @title Banana Repo Project /// @author Boonana /// @notice A smart contract that allows one to sell repos of whitelisted tokens. Liquidity (repo buyers) is provided by users. If a repo seller does not repurchase their asset, it defaults to liquidity providers contract BananaRepoProject is Ownable { ///////////////////////////////////////////////////////////////////////////// // Structs // ///////////////////////////////////////////////////////////////////////////// struct repo { uint256 expirationTime; uint256 repoTokenAmount; } struct pendingDeposit { uint256 activationTime; uint256 amount; } ///////////////////////////////////////////////////////////////////////////// // Constants // ///////////////////////////////////////////////////////////////////////////// address public immutable currencyToken; address public immutable repoToken; uint256 public constant PRECISION = 1e18; ///////////////////////////////////////////////////////////////////////////// // Storage // ///////////////////////////////////////////////////////////////////////////// mapping (address => uint256) public userCurrencyBalances; // how much each address is owed in Currency Token address[] public clearedDepositorsList; uint256 public totalClearedBalance; // sum of the balances in userCurrencyBalances uint256 public totalEligibleBalance; // sum of the balances in userCurrencyBalances, minus any that are "reserved" in other repos. i.e. the amount that can be used for another repo at the moment mapping (address => uint256) public userDefaultBalances; // how much each address is owed in repoToken // Implementation notes: // 1. When you deposit, your deposit will go in as a pending deposit with an activation time set pendingTime seconds in the future. In other words, your deposit will only be able to participate after the activation time. // 2. If you deposit and you have previous pending deposits whose pending time has not yet passed, your deposited amount will be added to the pending balance and the activation time will reset. // THEY WILL NOT BE CONSIDERED TWO DIFFERENT DEPOSITS WITH TWO DIFFERENT PENDING TIMES. // 3. However, every time you deposit, the code will first clear any pending deposits of yours whose activationTime have passed automatically, so rest assured cleared pending deposits will not be reset. // 4. Whenever a repo is initiated, all pending deposits whose activation time have passed will be cleared. // 5. You can withdraw before activationTime has passed on a pending deposit; the activation time only applies to if the money can be used in repos. // 6. If you have both pending and active balances, and withdraw, the balance will first be taken out of your pending balance - LIFO. mapping (address => pendingDeposit) public pendingBalances; // deposits that are not eligible to participate in repo transactions yet. If you deposit again before these activate, your time will restart. address[] public pendingAddressList; // list of addresses currently with pending balances mapping (address => repo) public activeRepos; // stores all active repos at the moment address[] public reposUsersList; uint256 public repoSellPrice; uint256 public repoBuybackPrice; uint256 public repoTimeLength; uint256 public pendingTime; // a pending time can be set to prevent users from only depositing when repos have been bought and not depositing otherwise. Would make sense to be set equal to repoTimeLength bool public reposPaused; // default set to false; ///////////////////////////////////////////////////////////////////////////// // Events // ///////////////////////////////////////////////////////////////////////////// event Deposit(address indexed user, uint256 amount); event Withdraw(address indexed user, address indexed token, uint256 amount); event repoSold(address indexed seller, uint256 repoTokenAmount, uint256 currencyTokenAmount); event repoBoughtBack(address indexed buyer, uint256 repoTokenAmount, uint256 currencyTokenAmount); event repoDefault(address indexed defaulter, uint256 repoTokenAmount, uint256 currencyTokenAmount); ///////////////////////////////////////////////////////////////////////////// // CONSTRUCTOR // ///////////////////////////////////////////////////////////////////////////// // @notice constructor // @param _currencyToken Address of the currency token. // @param _repoToken Address of the repo token - the token the contract has open repo bids for. // @param _repoSellPrice Sell price for 1e18 units of repoToken. Necessary since you can't store fractions :(. // @param _repoBuybackPrice Buyback price for 1e18 shares of repoToken. Necessary to do price fo 1e18 since you can't store fractions :(. // @param _repoTimeLength Amount of time in seconds before a buyback option for a repo expires. // @param _pendingTime Amount of time in seconds before deposited funds can participate in repo transactions constructor(address _currencyToken, address _repoToken, uint256 _repoSellPrice, uint256 _repoBuybackPrice, uint256 _repoTimeLength, uint256 _pendingTime) Ownable(_msgSender()) { currencyToken = _currencyToken; repoToken = _repoToken; repoSellPrice = _repoSellPrice; repoBuybackPrice = _repoBuybackPrice; repoTimeLength = _repoTimeLength; pendingTime = _pendingTime; } ///////////////////////////////////////////////////////////////////////////// // VIEWS // ///////////////////////////////////////////////////////////////////////////// // @notice shows how much _address is owed in both tokens (currencyToken and repoToken). Note this may not be withdrawable at the moment if some funds were used to buy repos. // @param _address the address to check owed balance for function getOwedBalance(address _address) external view returns (uint256, uint256) { return (pendingBalances[_address].amount + userCurrencyBalances[_address], userDefaultBalances[_address]); } // @notice shows how much _address can currently withdraw in both tokens (currencyToken and repoToken). Note they may be owed more, but some is not withdrawable because it is currently used to buy repos. Both pending and cleared balances are withdrawable. // @param _address the address to check withdrawable balance for function getWithdrawableBalance(address _address) external view returns (uint256, uint256) { uint256 withdrawableCurrencyBalance = (totalEligibleBalance > userCurrencyBalances[_address]) ? (userCurrencyBalances[_address] + pendingBalances[_address].amount) : (totalEligibleBalance + pendingBalances[_address].amount); return (withdrawableCurrencyBalance, userDefaultBalances[_address]); } // @notice returns true if the repo for a given address is expired, returns false if not or if the user does not have an active repo // @param the user whose repos to check function checkRepoExpiration(address _address) external view returns (bool) { uint256 expirationTime = activeRepos[_address].expirationTime; if (expirationTime == 0) { return false; } else if (block.timestamp > expirationTime) { return true; } else { return false; } } // @notice protocol earns money off rounding errors (i.e. if 10 tokens are split 1/3 1/3 1/3, then those are all roudned down and the remainder is protocol profit) or when there are no cleared balances when a repo is bought back (rare) // protocol profit as a result should be very small, but is always borrowable as it is included in totalEligibleBalance function getProtocolProfits() public view returns (uint256, uint256) { uint256 contractCurrencyBalance = IERC20(currencyToken).balanceOf(address(this)); uint256 pendingListLength = pendingAddressList.length; uint256 pendingBalanceSum; for (uint256 i=0; i<pendingListLength; ++i) { pendingBalanceSum += pendingBalances[pendingAddressList[i]].amount; } uint256 currencyTokenProfit = contractCurrencyBalance - totalEligibleBalance - pendingBalanceSum; uint256 repoListLength = reposUsersList.length; uint256 repoTokenAmountSum; for (uint256 i = 0; i<repoListLength; ++i) { repoTokenAmountSum += activeRepos[reposUsersList[i]].repoTokenAmount; } uint256 clearedDepositorLength = clearedDepositorsList.length; uint256 defaultBalanceSum; for (uint256 i = 0; i<clearedDepositorLength; ++i) { defaultBalanceSum += userDefaultBalances[clearedDepositorsList[i]]; } return (currencyTokenProfit, IERC20(repoToken).balanceOf(address(this)) - repoTokenAmountSum - defaultBalanceSum); } // @notice Checks the maximum amount of currencyToken available to buy a repo with at the moment in the protocol function getTotalCurrencyAvailableForRepo() external view returns (uint256) { uint256 pendingListLength = pendingAddressList.length; uint256 pendingBalanceSum; uint256 curTimestamp = block.timestamp; for (uint256 i=0; i<pendingListLength; ++i) { if (curTimestamp > pendingBalances[pendingAddressList[i]].activationTime) { pendingBalanceSum += pendingBalances[pendingAddressList[i]].amount; } } return (pendingBalanceSum + totalEligibleBalance); } ///////////////////////////////////////////////////////////////////////////// // CORE - DEPOSITORS // ///////////////////////////////////////////////////////////////////////////// /// @notice Deposit funds into the user's balance. /// @param _amount The amount to deposit. function depositFunds(uint256 _amount) external { IERC20(currencyToken).transferFrom(msg.sender, address(this), _amount); _clearPendingBalanceUser(msg.sender); // if there are any pending balances by this user, first clear them uint256 currentTime = block.timestamp; if (_amount == 0) return; // nothing to do if (pendingBalances[msg.sender].amount == 0) { // if it doesn't exist yet, add to pendingAddressList pendingAddressList.push(msg.sender); } pendingBalances[msg.sender].amount += _amount; // pendingBalances[msg.sender].activationTime = currentTime + pendingTime; // activation time reset emit Deposit(msg.sender, _amount); } /// @notice /// @param _token is the token to withdraw /// @param _amount is the amount to withdraw function withdrawFunds(address _token, uint256 _amount) external { if (_token == currencyToken) { uint256 currentBalance = userCurrencyBalances[msg.sender]; uint256 pendingBalance = pendingBalances[msg.sender].amount; if (currentBalance + pendingBalance < _amount) revert RepoErrors.InsufficientBalance(); if (_amount >= pendingBalance) { // we subtract from pending balance first to benefit depositor the most; LIFO uint256 remainingBalance = (_amount- pendingBalance); // see RepoErros.sol for explanation if (totalEligibleBalance < remainingBalance) revert RepoErrors.WithdrawingMoreThanEligible(); userCurrencyBalances[msg.sender] -= remainingBalance; if (currentBalance == remainingBalance) { _removeClearedDepositorsListUser(msg.sender); // the user has no cleared balance left } totalClearedBalance -= remainingBalance; totalEligibleBalance -= remainingBalance; delete pendingBalances[msg.sender]; _removePendingAddressListUser(msg.sender); } else { pendingBalances[msg.sender].amount -= _amount; // we subtract from pending balance first to benefit depositor the most; LIFO } IERC20(currencyToken).transfer(msg.sender, _amount); } else if (_token == repoToken) { // amount of tokens user currently is owed if (userDefaultBalances[msg.sender] < _amount) revert RepoErrors.InsufficientBalance(); userDefaultBalances[msg.sender] -= _amount; IERC20(repoToken).transfer(msg.sender, _amount); } else { revert RepoErrors.InvalidToken(); } emit Withdraw(msg.sender, _token, _amount); } /// @notice Allows users to withdraw their total owed balance. Note: may fail if some owed balance is currently part of a sold repo /// @dev External function that allows users to withdraw their available balances for currency and repo tokens. function withdrawAll() external { uint256 userClearedBalance = userCurrencyBalances[msg.sender]; uint256 userPendingBalance = pendingBalances[msg.sender].amount; uint256 userCurrencyBalance = userClearedBalance + userPendingBalance; uint256 userDefaultBalance = userDefaultBalances[msg.sender]; if (userCurrencyBalance == 0 && userDefaultBalance == 0) revert RepoErrors.NothingToWithdraw(); if (userClearedBalance > totalEligibleBalance) revert RepoErrors.WithdrawingMoreThanEligible(); if (userCurrencyBalance > 0) { if (userPendingBalance > 0) { delete pendingBalances[msg.sender]; _removePendingAddressListUser(msg.sender); } totalClearedBalance -= userClearedBalance; totalEligibleBalance -= userClearedBalance; delete userCurrencyBalances[msg.sender]; IERC20(currencyToken).transfer(msg.sender, userCurrencyBalance); emit Withdraw(msg.sender, currencyToken, userCurrencyBalance); } if (userDefaultBalance > 0) { delete userDefaultBalances[msg.sender]; IERC20(repoToken).transfer(msg.sender, userDefaultBalance); emit Withdraw(msg.sender, repoToken, userDefaultBalance); } _removeClearedDepositorsListUser(msg.sender); } ///////////////////////////////////////////////////////////////////////////// // CORE - REPO SELLERS // ///////////////////////////////////////////////////////////////////////////// /// @notice Allows the selling of repo tokens. /// @param _repoTokenAmount The amount of repo tokens to sell. function sellRepo(uint256 _repoTokenAmount) external { if (reposPaused) revert RepoErrors.RepoPaused(); if (activeRepos[msg.sender].repoTokenAmount > 0) revert RepoErrors.RepoAlreadyExistsForUser(); clearPendingBalances(); // any pending balances, clear, to maximize totalEligibleBalance uint256 currencyTokenAmount = _repoTokenAmount*repoSellPrice/PRECISION; if (currencyTokenAmount > totalEligibleBalance) revert RepoErrors.InsufficientEligibleCurrencyAmount(); IERC20(repoToken).transferFrom(msg.sender, address(this), _repoTokenAmount); IERC20(currencyToken).transfer(msg.sender, currencyTokenAmount); uint256 expirationTime = block.timestamp + repoTimeLength; activeRepos[msg.sender].expirationTime = expirationTime; activeRepos[msg.sender].repoTokenAmount = _repoTokenAmount; reposUsersList.push(msg.sender); totalEligibleBalance -= currencyTokenAmount; emit repoSold(msg.sender, _repoTokenAmount, currencyTokenAmount); } // @notice Allows the buyback of repo tokens. function buybackRepo() external { repo memory userRepo = activeRepos[msg.sender]; if (userRepo.expirationTime == 0) revert RepoErrors.NoActiveRepoForUser(); if (block.timestamp > userRepo.expirationTime) revert RepoErrors.RepoExpired(); uint256 currencyTokenAmount = userRepo.repoTokenAmount*repoBuybackPrice/PRECISION; IERC20(currencyToken).transferFrom(msg.sender, address(this), currencyTokenAmount); IERC20(repoToken).transfer(msg.sender, userRepo.repoTokenAmount); clearPendingBalances(); // a bit of a tough design decision. But basically if your balance is cleared before the repo is bought back, then you benefit, even if you weren't cleared at time of deposit // the alternative is to only _clearPendingBalances() before the repo is sold. But then deposit also clears pending repos, so say, if pendingTime and repoTimeLength are 24 hours: // first deposit is T-23 hours. // repo is sold at T+0 hours // if repo isn't repaid at T+1 hour, someone could deposit again $1 and cause their balance to clear // so the criteria then becomes "to guarantee eligibility without extra work deposit 24 hours before repo is sold, otherwise, with extra work you can guarantee eligibility if you deposit 24 hours before repo is repaid" uint256 newBalanceAdded = currencyTokenAmount - userRepo.repoTokenAmount*repoSellPrice/PRECISION; uint256 totalClearedBalanceCopy = totalClearedBalance; // don't call this each time from storage, just save a copy in memory to save gas uint256 listLength = clearedDepositorsList.length; for (uint256 i=0; i < listLength; ++i) { uint256 balanceToAdd = userCurrencyBalances[clearedDepositorsList[i]] * newBalanceAdded / totalClearedBalanceCopy; userCurrencyBalances[clearedDepositorsList[i]] += balanceToAdd; // everyone's balance increases yay totalClearedBalance += balanceToAdd; } totalEligibleBalance += (userRepo.repoTokenAmount*repoSellPrice/PRECISION + totalClearedBalance - totalClearedBalanceCopy); delete activeRepos[msg.sender]; _removeReposUsersListUser(msg.sender); emit repoBoughtBack(msg.sender, userRepo.repoTokenAmount, currencyTokenAmount); } function defaultRepo(address _address) external { repo memory userRepo = activeRepos[_address]; if (userRepo.expirationTime == 0) revert RepoErrors.NoActiveRepo(); // nothing to do if (block.timestamp <= userRepo.expirationTime) revert RepoErrors.RepoStillActive(); if (block.timestamp > userRepo.expirationTime) { // repo has indeed expired clearPendingBalances(); uint256 lostCurrencyBalance = userRepo.repoTokenAmount*repoSellPrice/PRECISION; uint256 totalClearedBalanceCopy = totalClearedBalance; // don't call this each time from storage, just save a copy in memory to save gas // see error explanation in RepoErrors.sols. This *shouldn't* be possible at the moment because withdrawals greater than totalEligibleBalance are not permitted, but having this check just in case if (totalClearedBalanceCopy < lostCurrencyBalance) revert RepoErrors.InsufficientClearedDeposits(); // high level this is what we want to do: // We want to remove lostCurrencyBalance across the cleared balances, and distribute userRepo.repoTokenAmount across those // Solidity division rounds towards zero. So we must be careful and err towards removing AT LEAST lostCurrencyBalance, and distributing AT MOST userRepo.repoTokenAmount, with excess going to protocol // 1. we can do this for currency balances by adjusting each to be the same percentage as [totalClearedBalance - lostCurrencyBalance] as they are now of totalClearedBalance // NOTE: we CANNOT subtract the amount to be subtracted multiplied proportionately across the balances, as rounding in that case may result in removing LESS than lostCurrencyBalance in total // 2. we can do this for repo balances by assigning the same current percentage to the repo tokens being distributed uint256 tempNewTotalClearedBalance = totalClearedBalance - lostCurrencyBalance; uint256 listLength = clearedDepositorsList.length; for (uint256 i=0; i < listLength; ++i) { address userAddress = clearedDepositorsList[i]; userDefaultBalances[userAddress] += userCurrencyBalances[userAddress]*userRepo.repoTokenAmount /totalClearedBalanceCopy; // as noted in 2) uint256 currencyBalanceOld = userCurrencyBalances[userAddress]; userCurrencyBalances[userAddress] = currencyBalanceOld * tempNewTotalClearedBalance / totalClearedBalanceCopy; // as noted in 1) uint256 currencyBalanceSubtracted = currencyBalanceOld - userCurrencyBalances[userAddress]; totalClearedBalance -= currencyBalanceSubtracted; if ((userCurrencyBalances[userAddress] == 0) && (userDefaultBalances[userAddress] ==0)) _removeClearedDepositorsListUser(userAddress); } // if removed MORE than lostCurrencyBalance, subtract excess from totalEligibleBalance totalEligibleBalance -= (tempNewTotalClearedBalance - totalClearedBalance); delete activeRepos[_address]; _removeReposUsersListUser(_address); emit repoDefault(_address, userRepo.repoTokenAmount, lostCurrencyBalance); } } // @notice Allows the owner to modify the repo sell price. function editRepoSellPrice(uint256 _price) external onlyOwner { repoSellPrice = _price; } // @notice Allows the owner to modify the repo buyback price. function editRepoBuybackPrice(uint256 _price) external onlyOwner { repoBuybackPrice = _price; } // @notice Allows the owner to modify the repo time length. function editRepoTimeLength(uint256 _seconds) external onlyOwner { repoTimeLength = _seconds; } // @notice Allows the owner to modify the pending time. function editPendingTime(uint256 _seconds) external onlyOwner { pendingTime = _seconds; } // @notice Allows the owner to pause repos. function pauseRepos() external onlyOwner { reposPaused = true; } // @notice Allows the owner to unpause repos. function unpauseRepos() external onlyOwner { reposPaused = false; } ///////////////////////////////////////////////////////////////////////////// // CORE - HELPERS // ///////////////////////////////////////////////////////////////////////////// /// @notice Clears pending balances by checking their activation time and moving the amounts to the users' currency balances if the activation time has passed. function clearPendingBalances() public { uint256 i = 0; uint256 currentTimestamp = block.timestamp; uint256 totalAmountCleared; uint256 pendingAddressListLength = pendingAddressList.length; // saves gas to store in memory vs read each time from storage while (i < pendingAddressListLength) { address userAddress = pendingAddressList[i]; pendingDeposit memory pendingDepositUser = pendingBalances[userAddress]; if (currentTimestamp >= pendingDepositUser.activationTime) { // add to list if first time depositor if (userCurrencyBalances[userAddress] == 0) { clearedDepositorsList.push(userAddress); } userCurrencyBalances[userAddress] += pendingDepositUser.amount; totalAmountCleared += pendingDepositUser.amount; delete pendingBalances[userAddress]; _removePendingAddressList(i); pendingAddressListLength -= 1; //reassign now that length is different } else { // Increment 'i' only when not removing the element to stay at the same index ++i; } } // write to storage just once at the end to save gas totalClearedBalance += totalAmountCleared; totalEligibleBalance += totalAmountCleared; } /// @notice Clears pending balances for a specified user address based on the activation time; if current timestamp is past activation time, the user's pending balance is added to their userCurrencyBalance. /// @param _address The address for which pending balances need to be cleared. function _clearPendingBalanceUser(address _address) internal { pendingDeposit memory pendingDepositUser = pendingBalances[_address]; // saves gas to cache in memory if (pendingDepositUser.amount == 0) return; // user does not have pending Balance uint256 currentTimestamp = block.timestamp; if (currentTimestamp >= pendingDepositUser.activationTime) { // add to list if first time depositor if (userCurrencyBalances[_address] == 0) clearedDepositorsList.push(_address); userCurrencyBalances[_address] += pendingDepositUser.amount; totalClearedBalance += pendingDepositUser.amount; totalEligibleBalance += pendingDepositUser.amount; delete pendingBalances[_address]; _removePendingAddressListUser(_address); } } /// @notice Removes a specific user address from pending balance list. /// @param _address The address to remove from pending balance list. function _removePendingAddressListUser(address _address) internal { uint256 listLength = pendingAddressList.length; for (uint256 i = 0; i < listLength; ++i) { if (pendingAddressList[i] == _address) { _removePendingAddressList(i); break; // end the function after we find the user, as they should only appear once } } } /// @notice Remove a user from the pending balance list at the specified index in the array. /// @param _index The index of the address in pendingAddressList to remove function _removePendingAddressList(uint256 _index) internal { uint256 listLength = pendingAddressList.length; if (_index > listLength) revert RepoErrors.InvalidIndex(); // remove from pending address list pendingAddressList[_index] = pendingAddressList[listLength - 1]; pendingAddressList.pop(); } /// @notice Removes a specific user address from cleared deposit list. /// @param _address The address to remove from cleared deposit list. function _removeClearedDepositorsListUser(address _address) internal { uint256 listLength = clearedDepositorsList.length; for (uint256 i = 0; i < listLength; ++i) { if (clearedDepositorsList[i] == _address) { _removeClearedDepositorsList(i); break; // end the function after we find the user, as they should only appear once } } } /// @notice Remove a user from the cleared Deposit list at the specified index in the array. /// @param _index The index of the address in clearedDepositList to remove function _removeClearedDepositorsList(uint256 _index) internal { uint256 listLength = clearedDepositorsList.length; if (_index > listLength) revert RepoErrors.InvalidIndex(); // remove from pending address list clearedDepositorsList[_index] = clearedDepositorsList[listLength - 1]; clearedDepositorsList.pop(); } /// @notice Removes a specific user address from repos users list. /// @param _address The address to remove from repos users list. function _removeReposUsersListUser(address _address) internal { uint256 listLength = reposUsersList.length; for (uint256 i = 0; i < listLength; ++i) { if (reposUsersList[i] == _address) { _removeReposUsersList(i); break; // end the function after we find the user, as they should only appear once } } } /// @notice Remove a user from the repos users list at the specified index in the array. /// @param _index The index of the address in reposUsersList to remove function _removeReposUsersList(uint256 _index) internal { uint256 listLength = reposUsersList.length; if (_index > listLength) revert RepoErrors.InvalidIndex(); // remove from pending address list reposUsersList[_index] = reposUsersList[listLength - 1]; reposUsersList.pop(); } ///////////////////////////////////////////////////////////////////////////// // CORE - ADMIN // ///////////////////////////////////////////////////////////////////////////// /// @notice for emergency use if the owner needs to withdraw all tokens. If withdrawing protocol profits, should use protocolProfitWithdraw /// @param _tokenAddress the address of the token to withdraw /// @param _amount the amount of the token to withdraw function adminWithdraw(address _tokenAddress, uint256 _amount) external onlyOwner { IERC20(_tokenAddress).transfer(msg.sender, _amount); } /// For withdrawing protocol profits function protocolProfitWithdraw() external onlyOwner { (uint256 protocolCurrencyProfits, uint256 protocolRepoProfits) = getProtocolProfits(); IERC20(currencyToken).transfer(msg.sender, protocolCurrencyProfits); IERC20(repoToken).transfer(msg.sender, protocolRepoProfits); } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v5.0.0) (token/ERC20/IERC20.sol) pragma solidity ^0.8.20; /** * @dev Interface of the ERC20 standard as defined in the EIP. */ interface IERC20 { /** * @dev Emitted when `value` tokens are moved from one account (`from`) to * another (`to`). * * Note that `value` may be zero. */ event Transfer(address indexed from, address indexed to, uint256 value); /** * @dev Emitted when the allowance of a `spender` for an `owner` is set by * a call to {approve}. `value` is the new allowance. */ event Approval(address indexed owner, address indexed spender, uint256 value); /** * @dev Returns the value of tokens in existence. */ function totalSupply() external view returns (uint256); /** * @dev Returns the value of tokens owned by `account`. */ function balanceOf(address account) external view returns (uint256); /** * @dev Moves a `value` amount of tokens from the caller's account to `to`. * * Returns a boolean value indicating whether the operation succeeded. * * Emits a {Transfer} event. */ function transfer(address to, uint256 value) external returns (bool); /** * @dev Returns the remaining number of tokens that `spender` will be * allowed to spend on behalf of `owner` through {transferFrom}. This is * zero by default. * * This value changes when {approve} or {transferFrom} are called. */ function allowance(address owner, address spender) external view returns (uint256); /** * @dev Sets a `value` amount of tokens as the allowance of `spender` over the * caller's tokens. * * Returns a boolean value indicating whether the operation succeeded. * * IMPORTANT: Beware that changing an allowance with this method brings the risk * that someone may use both the old and the new allowance by unfortunate * transaction ordering. One possible solution to mitigate this race * condition is to first reduce the spender's allowance to 0 and set the * desired value afterwards: * https://github.com/ethereum/EIPs/issues/20#issuecomment-263524729 * * Emits an {Approval} event. */ function approve(address spender, uint256 value) external returns (bool); /** * @dev Moves a `value` amount of tokens from `from` to `to` using the * allowance mechanism. `value` is then deducted from the caller's * allowance. * * Returns a boolean value indicating whether the operation succeeded. * * Emits a {Transfer} event. */ function transferFrom(address from, address to, uint256 value) external returns (bool); }
// SPDX-License-Identifier: MIT pragma solidity 0.8.21; library RepoErrors { error InsufficientBalance(); error NothingToWithdraw(); error InvalidToken(); error InvalidIndex(); // issue with code error RepoPaused(); error RepoAlreadyExistsForUser(); error InsufficientEligibleCurrencyAmount(); error RepoExpired(); error NoActiveRepo(); error NoActiveRepoForUser(); error RepoStillActive(); // so basically there is a possible case where there are no cleared Depositors. // // Example: // 1. At the time of sellRepo, there is one cleared Depositor for 1000 currencyToken. // 2. sellRepo is called, and repoToken is sold for all 1000 currencyToken. // 3. The repo expires. // 4. A new depositor deposits 1000 more currencyToken. At this point this is 1000 currencyToken in the contract, and 1 address each in clearedDepositorsList and pendingAddressList // 5. Before the new depositor's deposit clears, the cleared depositor withdraws all 1000 currency tokens. // 6. Now, there are 0 cleared depositors, and an expired repo! // // If defaultRepo is called in this case, the pending depositor would still be owed 1000 currency tokens, and no one would have ownership of all the defaulted repo tokens. // Therefore we need at least one cleared depositor to be present before defaultRepo() can be called; furthermore, we need the total number of cleared Deposits to be greater than or equal to 1000 currencyTokens. // Note this is guaranteed to happen within pendingTime as all pendingDeposits will be cleared then, and no one else can withdraw since there are no more tokens left in the contract. // Note it is fine if there are no cleared depositors present when buybackRepo() is called, as the extra currencyToken accrued would just sit in the contract for the owner presumably. error InsufficientClearedDeposits(); // this occurs when the amount of currencyToken you are trying to withdraw exceeds totalEligibleBalance, which is the unborrowed cleared balance at the moment // if you receive this error, this likely means the contract has enough currencyToken to pay you out, but some of that belongs to pending balances which do not belong to you. Your funds are locked in a repo at the moment, while the // pending funds belonging to someone else are not error WithdrawingMoreThanEligible(); }
// 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: MIT // OpenZeppelin Contracts (last updated v5.0.0) (access/Ownable.sol) pragma solidity ^0.8.20; import {Context} from "../utils/Context.sol"; /** * @dev Contract module which provides a basic access control mechanism, where * there is an account (an owner) that can be granted exclusive access to * specific functions. * * The initial owner is set to the address provided by the deployer. This can * later be changed with {transferOwnership}. * * This module is used through inheritance. It will make available the modifier * `onlyOwner`, which can be applied to your functions to restrict their use to * the owner. */ abstract contract Ownable is Context { address private _owner; /** * @dev The caller account is not authorized to perform an operation. */ error OwnableUnauthorizedAccount(address account); /** * @dev The owner is not a valid owner account. (eg. `address(0)`) */ error OwnableInvalidOwner(address owner); event OwnershipTransferred(address indexed previousOwner, address indexed newOwner); /** * @dev Initializes the contract setting the address provided by the deployer as the initial owner. */ constructor(address initialOwner) { if (initialOwner == address(0)) { revert OwnableInvalidOwner(address(0)); } _transferOwnership(initialOwner); } /** * @dev Throws if called by any account other than the owner. */ modifier onlyOwner() { _checkOwner(); _; } /** * @dev Returns the address of the current owner. */ function owner() public view virtual returns (address) { return _owner; } /** * @dev Throws if the sender is not the owner. */ function _checkOwner() internal view virtual { if (owner() != _msgSender()) { revert OwnableUnauthorizedAccount(_msgSender()); } } /** * @dev Leaves the contract without owner. It will not be possible to call * `onlyOwner` functions. Can only be called by the current owner. * * NOTE: Renouncing ownership will leave the contract without an owner, * thereby disabling any functionality that is only available to the owner. */ function renounceOwnership() public virtual onlyOwner { _transferOwnership(address(0)); } /** * @dev Transfers ownership of the contract to a new account (`newOwner`). * Can only be called by the current owner. */ function transferOwnership(address newOwner) public virtual onlyOwner { if (newOwner == address(0)) { revert OwnableInvalidOwner(address(0)); } _transferOwnership(newOwner); } /** * @dev Transfers ownership of the contract to a new account (`newOwner`). * Internal function without access restriction. */ function _transferOwnership(address newOwner) internal virtual { address oldOwner = _owner; _owner = newOwner; emit OwnershipTransferred(oldOwner, newOwner); } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v5.0.0) (utils/Context.sol) pragma solidity ^0.8.20; /** * @dev Provides information about the current execution context, including the * sender of the transaction and its data. While these are generally available * via msg.sender and msg.data, they should not be accessed in such a direct * manner, since when dealing with meta-transactions the account sending and * paying for execution may not be the actual sender (as far as an application * is concerned). * * This contract is only required for intermediate, library-like contracts. */ abstract contract Context { function _msgSender() internal view virtual returns (address) { return msg.sender; } function _msgData() internal view virtual returns (bytes calldata) { return msg.data; } }
{ "optimizer": { "enabled": true, "runs": 200 }, "outputSelection": { "*": { "*": [ "evm.bytecode", "evm.deployedBytecode", "devdoc", "userdoc", "metadata", "abi" ] } } }
Contract Security Audit
- No Contract Security Audit Submitted- Submit Audit Here
Contract ABI
API[{"inputs":[{"internalType":"address","name":"_currencyToken","type":"address"},{"internalType":"address","name":"_repoToken","type":"address"},{"internalType":"uint256","name":"_repoSellPrice","type":"uint256"},{"internalType":"uint256","name":"_repoBuybackPrice","type":"uint256"},{"internalType":"uint256","name":"_repoTimeLength","type":"uint256"},{"internalType":"uint256","name":"_pendingTime","type":"uint256"}],"stateMutability":"nonpayable","type":"constructor"},{"inputs":[],"name":"InsufficientBalance","type":"error"},{"inputs":[],"name":"InsufficientClearedDeposits","type":"error"},{"inputs":[],"name":"InsufficientEligibleCurrencyAmount","type":"error"},{"inputs":[],"name":"InvalidIndex","type":"error"},{"inputs":[],"name":"InvalidToken","type":"error"},{"inputs":[],"name":"NoActiveRepo","type":"error"},{"inputs":[],"name":"NoActiveRepoForUser","type":"error"},{"inputs":[],"name":"NothingToWithdraw","type":"error"},{"inputs":[{"internalType":"address","name":"owner","type":"address"}],"name":"OwnableInvalidOwner","type":"error"},{"inputs":[{"internalType":"address","name":"account","type":"address"}],"name":"OwnableUnauthorizedAccount","type":"error"},{"inputs":[],"name":"RepoAlreadyExistsForUser","type":"error"},{"inputs":[],"name":"RepoExpired","type":"error"},{"inputs":[],"name":"RepoPaused","type":"error"},{"inputs":[],"name":"RepoStillActive","type":"error"},{"inputs":[],"name":"WithdrawingMoreThanEligible","type":"error"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"user","type":"address"},{"indexed":false,"internalType":"uint256","name":"amount","type":"uint256"}],"name":"Deposit","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"previousOwner","type":"address"},{"indexed":true,"internalType":"address","name":"newOwner","type":"address"}],"name":"OwnershipTransferred","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"user","type":"address"},{"indexed":true,"internalType":"address","name":"token","type":"address"},{"indexed":false,"internalType":"uint256","name":"amount","type":"uint256"}],"name":"Withdraw","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"buyer","type":"address"},{"indexed":false,"internalType":"uint256","name":"repoTokenAmount","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"currencyTokenAmount","type":"uint256"}],"name":"repoBoughtBack","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"defaulter","type":"address"},{"indexed":false,"internalType":"uint256","name":"repoTokenAmount","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"currencyTokenAmount","type":"uint256"}],"name":"repoDefault","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"seller","type":"address"},{"indexed":false,"internalType":"uint256","name":"repoTokenAmount","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"currencyTokenAmount","type":"uint256"}],"name":"repoSold","type":"event"},{"inputs":[],"name":"PRECISION","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"activeRepos","outputs":[{"internalType":"uint256","name":"expirationTime","type":"uint256"},{"internalType":"uint256","name":"repoTokenAmount","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_tokenAddress","type":"address"},{"internalType":"uint256","name":"_amount","type":"uint256"}],"name":"adminWithdraw","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"buybackRepo","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_address","type":"address"}],"name":"checkRepoExpiration","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"clearPendingBalances","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"}],"name":"clearedDepositorsList","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"currencyToken","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_address","type":"address"}],"name":"defaultRepo","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_amount","type":"uint256"}],"name":"depositFunds","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_seconds","type":"uint256"}],"name":"editPendingTime","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_price","type":"uint256"}],"name":"editRepoBuybackPrice","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_price","type":"uint256"}],"name":"editRepoSellPrice","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_seconds","type":"uint256"}],"name":"editRepoTimeLength","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_address","type":"address"}],"name":"getOwedBalance","outputs":[{"internalType":"uint256","name":"","type":"uint256"},{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"getProtocolProfits","outputs":[{"internalType":"uint256","name":"","type":"uint256"},{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"getTotalCurrencyAvailableForRepo","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_address","type":"address"}],"name":"getWithdrawableBalance","outputs":[{"internalType":"uint256","name":"","type":"uint256"},{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"owner","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"pauseRepos","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"}],"name":"pendingAddressList","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"pendingBalances","outputs":[{"internalType":"uint256","name":"activationTime","type":"uint256"},{"internalType":"uint256","name":"amount","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"pendingTime","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"protocolProfitWithdraw","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"renounceOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"repoBuybackPrice","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"repoSellPrice","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"repoTimeLength","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"repoToken","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"reposPaused","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"}],"name":"reposUsersList","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"_repoTokenAmount","type":"uint256"}],"name":"sellRepo","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"totalClearedBalance","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"totalEligibleBalance","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"newOwner","type":"address"}],"name":"transferOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"unpauseRepos","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"userCurrencyBalances","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"userDefaultBalances","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"withdrawAll","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_token","type":"address"},{"internalType":"uint256","name":"_amount","type":"uint256"}],"name":"withdrawFunds","outputs":[],"stateMutability":"nonpayable","type":"function"}]
Contract Creation Code
60c060405234801562000010575f80fd5b5060405162002746380380620027468339810160408190526200003391620000fd565b33806200005957604051631e4fbdf760e01b81525f600482015260240160405180910390fd5b620000648162000092565b506001600160a01b039586166080529390941660a052600a91909155600b55600c91909155600d5562000156565b5f80546001600160a01b038381166001600160a01b0319831681178455604051919092169283917f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e09190a35050565b80516001600160a01b0381168114620000f8575f80fd5b919050565b5f805f805f8060c0878903121562000113575f80fd5b6200011e87620000e1565b95506200012e60208801620000e1565b945060408701519350606087015192506080870151915060a087015190509295509295509295565b60805160a051612557620001ef5f395f818161036f015281816106090152818161088501528181610ad0015281816116bb015281816117360152818161188e01528181611c4d0152611cf201525f81816103de0152818161057a015281816106a401528181610a3b01528181610d6d015281816115b20152818161162d0152818161191d01528181611a550152611bd101526125575ff3fe608060405234801561000f575f80fd5b506004361061023f575f3560e01c80636b2fa374116101355780639f8b20a5116100b4578063d8ae6f3111610079578063d8ae6f31146104da578063ecdae41b146104e2578063ee2ae94114610508578063f2fde38b14610515578063f8c716f914610528575f80fd5b80639f8b20a514610493578063aaf5eb68146104a6578063b60a266c146104b5578063b9bfebfb146104be578063c1075329146104c7575f80fd5b8063853828b6116100fa578063853828b61461044d5780638da5cb5b1461045557806394c174be1461046557806398e1e2b71461046d57806398f1196c14610480575f80fd5b80636b2fa374146103d9578063715018a61461040057806379c4855c146104085780638108add41461041b578063843592d31461043a575f80fd5b8063401d4482116101c157806360a859991161018657806360a859991461036a578063626f56f814610391578063631d54df146103a457806364e5a951146103c75780636a50c448146103d0575f80fd5b8063401d44821461030d57806350e041ea146103205780635d24b1e9146103335780635d73aa53146103595780635dc172ac14610362575f80fd5b806320cdc7b31161020757806320cdc7b31461029d578063252323e4146102a557806327163639146102bc5780632b4ccdb4146102e75780633b76594d146102fa575f80fd5b806301ea9bcd1461024357806304f6fe511461024d5780630a8239511461026f5780630dcaf64b146102775780631322adac1461028a575b5f80fd5b61024b610547565b005b610255610681565b604080519283526020830191909152015b60405180910390f35b61024b610912565b61024b6102853660046123bb565b610926565b6102556102983660046123ed565b610933565b61024b610984565b6102ae60045481565b604051908152602001610266565b6102cf6102ca3660046123bb565b610d14565b6040516001600160a01b039091168152602001610266565b6102cf6102f53660046123bb565b610d3c565b61024b6103083660046123bb565b610d4b565b61024b61031b36600461240d565b610ed4565b61024b61032e3660046123ed565b610f0f565b6102556103413660046123ed565b60086020525f90815260409020805460019091015482565b6102ae600a5481565b61024b6111eb565b6102cf7f000000000000000000000000000000000000000000000000000000000000000081565b61024b61039f3660046123bb565b61138c565b6103b76103b23660046123ed565b611399565b6040519015158152602001610266565b6102ae60035481565b6102ae600b5481565b6102cf7f000000000000000000000000000000000000000000000000000000000000000081565b61024b6113d9565b61024b6104163660046123bb565b6113ec565b6102ae6104293660046123ed565b60016020525f908152604090205481565b6102556104483660046123ed565b6113f9565b61024b61149b565b5f546001600160a01b03166102cf565b61024b611796565b61024b61047b3660046123bb565b6117ad565b6102cf61048e3660046123bb565b6117ba565b61024b6104a13660046123bb565b6117c9565b6102ae670de0b6b3a764000081565b6102ae600d5481565b6102ae600c5481565b61024b6104d536600461240d565b611a53565b6102ae611dc7565b6102556104f03660046123ed565b60066020525f90815260409020805460019091015482565b600e546103b79060ff1681565b61024b6105233660046123ed565b611e89565b6102ae6105363660046123ed565b60056020525f908152604090205481565b61054f611ecb565b5f80610559610681565b60405163a9059cbb60e01b81523360048201526024810183905291935091507f00000000000000000000000000000000000000000000000000000000000000006001600160a01b03169063a9059cbb906044016020604051808303815f875af11580156105c8573d5f803e3d5ffd5b505050506040513d601f19601f820116820180604052508101906105ec9190612435565b5060405163a9059cbb60e01b8152336004820152602481018290527f00000000000000000000000000000000000000000000000000000000000000006001600160a01b03169063a9059cbb906044015b6020604051808303815f875af1158015610658573d5f803e3d5ffd5b505050506040513d601f19601f8201168201806040525081019061067c9190612435565b505050565b6040516370a0823160e01b81523060048201525f90819081906001600160a01b037f000000000000000000000000000000000000000000000000000000000000000016906370a0823190602401602060405180830381865afa1580156106e9573d5f803e3d5ffd5b505050506040513d601f19601f8201168201806040525081019061070d9190612454565b6007549091505f805b828110156107775760065f600783815481106107345761073461246b565b5f9182526020808320909101546001600160a01b031683528201929092526040019020600101546107659083612493565b9150610770816124ac565b9050610716565b505f816004548561078891906124c4565b61079291906124c4565b6009549091505f805b828110156107fc5760085f600983815481106107b9576107b961246b565b5f9182526020808320909101546001600160a01b031683528201929092526040019020600101546107ea9083612493565b91506107f5816124ac565b905061079b565b506002545f805b828110156108615760055f600283815481106108215761082161246b565b5f9182526020808320909101546001600160a01b0316835282019290925260400190205461084f9083612493565b915061085a816124ac565b9050610803565b506040516370a0823160e01b81523060048201528590829085906001600160a01b037f000000000000000000000000000000000000000000000000000000000000000016906370a0823190602401602060405180830381865afa1580156108ca573d5f803e3d5ffd5b505050506040513d601f19601f820116820180604052508101906108ee9190612454565b6108f891906124c4565b61090291906124c4565b9950995050505050505050509091565b61091a611ecb565b600e805460ff19169055565b61092e611ecb565b600a55565b6001600160a01b0381165f908152600160208181526040808420546006909252832090910154829161096491612493565b6001600160a01b039093165f908152600560205260409020549293915050565b335f908152600860209081526040808320815180830190925280548083526001909101549282019290925291036109ce5760405163f71855a360e01b815260040160405180910390fd5b80514211156109f057604051633e6ba29f60e11b815260040160405180910390fd5b5f670de0b6b3a7640000600b548360200151610a0c91906124d7565b610a1691906124ee565b6040516323b872dd60e01b8152336004820152306024820152604481018290529091507f00000000000000000000000000000000000000000000000000000000000000006001600160a01b0316906323b872dd906064016020604051808303815f875af1158015610a89573d5f803e3d5ffd5b505050506040513d601f19601f82011682018060405250810190610aad9190612435565b50602082015160405163a9059cbb60e01b815233600482015260248101919091527f00000000000000000000000000000000000000000000000000000000000000006001600160a01b03169063a9059cbb906044016020604051808303815f875af1158015610b1e573d5f803e3d5ffd5b505050506040513d601f19601f82011682018060405250810190610b429190612435565b50610b4b6111eb565b5f670de0b6b3a7640000600a548460200151610b6791906124d7565b610b7191906124ee565b610b7b90836124c4565b600354600254919250905f5b81811015610c5c575f838560015f60028681548110610ba857610ba861246b565b5f9182526020808320909101546001600160a01b03168352820192909252604001902054610bd691906124d7565b610be091906124ee565b90508060015f60028581548110610bf957610bf961246b565b5f9182526020808320909101546001600160a01b0316835282019290925260400181208054909190610c2c908490612493565b925050819055508060035f828254610c449190612493565b90915550610c5591508290506124ac565b9050610b87565b5081600354670de0b6b3a7640000600a548860200151610c7c91906124d7565b610c8691906124ee565b610c909190612493565b610c9a91906124c4565b60045f828254610caa9190612493565b9091555050335f81815260086020526040812081815560010155610ccd90611ef7565b6020808601516040805191825291810186905233917f4ad4b82ea376b5cd1bddcd46a1a6885f09a01839ffed08a89603082412251d5d910160405180910390a25050505050565b60078181548110610d23575f80fd5b5f918252602090912001546001600160a01b0316905081565b60028181548110610d23575f80fd5b6040516323b872dd60e01b8152336004820152306024820152604481018290527f00000000000000000000000000000000000000000000000000000000000000006001600160a01b0316906323b872dd906064016020604051808303815f875af1158015610dbb573d5f803e3d5ffd5b505050506040513d601f19601f82011682018060405250810190610ddf9190612435565b50610de933611f53565b425f829003610df6575050565b335f908152600660205260408120600101549003610e5057600780546001810182555f919091527fa66cc928b5edb82af9bd49922954155ab7b0942694bea4ce44661d9a8736c6880180546001600160a01b031916331790555b335f9081526006602052604081206001018054849290610e71908490612493565b9091555050600d54610e839082612493565b335f81815260066020526040908190209290925590517fe1fffcc4923d04b559f4d29a8bfc6cda04eb5b0d3c460751c2402c5c5cc9109c90610ec89085815260200190565b60405180910390a25050565b610edc611ecb565b60405163a9059cbb60e01b8152336004820152602481018290526001600160a01b0383169063a9059cbb9060440161063c565b6001600160a01b0381165f90815260086020908152604080832081518083019092528054808352600190910154928201929092529103610f625760405163f360fff160e01b815260040160405180910390fd5b80514211610f835760405163267ecc6560e21b815260040160405180910390fd5b80514211156111e757610f946111eb565b5f670de0b6b3a7640000600a548360200151610fb091906124d7565b610fba91906124ee565b60035490915081811015610fe157604051635492981d60e01b815260040160405180910390fd5b5f82600354610ff091906124c4565b6002549091505f5b81811015611148575f600282815481106110145761101461246b565b5f918252602080832090910154898201516001600160a01b039091168084526001909252604090922054909250869161104c916124d7565b61105691906124ee565b6001600160a01b0382165f908152600560205260408120805490919061107d908490612493565b90915550506001600160a01b0381165f90815260016020526040902054856110a586836124d7565b6110af91906124ee565b6001600160a01b0383165f908152600160205260408120829055906110d490836124c4565b90508060035f8282546110e791906124c4565b90915550506001600160a01b0383165f9081526001602052604090205415801561112657506001600160a01b0383165f90815260056020526040902054155b156111345761113483612098565b50505080611141906124ac565b9050610ff8565b5060035461115690836124c4565b60045f82825461116691906124c4565b90915550506001600160a01b0386165f9081526008602052604081208181556001015561119286611ef7565b856001600160a01b03167fbfb9d2d37b2bc565c5ad7f31d7d761ef5289234396d33c3e6ad2ef1a8a86b2bb8660200151866040516111da929190918252602082015260400190565b60405180910390a2505050505b5050565b6007545f90429082905b80841015611358575f600785815481106112115761121161246b565b5f9182526020808320909101546001600160a01b03168083526006825260409283902083518085019094528054808552600191909101549284019290925292508510611345576001600160a01b0382165f9081526001602052604081205490036112c057600280546001810182555f919091527f405787fa12a823e0f2b7631cc41b3ba8828b3321ca811111fa75cd3aa3bb5ace0180546001600160a01b0319166001600160a01b0384161790555b806020015160015f846001600160a01b03166001600160a01b031681526020019081526020015f205f8282546112f69190612493565b9091555050602081015161130a9085612493565b6001600160a01b0383165f908152600660205260408120818155600101559350611333866120f4565b61133e6001846124c4565b9250611351565b61134e866124ac565b95505b50506111f5565b8160035f8282546113699190612493565b925050819055508160045f8282546113819190612493565b909155505050505050565b611394611ecb565b600b55565b6001600160a01b0381165f908152600860205260408120548082036113c057505f92915050565b804211156113d15750600192915050565b505f92915050565b6113e1611ecb565b6113ea5f6121c0565b565b6113f4611ecb565b600c55565b6001600160a01b0381165f90815260016020526040812054600454829182911161144a576001600160a01b0384165f908152600660205260409020600101546004546114459190612493565b61147a565b6001600160a01b0384165f90815260066020908152604080832060019081015492529091205461147a9190612493565b6001600160a01b039094165f90815260056020526040902054939492505050565b335f90815260016020818152604080842054600690925283209091015490916114c48284612493565b335f90815260056020526040902054909150811580156114e2575080155b1561150057604051630686827b60e51b815260040160405180910390fd5b60045484111561152357604051631d620e8760e31b815260040160405180910390fd5b811561167e57821561154d57335f8181526006602052604081208181556001015561154d9061220f565b8360035f82825461155e91906124c4565b925050819055508360045f82825461157691906124c4565b9091555050335f81815260016020526040808220919091555163a9059cbb60e01b81526004810191909152602481018390526001600160a01b037f0000000000000000000000000000000000000000000000000000000000000000169063a9059cbb906044016020604051808303815f875af11580156115f8573d5f803e3d5ffd5b505050506040513d601f19601f8201168201806040525081019061161c9190612435565b506040518281526001600160a01b037f0000000000000000000000000000000000000000000000000000000000000000169033907f9b1bfa7fa9ee420a16e124f794c35ac9f90472acc99140eb2f6447c714cad8eb9060200160405180910390a35b801561178757335f81815260056020526040808220919091555163a9059cbb60e01b81526004810191909152602481018290526001600160a01b037f0000000000000000000000000000000000000000000000000000000000000000169063a9059cbb906044016020604051808303815f875af1158015611701573d5f803e3d5ffd5b505050506040513d601f19601f820116820180604052508101906117259190612435565b506040518181526001600160a01b037f0000000000000000000000000000000000000000000000000000000000000000169033907f9b1bfa7fa9ee420a16e124f794c35ac9f90472acc99140eb2f6447c714cad8eb9060200160405180910390a35b61179033612098565b50505050565b61179e611ecb565b600e805460ff19166001179055565b6117b5611ecb565b600d55565b60098181548110610d23575f80fd5b600e5460ff16156117ed576040516378bbd0ff60e11b815260040160405180910390fd5b335f908152600860205260409020600101541561181d57604051636cfe452760e11b815260040160405180910390fd5b6118256111eb565b5f670de0b6b3a7640000600a548361183d91906124d7565b61184791906124ee565b905060045481111561186c57604051632505c72b60e21b815260040160405180910390fd5b6040516323b872dd60e01b8152336004820152306024820152604481018390527f00000000000000000000000000000000000000000000000000000000000000006001600160a01b0316906323b872dd906064016020604051808303815f875af11580156118dc573d5f803e3d5ffd5b505050506040513d601f19601f820116820180604052508101906119009190612435565b5060405163a9059cbb60e01b8152336004820152602481018290527f00000000000000000000000000000000000000000000000000000000000000006001600160a01b03169063a9059cbb906044016020604051808303815f875af115801561196b573d5f803e3d5ffd5b505050506040513d601f19601f8201168201806040525081019061198f9190612435565b505f600c544261199f9190612493565b335f818152600860205260408120838155600190810187905560098054918201815582527f6e1540171b6c0c960b71a7020d9f60077f6af931a8bbf590da0223dacf75c7af0180546001600160a01b031916909217909155600480549293508492909190611a0e9084906124c4565b9091555050604080518481526020810184905233917f7b8d1106b7737736b442df90429adac60d78d42c6ca5987bf19db3733277f1d4910160405180910390a2505050565b7f00000000000000000000000000000000000000000000000000000000000000006001600160a01b0316826001600160a01b031603611c4b57335f908152600160208181526040808420546006909252909220015482611ab38284612493565b1015611ad257604051631e9acf1760e31b815260040160405180910390fd5b808310611b8e575f611ae482856124c4565b9050806004541015611b0957604051631d620e8760e31b815260040160405180910390fd5b335f9081526001602052604081208054839290611b279084906124c4565b9091555050808303611b3c57611b3c33612098565b8060035f828254611b4d91906124c4565b925050819055508060045f828254611b6591906124c4565b9091555050335f81815260066020526040812081815560010155611b889061220f565b50611bb5565b335f9081526006602052604081206001018054859290611baf9084906124c4565b90915550505b60405163a9059cbb60e01b8152336004820152602481018490527f00000000000000000000000000000000000000000000000000000000000000006001600160a01b03169063a9059cbb906044016020604051808303815f875af1158015611c1f573d5f803e3d5ffd5b505050506040513d601f19601f82011682018060405250810190611c439190612435565b505050611d83565b7f00000000000000000000000000000000000000000000000000000000000000006001600160a01b0316826001600160a01b031603611d6a57335f90815260056020526040902054811115611cb357604051631e9acf1760e31b815260040160405180910390fd5b335f9081526005602052604081208054839290611cd19084906124c4565b909155505060405163a9059cbb60e01b8152336004820152602481018290527f00000000000000000000000000000000000000000000000000000000000000006001600160a01b03169063a9059cbb906044016020604051808303815f875af1158015611d40573d5f803e3d5ffd5b505050506040513d601f19601f82011682018060405250810190611d649190612435565b50611d83565b60405163c1ab6dc160e01b815260040160405180910390fd5b6040518181526001600160a01b0383169033907f9b1bfa7fa9ee420a16e124f794c35ac9f90472acc99140eb2f6447c714cad8eb9060200160405180910390a35050565b6007545f908142815b83811015611e735760065f60078381548110611dee57611dee61246b565b5f9182526020808320909101546001600160a01b03168352820192909252604001902054821115611e635760065f60078381548110611e2f57611e2f61246b565b5f9182526020808320909101546001600160a01b03168352820192909252604001902060010154611e609084612493565b92505b611e6c816124ac565b9050611dd0565b50600454611e819083612493565b935050505090565b611e91611ecb565b6001600160a01b038116611ebf57604051631e4fbdf760e01b81525f60048201526024015b60405180910390fd5b611ec8816121c0565b50565b5f546001600160a01b031633146113ea5760405163118cdaa760e01b8152336004820152602401611eb6565b6009545f5b8181101561067c57826001600160a01b031660098281548110611f2157611f2161246b565b5f918252602090912001546001600160a01b031603611f435761067c8161226b565b611f4c816124ac565b9050611efc565b6001600160a01b0381165f908152600660209081526040808320815180830190925280548252600101549181018290529103611f8d575050565b80514290811061067c576001600160a01b0383165f90815260016020526040812054900361200057600280546001810182555f919091527f405787fa12a823e0f2b7631cc41b3ba8828b3321ca811111fa75cd3aa3bb5ace0180546001600160a01b0319166001600160a01b0385161790555b816020015160015f856001600160a01b03166001600160a01b031681526020019081526020015f205f8282546120369190612493565b90915550506020820151600380545f90612051908490612493565b90915550506020820151600480545f9061206c908490612493565b90915550506001600160a01b0383165f9081526006602052604081208181556001015561067c8361220f565b6002545f5b8181101561067c57826001600160a01b0316600282815481106120c2576120c261246b565b5f918252602090912001546001600160a01b0316036120e45761067c81612313565b6120ed816124ac565b905061209d565b60075480821115612118576040516363df817160e01b815260040160405180910390fd5b60076121256001836124c4565b815481106121355761213561246b565b5f91825260209091200154600780546001600160a01b0390921691849081106121605761216061246b565b905f5260205f20015f6101000a8154816001600160a01b0302191690836001600160a01b03160217905550600780548061219c5761219c61250d565b5f8281526020902081015f1990810180546001600160a01b03191690550190555050565b5f80546001600160a01b038381166001600160a01b0319831681178455604051919092169283917f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e09190a35050565b6007545f5b8181101561067c57826001600160a01b0316600782815481106122395761223961246b565b5f918252602090912001546001600160a01b03160361225b5761067c816120f4565b612264816124ac565b9050612214565b6009548082111561228f576040516363df817160e01b815260040160405180910390fd5b600961229c6001836124c4565b815481106122ac576122ac61246b565b5f91825260209091200154600980546001600160a01b0390921691849081106122d7576122d761246b565b905f5260205f20015f6101000a8154816001600160a01b0302191690836001600160a01b03160217905550600980548061219c5761219c61250d565b60025480821115612337576040516363df817160e01b815260040160405180910390fd5b60026123446001836124c4565b815481106123545761235461246b565b5f91825260209091200154600280546001600160a01b03909216918490811061237f5761237f61246b565b905f5260205f20015f6101000a8154816001600160a01b0302191690836001600160a01b03160217905550600280548061219c5761219c61250d565b5f602082840312156123cb575f80fd5b5035919050565b80356001600160a01b03811681146123e8575f80fd5b919050565b5f602082840312156123fd575f80fd5b612406826123d2565b9392505050565b5f806040838503121561241e575f80fd5b612427836123d2565b946020939093013593505050565b5f60208284031215612445575f80fd5b81518015158114612406575f80fd5b5f60208284031215612464575f80fd5b5051919050565b634e487b7160e01b5f52603260045260245ffd5b634e487b7160e01b5f52601160045260245ffd5b808201808211156124a6576124a661247f565b92915050565b5f600182016124bd576124bd61247f565b5060010190565b818103818111156124a6576124a661247f565b80820281158282048414176124a6576124a661247f565b5f8261250857634e487b7160e01b5f52601260045260245ffd5b500490565b634e487b7160e01b5f52603160045260245ffdfea264697066735822122080255cb23ac42565067335ec9bd0e6ccda2b470583d6a896cccca935fb61bb2e64736f6c634300081500330000000000000000000000006b175474e89094c44da98b954eedeac495271d0f0000000000000000000000000ab87046fbb341d058f17cbc4c1133f25a20a52f0000000000000000000000000000000000000000000000056bc75e2d6310000000000000000000000000000000000000000000000000000595698248593c000000000000000000000000000000000000000000000000000000000000000151800000000000000000000000000000000000000000000000000000000000000e10
Deployed Bytecode
0x608060405234801561000f575f80fd5b506004361061023f575f3560e01c80636b2fa374116101355780639f8b20a5116100b4578063d8ae6f3111610079578063d8ae6f31146104da578063ecdae41b146104e2578063ee2ae94114610508578063f2fde38b14610515578063f8c716f914610528575f80fd5b80639f8b20a514610493578063aaf5eb68146104a6578063b60a266c146104b5578063b9bfebfb146104be578063c1075329146104c7575f80fd5b8063853828b6116100fa578063853828b61461044d5780638da5cb5b1461045557806394c174be1461046557806398e1e2b71461046d57806398f1196c14610480575f80fd5b80636b2fa374146103d9578063715018a61461040057806379c4855c146104085780638108add41461041b578063843592d31461043a575f80fd5b8063401d4482116101c157806360a859991161018657806360a859991461036a578063626f56f814610391578063631d54df146103a457806364e5a951146103c75780636a50c448146103d0575f80fd5b8063401d44821461030d57806350e041ea146103205780635d24b1e9146103335780635d73aa53146103595780635dc172ac14610362575f80fd5b806320cdc7b31161020757806320cdc7b31461029d578063252323e4146102a557806327163639146102bc5780632b4ccdb4146102e75780633b76594d146102fa575f80fd5b806301ea9bcd1461024357806304f6fe511461024d5780630a8239511461026f5780630dcaf64b146102775780631322adac1461028a575b5f80fd5b61024b610547565b005b610255610681565b604080519283526020830191909152015b60405180910390f35b61024b610912565b61024b6102853660046123bb565b610926565b6102556102983660046123ed565b610933565b61024b610984565b6102ae60045481565b604051908152602001610266565b6102cf6102ca3660046123bb565b610d14565b6040516001600160a01b039091168152602001610266565b6102cf6102f53660046123bb565b610d3c565b61024b6103083660046123bb565b610d4b565b61024b61031b36600461240d565b610ed4565b61024b61032e3660046123ed565b610f0f565b6102556103413660046123ed565b60086020525f90815260409020805460019091015482565b6102ae600a5481565b61024b6111eb565b6102cf7f0000000000000000000000000ab87046fbb341d058f17cbc4c1133f25a20a52f81565b61024b61039f3660046123bb565b61138c565b6103b76103b23660046123ed565b611399565b6040519015158152602001610266565b6102ae60035481565b6102ae600b5481565b6102cf7f0000000000000000000000006b175474e89094c44da98b954eedeac495271d0f81565b61024b6113d9565b61024b6104163660046123bb565b6113ec565b6102ae6104293660046123ed565b60016020525f908152604090205481565b6102556104483660046123ed565b6113f9565b61024b61149b565b5f546001600160a01b03166102cf565b61024b611796565b61024b61047b3660046123bb565b6117ad565b6102cf61048e3660046123bb565b6117ba565b61024b6104a13660046123bb565b6117c9565b6102ae670de0b6b3a764000081565b6102ae600d5481565b6102ae600c5481565b61024b6104d536600461240d565b611a53565b6102ae611dc7565b6102556104f03660046123ed565b60066020525f90815260409020805460019091015482565b600e546103b79060ff1681565b61024b6105233660046123ed565b611e89565b6102ae6105363660046123ed565b60056020525f908152604090205481565b61054f611ecb565b5f80610559610681565b60405163a9059cbb60e01b81523360048201526024810183905291935091507f0000000000000000000000006b175474e89094c44da98b954eedeac495271d0f6001600160a01b03169063a9059cbb906044016020604051808303815f875af11580156105c8573d5f803e3d5ffd5b505050506040513d601f19601f820116820180604052508101906105ec9190612435565b5060405163a9059cbb60e01b8152336004820152602481018290527f0000000000000000000000000ab87046fbb341d058f17cbc4c1133f25a20a52f6001600160a01b03169063a9059cbb906044015b6020604051808303815f875af1158015610658573d5f803e3d5ffd5b505050506040513d601f19601f8201168201806040525081019061067c9190612435565b505050565b6040516370a0823160e01b81523060048201525f90819081906001600160a01b037f0000000000000000000000006b175474e89094c44da98b954eedeac495271d0f16906370a0823190602401602060405180830381865afa1580156106e9573d5f803e3d5ffd5b505050506040513d601f19601f8201168201806040525081019061070d9190612454565b6007549091505f805b828110156107775760065f600783815481106107345761073461246b565b5f9182526020808320909101546001600160a01b031683528201929092526040019020600101546107659083612493565b9150610770816124ac565b9050610716565b505f816004548561078891906124c4565b61079291906124c4565b6009549091505f805b828110156107fc5760085f600983815481106107b9576107b961246b565b5f9182526020808320909101546001600160a01b031683528201929092526040019020600101546107ea9083612493565b91506107f5816124ac565b905061079b565b506002545f805b828110156108615760055f600283815481106108215761082161246b565b5f9182526020808320909101546001600160a01b0316835282019290925260400190205461084f9083612493565b915061085a816124ac565b9050610803565b506040516370a0823160e01b81523060048201528590829085906001600160a01b037f0000000000000000000000000ab87046fbb341d058f17cbc4c1133f25a20a52f16906370a0823190602401602060405180830381865afa1580156108ca573d5f803e3d5ffd5b505050506040513d601f19601f820116820180604052508101906108ee9190612454565b6108f891906124c4565b61090291906124c4565b9950995050505050505050509091565b61091a611ecb565b600e805460ff19169055565b61092e611ecb565b600a55565b6001600160a01b0381165f908152600160208181526040808420546006909252832090910154829161096491612493565b6001600160a01b039093165f908152600560205260409020549293915050565b335f908152600860209081526040808320815180830190925280548083526001909101549282019290925291036109ce5760405163f71855a360e01b815260040160405180910390fd5b80514211156109f057604051633e6ba29f60e11b815260040160405180910390fd5b5f670de0b6b3a7640000600b548360200151610a0c91906124d7565b610a1691906124ee565b6040516323b872dd60e01b8152336004820152306024820152604481018290529091507f0000000000000000000000006b175474e89094c44da98b954eedeac495271d0f6001600160a01b0316906323b872dd906064016020604051808303815f875af1158015610a89573d5f803e3d5ffd5b505050506040513d601f19601f82011682018060405250810190610aad9190612435565b50602082015160405163a9059cbb60e01b815233600482015260248101919091527f0000000000000000000000000ab87046fbb341d058f17cbc4c1133f25a20a52f6001600160a01b03169063a9059cbb906044016020604051808303815f875af1158015610b1e573d5f803e3d5ffd5b505050506040513d601f19601f82011682018060405250810190610b429190612435565b50610b4b6111eb565b5f670de0b6b3a7640000600a548460200151610b6791906124d7565b610b7191906124ee565b610b7b90836124c4565b600354600254919250905f5b81811015610c5c575f838560015f60028681548110610ba857610ba861246b565b5f9182526020808320909101546001600160a01b03168352820192909252604001902054610bd691906124d7565b610be091906124ee565b90508060015f60028581548110610bf957610bf961246b565b5f9182526020808320909101546001600160a01b0316835282019290925260400181208054909190610c2c908490612493565b925050819055508060035f828254610c449190612493565b90915550610c5591508290506124ac565b9050610b87565b5081600354670de0b6b3a7640000600a548860200151610c7c91906124d7565b610c8691906124ee565b610c909190612493565b610c9a91906124c4565b60045f828254610caa9190612493565b9091555050335f81815260086020526040812081815560010155610ccd90611ef7565b6020808601516040805191825291810186905233917f4ad4b82ea376b5cd1bddcd46a1a6885f09a01839ffed08a89603082412251d5d910160405180910390a25050505050565b60078181548110610d23575f80fd5b5f918252602090912001546001600160a01b0316905081565b60028181548110610d23575f80fd5b6040516323b872dd60e01b8152336004820152306024820152604481018290527f0000000000000000000000006b175474e89094c44da98b954eedeac495271d0f6001600160a01b0316906323b872dd906064016020604051808303815f875af1158015610dbb573d5f803e3d5ffd5b505050506040513d601f19601f82011682018060405250810190610ddf9190612435565b50610de933611f53565b425f829003610df6575050565b335f908152600660205260408120600101549003610e5057600780546001810182555f919091527fa66cc928b5edb82af9bd49922954155ab7b0942694bea4ce44661d9a8736c6880180546001600160a01b031916331790555b335f9081526006602052604081206001018054849290610e71908490612493565b9091555050600d54610e839082612493565b335f81815260066020526040908190209290925590517fe1fffcc4923d04b559f4d29a8bfc6cda04eb5b0d3c460751c2402c5c5cc9109c90610ec89085815260200190565b60405180910390a25050565b610edc611ecb565b60405163a9059cbb60e01b8152336004820152602481018290526001600160a01b0383169063a9059cbb9060440161063c565b6001600160a01b0381165f90815260086020908152604080832081518083019092528054808352600190910154928201929092529103610f625760405163f360fff160e01b815260040160405180910390fd5b80514211610f835760405163267ecc6560e21b815260040160405180910390fd5b80514211156111e757610f946111eb565b5f670de0b6b3a7640000600a548360200151610fb091906124d7565b610fba91906124ee565b60035490915081811015610fe157604051635492981d60e01b815260040160405180910390fd5b5f82600354610ff091906124c4565b6002549091505f5b81811015611148575f600282815481106110145761101461246b565b5f918252602080832090910154898201516001600160a01b039091168084526001909252604090922054909250869161104c916124d7565b61105691906124ee565b6001600160a01b0382165f908152600560205260408120805490919061107d908490612493565b90915550506001600160a01b0381165f90815260016020526040902054856110a586836124d7565b6110af91906124ee565b6001600160a01b0383165f908152600160205260408120829055906110d490836124c4565b90508060035f8282546110e791906124c4565b90915550506001600160a01b0383165f9081526001602052604090205415801561112657506001600160a01b0383165f90815260056020526040902054155b156111345761113483612098565b50505080611141906124ac565b9050610ff8565b5060035461115690836124c4565b60045f82825461116691906124c4565b90915550506001600160a01b0386165f9081526008602052604081208181556001015561119286611ef7565b856001600160a01b03167fbfb9d2d37b2bc565c5ad7f31d7d761ef5289234396d33c3e6ad2ef1a8a86b2bb8660200151866040516111da929190918252602082015260400190565b60405180910390a2505050505b5050565b6007545f90429082905b80841015611358575f600785815481106112115761121161246b565b5f9182526020808320909101546001600160a01b03168083526006825260409283902083518085019094528054808552600191909101549284019290925292508510611345576001600160a01b0382165f9081526001602052604081205490036112c057600280546001810182555f919091527f405787fa12a823e0f2b7631cc41b3ba8828b3321ca811111fa75cd3aa3bb5ace0180546001600160a01b0319166001600160a01b0384161790555b806020015160015f846001600160a01b03166001600160a01b031681526020019081526020015f205f8282546112f69190612493565b9091555050602081015161130a9085612493565b6001600160a01b0383165f908152600660205260408120818155600101559350611333866120f4565b61133e6001846124c4565b9250611351565b61134e866124ac565b95505b50506111f5565b8160035f8282546113699190612493565b925050819055508160045f8282546113819190612493565b909155505050505050565b611394611ecb565b600b55565b6001600160a01b0381165f908152600860205260408120548082036113c057505f92915050565b804211156113d15750600192915050565b505f92915050565b6113e1611ecb565b6113ea5f6121c0565b565b6113f4611ecb565b600c55565b6001600160a01b0381165f90815260016020526040812054600454829182911161144a576001600160a01b0384165f908152600660205260409020600101546004546114459190612493565b61147a565b6001600160a01b0384165f90815260066020908152604080832060019081015492529091205461147a9190612493565b6001600160a01b039094165f90815260056020526040902054939492505050565b335f90815260016020818152604080842054600690925283209091015490916114c48284612493565b335f90815260056020526040902054909150811580156114e2575080155b1561150057604051630686827b60e51b815260040160405180910390fd5b60045484111561152357604051631d620e8760e31b815260040160405180910390fd5b811561167e57821561154d57335f8181526006602052604081208181556001015561154d9061220f565b8360035f82825461155e91906124c4565b925050819055508360045f82825461157691906124c4565b9091555050335f81815260016020526040808220919091555163a9059cbb60e01b81526004810191909152602481018390526001600160a01b037f0000000000000000000000006b175474e89094c44da98b954eedeac495271d0f169063a9059cbb906044016020604051808303815f875af11580156115f8573d5f803e3d5ffd5b505050506040513d601f19601f8201168201806040525081019061161c9190612435565b506040518281526001600160a01b037f0000000000000000000000006b175474e89094c44da98b954eedeac495271d0f169033907f9b1bfa7fa9ee420a16e124f794c35ac9f90472acc99140eb2f6447c714cad8eb9060200160405180910390a35b801561178757335f81815260056020526040808220919091555163a9059cbb60e01b81526004810191909152602481018290526001600160a01b037f0000000000000000000000000ab87046fbb341d058f17cbc4c1133f25a20a52f169063a9059cbb906044016020604051808303815f875af1158015611701573d5f803e3d5ffd5b505050506040513d601f19601f820116820180604052508101906117259190612435565b506040518181526001600160a01b037f0000000000000000000000000ab87046fbb341d058f17cbc4c1133f25a20a52f169033907f9b1bfa7fa9ee420a16e124f794c35ac9f90472acc99140eb2f6447c714cad8eb9060200160405180910390a35b61179033612098565b50505050565b61179e611ecb565b600e805460ff19166001179055565b6117b5611ecb565b600d55565b60098181548110610d23575f80fd5b600e5460ff16156117ed576040516378bbd0ff60e11b815260040160405180910390fd5b335f908152600860205260409020600101541561181d57604051636cfe452760e11b815260040160405180910390fd5b6118256111eb565b5f670de0b6b3a7640000600a548361183d91906124d7565b61184791906124ee565b905060045481111561186c57604051632505c72b60e21b815260040160405180910390fd5b6040516323b872dd60e01b8152336004820152306024820152604481018390527f0000000000000000000000000ab87046fbb341d058f17cbc4c1133f25a20a52f6001600160a01b0316906323b872dd906064016020604051808303815f875af11580156118dc573d5f803e3d5ffd5b505050506040513d601f19601f820116820180604052508101906119009190612435565b5060405163a9059cbb60e01b8152336004820152602481018290527f0000000000000000000000006b175474e89094c44da98b954eedeac495271d0f6001600160a01b03169063a9059cbb906044016020604051808303815f875af115801561196b573d5f803e3d5ffd5b505050506040513d601f19601f8201168201806040525081019061198f9190612435565b505f600c544261199f9190612493565b335f818152600860205260408120838155600190810187905560098054918201815582527f6e1540171b6c0c960b71a7020d9f60077f6af931a8bbf590da0223dacf75c7af0180546001600160a01b031916909217909155600480549293508492909190611a0e9084906124c4565b9091555050604080518481526020810184905233917f7b8d1106b7737736b442df90429adac60d78d42c6ca5987bf19db3733277f1d4910160405180910390a2505050565b7f0000000000000000000000006b175474e89094c44da98b954eedeac495271d0f6001600160a01b0316826001600160a01b031603611c4b57335f908152600160208181526040808420546006909252909220015482611ab38284612493565b1015611ad257604051631e9acf1760e31b815260040160405180910390fd5b808310611b8e575f611ae482856124c4565b9050806004541015611b0957604051631d620e8760e31b815260040160405180910390fd5b335f9081526001602052604081208054839290611b279084906124c4565b9091555050808303611b3c57611b3c33612098565b8060035f828254611b4d91906124c4565b925050819055508060045f828254611b6591906124c4565b9091555050335f81815260066020526040812081815560010155611b889061220f565b50611bb5565b335f9081526006602052604081206001018054859290611baf9084906124c4565b90915550505b60405163a9059cbb60e01b8152336004820152602481018490527f0000000000000000000000006b175474e89094c44da98b954eedeac495271d0f6001600160a01b03169063a9059cbb906044016020604051808303815f875af1158015611c1f573d5f803e3d5ffd5b505050506040513d601f19601f82011682018060405250810190611c439190612435565b505050611d83565b7f0000000000000000000000000ab87046fbb341d058f17cbc4c1133f25a20a52f6001600160a01b0316826001600160a01b031603611d6a57335f90815260056020526040902054811115611cb357604051631e9acf1760e31b815260040160405180910390fd5b335f9081526005602052604081208054839290611cd19084906124c4565b909155505060405163a9059cbb60e01b8152336004820152602481018290527f0000000000000000000000000ab87046fbb341d058f17cbc4c1133f25a20a52f6001600160a01b03169063a9059cbb906044016020604051808303815f875af1158015611d40573d5f803e3d5ffd5b505050506040513d601f19601f82011682018060405250810190611d649190612435565b50611d83565b60405163c1ab6dc160e01b815260040160405180910390fd5b6040518181526001600160a01b0383169033907f9b1bfa7fa9ee420a16e124f794c35ac9f90472acc99140eb2f6447c714cad8eb9060200160405180910390a35050565b6007545f908142815b83811015611e735760065f60078381548110611dee57611dee61246b565b5f9182526020808320909101546001600160a01b03168352820192909252604001902054821115611e635760065f60078381548110611e2f57611e2f61246b565b5f9182526020808320909101546001600160a01b03168352820192909252604001902060010154611e609084612493565b92505b611e6c816124ac565b9050611dd0565b50600454611e819083612493565b935050505090565b611e91611ecb565b6001600160a01b038116611ebf57604051631e4fbdf760e01b81525f60048201526024015b60405180910390fd5b611ec8816121c0565b50565b5f546001600160a01b031633146113ea5760405163118cdaa760e01b8152336004820152602401611eb6565b6009545f5b8181101561067c57826001600160a01b031660098281548110611f2157611f2161246b565b5f918252602090912001546001600160a01b031603611f435761067c8161226b565b611f4c816124ac565b9050611efc565b6001600160a01b0381165f908152600660209081526040808320815180830190925280548252600101549181018290529103611f8d575050565b80514290811061067c576001600160a01b0383165f90815260016020526040812054900361200057600280546001810182555f919091527f405787fa12a823e0f2b7631cc41b3ba8828b3321ca811111fa75cd3aa3bb5ace0180546001600160a01b0319166001600160a01b0385161790555b816020015160015f856001600160a01b03166001600160a01b031681526020019081526020015f205f8282546120369190612493565b90915550506020820151600380545f90612051908490612493565b90915550506020820151600480545f9061206c908490612493565b90915550506001600160a01b0383165f9081526006602052604081208181556001015561067c8361220f565b6002545f5b8181101561067c57826001600160a01b0316600282815481106120c2576120c261246b565b5f918252602090912001546001600160a01b0316036120e45761067c81612313565b6120ed816124ac565b905061209d565b60075480821115612118576040516363df817160e01b815260040160405180910390fd5b60076121256001836124c4565b815481106121355761213561246b565b5f91825260209091200154600780546001600160a01b0390921691849081106121605761216061246b565b905f5260205f20015f6101000a8154816001600160a01b0302191690836001600160a01b03160217905550600780548061219c5761219c61250d565b5f8281526020902081015f1990810180546001600160a01b03191690550190555050565b5f80546001600160a01b038381166001600160a01b0319831681178455604051919092169283917f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e09190a35050565b6007545f5b8181101561067c57826001600160a01b0316600782815481106122395761223961246b565b5f918252602090912001546001600160a01b03160361225b5761067c816120f4565b612264816124ac565b9050612214565b6009548082111561228f576040516363df817160e01b815260040160405180910390fd5b600961229c6001836124c4565b815481106122ac576122ac61246b565b5f91825260209091200154600980546001600160a01b0390921691849081106122d7576122d761246b565b905f5260205f20015f6101000a8154816001600160a01b0302191690836001600160a01b03160217905550600980548061219c5761219c61250d565b60025480821115612337576040516363df817160e01b815260040160405180910390fd5b60026123446001836124c4565b815481106123545761235461246b565b5f91825260209091200154600280546001600160a01b03909216918490811061237f5761237f61246b565b905f5260205f20015f6101000a8154816001600160a01b0302191690836001600160a01b03160217905550600280548061219c5761219c61250d565b5f602082840312156123cb575f80fd5b5035919050565b80356001600160a01b03811681146123e8575f80fd5b919050565b5f602082840312156123fd575f80fd5b612406826123d2565b9392505050565b5f806040838503121561241e575f80fd5b612427836123d2565b946020939093013593505050565b5f60208284031215612445575f80fd5b81518015158114612406575f80fd5b5f60208284031215612464575f80fd5b5051919050565b634e487b7160e01b5f52603260045260245ffd5b634e487b7160e01b5f52601160045260245ffd5b808201808211156124a6576124a661247f565b92915050565b5f600182016124bd576124bd61247f565b5060010190565b818103818111156124a6576124a661247f565b80820281158282048414176124a6576124a661247f565b5f8261250857634e487b7160e01b5f52601260045260245ffd5b500490565b634e487b7160e01b5f52603160045260245ffdfea264697066735822122080255cb23ac42565067335ec9bd0e6ccda2b470583d6a896cccca935fb61bb2e64736f6c63430008150033
Constructor Arguments (ABI-Encoded and is the last bytes of the Contract Creation Code above)
0000000000000000000000006b175474e89094c44da98b954eedeac495271d0f0000000000000000000000000ab87046fbb341d058f17cbc4c1133f25a20a52f0000000000000000000000000000000000000000000000056bc75e2d6310000000000000000000000000000000000000000000000000000595698248593c000000000000000000000000000000000000000000000000000000000000000151800000000000000000000000000000000000000000000000000000000000000e10
-----Decoded View---------------
Arg [0] : _currencyToken (address): 0x6B175474E89094C44Da98b954EedeAC495271d0F
Arg [1] : _repoToken (address): 0x0ab87046fBb341D058F17CBC4c1133F25a20a52f
Arg [2] : _repoSellPrice (uint256): 100000000000000000000
Arg [3] : _repoBuybackPrice (uint256): 103000000000000000000
Arg [4] : _repoTimeLength (uint256): 86400
Arg [5] : _pendingTime (uint256): 3600
-----Encoded View---------------
6 Constructor Arguments found :
Arg [0] : 0000000000000000000000006b175474e89094c44da98b954eedeac495271d0f
Arg [1] : 0000000000000000000000000ab87046fbb341d058f17cbc4c1133f25a20a52f
Arg [2] : 0000000000000000000000000000000000000000000000056bc75e2d63100000
Arg [3] : 00000000000000000000000000000000000000000000000595698248593c0000
Arg [4] : 0000000000000000000000000000000000000000000000000000000000015180
Arg [5] : 0000000000000000000000000000000000000000000000000000000000000e10
Loading...
Loading
Loading...
Loading
Multichain Portfolio | 34 Chains
Chain | Token | Portfolio % | Price | Amount | Value |
---|
Loading...
Loading
Loading...
Loading
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.