Feature Tip: Add private address tag to any address under My Name Tag !
Overview
ETH Balance
0 ETH
Eth Value
$0.00More Info
Private Name Tags
ContractCreator
Latest 25 from a total of 185 transactions
Transaction Hash |
Method
|
Block
|
From
|
To
|
|||||
---|---|---|---|---|---|---|---|---|---|
Swap | 19452580 | 121 days ago | IN | 0.004 ETH | 0.00859037 | ||||
Swap | 17803860 | 352 days ago | IN | 0 ETH | 0.00285487 | ||||
Swap | 17547652 | 388 days ago | IN | 0 ETH | 0.00366482 | ||||
Swap | 17376513 | 412 days ago | IN | 0 ETH | 0.00430892 | ||||
Swap | 17282470 | 425 days ago | IN | 0 ETH | 0.00957814 | ||||
Swap | 17279878 | 425 days ago | IN | 0 ETH | 0.00932633 | ||||
Swap | 17279639 | 425 days ago | IN | 0 ETH | 0.01341254 | ||||
Swap | 17279622 | 425 days ago | IN | 0 ETH | 0.01090708 | ||||
Swap | 17279610 | 425 days ago | IN | 0 ETH | 0.01104414 | ||||
Transfer Ownersh... | 17279223 | 425 days ago | IN | 0 ETH | 0.00150112 | ||||
Swap | 17279205 | 425 days ago | IN | 0 ETH | 0.00820773 | ||||
Swap | 17234593 | 432 days ago | IN | 0 ETH | 0.01322768 | ||||
Swap | 17222478 | 433 days ago | IN | 0 ETH | 0.00952072 | ||||
Swap | 17156715 | 443 days ago | IN | 0 ETH | 0.00876065 | ||||
Swap | 17137856 | 445 days ago | IN | 0 ETH | 0.00909007 | ||||
Swap | 17135352 | 446 days ago | IN | 0 ETH | 0.00589776 | ||||
Swap | 17134517 | 446 days ago | IN | 0.007 ETH | 0.007771 | ||||
Swap | 17134510 | 446 days ago | IN | 0 ETH | 0.0057324 | ||||
Swap | 17133132 | 446 days ago | IN | 0 ETH | 0.00547061 | ||||
Swap | 17133097 | 446 days ago | IN | 0 ETH | 0.00716232 | ||||
Swap | 17132924 | 446 days ago | IN | 0 ETH | 0.00855096 | ||||
Swap | 17132815 | 446 days ago | IN | 0 ETH | 0.00693407 | ||||
Swap | 17132752 | 446 days ago | IN | 0 ETH | 0.00907719 | ||||
Swap | 17132675 | 446 days ago | IN | 0 ETH | 0.01326609 | ||||
Swap | 17132618 | 446 days ago | IN | 0 ETH | 0.00868586 |
Latest 25 internal transactions (View All)
Advanced mode:
Parent Transaction Hash | Block | From | To | |||
---|---|---|---|---|---|---|
19452580 | 121 days ago | 0.004 ETH | ||||
17547652 | 388 days ago | 0.01223591 ETH | ||||
17547652 | 388 days ago | 0.01223591 ETH | ||||
17279639 | 425 days ago | 0.06866517 ETH | ||||
17279639 | 425 days ago | 0.06866517 ETH | ||||
17222478 | 433 days ago | 0.0405526 ETH | ||||
17222478 | 433 days ago | 0.0405526 ETH | ||||
17137856 | 445 days ago | 0.05330609 ETH | ||||
17137856 | 445 days ago | 0.05330609 ETH | ||||
17134517 | 446 days ago | 0.007 ETH | ||||
17130783 | 446 days ago | 0.01499999 ETH | ||||
17122048 | 447 days ago | 0.0500758 ETH | ||||
17080637 | 453 days ago | 0.48039028 ETH | ||||
17080637 | 453 days ago | 0.48039028 ETH | ||||
17070187 | 455 days ago | 0.19217946 ETH | ||||
17070187 | 455 days ago | 0.19217946 ETH | ||||
17064651 | 456 days ago | 0.001 ETH | ||||
17002277 | 464 days ago | 0.004 ETH | ||||
16909758 | 478 days ago | 0.05689542 ETH | ||||
16909758 | 478 days ago | 0.05689542 ETH | ||||
16861201 | 484 days ago | 0.05612724 ETH | ||||
16861201 | 484 days ago | 0.05612724 ETH | ||||
16817486 | 491 days ago | 0.36425797 ETH | ||||
16817486 | 491 days ago | 0.36425797 ETH | ||||
16751228 | 500 days ago | 0.11111699 ETH |
Loading...
Loading
Contract Name:
NarfexExchangerRouter2
Compiler Version
v0.8.17+commit.8df45f5f
Optimization Enabled:
Yes with 200 runs
Other Settings:
default evmVersion
Contract Source Code (Solidity Standard Json-Input format)
//SPDX-License-Identifier: Unlicense pragma solidity ^0.8.17; import './PancakeLibrary.sol'; import './INarfexOracle.sol'; import "@openzeppelin/contracts/token/ERC20/utils/SafeERC20.sol"; import "@openzeppelin/contracts/access/Ownable.sol"; interface INarfexFiat is IERC20 { function burnFrom(address _address, uint _amount) external; function mintTo(address _address, uint _amount) external; } interface INarfexExchangerPool { function getBalance() external view returns (uint); function approveRouter() external; } interface IWBNB { function deposit() external payable; function transfer(address to, uint value) external returns (bool); function withdraw(uint) external; } /// @title DEX Router for Narfex Fiats /// @author Danil Sakhinov /// @dev Allows to exchange between fiats and crypto coins /// @dev Exchanges using USDT liquidity pool /// @dev Uses Narfex oracle to get commissions and prices /// @dev Supports tokens with a transfer fee contract NarfexExchangerRouter2 is Ownable { using Address for address; /// Structures for solving the problem of limiting the number of variables struct ExchangeData { uint rate; int commission; uint inAmountClear; uint outAmountClear; uint inAmount; uint outAmount; address commToken; int commAmount; uint referReward; int profitUSDT; } struct SwapData { address[] path; uint[] amounts; bool isExactOut; uint amount; uint inAmount; uint inAmountMax; uint outAmount; uint outAmountMin; uint deadline; address refer; } struct Token { address addr; bool isFiat; int commission; uint price; uint reward; uint transferFee; } IERC20 public USDT; IWBNB public WBNB; INarfexOracle public oracle; INarfexExchangerPool public pool; uint constant PRECISION = 10**18; uint private USDT_PRECISION = 10**6; uint constant PERCENT_PRECISION = 10**4; uint constant MAX_INT = 2**256 - 1; /// @param _oracleAddress NarfexOracle address /// @param _poolAddress NarfexExchangerPool address /// @param _usdtAddress USDT address /// @param _wbnbAddress WrapBNB address constructor ( address _oracleAddress, address _poolAddress, address _usdtAddress, address _wbnbAddress ) { oracle = INarfexOracle(_oracleAddress); USDT = IERC20(_usdtAddress); WBNB = IWBNB(_wbnbAddress); pool = INarfexExchangerPool(_poolAddress); if (block.chainid == 56 || block.chainid == 97) { USDT_PRECISION = 10**18; } } /// @notice Checking for an outdated transaction /// @param deadline Limit block timestamp modifier ensure(uint deadline) { require(deadline >= block.timestamp, "Transaction expired"); _; } event SwapFiat(address indexed _account, address _fromToken, address _toToken, ExchangeData _exchange); event SwapDEX(address indexed _account, address _fromToken, address _toToken, uint inAmount, uint outAmount); event ReferralReward(address _token, uint _amount, address indexed _receiver); /// @notice Default function for BNB receive. Accepts BNB only from WBNB contract receive() external payable { assert(msg.sender == address(WBNB)); } /// @notice Assigns token data from oracle to structure with token address /// @param addr Token address /// @param t Token data from the oracle /// @return New structure with addr function _assignTokenData(address addr, INarfexOracle.TokenData memory t) internal pure returns (Token memory) { return Token(addr, t.isFiat, t.commission, t.price, t.reward, t.transferFee); } /// @notice Returns the price of the token quantity in USDT equivalent /// @param _token Token address /// @param _amount Token amount /// @return USDT amount function _getUSDTValue(address _token, int _amount) internal view returns (int) { if (_amount == 0) return 0; uint uintValue = oracle.getPrice(_token) * uint(_amount) / USDT_PRECISION; return _amount >= 0 ? int(uintValue) : -int(uintValue); } /// @notice Calculates prices and commissions when exchanging with fiat /// @param A First token /// @param B Second token /// @param _amount The amount of one of the tokens. Depends on _isExactOut /// @param _isExactOut Is the specified amount an output value /// @dev The last parameter shows the direction of the exchange function _getExchangeValues(Token memory A, Token memory B, uint _amount, bool _isExactOut) internal view returns (ExchangeData memory exchange) { /// Calculate price { uint priceA = A.addr == address(USDT) ? USDT_PRECISION : A.price; uint priceB = B.addr == address(USDT) ? USDT_PRECISION : B.price; exchange.rate = priceA * USDT_PRECISION / priceB; } /// Calculate commission { int unit = int(PERCENT_PRECISION); exchange.commission = (A.commission + unit) * (B.commission + unit) / unit - unit; } /// Calculate clear amounts exchange.inAmountClear = _isExactOut ? _amount * USDT_PRECISION / exchange.rate : _amount; exchange.outAmountClear = _isExactOut ? _amount : _amount * exchange.rate / USDT_PRECISION; /// Calculate amounts with commission if (_isExactOut) { exchange.inAmount = exchange.inAmountClear * uint(int(PERCENT_PRECISION) + exchange.commission) / PERCENT_PRECISION; exchange.outAmount = _amount; } else { exchange.inAmount = _amount; exchange.outAmount = exchange.outAmountClear * uint(int(PERCENT_PRECISION) - exchange.commission) / PERCENT_PRECISION; } /// Calculate commission and profit amount exchange.commToken = A.isFiat ? A.addr : B.addr; exchange.commAmount = int(A.isFiat ? exchange.inAmount : exchange.outAmount) * exchange.commission / int(PERCENT_PRECISION); exchange.profitUSDT = _getUSDTValue(exchange.commToken, exchange.commAmount); } /// @notice Sends the referral agent his reward /// @param A Reward token /// @param _amount Quantity from which the amount of the reward should be calculated /// @param _receiver Referral agent address function _sendReferReward(Token memory A, uint _amount, address _receiver) internal returns (uint) { if (_receiver != address(0)) { uint refPercent = A.reward; if (refPercent > 0) { uint refAmount = refPercent * _amount / PERCENT_PRECISION; INarfexFiat(A.addr).mintTo(_receiver, refAmount); emit ReferralReward(A.addr, refAmount, _receiver); return refAmount; } } return 0; } /// @notice Only exchanges between fiats /// @param _accountAddress Recipient address /// @param A First token /// @param B Second token /// @param exchange Calculated values to exchange /// @param _refer Referral agent address function _swapFiats( address _accountAddress, Token memory A, Token memory B, ExchangeData memory exchange, address _refer ) private { require(INarfexFiat(A.addr).balanceOf(_accountAddress) >= exchange.inAmount, "Not enough balance"); /// Exchange tokens INarfexFiat(A.addr).burnFrom(_accountAddress, exchange.inAmount); INarfexFiat(B.addr).mintTo(_accountAddress, exchange.outAmount); /// Send referral reward Token memory C = A.addr == exchange.commToken ? A : B; exchange.referReward = _sendReferReward(C, exchange.inAmountClear, _refer); exchange.profitUSDT -= _getUSDTValue(C.addr, int(exchange.referReward)); emit SwapFiat(_accountAddress, A.addr, B.addr, exchange); } /// @notice Fiat and USDT Pair Exchange /// @param _accountAddress Recipient address /// @param A First token /// @param B Second token /// @param exchange Calculated values to exchange /// @param _refer Referral agent address /// @param _isItSwapWithDEX Cancels sending USDT to the user /// @dev The last parameter is needed for further or upcoming work with DEX function _swapFiatAndUSDT( address _accountAddress, Token memory A, Token memory B, ExchangeData memory exchange, address _refer, bool _isItSwapWithDEX ) private returns (uint usdtAmount) { Token memory C = A.addr == exchange.commToken ? A : B; if (A.addr == address(USDT)) { /// If conversion from USDT to fiat if (!_isItSwapWithDEX) { /// Transfer from the account USDT.transferFrom(_accountAddress, address(pool), exchange.inAmount); } /// ELSE: USDT must be already transferred to the pool by DEX /// Mint fiat to the final account INarfexFiat(B.addr).mintTo(_accountAddress, exchange.outAmount); /// Send refer reward exchange.referReward = _sendReferReward(C, exchange.outAmountClear, _refer); exchange.profitUSDT -= _getUSDTValue(C.addr, int(exchange.referReward)); } else { /// If conversion from fiat to usdt require(pool.getBalance() >= exchange.outAmount, "Not enough liquidity pool amount"); /// Burn fiat from account INarfexFiat(A.addr).burnFrom(_accountAddress, exchange.inAmount); /// Send refer reward exchange.referReward = _sendReferReward(C, exchange.inAmountClear, _refer); exchange.profitUSDT -= _getUSDTValue(C.addr, int(exchange.referReward)); /// Then transfer USDT if (!_isItSwapWithDEX) { /// Transfer USDT to the final account USDT.transferFrom(address(pool), _accountAddress, exchange.outAmount); } usdtAmount = exchange.outAmount; } emit SwapFiat(_accountAddress, A.addr, B.addr, exchange); } /// @notice Truncates the path, excluding the fiat from it /// @param _path An array of addresses representing the exchange path /// @param isFromFiat Indicates the direction of the route (Fiat>DEX of DEX>Fiat) function _getDEXSubPath(address[] memory _path, bool isFromFiat) internal pure returns (address[] memory) { address[] memory path = new address[](_path.length - 1); for (uint i = 0; i < path.length; i++) { path[i] = _path[isFromFiat ? i + 1 : i]; } return path; } /// @notice Gets the reserves of tokens in the path and calculates the final value /// @param data Prepared swap data /// @dev Updates the data in the structure passed as a parameter function _processSwapData(SwapData memory data) internal view { if (data.isExactOut) { data.amounts = PancakeLibrary.getAmountsIn(data.outAmount, data.path); data.inAmount = data.amounts[0]; } else { data.amounts = PancakeLibrary.getAmountsOut(data.inAmount, data.path); data.outAmount = data.amounts[data.amounts.length - 1]; } } /// @notice Exchange only between crypto Сoins through liquidity pairs /// @param _account Recipient account address /// @param data Prepared swap data /// @param A Input token data /// @param B Output token data function _swapOnlyDEX( address payable _account, SwapData memory data, Token memory A, Token memory B ) private { uint transferInAmount; if (data.isExactOut) { /// Increase output amount by outgoing token fee for calculations data.outAmount = B.transferFee > 0 ? data.amount * (PERCENT_PRECISION + B.transferFee) / PERCENT_PRECISION : data.amount; } else { transferInAmount = data.amount; /// Decrease input amount for calculations data.inAmount = A.transferFee > 0 ? data.amount * (PERCENT_PRECISION - A.transferFee) / PERCENT_PRECISION : data.amount; } /// Calculate the opposite value _processSwapData(data); if (data.isExactOut) { /// Increase input amount by inbound token fee transferInAmount = A.transferFee > 0 ? data.inAmount * (PERCENT_PRECISION + A.transferFee) / PERCENT_PRECISION : data.inAmount; require(data.inAmount <= data.inAmountMax, "Input amount is higher than maximum"); } else { require(data.outAmount >= data.outAmountMin, "Output amount is lower than minimum"); } address firstPair = PancakeLibrary.pairFor(data.path[0], data.path[1]); if (A.addr == address(WBNB)) { /// BNB insert require(msg.value >= transferInAmount, "BNB is not sended"); WBNB.deposit{value: transferInAmount}(); assert(WBNB.transfer(firstPair, transferInAmount)); if (msg.value > transferInAmount) { /// Return unused BNB _account.transfer(msg.value - transferInAmount); } } else { /// Coin insert SafeERC20.safeTransferFrom(IERC20(data.path[0]), _account, firstPair, transferInAmount); } if (B.addr == address(WBNB)) { /// Send BNB after swap _swapDEX(data.amounts, data.path, address(this)); WBNB.withdraw(data.outAmount); _account.transfer(data.outAmount); } else { /// Send Coin after swap _swapDEX(data.amounts, data.path, _account); } emit SwapDEX(_account, A.addr, B.addr, data.inAmount, data.outAmount); } /// @notice Exchange through liquidity pairs along the route /// @param amounts Pre-read reserves in liquidity pairs /// @param path An array of addresses representing the exchange path /// @param _to Address of the recipient function _swapDEX(uint[] memory amounts, address[] memory path, address _to) internal { for (uint i; i < path.length - 1; i++) { (address input, address output) = (path[i], path[i + 1]); (address token0,) = PancakeLibrary.sortTokens(input, output); uint amountOut = amounts[i + 1]; (uint amount0Out, uint amount1Out) = input == token0 ? (uint(0), amountOut) : (amountOut, uint(0)); address to = i < path.length - 2 ? PancakeLibrary.pairFor(output, path[i + 2]) : _to; IPancakePair(PancakeLibrary.pairFor(input, output)).swap( amount0Out, amount1Out, to, new bytes(0) ); } } /// @notice Fiat to crypto Coin exchange and vice versa /// @param _account Recipient address /// @param data Prepared swap data /// @param F Fiat token data /// @param C Coin token data /// @param isFromFiat Exchange direction /// @dev Takes into account tokens with transfer fees function _swapFiatWithDEX( address payable _account, SwapData memory data, Token memory F, // Fiat Token memory C, // Coin bool isFromFiat ) private { /// USDT token data Token memory U = _assignTokenData(address(USDT), oracle.getTokenData(address(USDT), false)); uint lastIndex = data.path.length - 1; require((isFromFiat && data.path[0] == U.addr) || (!isFromFiat && data.path[lastIndex] == U.addr), "The exchange between fiat and crypto must be done via USDT"); ExchangeData memory exchange; if (data.isExactOut) { /// If exact OUT if (isFromFiat) { /// FIAT > USDT > DEX > COIN!! /// Calculate other amounts from the start amount data data.outAmount = data.amount; if (C.transferFee > 0) { /// Increasing the output amount offsets the loss from the fee data.outAmount = data.outAmount * (PERCENT_PRECISION + C.transferFee) / PERCENT_PRECISION; } _processSwapData(data); exchange = _getExchangeValues(F, U, data.inAmount, true); require(exchange.inAmount <= data.inAmountMax, "Input amount is higher than maximum"); /// Swap Fiat with USDT _swapFiatAndUSDT(_account, F, U, exchange, data.refer, true); /// Transfer USDT from the Pool to the first pair { address firstPair = PancakeLibrary.pairFor(U.addr, data.path[1]); SafeERC20.safeTransferFrom(USDT, address(pool), firstPair, data.inAmount); } /// Swap and send to the account if (C.addr == address(WBNB)) { /// Swap with BNB out _swapDEX(data.amounts, data.path, address(this)); WBNB.withdraw(data.outAmount); _account.transfer(data.outAmount); } else { /// Swap with coin out _swapDEX(data.amounts, data.path, _account); } emit SwapDEX(_account, data.path[0], data.path[lastIndex], data.inAmount, data.outAmount); } else { /// COIN > DEX > USDT > FIAT!! /// Calculate other amounts from the start amount data exchange = _getExchangeValues(U, F, data.amount, true); data.outAmount = exchange.inAmount; _processSwapData(data); require(data.inAmount <= data.inAmountMax, "Input amount is higher than maximum"); /// Transfer Coin from the account to the first pair { address firstPair = PancakeLibrary.pairFor(C.addr, data.path[1]); if (C.addr == address(WBNB)) { /// BNB transfer require(msg.value >= data.inAmount, "BNB is not sended"); WBNB.deposit{value: data.inAmount}(); assert(WBNB.transfer(firstPair, data.inAmount)); if (msg.value > data.inAmount) { /// Return unused BNB _account.transfer(msg.value - data.inAmount); } } else { /// Send increased coin amount from the account to DEX uint inAmountWithFee = C.transferFee > 0 ? data.inAmount * (PERCENT_PRECISION + C.transferFee) / PERCENT_PRECISION : data.inAmount; SafeERC20.safeTransferFrom(IERC20(C.addr), _account, firstPair, inAmountWithFee); } } /// Swap and send USDT to the pool _swapDEX(data.amounts, data.path, address(pool)); emit SwapDEX(_account, data.path[0], data.path[lastIndex], data.inAmount, data.outAmount); /// Swap USDT and Fiat _swapFiatAndUSDT(_account, U, F, exchange, data.refer, true); } } else { /// If exact IN if (isFromFiat) { /// FIAT!! > USDT > DEX > COIN /// Calculate other amounts from the start amount data exchange = _getExchangeValues(F, U, data.amount, false); data.inAmount = exchange.outAmount; _processSwapData(data); require(data.outAmount >= data.outAmountMin, "Output amount is lower than minimum"); /// Swap Fiat with USDT _swapFiatAndUSDT(_account, F, U, exchange, data.refer, true); /// Transfer USDT from the Pool to the first pair { address firstPair = PancakeLibrary.pairFor(U.addr, data.path[1]); SafeERC20.safeTransferFrom(USDT, address(pool), firstPair, data.inAmount); /// TransferFee only affects delivered amount } /// Swap and send to the account if (C.addr == address(WBNB)) { /// Swap with BNB transfer _swapDEX(data.amounts, data.path, address(this)); WBNB.withdraw(data.outAmount); _account.transfer(data.outAmount); } else { /// Swap with coin transfer _swapDEX(data.amounts, data.path, _account); } emit SwapDEX(_account, data.path[0], data.path[lastIndex], data.inAmount, data.outAmount); } else { /// COIN!! > DEX > USDT > FIAT /// Calculate other amounts from the start amount data data.inAmount = data.amount; if (C.transferFee > 0) { /// DEX swap with get a reduced value data.inAmount = data.inAmount * (PERCENT_PRECISION - C.transferFee) / PERCENT_PRECISION; } _processSwapData(data); exchange = _getExchangeValues(U, F, data.outAmount, false); require(exchange.outAmount >= data.outAmountMin, "Output amount is lower than minimum"); /// Transfer Coin from the account to the first pair { address firstPair = PancakeLibrary.pairFor(C.addr, data.path[1]); if (C.addr == address(WBNB)) { /// BNB transfer require(msg.value >= data.amount, "BNB is not sended"); WBNB.deposit{value: data.amount}(); assert(WBNB.transfer(firstPair, data.amount)); } else { /// Coin transfer SafeERC20.safeTransferFrom(IERC20(C.addr), _account, firstPair, data.amount); /// Full amount } } /// Swap and send USDT to the pool _swapDEX(data.amounts, data.path, address(pool)); emit SwapDEX(_account, data.path[0], data.path[lastIndex], data.inAmount, data.outAmount); /// Swap USDT and Fiat _swapFiatAndUSDT(_account, U, F, exchange, data.refer, true); } } } /// @notice Main Routing Exchange Function /// @param _account Recipient address /// @param data Prepared data for exchange function _swap( address payable _account, SwapData memory data ) private { require(data.path.length > 1, "Path length must be at least 2 addresses"); uint lastIndex = data.path.length - 1; Token memory A; /// First token Token memory B; /// Last token { /// Get the oracle data for the first and last tokens address[] memory sideTokens = new address[](2); sideTokens[0] = data.path[0]; sideTokens[1] = data.path[lastIndex]; INarfexOracle.TokenData[] memory tokensData = oracle.getTokensData(sideTokens, true); A = _assignTokenData(sideTokens[0], tokensData[0]); B = _assignTokenData(sideTokens[1], tokensData[1]); } require(A.addr != B.addr, "Can't swap the same tokens"); if (A.isFiat && B.isFiat) { /// If swap between fiats ExchangeData memory exchange = _getExchangeValues(A, B, data.amount, data.isExactOut); _swapFiats(_account, A, B, exchange, data.refer); return; } if (!A.isFiat && !B.isFiat) { /// Swap on DEX only _swapOnlyDEX(_account, data, A, B); return; } if ((A.isFiat && B.addr == address(USDT)) || (B.isFiat && A.addr == address(USDT))) { /// If swap between fiat and USDT in the pool ExchangeData memory exchange = _getExchangeValues(A, B, data.amount, data.isExactOut); _swapFiatAndUSDT(_account, A, B, exchange, data.refer, false); return; } /// Swap with DEX and Fiats data.path = _getDEXSubPath(data.path, A.isFiat); _swapFiatWithDEX(_account, data, A.isFiat ? A : B, A.isFiat ? B : A, A.isFiat); } /// @notice Set a new pool address /// @param _newPoolAddress Another pool address /// @param _decimals Pool token decimals function setPool(address _newPoolAddress, uint8 _decimals) public onlyOwner { pool = INarfexExchangerPool(_newPoolAddress); USDT_PRECISION = 10**_decimals; } /// @notice Set a new oracle address /// @param _newOracleAddress Another oracle address function setOracle(address _newOracleAddress) public onlyOwner { oracle = INarfexOracle(_newOracleAddress); } /// @notice Swap tokens public function /// @param path An array of addresses representing the exchange path /// @param isExactOut Is the amount an output value /// @param amountLimit Becomes the min output amount for isExactOut=true, and max input for false /// @param deadline The transaction must be completed no later than the specified time /// @param refer Referral agent address /// @dev If the user wants to get an exact amount in the output, isExactOut should be true /// @dev Fiat to crypto must be exchanged via USDT function swap( address[] memory path, bool isExactOut, uint amount, uint amountLimit, uint deadline, address refer) public payable ensure(deadline) { SwapData memory data; data.path = path; data.isExactOut = isExactOut; data.amount = amount; data.inAmount = isExactOut ? 0 : amount; data.inAmountMax = isExactOut ? amountLimit : MAX_INT; data.outAmount = isExactOut ? amount : 0; data.outAmountMin = isExactOut ? 0 : amountLimit; data.refer = refer; _swap(payable(msg.sender), data); } }
//SPDX-License-Identifier: MIT pragma solidity ^0.8.17; interface IPancakePair { function name() external pure returns (string memory); function symbol() external pure returns (string memory); function decimals() external pure returns (uint8); function totalSupply() external view returns (uint); function balanceOf(address owner) external view returns (uint); function allowance(address owner, address spender) external view returns (uint); function approve(address spender, uint value) external returns (bool); function transfer(address to, uint value) external returns (bool); function transferFrom(address from, address to, uint value) external returns (bool); function DOMAIN_SEPARATOR() external view returns (bytes32); function PERMIT_TYPEHASH() external pure returns (bytes32); function nonces(address owner) external view returns (uint); function permit(address owner, address spender, uint value, uint deadline, uint8 v, bytes32 r, bytes32 s) external; function MINIMUM_LIQUIDITY() external pure returns (uint); function factory() external view returns (address); function token0() external view returns (address); function token1() external view returns (address); function getReserves() external view returns (uint112 reserve0, uint112 reserve1, uint32 blockTimestampLast); function price0CumulativeLast() external view returns (uint); function price1CumulativeLast() external view returns (uint); function kLast() external view returns (uint); function swap(uint amount0Out, uint amount1Out, address to, bytes calldata data) external; function skim(address to) external; function sync() external; function initialize(address, address) external; } library PancakeLibrary { /// @notice Returns sorted token addresses, used to handle return values from pairs sorted in this order /// @param tokenA First token address /// @param tokenB Second token address function sortTokens(address tokenA, address tokenB) internal pure returns (address token0, address token1) { require(tokenA != tokenB, 'PancakeLibrary: IDENTICAL_ADDRESSES'); (token0, token1) = tokenA < tokenB ? (tokenA, tokenB) : (tokenB, tokenA); require(token0 != address(0), 'PancakeLibrary: ZERO_ADDRESS'); } /// @notice Calculates address for a pair without making any external calls /// @param tokenA First token address /// @param tokenB Second token address function pairFor(address tokenA, address tokenB) internal view returns (address pair) { (address token0, address token1) = sortTokens(tokenA, tokenB); /// ETH data bytes memory factory = hex'5C69bEe701ef814a2B6a3EDD4B1652CB9cc5aA6f'; bytes memory initCodeHash = hex'96e8ac4277198ff8b6f785478aa9a39f403cb768dd02cbee326c3e7da348845f'; if (block.chainid == 56) { /// BSC factory = hex'cA143Ce32Fe78f1f7019d7d551a6402fC5350c73'; initCodeHash = hex'00fb7f630766e6a796048ea87d01acd3068e8ff67d078148a3fa3f4a84f69bd5'; } if (block.chainid == 97) { /// BSC testnet factory = hex'b7926c0430afb07aa7defde6da862ae0bde767bc'; initCodeHash = hex'ecba335299a6693cb2ebc4782e74669b84290b6378ea3a3873c7231a8d7d1074'; } pair = address(uint160(uint256(keccak256(abi.encodePacked( hex'ff', factory, keccak256(abi.encodePacked(token0, token1)), initCodeHash ))))); } // fetches and sorts the reserves for a pair function getReserves(address tokenA, address tokenB) internal view returns (uint reserveA, uint reserveB) { (address token0,) = sortTokens(tokenA, tokenB); pairFor(tokenA, tokenB); (uint reserve0, uint reserve1,) = IPancakePair(pairFor(tokenA, tokenB)).getReserves(); (reserveA, reserveB) = tokenA == token0 ? (reserve0, reserve1) : (reserve1, reserve0); } // given some amount of an asset and pair reserves, returns an equivalent amount of the other asset function quote(uint amountA, uint reserveA, uint reserveB) internal pure returns (uint amountB) { require(amountA > 0, 'PancakeLibrary: INSUFFICIENT_AMOUNT'); require(reserveA > 0 && reserveB > 0, 'PancakeLibrary: INSUFFICIENT_LIQUIDITY'); amountB = amountA * reserveB / reserveA; } // given an input amount of an asset and pair reserves, returns the maximum output amount of the other asset function getAmountOut(uint amountIn, uint reserveIn, uint reserveOut) internal view returns (uint amountOut) { require(amountIn > 0, 'PancakeLibrary: INSUFFICIENT_INPUT_AMOUNT'); require(reserveIn > 0 && reserveOut > 0, 'PancakeLibrary: INSUFFICIENT_LIQUIDITY'); uint pancakeCommission = 9970; /// ETH commission if (block.chainid == 56) { pancakeCommission = 9975; /// BSC commission } if (block.chainid == 97) { pancakeCommission = 9980; /// BSC testnet commission } uint amountInWithFee = amountIn * pancakeCommission; uint numerator = amountInWithFee * reserveOut; uint denominator = reserveIn * 10000 + amountInWithFee; amountOut = numerator / denominator; } // given an output amount of an asset and pair reserves, returns a required input amount of the other asset function getAmountIn(uint amountOut, uint reserveIn, uint reserveOut) internal view returns (uint amountIn) { require(amountOut > 0, 'PancakeLibrary: INSUFFICIENT_OUTPUT_AMOUNT'); require(reserveIn > 0 && reserveOut > 0, 'PancakeLibrary: INSUFFICIENT_LIQUIDITY'); uint pancakeCommission = 9970; /// ETH commission if (block.chainid == 56) { pancakeCommission = 9975; /// BSC commission } if (block.chainid == 97) { pancakeCommission = 9980; /// BSC testnet commission } uint numerator = reserveIn * amountOut * 10000; uint denominator = (reserveOut - amountOut) * pancakeCommission; amountIn = (numerator / denominator) + 1; } // performs chained getAmountOut calculations on any number of pairs function getAmountsOut(uint amountIn, address[] memory path) internal view returns (uint[] memory amounts) { require(path.length >= 2, 'PancakeLibrary: INVALID_PATH'); amounts = new uint[](path.length); amounts[0] = amountIn; for (uint i; i < path.length - 1; i++) { (uint reserveIn, uint reserveOut) = getReserves(path[i], path[i + 1]); amounts[i + 1] = getAmountOut(amounts[i], reserveIn, reserveOut); } } // performs chained getAmountIn calculations on any number of pairs function getAmountsIn(uint amountOut, address[] memory path) internal view returns (uint[] memory amounts) { require(path.length >= 2, 'PancakeLibrary: INVALID_PATH'); amounts = new uint[](path.length); amounts[amounts.length - 1] = amountOut; for (uint i = path.length - 1; i > 0; i--) { (uint reserveIn, uint reserveOut) = getReserves(path[i - 1], path[i]); amounts[i - 1] = getAmountIn(amounts[i], reserveIn, reserveOut); } } }
//SPDX-License-Identifier: MIT pragma solidity ^0.8.13; interface INarfexOracle { struct Token { bool isFiat; bool isCustomCommission; // Use default commission on false bool isCustomReward; // Use defalt referral percent on false uint price; // USD price only for fiats uint reward; // Referral percent only for fiats int commission; // Commission percent. Can be lower than zero uint transferFee; // Token transfer fee with 1000 decimals precision (20 for NRFX is 2%) } /// Calculated Token data struct TokenData { bool isFiat; int commission; uint price; uint reward; uint transferFee; } function defaultFiatCommission() external pure returns (int); function defaultCryptoCommission() external pure returns (int); function defaultReward() external pure returns (uint); function tokens(address _address) external returns (Token memory); function getPrice(address _address) external view returns (uint); function getIsFiat(address _address) external view returns (bool); function getCommission(address _address) external view returns (int); function getReferralPercent(address _address) external view returns (uint); function getTokenTransferFee(address _address) external view returns (uint); function getTokenData(address _address, bool _skipCoinPrice) external view returns (TokenData memory tokenData); function getTokensData(address[] calldata _tokens, bool _skipCoinPrice) external view returns (TokenData[] memory); }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.8.0) (token/ERC20/utils/SafeERC20.sol) pragma solidity ^0.8.0; import "../IERC20.sol"; import "../extensions/draft-IERC20Permit.sol"; import "../../../utils/Address.sol"; /** * @title SafeERC20 * @dev Wrappers around ERC20 operations that throw on failure (when the token * contract returns false). Tokens that return no value (and instead revert or * throw on failure) are also supported, non-reverting calls are assumed to be * successful. * To use this library you can add a `using SafeERC20 for IERC20;` statement to your contract, * which allows you to call the safe operations as `token.safeTransfer(...)`, etc. */ library SafeERC20 { using Address for address; function safeTransfer( IERC20 token, address to, uint256 value ) internal { _callOptionalReturn(token, abi.encodeWithSelector(token.transfer.selector, to, value)); } function safeTransferFrom( IERC20 token, address from, address to, uint256 value ) internal { _callOptionalReturn(token, abi.encodeWithSelector(token.transferFrom.selector, from, to, value)); } /** * @dev Deprecated. This function has issues similar to the ones found in * {IERC20-approve}, and its usage is discouraged. * * Whenever possible, use {safeIncreaseAllowance} and * {safeDecreaseAllowance} instead. */ function safeApprove( IERC20 token, address spender, uint256 value ) internal { // safeApprove should only be called when setting an initial allowance, // or when resetting it to zero. To increase and decrease it, use // 'safeIncreaseAllowance' and 'safeDecreaseAllowance' require( (value == 0) || (token.allowance(address(this), spender) == 0), "SafeERC20: approve from non-zero to non-zero allowance" ); _callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, value)); } function safeIncreaseAllowance( IERC20 token, address spender, uint256 value ) internal { uint256 newAllowance = token.allowance(address(this), spender) + value; _callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, newAllowance)); } function safeDecreaseAllowance( IERC20 token, address spender, uint256 value ) internal { unchecked { uint256 oldAllowance = token.allowance(address(this), spender); require(oldAllowance >= value, "SafeERC20: decreased allowance below zero"); uint256 newAllowance = oldAllowance - value; _callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, newAllowance)); } } function safePermit( IERC20Permit token, address owner, address spender, uint256 value, uint256 deadline, uint8 v, bytes32 r, bytes32 s ) internal { uint256 nonceBefore = token.nonces(owner); token.permit(owner, spender, value, deadline, v, r, s); uint256 nonceAfter = token.nonces(owner); require(nonceAfter == nonceBefore + 1, "SafeERC20: permit did not succeed"); } /** * @dev Imitates a Solidity high-level call (i.e. a regular function call to a contract), relaxing the requirement * on the return value: the return value is optional (but if data is returned, it must not be false). * @param token The token targeted by the call. * @param data The call data (encoded using abi.encode or one of its variants). */ function _callOptionalReturn(IERC20 token, bytes memory data) private { // We need to perform a low level call here, to bypass Solidity's return data size checking mechanism, since // we're implementing it ourselves. We use {Address-functionCall} to perform this call, which verifies that // the target address contains contract code and also asserts for success in the low-level call. bytes memory returndata = address(token).functionCall(data, "SafeERC20: low-level call failed"); if (returndata.length > 0) { // Return data is optional require(abi.decode(returndata, (bool)), "SafeERC20: ERC20 operation did not succeed"); } } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.7.0) (access/Ownable.sol) pragma solidity ^0.8.0; import "../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. * * By default, the owner account will be the one that deploys the contract. 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; event OwnershipTransferred(address indexed previousOwner, address indexed newOwner); /** * @dev Initializes the contract setting the deployer as the initial owner. */ constructor() { _transferOwnership(_msgSender()); } /** * @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 { require(owner() == _msgSender(), "Ownable: caller is not the owner"); } /** * @dev Leaves the contract without owner. It will not be possible to call * `onlyOwner` functions anymore. Can only be called by the current owner. * * NOTE: Renouncing ownership will leave the contract without an owner, * thereby removing 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 { require(newOwner != address(0), "Ownable: new owner is the zero address"); _transferOwnership(newOwner); } /** * @dev Transfers ownership of the contract to a new account (`newOwner`). * Internal function without access restriction. */ function _transferOwnership(address newOwner) internal virtual { address oldOwner = _owner; _owner = newOwner; emit OwnershipTransferred(oldOwner, newOwner); } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.6.0) (token/ERC20/IERC20.sol) pragma solidity ^0.8.0; /** * @dev Interface of the ERC20 standard as defined in the EIP. */ interface IERC20 { /** * @dev Emitted when `value` tokens are moved from one account (`from`) to * another (`to`). * * Note that `value` may be zero. */ event Transfer(address indexed from, address indexed to, uint256 value); /** * @dev Emitted when the allowance of a `spender` for an `owner` is set by * a call to {approve}. `value` is the new allowance. */ event Approval(address indexed owner, address indexed spender, uint256 value); /** * @dev Returns the amount of tokens in existence. */ function totalSupply() external view returns (uint256); /** * @dev Returns the amount of tokens owned by `account`. */ function balanceOf(address account) external view returns (uint256); /** * @dev Moves `amount` tokens from the caller's account to `to`. * * Returns a boolean value indicating whether the operation succeeded. * * Emits a {Transfer} event. */ function transfer(address to, uint256 amount) external returns (bool); /** * @dev Returns the remaining number of tokens that `spender` will be * allowed to spend on behalf of `owner` through {transferFrom}. This is * zero by default. * * This value changes when {approve} or {transferFrom} are called. */ function allowance(address owner, address spender) external view returns (uint256); /** * @dev Sets `amount` as the allowance of `spender` over the caller's tokens. * * Returns a boolean value indicating whether the operation succeeded. * * IMPORTANT: Beware that changing an allowance with this method brings the risk * that someone may use both the old and the new allowance by unfortunate * transaction ordering. One possible solution to mitigate this race * condition is to first reduce the spender's allowance to 0 and set the * desired value afterwards: * https://github.com/ethereum/EIPs/issues/20#issuecomment-263524729 * * Emits an {Approval} event. */ function approve(address spender, uint256 amount) external returns (bool); /** * @dev Moves `amount` tokens from `from` to `to` using the * allowance mechanism. `amount` is then deducted from the caller's * allowance. * * Returns a boolean value indicating whether the operation succeeded. * * Emits a {Transfer} event. */ function transferFrom( address from, address to, uint256 amount ) external returns (bool); }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts v4.4.1 (token/ERC20/extensions/draft-IERC20Permit.sol) pragma solidity ^0.8.0; /** * @dev Interface of the ERC20 Permit extension allowing approvals to be made via signatures, as defined in * https://eips.ethereum.org/EIPS/eip-2612[EIP-2612]. * * Adds the {permit} method, which can be used to change an account's ERC20 allowance (see {IERC20-allowance}) by * presenting a message signed by the account. By not relying on {IERC20-approve}, the token holder account doesn't * need to send a transaction, and thus is not required to hold Ether at all. */ interface IERC20Permit { /** * @dev Sets `value` as the allowance of `spender` over ``owner``'s tokens, * given ``owner``'s signed approval. * * IMPORTANT: The same issues {IERC20-approve} has related to transaction * ordering also apply here. * * Emits an {Approval} event. * * Requirements: * * - `spender` cannot be the zero address. * - `deadline` must be a timestamp in the future. * - `v`, `r` and `s` must be a valid `secp256k1` signature from `owner` * over the EIP712-formatted function arguments. * - the signature must use ``owner``'s current nonce (see {nonces}). * * For more information on the signature format, see the * https://eips.ethereum.org/EIPS/eip-2612#specification[relevant EIP * section]. */ function permit( address owner, address spender, uint256 value, uint256 deadline, uint8 v, bytes32 r, bytes32 s ) external; /** * @dev Returns the current nonce for `owner`. This value must be * included whenever a signature is generated for {permit}. * * Every successful call to {permit} increases ``owner``'s nonce by one. This * prevents a signature from being used multiple times. */ function nonces(address owner) external view returns (uint256); /** * @dev Returns the domain separator used in the encoding of the signature for {permit}, as defined by {EIP712}. */ // solhint-disable-next-line func-name-mixedcase function DOMAIN_SEPARATOR() external view returns (bytes32); }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.8.0) (utils/Address.sol) pragma solidity ^0.8.1; /** * @dev Collection of functions related to the address type */ library Address { /** * @dev Returns true if `account` is a contract. * * [IMPORTANT] * ==== * It is unsafe to assume that an address for which this function returns * false is an externally-owned account (EOA) and not a contract. * * Among others, `isContract` will return false for the following * types of addresses: * * - an externally-owned account * - a contract in construction * - an address where a contract will be created * - an address where a contract lived, but was destroyed * ==== * * [IMPORTANT] * ==== * You shouldn't rely on `isContract` to protect against flash loan attacks! * * Preventing calls from contracts is highly discouraged. It breaks composability, breaks support for smart wallets * like Gnosis Safe, and does not provide security since it can be circumvented by calling from a contract * constructor. * ==== */ function isContract(address account) internal view returns (bool) { // This method relies on extcodesize/address.code.length, which returns 0 // for contracts in construction, since the code is only stored at the end // of the constructor execution. return account.code.length > 0; } /** * @dev Replacement for Solidity's `transfer`: sends `amount` wei to * `recipient`, forwarding all available gas and reverting on errors. * * https://eips.ethereum.org/EIPS/eip-1884[EIP1884] increases the gas cost * of certain opcodes, possibly making contracts go over the 2300 gas limit * imposed by `transfer`, making them unable to receive funds via * `transfer`. {sendValue} removes this limitation. * * https://diligence.consensys.net/posts/2019/09/stop-using-soliditys-transfer-now/[Learn more]. * * IMPORTANT: because control is transferred to `recipient`, care must be * taken to not create reentrancy vulnerabilities. Consider using * {ReentrancyGuard} or the * https://solidity.readthedocs.io/en/v0.5.11/security-considerations.html#use-the-checks-effects-interactions-pattern[checks-effects-interactions pattern]. */ function sendValue(address payable recipient, uint256 amount) internal { require(address(this).balance >= amount, "Address: insufficient balance"); (bool success, ) = recipient.call{value: amount}(""); require(success, "Address: unable to send value, recipient may have reverted"); } /** * @dev Performs a Solidity function call using a low level `call`. A * plain `call` is an unsafe replacement for a function call: use this * function instead. * * If `target` reverts with a revert reason, it is bubbled up by this * function (like regular Solidity function calls). * * Returns the raw returned data. To convert to the expected return value, * use https://solidity.readthedocs.io/en/latest/units-and-global-variables.html?highlight=abi.decode#abi-encoding-and-decoding-functions[`abi.decode`]. * * Requirements: * * - `target` must be a contract. * - calling `target` with `data` must not revert. * * _Available since v3.1._ */ function functionCall(address target, bytes memory data) internal returns (bytes memory) { return functionCallWithValue(target, data, 0, "Address: low-level call failed"); } /** * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], but with * `errorMessage` as a fallback revert reason when `target` reverts. * * _Available since v3.1._ */ function functionCall( address target, bytes memory data, string memory errorMessage ) internal returns (bytes memory) { return functionCallWithValue(target, data, 0, errorMessage); } /** * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], * but also transferring `value` wei to `target`. * * Requirements: * * - the calling contract must have an ETH balance of at least `value`. * - the called Solidity function must be `payable`. * * _Available since v3.1._ */ function functionCallWithValue( address target, bytes memory data, uint256 value ) internal returns (bytes memory) { return functionCallWithValue(target, data, value, "Address: low-level call with value failed"); } /** * @dev Same as {xref-Address-functionCallWithValue-address-bytes-uint256-}[`functionCallWithValue`], but * with `errorMessage` as a fallback revert reason when `target` reverts. * * _Available since v3.1._ */ function functionCallWithValue( address target, bytes memory data, uint256 value, string memory errorMessage ) internal returns (bytes memory) { require(address(this).balance >= value, "Address: insufficient balance for call"); (bool success, bytes memory returndata) = target.call{value: value}(data); return verifyCallResultFromTarget(target, success, returndata, errorMessage); } /** * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], * but performing a static call. * * _Available since v3.3._ */ function functionStaticCall(address target, bytes memory data) internal view returns (bytes memory) { return functionStaticCall(target, data, "Address: low-level static call failed"); } /** * @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`], * but performing a static call. * * _Available since v3.3._ */ function functionStaticCall( address target, bytes memory data, string memory errorMessage ) internal view returns (bytes memory) { (bool success, bytes memory returndata) = target.staticcall(data); return verifyCallResultFromTarget(target, success, returndata, errorMessage); } /** * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], * but performing a delegate call. * * _Available since v3.4._ */ function functionDelegateCall(address target, bytes memory data) internal returns (bytes memory) { return functionDelegateCall(target, data, "Address: low-level delegate call failed"); } /** * @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`], * but performing a delegate call. * * _Available since v3.4._ */ function functionDelegateCall( address target, bytes memory data, string memory errorMessage ) internal returns (bytes memory) { (bool success, bytes memory returndata) = target.delegatecall(data); return verifyCallResultFromTarget(target, success, returndata, errorMessage); } /** * @dev Tool to verify that a low level call to smart-contract was successful, and revert (either by bubbling * the revert reason or using the provided one) in case of unsuccessful call or if target was not a contract. * * _Available since v4.8._ */ function verifyCallResultFromTarget( address target, bool success, bytes memory returndata, string memory errorMessage ) internal view returns (bytes memory) { if (success) { if (returndata.length == 0) { // only check isContract if the call was successful and the return data is empty // otherwise we already know that it was a contract require(isContract(target), "Address: call to non-contract"); } return returndata; } else { _revert(returndata, errorMessage); } } /** * @dev Tool to verify that a low level call was successful, and revert if it wasn't, either by bubbling the * revert reason or using the provided one. * * _Available since v4.3._ */ function verifyCallResult( bool success, bytes memory returndata, string memory errorMessage ) internal pure returns (bytes memory) { if (success) { return returndata; } else { _revert(returndata, errorMessage); } } function _revert(bytes memory returndata, string memory errorMessage) private pure { // Look for revert reason and bubble it up if present if (returndata.length > 0) { // The easiest way to bubble the revert reason is using memory via assembly /// @solidity memory-safe-assembly assembly { let returndata_size := mload(returndata) revert(add(32, returndata), returndata_size) } } else { revert(errorMessage); } } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts v4.4.1 (utils/Context.sol) pragma solidity ^0.8.0; /** * @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" ] } }, "libraries": {} }
Contract Security Audit
- No Contract Security Audit Submitted- Submit Audit Here
[{"inputs":[{"internalType":"address","name":"_oracleAddress","type":"address"},{"internalType":"address","name":"_poolAddress","type":"address"},{"internalType":"address","name":"_usdtAddress","type":"address"},{"internalType":"address","name":"_wbnbAddress","type":"address"}],"stateMutability":"nonpayable","type":"constructor"},{"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":false,"internalType":"address","name":"_token","type":"address"},{"indexed":false,"internalType":"uint256","name":"_amount","type":"uint256"},{"indexed":true,"internalType":"address","name":"_receiver","type":"address"}],"name":"ReferralReward","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"_account","type":"address"},{"indexed":false,"internalType":"address","name":"_fromToken","type":"address"},{"indexed":false,"internalType":"address","name":"_toToken","type":"address"},{"indexed":false,"internalType":"uint256","name":"inAmount","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"outAmount","type":"uint256"}],"name":"SwapDEX","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"_account","type":"address"},{"indexed":false,"internalType":"address","name":"_fromToken","type":"address"},{"indexed":false,"internalType":"address","name":"_toToken","type":"address"},{"components":[{"internalType":"uint256","name":"rate","type":"uint256"},{"internalType":"int256","name":"commission","type":"int256"},{"internalType":"uint256","name":"inAmountClear","type":"uint256"},{"internalType":"uint256","name":"outAmountClear","type":"uint256"},{"internalType":"uint256","name":"inAmount","type":"uint256"},{"internalType":"uint256","name":"outAmount","type":"uint256"},{"internalType":"address","name":"commToken","type":"address"},{"internalType":"int256","name":"commAmount","type":"int256"},{"internalType":"uint256","name":"referReward","type":"uint256"},{"internalType":"int256","name":"profitUSDT","type":"int256"}],"indexed":false,"internalType":"struct NarfexExchangerRouter2.ExchangeData","name":"_exchange","type":"tuple"}],"name":"SwapFiat","type":"event"},{"inputs":[],"name":"USDT","outputs":[{"internalType":"contract IERC20","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"WBNB","outputs":[{"internalType":"contract IWBNB","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"oracle","outputs":[{"internalType":"contract INarfexOracle","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"owner","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"pool","outputs":[{"internalType":"contract INarfexExchangerPool","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"renounceOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_newOracleAddress","type":"address"}],"name":"setOracle","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_newPoolAddress","type":"address"},{"internalType":"uint8","name":"_decimals","type":"uint8"}],"name":"setPool","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address[]","name":"path","type":"address[]"},{"internalType":"bool","name":"isExactOut","type":"bool"},{"internalType":"uint256","name":"amount","type":"uint256"},{"internalType":"uint256","name":"amountLimit","type":"uint256"},{"internalType":"uint256","name":"deadline","type":"uint256"},{"internalType":"address","name":"refer","type":"address"}],"name":"swap","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[{"internalType":"address","name":"newOwner","type":"address"}],"name":"transferOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"stateMutability":"payable","type":"receive"}]
Contract Creation Code
6080604052620f42406005553480156200001857600080fd5b5060405162003ad938038062003ad98339810160408190526200003b916200012a565b6200004633620000bd565b600380546001600160a01b038087166001600160a01b031992831617909255600180548584169083161790556002805484841690831617905560048054928616929091169190911790556038461480620000a05750466061145b15620000b357670de0b6b3a76400006005555b5050505062000187565b600080546001600160a01b038381166001600160a01b0319831681178455604051919092169283917f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e09190a35050565b80516001600160a01b03811681146200012557600080fd5b919050565b600080600080608085870312156200014157600080fd5b6200014c856200010d565b93506200015c602086016200010d565b92506200016c604086016200010d565b91506200017c606086016200010d565b905092959194509250565b61394280620001976000396000f3fe6080604052600436106100955760003560e01c80638da5cb5b116100595780638da5cb5b1461016c5780638dd950021461018a57806395001298146101aa578063c54e44eb146101bd578063f2fde38b146101dd57600080fd5b806316f0115b146100bb5780632c9dbd35146100f7578063715018a6146101175780637adbf9731461012c5780637dc0d1d01461014c57600080fd5b366100b6576002546001600160a01b031633146100b4576100b4612fa6565b005b600080fd5b3480156100c757600080fd5b506004546100db906001600160a01b031681565b6040516001600160a01b03909116815260200160405180910390f35b34801561010357600080fd5b506100b4610112366004612fd8565b6101fd565b34801561012357600080fd5b506100b4610232565b34801561013857600080fd5b506100b4610147366004613015565b610246565b34801561015857600080fd5b506003546100db906001600160a01b031681565b34801561017857600080fd5b506000546001600160a01b03166100db565b34801561019657600080fd5b506002546100db906001600160a01b031681565b6100b46101b83660046130b4565b610270565b3480156101c957600080fd5b506001546100db906001600160a01b031681565b3480156101e957600080fd5b506100b46101f8366004613015565b6103a0565b610205610419565b600480546001600160a01b0319166001600160a01b03841617905561022b81600a613289565b6005555050565b61023a610419565b6102446000610473565b565b61024e610419565b600380546001600160a01b0319166001600160a01b0392909216919091179055565b81428110156102bc5760405162461bcd60e51b8152602060048201526013602482015272151c985b9cd858dd1a5bdb88195e1c1a5c9959606a1b60448201526064015b60405180910390fd5b61031d604051806101400160405280606081526020016060815260200160001515815260200160008152602001600081526020016000815260200160008152602001600081526020016000815260200160006001600160a01b031681525090565b8781528615156040820152606081018690528661033a578561033d565b60005b60808201528661034f57600019610351565b845b60a082015286610362576000610364565b855b60c0820152866103745784610377565b60005b60e08201526001600160a01b03831661012082015261039633826104c3565b5050505050505050565b6103a8610419565b6001600160a01b03811661040d5760405162461bcd60e51b815260206004820152602660248201527f4f776e61626c653a206e6577206f776e657220697320746865207a65726f206160448201526564647265737360d01b60648201526084016102b3565b61041681610473565b50565b6000546001600160a01b031633146102445760405162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e657260448201526064016102b3565b600080546001600160a01b038381166001600160a01b0319831681178455604051919092169283917f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e09190a35050565b8051516001106105265760405162461bcd60e51b815260206004820152602860248201527f50617468206c656e677468206d757374206265206174206c6561737420322061604482015267646472657373657360c01b60648201526084016102b3565b80515160009061053890600190613298565b9050610542612f09565b61054a612f09565b6040805160028082526060820183526000926020830190803683370190505090508460000151600081518110610582576105826132ab565b60200260200101518160008151811061059d5761059d6132ab565b6001600160a01b039092166020928302919091019091015284518051859081106105c9576105c96132ab565b6020026020010151816001815181106105e4576105e46132ab565b6001600160a01b03928316602091820292909201015260035460405163ed0763bd60e01b8152600092919091169063ed0763bd906106299085906001906004016132c1565b600060405180830381865afa158015610646573d6000803e3d6000fd5b505050506040513d6000823e601f3d908101601f1916820160405261066e9190810190613390565b90506106ae82600081518110610686576106866132ab565b6020026020010151826000815181106106a1576106a16132ab565b602002602001015161088b565b93506106e1826001815181106106c6576106c66132ab565b6020026020010151826001815181106106a1576106a16132ab565b9250505080600001516001600160a01b031682600001516001600160a01b03160361074e5760405162461bcd60e51b815260206004820152601a60248201527f43616e27742073776170207468652073616d6520746f6b656e7300000000000060448201526064016102b3565b81602001518015610760575080602001515b1561079757600061077b8383876060015188604001516108e8565b905061078f86848484896101200151610afa565b505050505050565b81602001511580156107ab57508060200151155b156107c3576107bc85858484610d36565b5050505050565b816020015180156107e3575060015481516001600160a01b039081169116145b80610809575080602001518015610809575060015482516001600160a01b039081169116145b156108435760006108248383876060015188604001516108e8565b905061083a868484848961012001516000611190565b50505050505050565b61085584600001518360200151611551565b845260208201516107bc908690869061086e5783610870565b845b856020015161087f5785610881565b845b8660200151611626565b610893612f09565b6040518060c00160405280846001600160a01b03168152602001836000015115158152602001836020015181526020018360400151815260200183606001518152602001836080015181525090505b92915050565b6108f0612f4a565b60015485516000916001600160a01b03918216911614610914578560600151610918565b6005545b60015486519192506000916001600160a01b0390811691161461093f578560600151610943565b6005545b90508060055483610954919061342a565b61095e9190613457565b835250506040840151612710908190819061097a90829061346b565b83896040015161098a919061346b565b6109949190613493565b61099e91906134c3565b6109a891906134f1565b602083015250816109b957826109d2565b80516005546109c8908561342a565b6109d29190613457565b6040820152816109fa5760055481516109eb908561342a565b6109f59190613457565b6109fc565b825b60608201528115610a46576127108160200151612710610a1c919061346b565b8260400151610a2b919061342a565b610a359190613457565b608082015260a08101839052610a7f565b60808101839052602081015161271090610a6090826134f1565b8260600151610a6f919061342a565b610a799190613457565b60a08201525b8460200151610a8f578351610a92565b84515b6001600160a01b031660c0820152602080820151908601516127109190610abd578260a00151610ac3565b82608001515b610acd9190613493565b610ad791906134c3565b60e0820181905260c0820151610aec9161202c565b610120820152949350505050565b608082015184516040516370a0823160e01b81526001600160a01b038881166004830152909116906370a0823190602401602060405180830381865afa158015610b48573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610b6c9190613511565b1015610baf5760405162461bcd60e51b81526020600482015260126024820152714e6f7420656e6f7567682062616c616e636560701b60448201526064016102b3565b8351608083015160405163079cc67960e41b81526001600160a01b03888116600483015260248201929092529116906379cc679090604401600060405180830381600087803b158015610c0157600080fd5b505af1158015610c15573d6000803e3d6000fd5b5050845160a08501516040516308934a5f60e31b81526001600160a01b038a8116600483015260248201929092529116925063449a52f89150604401600060405180830381600087803b158015610c6b57600080fd5b505af1158015610c7f573d6000803e3d6000fd5b5050505060008260c001516001600160a01b031685600001516001600160a01b031614610cac5783610cae565b845b9050610cbf818460400151846120eb565b61010084018190528151610cd29161202c565b8361012001818151610ce491906134f1565b905250845184516040516001600160a01b038916927f1ab086cd308e1d16ad36455000c658ca32a39d4d8520babe71c257c74853e0d292610d2692889061352a565b60405180910390a2505050505050565b6000836040015115610d905760008260a0015111610d58578360600151610d86565b6127108260a00151612710610d6d91906135cb565b8560600151610d7c919061342a565b610d869190613457565b60c0850152610ddd565b50606083015160a0830151610da9578360600151610dd7565b6127108360a00151612710610dbe9190613298565b8560600151610dcd919061342a565b610dd79190613457565b60808501525b610de6846121e6565b836040015115610e635760008360a0015111610e06578360800151610e34565b6127108360a00151612710610e1b91906135cb565b8560800151610e2a919061342a565b610e349190613457565b90508360a0015184608001511115610e5e5760405162461bcd60e51b81526004016102b3906135de565b610e8b565b8360e001518460c001511015610e8b5760405162461bcd60e51b81526004016102b390613621565b6000610ed38560000151600081518110610ea757610ea76132ab565b60200260200101518660000151600181518110610ec657610ec66132ab565b6020026020010151612279565b60025485519192506001600160a01b039081169116036110485781341015610f0d5760405162461bcd60e51b81526004016102b390613664565b600260009054906101000a90046001600160a01b03166001600160a01b031663d0e30db0836040518263ffffffff1660e01b81526004016000604051808303818588803b158015610f5d57600080fd5b505af1158015610f71573d6000803e3d6000fd5b505060025460405163a9059cbb60e01b81526001600160a01b03868116600483015260248201889052909116935063a9059cbb925060440190506020604051808303816000875af1158015610fca573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610fee919061368f565b610ffa57610ffa612fa6565b81341115611043576001600160a01b0386166108fc6110198434613298565b6040518115909202916000818181858888f19350505050158015611041573d6000803e3d6000fd5b505b611072565b6110728560000151600081518110611062576110626132ab565b6020026020010151878385612442565b60025483516001600160a01b039182169116036111445761109c85602001518660000151306124a2565b60025460c0860151604051632e1a7d4d60e01b81526001600160a01b0390921691632e1a7d4d916110d39160040190815260200190565b600060405180830381600087803b1580156110ed57600080fd5b505af1158015611101573d6000803e3d6000fd5b5050505060c08501516040516001600160a01b038816916108fc811502916000818181858888f1935050505015801561113e573d6000803e3d6000fd5b50611157565b61115785602001518660000151886124a2565b83518351608087015160c08801516040516001600160a01b038b16946000805160206138ed83398151915294610d2694919390926136ac565b6000808460c001516001600160a01b031687600001516001600160a01b0316146111ba57856111bc565b865b60015488519192506001600160a01b0390811691160361130a5782611264576001546004805460808801516040516323b872dd60e01b81526001600160a01b038d811694820194909452918316602483015260448201529116906323b872dd906064016020604051808303816000875af115801561123e573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190611262919061368f565b505b855160a08601516040516308934a5f60e31b81526001600160a01b038b81166004830152602482019290925291169063449a52f890604401600060405180830381600087803b1580156112b657600080fd5b505af11580156112ca573d6000803e3d6000fd5b505050506112dd818660600151866120eb565b610100860181905281516112f09161202c565b856101200181815161130291906134f1565b9052506114ff565b60a08501516004805460408051629032ff60e51b815290516001600160a01b03909216926312065fe09282820192602092908290030181865afa158015611355573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906113799190613511565b10156113c75760405162461bcd60e51b815260206004820181905260248201527f4e6f7420656e6f756768206c697175696469747920706f6f6c20616d6f756e7460448201526064016102b3565b8651608086015160405163079cc67960e41b81526001600160a01b038b8116600483015260248201929092529116906379cc679090604401600060405180830381600087803b15801561141957600080fd5b505af115801561142d573d6000803e3d6000fd5b50505050611440818660400151866120eb565b610100860181905281516114539161202c565b856101200181815161146591906134f1565b905250826114f7576001546004805460a08801516040516323b872dd60e01b81526001600160a01b03928316938101939093528b821660248401526044830152909116906323b872dd906064016020604051808303816000875af11580156114d1573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906114f5919061368f565b505b8460a0015191505b865186516040516001600160a01b038b16927f1ab086cd308e1d16ad36455000c658ca32a39d4d8520babe71c257c74853e0d29261153e928a9061352a565b60405180910390a2509695505050505050565b60606000600184516115639190613298565b67ffffffffffffffff81111561157b5761157b613030565b6040519080825280602002602001820160405280156115a4578160200160208202803683370190505b50905060005b815181101561161e5784846115bf57816115ca565b6115ca8260016135cb565b815181106115da576115da6132ab565b60200260200101518282815181106115f4576115f46132ab565b6001600160a01b039092166020928302919091019091015280611616816136d5565b9150506115aa565b509392505050565b600154600354604051634ff4a2a760e11b81526001600160a01b0392831660048201819052600060248301819052936116ad939192911690639fe9454e9060440160a060405180830381865afa158015611684573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906116a891906136ee565b61088b565b9050600060018660000151516116c39190613298565b9050828015611705575081600001516001600160a01b031686600001516000815181106116f2576116f26132ab565b60200260200101516001600160a01b0316145b8061174b57508215801561174b575081600001516001600160a01b031686600001518281518110611738576117386132ab565b60200260200101516001600160a01b0316145b6117bd5760405162461bcd60e51b815260206004820152603a60248201527f5468652065786368616e6765206265747765656e206669617420616e6420637260448201527f7970746f206d75737420626520646f6e6520766961205553445400000000000060648201526084016102b3565b6117c5612f4a565b866040015115611d28578315611a2357606087015160c088015260a08501511561181d576127108560a001516127106117fe91906135cb565b8860c0015161180d919061342a565b6118179190613457565b60c08801525b611826876121e6565b6118378684896080015160016108e8565b90508660a00151816080015111156118615760405162461bcd60e51b81526004016102b3906135de565b611875888785848b61012001516001611190565b50600061189784600001518960000151600181518110610ec657610ec66132ab565b60015460045460808b01519293506118c0926001600160a01b0392831692909116908490612442565b5060025485516001600160a01b03918216911603611993576118eb87602001518860000151306124a2565b60025460c0880151604051632e1a7d4d60e01b81526001600160a01b0390921691632e1a7d4d916119229160040190815260200190565b600060405180830381600087803b15801561193c57600080fd5b505af1158015611950573d6000803e3d6000fd5b5050505060c08701516040516001600160a01b038a16916108fc811502916000818181858888f1935050505015801561198d573d6000803e3d6000fd5b506119a6565b6119a6876020015188600001518a6124a2565b876001600160a01b03166000805160206138ed83398151915288600001516000815181106119d6576119d66132ab565b6020026020010151896000015185815181106119f4576119f46132ab565b60200260200101518a608001518b60c00151604051611a1694939291906136ac565b60405180910390a2610396565b611a348387896060015160016108e8565b608081015160c08901529050611a49876121e6565b8660a0015187608001511115611a715760405162461bcd60e51b81526004016102b3906135de565b6000611a9286600001518960000151600181518110610ec657610ec66132ab565b60025487519192506001600160a01b03908116911603611c1e578760800151341015611ad05760405162461bcd60e51b81526004016102b390613664565b600260009054906101000a90046001600160a01b03166001600160a01b031663d0e30db089608001516040518263ffffffff1660e01b81526004016000604051808303818588803b158015611b2457600080fd5b505af1158015611b38573d6000803e3d6000fd5b505060025460808c015160405163a9059cbb60e01b81526001600160a01b03878116600483015260248201929092529116935063a9059cbb925060440190506020604051808303816000875af1158015611b96573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190611bba919061368f565b611bc657611bc6612fa6565b8760800151341115611c1957886001600160a01b03166108fc896080015134611bef9190613298565b6040518115909202916000818181858888f19350505050158015611c17573d6000803e3d6000fd5b505b611c77565b6000808760a0015111611c35578860800151611c63565b6127108760a00151612710611c4a91906135cb565b8a60800151611c59919061342a565b611c639190613457565b9050611c7587600001518b8484612442565b505b5060208701518751600454611c969291906001600160a01b03166124a2565b876001600160a01b03166000805160206138ed8339815191528860000151600081518110611cc657611cc66132ab565b602002602001015189600001518581518110611ce457611ce46132ab565b60200260200101518a608001518b60c00151604051611d0694939291906136ac565b60405180910390a2611d22888488848b61012001516001611190565b50610396565b8315611d7c57611d3f8684896060015160006108e8565b60a081015160808901529050611d54876121e6565b8660e001518760c0015110156118615760405162461bcd60e51b81526004016102b390613621565b6060870151608088015260a085015115611dc4576127108560a00151612710611da59190613298565b8860800151611db4919061342a565b611dbe9190613457565b60808801525b611dcd876121e6565b611dde83878960c0015160006108e8565b90508660e001518160a001511015611e085760405162461bcd60e51b81526004016102b390613621565b6000611e2986600001518960000151600181518110610ec657610ec66132ab565b60025487519192506001600160a01b03908116911603611f62578760600151341015611e675760405162461bcd60e51b81526004016102b390613664565b600260009054906101000a90046001600160a01b03166001600160a01b031663d0e30db089606001516040518263ffffffff1660e01b81526004016000604051808303818588803b158015611ebb57600080fd5b505af1158015611ecf573d6000803e3d6000fd5b505060025460608c015160405163a9059cbb60e01b81526001600160a01b03878116600483015260248201929092529116935063a9059cbb925060440190506020604051808303816000875af1158015611f2d573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190611f51919061368f565b611f5d57611f5d612fa6565b611f76565b611f7686600001518a838b60600151612442565b5060208701518751600454611f959291906001600160a01b03166124a2565b876001600160a01b03166000805160206138ed8339815191528860000151600081518110611fc557611fc56132ab565b602002602001015189600001518581518110611fe357611fe36132ab565b60200260200101518a608001518b60c0015160405161200594939291906136ac565b60405180910390a2612021888488848b61012001516001611190565b505050505050505050565b60008160000361203e575060006108e2565b6005546003546040516341976e0960e01b81526001600160a01b0386811660048301526000939286929116906341976e0990602401602060405180830381865afa158015612090573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906120b49190613511565b6120be919061342a565b6120c89190613457565b905060008312156120e1576120dc8161370a565b6120e3565b805b949350505050565b60006001600160a01b038216156121db57608084015180156121d9576000612710612116868461342a565b6121209190613457565b86516040516308934a5f60e31b81526001600160a01b0387811660048301526024820184905292935091169063449a52f890604401600060405180830381600087803b15801561216f57600080fd5b505af1158015612183573d6000803e3d6000fd5b50508751604080516001600160a01b0392831681526020810186905291881693507f44db3609235da669e4ce17e6c91be1fadce4923ec8b39f2e3b9e33dffd3214cf92500160405180910390a291506121df9050565b505b5060005b9392505050565b80604001511561222e576122028160c001518260000151612655565b60208201819052805160009061221a5761221a6132ab565b602002602001015181608001818152505050565b612240816080015182600001516127e9565b60208201819052805161225590600190613298565b81518110612265576122656132ab565b60200260200101518160c001818152505050565b6000806000612288858561295e565b60408051808201825260148152735c69bee701ef814a2b6a3edd4b1652cb9cc5aa6f60601b60208083019190915282518084019093528083527f96e8ac4277198ff8b6f785478aa9a39f403cb768dd02cbee326c3e7da348845f908301529294509092504660380361235b5760405180604001604052806014815260200173ca143ce32fe78f1f7019d7d551a6402fc5350c7360601b81525091506040518060400160405280602081526020017efb7f630766e6a796048ea87d01acd3068e8ff67d078148a3fa3f4a84f69bd581525090505b466061036123cb57604051806040016040528060148152602001732de49b010c2bec1ea9f7bf79b6a18ab82f79d9ef60621b81525091506040518060400160405280602081526020017fecba335299a6693cb2ebc4782e74669b84290b6378ea3a3873c7231a8d7d107481525090505b6040516bffffffffffffffffffffffff19606086811b8216602084015285901b1660348201528290604801604051602081830303815290604052805190602001208260405160200161241f9392919061374a565b60408051601f198184030181529190528051602090910120979650505050505050565b604080516001600160a01b0385811660248301528416604482015260648082018490528251808303909101815260849091019091526020810180516001600160e01b03166323b872dd60e01b17905261249c908590612a53565b50505050565b60005b600183516124b39190613298565b81101561249c576000808483815181106124cf576124cf6132ab565b6020026020010151858460016124e591906135cb565b815181106124f5576124f56132ab565b602002602001015191509150600061250d838361295e565b50905060008761251e8660016135cb565b8151811061252e5761252e6132ab565b60200260200101519050600080836001600160a01b0316866001600160a01b03161461255c57826000612560565b6000835b91509150600060028a516125749190613298565b881061258057886125a0565b6125a0868b6125908b60026135cb565b81518110610ec657610ec66132ab565b90506125ac8787612279565b6001600160a01b031663022c0d9f84848460006040519080825280601f01601f1916602001820160405280156125e9576020820181803683370190505b506040518563ffffffff1660e01b815260040161260994939291906137c2565b600060405180830381600087803b15801561262357600080fd5b505af1158015612637573d6000803e3d6000fd5b5050505050505050505050808061264d906136d5565b9150506124a5565b60606002825110156126a95760405162461bcd60e51b815260206004820152601c60248201527f50616e63616b654c6962726172793a20494e56414c49445f504154480000000060448201526064016102b3565b815167ffffffffffffffff8111156126c3576126c3613030565b6040519080825280602002602001820160405280156126ec578160200160208202803683370190505b5090508281600183516126ff9190613298565b8151811061270f5761270f6132ab565b60200260200101818152505060006001835161272b9190613298565b90505b80156127e25760008061277d85612746600186613298565b81518110612756576127566132ab565b6020026020010151868581518110612770576127706132ab565b6020026020010151612b2a565b915091506127a5848481518110612796576127966132ab565b60200260200101518383612bfd565b846127b1600186613298565b815181106127c1576127c16132ab565b602002602001018181525050505080806127da906137f9565b91505061272e565b5092915050565b606060028251101561283d5760405162461bcd60e51b815260206004820152601c60248201527f50616e63616b654c6962726172793a20494e56414c49445f504154480000000060448201526064016102b3565b815167ffffffffffffffff81111561285757612857613030565b604051908082528060200260200182016040528015612880578160200160208202803683370190505b5090508281600081518110612897576128976132ab565b60200260200101818152505060005b600183516128b49190613298565b8110156127e2576000806128f98584815181106128d3576128d36132ab565b6020026020010151868560016128e991906135cb565b81518110612770576127706132ab565b91509150612921848481518110612912576129126132ab565b60200260200101518383612cfd565b8461292d8560016135cb565b8151811061293d5761293d6132ab565b60200260200101818152505050508080612956906136d5565b9150506128a6565b600080826001600160a01b0316846001600160a01b0316036129ce5760405162461bcd60e51b815260206004820152602360248201527f50616e63616b654c6962726172793a204944454e544943414c5f41444452455360448201526253455360e81b60648201526084016102b3565b826001600160a01b0316846001600160a01b0316106129ee5782846129f1565b83835b90925090506001600160a01b038216612a4c5760405162461bcd60e51b815260206004820152601c60248201527f50616e63616b654c6962726172793a205a45524f5f414444524553530000000060448201526064016102b3565b9250929050565b6000612aa8826040518060400160405280602081526020017f5361666545524332303a206c6f772d6c6576656c2063616c6c206661696c6564815250856001600160a01b0316612df69092919063ffffffff16565b805190915015612b255780806020019051810190612ac6919061368f565b612b255760405162461bcd60e51b815260206004820152602a60248201527f5361666545524332303a204552433230206f7065726174696f6e20646964206e6044820152691bdd081cdd58d8d9595960b21b60648201526084016102b3565b505050565b6000806000612b39858561295e565b509050612b468585612279565b50600080612b548787612279565b6001600160a01b0316630902f1ac6040518163ffffffff1660e01b8152600401606060405180830381865afa158015612b91573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190612bb59190613827565b506001600160701b031691506001600160701b03169150826001600160a01b0316876001600160a01b031614612bec578082612bef565b81815b909890975095505050505050565b6000808411612c615760405162461bcd60e51b815260206004820152602a60248201527f50616e63616b654c6962726172793a20494e53554646494349454e545f4f555460448201526914155517d05353d5539560b21b60648201526084016102b3565b600083118015612c715750600082115b612c8d5760405162461bcd60e51b81526004016102b390613877565b6126f246603803612c9d57506126f75b46606103612caa57506126fc5b6000612cb6868661342a565b612cc29061271061342a565b9050600082612cd18887613298565b612cdb919061342a565b9050612ce78183613457565b612cf29060016135cb565b979650505050505050565b6000808411612d605760405162461bcd60e51b815260206004820152602960248201527f50616e63616b654c6962726172793a20494e53554646494349454e545f494e50604482015268155517d05353d5539560ba1b60648201526084016102b3565b600083118015612d705750600082115b612d8c5760405162461bcd60e51b81526004016102b390613877565b6126f246603803612d9c57506126f75b46606103612da957506126fc5b6000612db5828761342a565b90506000612dc3858361342a565b9050600082612dd48861271061342a565b612dde91906135cb565b9050612dea8183613457565b98975050505050505050565b60606120e3848460008585600080866001600160a01b03168587604051612e1d91906138bd565b60006040518083038185875af1925050503d8060008114612e5a576040519150601f19603f3d011682016040523d82523d6000602084013e612e5f565b606091505b5091509150612cf28783838760608315612eda578251600003612ed3576001600160a01b0385163b612ed35760405162461bcd60e51b815260206004820152601d60248201527f416464726573733a2063616c6c20746f206e6f6e2d636f6e747261637400000060448201526064016102b3565b50816120e3565b6120e38383815115612eef5781518083602001fd5b8060405162461bcd60e51b81526004016102b391906138d9565b6040518060c0016040528060006001600160a01b03168152602001600015158152602001600081526020016000815260200160008152602001600081525090565b60405180610140016040528060008152602001600081526020016000815260200160008152602001600081526020016000815260200160006001600160a01b031681526020016000815260200160008152602001600081525090565b634e487b7160e01b600052600160045260246000fd5b80356001600160a01b0381168114612fd357600080fd5b919050565b60008060408385031215612feb57600080fd5b612ff483612fbc565b9150602083013560ff8116811461300a57600080fd5b809150509250929050565b60006020828403121561302757600080fd5b6121df82612fbc565b634e487b7160e01b600052604160045260246000fd5b604051601f8201601f1916810167ffffffffffffffff8111828210171561306f5761306f613030565b604052919050565b600067ffffffffffffffff82111561309157613091613030565b5060051b60200190565b801515811461041657600080fd5b8035612fd38161309b565b60008060008060008060c087890312156130cd57600080fd5b863567ffffffffffffffff8111156130e457600080fd5b8701601f810189136130f557600080fd5b8035602061310a61310583613077565b613046565b82815260059290921b8301810191818101908c84111561312957600080fd5b938201935b8385101561314e5761313f85612fbc565b8252938201939082019061312e565b995061315d90508a82016130a9565b975050505060408701359350606087013592506080870135915061318360a08801612fbc565b90509295509295509295565b634e487b7160e01b600052601160045260246000fd5b600181815b808511156131e05781600019048211156131c6576131c661318f565b808516156131d357918102915b93841c93908002906131aa565b509250929050565b6000826131f7575060016108e2565b81613204575060006108e2565b816001811461321a576002811461322457613240565b60019150506108e2565b60ff8411156132355761323561318f565b50506001821b6108e2565b5060208310610133831016604e8410600b8410161715613263575081810a6108e2565b61326d83836131a5565b80600019048211156132815761328161318f565b029392505050565b60006121df60ff8416836131e8565b818103818111156108e2576108e261318f565b634e487b7160e01b600052603260045260246000fd5b604080825283519082018190526000906020906060840190828701845b828110156133035781516001600160a01b0316845292840192908401906001016132de565b50505093151592019190915250919050565b600060a0828403121561332757600080fd5b60405160a0810181811067ffffffffffffffff8211171561334a5761334a613030565b8060405250809150825161335d8161309b565b80825250602083015160208201526040830151604082015260608301516060820152608083015160808201525092915050565b600060208083850312156133a357600080fd5b825167ffffffffffffffff8111156133ba57600080fd5b8301601f810185136133cb57600080fd5b80516133d961310582613077565b81815260a091820283018401918482019190888411156133f857600080fd5b938501935b8385101561341e5761340f8986613315565b835293840193918501916133fd565b50979650505050505050565b80820281158282048414176108e2576108e261318f565b634e487b7160e01b600052601260045260246000fd5b60008261346657613466613441565b500490565b808201828112600083128015821682158216171561348b5761348b61318f565b505092915050565b80820260008212600160ff1b841416156134af576134af61318f565b81810583148215176108e2576108e261318f565b6000826134d2576134d2613441565b600160ff1b8214600019841416156134ec576134ec61318f565b500590565b81810360008312801583831316838312821617156127e2576127e261318f565b60006020828403121561352357600080fd5b5051919050565b60006101808201905060018060a01b038086168352808516602084015250825160408301526020830151606083015260408301516080830152606083015160a0830152608083015160c083015260a083015160e083015260c083015161010061359d818501836001600160a01b03169052565b60e0850151915061012082818601528186015161014086015280860151610160860152505050949350505050565b808201808211156108e2576108e261318f565b60208082526023908201527f496e70757420616d6f756e7420697320686967686572207468616e206d6178696040820152626d756d60e81b606082015260800190565b60208082526023908201527f4f757470757420616d6f756e74206973206c6f776572207468616e206d696e696040820152626d756d60e81b606082015260800190565b602080825260119082015270109390881a5cc81b9bdd081cd95b991959607a1b604082015260600190565b6000602082840312156136a157600080fd5b81516120e18161309b565b6001600160a01b0394851681529290931660208301526040820152606081019190915260800190565b6000600182016136e7576136e761318f565b5060010190565b600060a0828403121561370057600080fd5b6121df8383613315565b6000600160ff1b820161371f5761371f61318f565b5060000390565b60005b83811015613741578181015183820152602001613729565b50506000910152565b6001600160f81b03198152835160009061376b816001850160208901613726565b80830190508460018201528351613789816021840160208801613726565b0160210195945050505050565b600081518084526137ae816020860160208601613726565b601f01601f19169290920160200192915050565b84815283602082015260018060a01b03831660408201526080606082015260006137ef6080830184613796565b9695505050505050565b6000816138085761380861318f565b506000190190565b80516001600160701b0381168114612fd357600080fd5b60008060006060848603121561383c57600080fd5b61384584613810565b925061385360208501613810565b9150604084015163ffffffff8116811461386c57600080fd5b809150509250925092565b60208082526026908201527f50616e63616b654c6962726172793a20494e53554646494349454e545f4c495160408201526555494449545960d01b606082015260800190565b600082516138cf818460208701613726565b9190910192915050565b6020815260006121df602083018461379656feac788891c0eecf130a98c52555b1482d6b5cc80dcea810b33b4da4efb7ed5feda2646970667358221220168cd644644575b7f6ffe4e80f7a3344c0549618c157a78140ab1b872c0e013f64736f6c63430008110033000000000000000000000000babffce575929ddd7ad29deeeb5b7a5f5dee4ab6000000000000000000000000ad1fc0e22c13159884cf9fd1d46e3c2ad60c8f36000000000000000000000000a0b86991c6218b36c1d19d4a2e9eb0ce3606eb48000000000000000000000000c02aaa39b223fe8d0a0e5c4f27ead9083c756cc2
Deployed Bytecode
0x6080604052600436106100955760003560e01c80638da5cb5b116100595780638da5cb5b1461016c5780638dd950021461018a57806395001298146101aa578063c54e44eb146101bd578063f2fde38b146101dd57600080fd5b806316f0115b146100bb5780632c9dbd35146100f7578063715018a6146101175780637adbf9731461012c5780637dc0d1d01461014c57600080fd5b366100b6576002546001600160a01b031633146100b4576100b4612fa6565b005b600080fd5b3480156100c757600080fd5b506004546100db906001600160a01b031681565b6040516001600160a01b03909116815260200160405180910390f35b34801561010357600080fd5b506100b4610112366004612fd8565b6101fd565b34801561012357600080fd5b506100b4610232565b34801561013857600080fd5b506100b4610147366004613015565b610246565b34801561015857600080fd5b506003546100db906001600160a01b031681565b34801561017857600080fd5b506000546001600160a01b03166100db565b34801561019657600080fd5b506002546100db906001600160a01b031681565b6100b46101b83660046130b4565b610270565b3480156101c957600080fd5b506001546100db906001600160a01b031681565b3480156101e957600080fd5b506100b46101f8366004613015565b6103a0565b610205610419565b600480546001600160a01b0319166001600160a01b03841617905561022b81600a613289565b6005555050565b61023a610419565b6102446000610473565b565b61024e610419565b600380546001600160a01b0319166001600160a01b0392909216919091179055565b81428110156102bc5760405162461bcd60e51b8152602060048201526013602482015272151c985b9cd858dd1a5bdb88195e1c1a5c9959606a1b60448201526064015b60405180910390fd5b61031d604051806101400160405280606081526020016060815260200160001515815260200160008152602001600081526020016000815260200160008152602001600081526020016000815260200160006001600160a01b031681525090565b8781528615156040820152606081018690528661033a578561033d565b60005b60808201528661034f57600019610351565b845b60a082015286610362576000610364565b855b60c0820152866103745784610377565b60005b60e08201526001600160a01b03831661012082015261039633826104c3565b5050505050505050565b6103a8610419565b6001600160a01b03811661040d5760405162461bcd60e51b815260206004820152602660248201527f4f776e61626c653a206e6577206f776e657220697320746865207a65726f206160448201526564647265737360d01b60648201526084016102b3565b61041681610473565b50565b6000546001600160a01b031633146102445760405162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e657260448201526064016102b3565b600080546001600160a01b038381166001600160a01b0319831681178455604051919092169283917f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e09190a35050565b8051516001106105265760405162461bcd60e51b815260206004820152602860248201527f50617468206c656e677468206d757374206265206174206c6561737420322061604482015267646472657373657360c01b60648201526084016102b3565b80515160009061053890600190613298565b9050610542612f09565b61054a612f09565b6040805160028082526060820183526000926020830190803683370190505090508460000151600081518110610582576105826132ab565b60200260200101518160008151811061059d5761059d6132ab565b6001600160a01b039092166020928302919091019091015284518051859081106105c9576105c96132ab565b6020026020010151816001815181106105e4576105e46132ab565b6001600160a01b03928316602091820292909201015260035460405163ed0763bd60e01b8152600092919091169063ed0763bd906106299085906001906004016132c1565b600060405180830381865afa158015610646573d6000803e3d6000fd5b505050506040513d6000823e601f3d908101601f1916820160405261066e9190810190613390565b90506106ae82600081518110610686576106866132ab565b6020026020010151826000815181106106a1576106a16132ab565b602002602001015161088b565b93506106e1826001815181106106c6576106c66132ab565b6020026020010151826001815181106106a1576106a16132ab565b9250505080600001516001600160a01b031682600001516001600160a01b03160361074e5760405162461bcd60e51b815260206004820152601a60248201527f43616e27742073776170207468652073616d6520746f6b656e7300000000000060448201526064016102b3565b81602001518015610760575080602001515b1561079757600061077b8383876060015188604001516108e8565b905061078f86848484896101200151610afa565b505050505050565b81602001511580156107ab57508060200151155b156107c3576107bc85858484610d36565b5050505050565b816020015180156107e3575060015481516001600160a01b039081169116145b80610809575080602001518015610809575060015482516001600160a01b039081169116145b156108435760006108248383876060015188604001516108e8565b905061083a868484848961012001516000611190565b50505050505050565b61085584600001518360200151611551565b845260208201516107bc908690869061086e5783610870565b845b856020015161087f5785610881565b845b8660200151611626565b610893612f09565b6040518060c00160405280846001600160a01b03168152602001836000015115158152602001836020015181526020018360400151815260200183606001518152602001836080015181525090505b92915050565b6108f0612f4a565b60015485516000916001600160a01b03918216911614610914578560600151610918565b6005545b60015486519192506000916001600160a01b0390811691161461093f578560600151610943565b6005545b90508060055483610954919061342a565b61095e9190613457565b835250506040840151612710908190819061097a90829061346b565b83896040015161098a919061346b565b6109949190613493565b61099e91906134c3565b6109a891906134f1565b602083015250816109b957826109d2565b80516005546109c8908561342a565b6109d29190613457565b6040820152816109fa5760055481516109eb908561342a565b6109f59190613457565b6109fc565b825b60608201528115610a46576127108160200151612710610a1c919061346b565b8260400151610a2b919061342a565b610a359190613457565b608082015260a08101839052610a7f565b60808101839052602081015161271090610a6090826134f1565b8260600151610a6f919061342a565b610a799190613457565b60a08201525b8460200151610a8f578351610a92565b84515b6001600160a01b031660c0820152602080820151908601516127109190610abd578260a00151610ac3565b82608001515b610acd9190613493565b610ad791906134c3565b60e0820181905260c0820151610aec9161202c565b610120820152949350505050565b608082015184516040516370a0823160e01b81526001600160a01b038881166004830152909116906370a0823190602401602060405180830381865afa158015610b48573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610b6c9190613511565b1015610baf5760405162461bcd60e51b81526020600482015260126024820152714e6f7420656e6f7567682062616c616e636560701b60448201526064016102b3565b8351608083015160405163079cc67960e41b81526001600160a01b03888116600483015260248201929092529116906379cc679090604401600060405180830381600087803b158015610c0157600080fd5b505af1158015610c15573d6000803e3d6000fd5b5050845160a08501516040516308934a5f60e31b81526001600160a01b038a8116600483015260248201929092529116925063449a52f89150604401600060405180830381600087803b158015610c6b57600080fd5b505af1158015610c7f573d6000803e3d6000fd5b5050505060008260c001516001600160a01b031685600001516001600160a01b031614610cac5783610cae565b845b9050610cbf818460400151846120eb565b61010084018190528151610cd29161202c565b8361012001818151610ce491906134f1565b905250845184516040516001600160a01b038916927f1ab086cd308e1d16ad36455000c658ca32a39d4d8520babe71c257c74853e0d292610d2692889061352a565b60405180910390a2505050505050565b6000836040015115610d905760008260a0015111610d58578360600151610d86565b6127108260a00151612710610d6d91906135cb565b8560600151610d7c919061342a565b610d869190613457565b60c0850152610ddd565b50606083015160a0830151610da9578360600151610dd7565b6127108360a00151612710610dbe9190613298565b8560600151610dcd919061342a565b610dd79190613457565b60808501525b610de6846121e6565b836040015115610e635760008360a0015111610e06578360800151610e34565b6127108360a00151612710610e1b91906135cb565b8560800151610e2a919061342a565b610e349190613457565b90508360a0015184608001511115610e5e5760405162461bcd60e51b81526004016102b3906135de565b610e8b565b8360e001518460c001511015610e8b5760405162461bcd60e51b81526004016102b390613621565b6000610ed38560000151600081518110610ea757610ea76132ab565b60200260200101518660000151600181518110610ec657610ec66132ab565b6020026020010151612279565b60025485519192506001600160a01b039081169116036110485781341015610f0d5760405162461bcd60e51b81526004016102b390613664565b600260009054906101000a90046001600160a01b03166001600160a01b031663d0e30db0836040518263ffffffff1660e01b81526004016000604051808303818588803b158015610f5d57600080fd5b505af1158015610f71573d6000803e3d6000fd5b505060025460405163a9059cbb60e01b81526001600160a01b03868116600483015260248201889052909116935063a9059cbb925060440190506020604051808303816000875af1158015610fca573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610fee919061368f565b610ffa57610ffa612fa6565b81341115611043576001600160a01b0386166108fc6110198434613298565b6040518115909202916000818181858888f19350505050158015611041573d6000803e3d6000fd5b505b611072565b6110728560000151600081518110611062576110626132ab565b6020026020010151878385612442565b60025483516001600160a01b039182169116036111445761109c85602001518660000151306124a2565b60025460c0860151604051632e1a7d4d60e01b81526001600160a01b0390921691632e1a7d4d916110d39160040190815260200190565b600060405180830381600087803b1580156110ed57600080fd5b505af1158015611101573d6000803e3d6000fd5b5050505060c08501516040516001600160a01b038816916108fc811502916000818181858888f1935050505015801561113e573d6000803e3d6000fd5b50611157565b61115785602001518660000151886124a2565b83518351608087015160c08801516040516001600160a01b038b16946000805160206138ed83398151915294610d2694919390926136ac565b6000808460c001516001600160a01b031687600001516001600160a01b0316146111ba57856111bc565b865b60015488519192506001600160a01b0390811691160361130a5782611264576001546004805460808801516040516323b872dd60e01b81526001600160a01b038d811694820194909452918316602483015260448201529116906323b872dd906064016020604051808303816000875af115801561123e573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190611262919061368f565b505b855160a08601516040516308934a5f60e31b81526001600160a01b038b81166004830152602482019290925291169063449a52f890604401600060405180830381600087803b1580156112b657600080fd5b505af11580156112ca573d6000803e3d6000fd5b505050506112dd818660600151866120eb565b610100860181905281516112f09161202c565b856101200181815161130291906134f1565b9052506114ff565b60a08501516004805460408051629032ff60e51b815290516001600160a01b03909216926312065fe09282820192602092908290030181865afa158015611355573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906113799190613511565b10156113c75760405162461bcd60e51b815260206004820181905260248201527f4e6f7420656e6f756768206c697175696469747920706f6f6c20616d6f756e7460448201526064016102b3565b8651608086015160405163079cc67960e41b81526001600160a01b038b8116600483015260248201929092529116906379cc679090604401600060405180830381600087803b15801561141957600080fd5b505af115801561142d573d6000803e3d6000fd5b50505050611440818660400151866120eb565b610100860181905281516114539161202c565b856101200181815161146591906134f1565b905250826114f7576001546004805460a08801516040516323b872dd60e01b81526001600160a01b03928316938101939093528b821660248401526044830152909116906323b872dd906064016020604051808303816000875af11580156114d1573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906114f5919061368f565b505b8460a0015191505b865186516040516001600160a01b038b16927f1ab086cd308e1d16ad36455000c658ca32a39d4d8520babe71c257c74853e0d29261153e928a9061352a565b60405180910390a2509695505050505050565b60606000600184516115639190613298565b67ffffffffffffffff81111561157b5761157b613030565b6040519080825280602002602001820160405280156115a4578160200160208202803683370190505b50905060005b815181101561161e5784846115bf57816115ca565b6115ca8260016135cb565b815181106115da576115da6132ab565b60200260200101518282815181106115f4576115f46132ab565b6001600160a01b039092166020928302919091019091015280611616816136d5565b9150506115aa565b509392505050565b600154600354604051634ff4a2a760e11b81526001600160a01b0392831660048201819052600060248301819052936116ad939192911690639fe9454e9060440160a060405180830381865afa158015611684573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906116a891906136ee565b61088b565b9050600060018660000151516116c39190613298565b9050828015611705575081600001516001600160a01b031686600001516000815181106116f2576116f26132ab565b60200260200101516001600160a01b0316145b8061174b57508215801561174b575081600001516001600160a01b031686600001518281518110611738576117386132ab565b60200260200101516001600160a01b0316145b6117bd5760405162461bcd60e51b815260206004820152603a60248201527f5468652065786368616e6765206265747765656e206669617420616e6420637260448201527f7970746f206d75737420626520646f6e6520766961205553445400000000000060648201526084016102b3565b6117c5612f4a565b866040015115611d28578315611a2357606087015160c088015260a08501511561181d576127108560a001516127106117fe91906135cb565b8860c0015161180d919061342a565b6118179190613457565b60c08801525b611826876121e6565b6118378684896080015160016108e8565b90508660a00151816080015111156118615760405162461bcd60e51b81526004016102b3906135de565b611875888785848b61012001516001611190565b50600061189784600001518960000151600181518110610ec657610ec66132ab565b60015460045460808b01519293506118c0926001600160a01b0392831692909116908490612442565b5060025485516001600160a01b03918216911603611993576118eb87602001518860000151306124a2565b60025460c0880151604051632e1a7d4d60e01b81526001600160a01b0390921691632e1a7d4d916119229160040190815260200190565b600060405180830381600087803b15801561193c57600080fd5b505af1158015611950573d6000803e3d6000fd5b5050505060c08701516040516001600160a01b038a16916108fc811502916000818181858888f1935050505015801561198d573d6000803e3d6000fd5b506119a6565b6119a6876020015188600001518a6124a2565b876001600160a01b03166000805160206138ed83398151915288600001516000815181106119d6576119d66132ab565b6020026020010151896000015185815181106119f4576119f46132ab565b60200260200101518a608001518b60c00151604051611a1694939291906136ac565b60405180910390a2610396565b611a348387896060015160016108e8565b608081015160c08901529050611a49876121e6565b8660a0015187608001511115611a715760405162461bcd60e51b81526004016102b3906135de565b6000611a9286600001518960000151600181518110610ec657610ec66132ab565b60025487519192506001600160a01b03908116911603611c1e578760800151341015611ad05760405162461bcd60e51b81526004016102b390613664565b600260009054906101000a90046001600160a01b03166001600160a01b031663d0e30db089608001516040518263ffffffff1660e01b81526004016000604051808303818588803b158015611b2457600080fd5b505af1158015611b38573d6000803e3d6000fd5b505060025460808c015160405163a9059cbb60e01b81526001600160a01b03878116600483015260248201929092529116935063a9059cbb925060440190506020604051808303816000875af1158015611b96573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190611bba919061368f565b611bc657611bc6612fa6565b8760800151341115611c1957886001600160a01b03166108fc896080015134611bef9190613298565b6040518115909202916000818181858888f19350505050158015611c17573d6000803e3d6000fd5b505b611c77565b6000808760a0015111611c35578860800151611c63565b6127108760a00151612710611c4a91906135cb565b8a60800151611c59919061342a565b611c639190613457565b9050611c7587600001518b8484612442565b505b5060208701518751600454611c969291906001600160a01b03166124a2565b876001600160a01b03166000805160206138ed8339815191528860000151600081518110611cc657611cc66132ab565b602002602001015189600001518581518110611ce457611ce46132ab565b60200260200101518a608001518b60c00151604051611d0694939291906136ac565b60405180910390a2611d22888488848b61012001516001611190565b50610396565b8315611d7c57611d3f8684896060015160006108e8565b60a081015160808901529050611d54876121e6565b8660e001518760c0015110156118615760405162461bcd60e51b81526004016102b390613621565b6060870151608088015260a085015115611dc4576127108560a00151612710611da59190613298565b8860800151611db4919061342a565b611dbe9190613457565b60808801525b611dcd876121e6565b611dde83878960c0015160006108e8565b90508660e001518160a001511015611e085760405162461bcd60e51b81526004016102b390613621565b6000611e2986600001518960000151600181518110610ec657610ec66132ab565b60025487519192506001600160a01b03908116911603611f62578760600151341015611e675760405162461bcd60e51b81526004016102b390613664565b600260009054906101000a90046001600160a01b03166001600160a01b031663d0e30db089606001516040518263ffffffff1660e01b81526004016000604051808303818588803b158015611ebb57600080fd5b505af1158015611ecf573d6000803e3d6000fd5b505060025460608c015160405163a9059cbb60e01b81526001600160a01b03878116600483015260248201929092529116935063a9059cbb925060440190506020604051808303816000875af1158015611f2d573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190611f51919061368f565b611f5d57611f5d612fa6565b611f76565b611f7686600001518a838b60600151612442565b5060208701518751600454611f959291906001600160a01b03166124a2565b876001600160a01b03166000805160206138ed8339815191528860000151600081518110611fc557611fc56132ab565b602002602001015189600001518581518110611fe357611fe36132ab565b60200260200101518a608001518b60c0015160405161200594939291906136ac565b60405180910390a2612021888488848b61012001516001611190565b505050505050505050565b60008160000361203e575060006108e2565b6005546003546040516341976e0960e01b81526001600160a01b0386811660048301526000939286929116906341976e0990602401602060405180830381865afa158015612090573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906120b49190613511565b6120be919061342a565b6120c89190613457565b905060008312156120e1576120dc8161370a565b6120e3565b805b949350505050565b60006001600160a01b038216156121db57608084015180156121d9576000612710612116868461342a565b6121209190613457565b86516040516308934a5f60e31b81526001600160a01b0387811660048301526024820184905292935091169063449a52f890604401600060405180830381600087803b15801561216f57600080fd5b505af1158015612183573d6000803e3d6000fd5b50508751604080516001600160a01b0392831681526020810186905291881693507f44db3609235da669e4ce17e6c91be1fadce4923ec8b39f2e3b9e33dffd3214cf92500160405180910390a291506121df9050565b505b5060005b9392505050565b80604001511561222e576122028160c001518260000151612655565b60208201819052805160009061221a5761221a6132ab565b602002602001015181608001818152505050565b612240816080015182600001516127e9565b60208201819052805161225590600190613298565b81518110612265576122656132ab565b60200260200101518160c001818152505050565b6000806000612288858561295e565b60408051808201825260148152735c69bee701ef814a2b6a3edd4b1652cb9cc5aa6f60601b60208083019190915282518084019093528083527f96e8ac4277198ff8b6f785478aa9a39f403cb768dd02cbee326c3e7da348845f908301529294509092504660380361235b5760405180604001604052806014815260200173ca143ce32fe78f1f7019d7d551a6402fc5350c7360601b81525091506040518060400160405280602081526020017efb7f630766e6a796048ea87d01acd3068e8ff67d078148a3fa3f4a84f69bd581525090505b466061036123cb57604051806040016040528060148152602001732de49b010c2bec1ea9f7bf79b6a18ab82f79d9ef60621b81525091506040518060400160405280602081526020017fecba335299a6693cb2ebc4782e74669b84290b6378ea3a3873c7231a8d7d107481525090505b6040516bffffffffffffffffffffffff19606086811b8216602084015285901b1660348201528290604801604051602081830303815290604052805190602001208260405160200161241f9392919061374a565b60408051601f198184030181529190528051602090910120979650505050505050565b604080516001600160a01b0385811660248301528416604482015260648082018490528251808303909101815260849091019091526020810180516001600160e01b03166323b872dd60e01b17905261249c908590612a53565b50505050565b60005b600183516124b39190613298565b81101561249c576000808483815181106124cf576124cf6132ab565b6020026020010151858460016124e591906135cb565b815181106124f5576124f56132ab565b602002602001015191509150600061250d838361295e565b50905060008761251e8660016135cb565b8151811061252e5761252e6132ab565b60200260200101519050600080836001600160a01b0316866001600160a01b03161461255c57826000612560565b6000835b91509150600060028a516125749190613298565b881061258057886125a0565b6125a0868b6125908b60026135cb565b81518110610ec657610ec66132ab565b90506125ac8787612279565b6001600160a01b031663022c0d9f84848460006040519080825280601f01601f1916602001820160405280156125e9576020820181803683370190505b506040518563ffffffff1660e01b815260040161260994939291906137c2565b600060405180830381600087803b15801561262357600080fd5b505af1158015612637573d6000803e3d6000fd5b5050505050505050505050808061264d906136d5565b9150506124a5565b60606002825110156126a95760405162461bcd60e51b815260206004820152601c60248201527f50616e63616b654c6962726172793a20494e56414c49445f504154480000000060448201526064016102b3565b815167ffffffffffffffff8111156126c3576126c3613030565b6040519080825280602002602001820160405280156126ec578160200160208202803683370190505b5090508281600183516126ff9190613298565b8151811061270f5761270f6132ab565b60200260200101818152505060006001835161272b9190613298565b90505b80156127e25760008061277d85612746600186613298565b81518110612756576127566132ab565b6020026020010151868581518110612770576127706132ab565b6020026020010151612b2a565b915091506127a5848481518110612796576127966132ab565b60200260200101518383612bfd565b846127b1600186613298565b815181106127c1576127c16132ab565b602002602001018181525050505080806127da906137f9565b91505061272e565b5092915050565b606060028251101561283d5760405162461bcd60e51b815260206004820152601c60248201527f50616e63616b654c6962726172793a20494e56414c49445f504154480000000060448201526064016102b3565b815167ffffffffffffffff81111561285757612857613030565b604051908082528060200260200182016040528015612880578160200160208202803683370190505b5090508281600081518110612897576128976132ab565b60200260200101818152505060005b600183516128b49190613298565b8110156127e2576000806128f98584815181106128d3576128d36132ab565b6020026020010151868560016128e991906135cb565b81518110612770576127706132ab565b91509150612921848481518110612912576129126132ab565b60200260200101518383612cfd565b8461292d8560016135cb565b8151811061293d5761293d6132ab565b60200260200101818152505050508080612956906136d5565b9150506128a6565b600080826001600160a01b0316846001600160a01b0316036129ce5760405162461bcd60e51b815260206004820152602360248201527f50616e63616b654c6962726172793a204944454e544943414c5f41444452455360448201526253455360e81b60648201526084016102b3565b826001600160a01b0316846001600160a01b0316106129ee5782846129f1565b83835b90925090506001600160a01b038216612a4c5760405162461bcd60e51b815260206004820152601c60248201527f50616e63616b654c6962726172793a205a45524f5f414444524553530000000060448201526064016102b3565b9250929050565b6000612aa8826040518060400160405280602081526020017f5361666545524332303a206c6f772d6c6576656c2063616c6c206661696c6564815250856001600160a01b0316612df69092919063ffffffff16565b805190915015612b255780806020019051810190612ac6919061368f565b612b255760405162461bcd60e51b815260206004820152602a60248201527f5361666545524332303a204552433230206f7065726174696f6e20646964206e6044820152691bdd081cdd58d8d9595960b21b60648201526084016102b3565b505050565b6000806000612b39858561295e565b509050612b468585612279565b50600080612b548787612279565b6001600160a01b0316630902f1ac6040518163ffffffff1660e01b8152600401606060405180830381865afa158015612b91573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190612bb59190613827565b506001600160701b031691506001600160701b03169150826001600160a01b0316876001600160a01b031614612bec578082612bef565b81815b909890975095505050505050565b6000808411612c615760405162461bcd60e51b815260206004820152602a60248201527f50616e63616b654c6962726172793a20494e53554646494349454e545f4f555460448201526914155517d05353d5539560b21b60648201526084016102b3565b600083118015612c715750600082115b612c8d5760405162461bcd60e51b81526004016102b390613877565b6126f246603803612c9d57506126f75b46606103612caa57506126fc5b6000612cb6868661342a565b612cc29061271061342a565b9050600082612cd18887613298565b612cdb919061342a565b9050612ce78183613457565b612cf29060016135cb565b979650505050505050565b6000808411612d605760405162461bcd60e51b815260206004820152602960248201527f50616e63616b654c6962726172793a20494e53554646494349454e545f494e50604482015268155517d05353d5539560ba1b60648201526084016102b3565b600083118015612d705750600082115b612d8c5760405162461bcd60e51b81526004016102b390613877565b6126f246603803612d9c57506126f75b46606103612da957506126fc5b6000612db5828761342a565b90506000612dc3858361342a565b9050600082612dd48861271061342a565b612dde91906135cb565b9050612dea8183613457565b98975050505050505050565b60606120e3848460008585600080866001600160a01b03168587604051612e1d91906138bd565b60006040518083038185875af1925050503d8060008114612e5a576040519150601f19603f3d011682016040523d82523d6000602084013e612e5f565b606091505b5091509150612cf28783838760608315612eda578251600003612ed3576001600160a01b0385163b612ed35760405162461bcd60e51b815260206004820152601d60248201527f416464726573733a2063616c6c20746f206e6f6e2d636f6e747261637400000060448201526064016102b3565b50816120e3565b6120e38383815115612eef5781518083602001fd5b8060405162461bcd60e51b81526004016102b391906138d9565b6040518060c0016040528060006001600160a01b03168152602001600015158152602001600081526020016000815260200160008152602001600081525090565b60405180610140016040528060008152602001600081526020016000815260200160008152602001600081526020016000815260200160006001600160a01b031681526020016000815260200160008152602001600081525090565b634e487b7160e01b600052600160045260246000fd5b80356001600160a01b0381168114612fd357600080fd5b919050565b60008060408385031215612feb57600080fd5b612ff483612fbc565b9150602083013560ff8116811461300a57600080fd5b809150509250929050565b60006020828403121561302757600080fd5b6121df82612fbc565b634e487b7160e01b600052604160045260246000fd5b604051601f8201601f1916810167ffffffffffffffff8111828210171561306f5761306f613030565b604052919050565b600067ffffffffffffffff82111561309157613091613030565b5060051b60200190565b801515811461041657600080fd5b8035612fd38161309b565b60008060008060008060c087890312156130cd57600080fd5b863567ffffffffffffffff8111156130e457600080fd5b8701601f810189136130f557600080fd5b8035602061310a61310583613077565b613046565b82815260059290921b8301810191818101908c84111561312957600080fd5b938201935b8385101561314e5761313f85612fbc565b8252938201939082019061312e565b995061315d90508a82016130a9565b975050505060408701359350606087013592506080870135915061318360a08801612fbc565b90509295509295509295565b634e487b7160e01b600052601160045260246000fd5b600181815b808511156131e05781600019048211156131c6576131c661318f565b808516156131d357918102915b93841c93908002906131aa565b509250929050565b6000826131f7575060016108e2565b81613204575060006108e2565b816001811461321a576002811461322457613240565b60019150506108e2565b60ff8411156132355761323561318f565b50506001821b6108e2565b5060208310610133831016604e8410600b8410161715613263575081810a6108e2565b61326d83836131a5565b80600019048211156132815761328161318f565b029392505050565b60006121df60ff8416836131e8565b818103818111156108e2576108e261318f565b634e487b7160e01b600052603260045260246000fd5b604080825283519082018190526000906020906060840190828701845b828110156133035781516001600160a01b0316845292840192908401906001016132de565b50505093151592019190915250919050565b600060a0828403121561332757600080fd5b60405160a0810181811067ffffffffffffffff8211171561334a5761334a613030565b8060405250809150825161335d8161309b565b80825250602083015160208201526040830151604082015260608301516060820152608083015160808201525092915050565b600060208083850312156133a357600080fd5b825167ffffffffffffffff8111156133ba57600080fd5b8301601f810185136133cb57600080fd5b80516133d961310582613077565b81815260a091820283018401918482019190888411156133f857600080fd5b938501935b8385101561341e5761340f8986613315565b835293840193918501916133fd565b50979650505050505050565b80820281158282048414176108e2576108e261318f565b634e487b7160e01b600052601260045260246000fd5b60008261346657613466613441565b500490565b808201828112600083128015821682158216171561348b5761348b61318f565b505092915050565b80820260008212600160ff1b841416156134af576134af61318f565b81810583148215176108e2576108e261318f565b6000826134d2576134d2613441565b600160ff1b8214600019841416156134ec576134ec61318f565b500590565b81810360008312801583831316838312821617156127e2576127e261318f565b60006020828403121561352357600080fd5b5051919050565b60006101808201905060018060a01b038086168352808516602084015250825160408301526020830151606083015260408301516080830152606083015160a0830152608083015160c083015260a083015160e083015260c083015161010061359d818501836001600160a01b03169052565b60e0850151915061012082818601528186015161014086015280860151610160860152505050949350505050565b808201808211156108e2576108e261318f565b60208082526023908201527f496e70757420616d6f756e7420697320686967686572207468616e206d6178696040820152626d756d60e81b606082015260800190565b60208082526023908201527f4f757470757420616d6f756e74206973206c6f776572207468616e206d696e696040820152626d756d60e81b606082015260800190565b602080825260119082015270109390881a5cc81b9bdd081cd95b991959607a1b604082015260600190565b6000602082840312156136a157600080fd5b81516120e18161309b565b6001600160a01b0394851681529290931660208301526040820152606081019190915260800190565b6000600182016136e7576136e761318f565b5060010190565b600060a0828403121561370057600080fd5b6121df8383613315565b6000600160ff1b820161371f5761371f61318f565b5060000390565b60005b83811015613741578181015183820152602001613729565b50506000910152565b6001600160f81b03198152835160009061376b816001850160208901613726565b80830190508460018201528351613789816021840160208801613726565b0160210195945050505050565b600081518084526137ae816020860160208601613726565b601f01601f19169290920160200192915050565b84815283602082015260018060a01b03831660408201526080606082015260006137ef6080830184613796565b9695505050505050565b6000816138085761380861318f565b506000190190565b80516001600160701b0381168114612fd357600080fd5b60008060006060848603121561383c57600080fd5b61384584613810565b925061385360208501613810565b9150604084015163ffffffff8116811461386c57600080fd5b809150509250925092565b60208082526026908201527f50616e63616b654c6962726172793a20494e53554646494349454e545f4c495160408201526555494449545960d01b606082015260800190565b600082516138cf818460208701613726565b9190910192915050565b6020815260006121df602083018461379656feac788891c0eecf130a98c52555b1482d6b5cc80dcea810b33b4da4efb7ed5feda2646970667358221220168cd644644575b7f6ffe4e80f7a3344c0549618c157a78140ab1b872c0e013f64736f6c63430008110033
Constructor Arguments (ABI-Encoded and is the last bytes of the Contract Creation Code above)
000000000000000000000000babffce575929ddd7ad29deeeb5b7a5f5dee4ab6000000000000000000000000ad1fc0e22c13159884cf9fd1d46e3c2ad60c8f36000000000000000000000000a0b86991c6218b36c1d19d4a2e9eb0ce3606eb48000000000000000000000000c02aaa39b223fe8d0a0e5c4f27ead9083c756cc2
-----Decoded View---------------
Arg [0] : _oracleAddress (address): 0xBaBfFCe575929DDd7aD29DEEeb5B7A5F5dee4Ab6
Arg [1] : _poolAddress (address): 0xAD1Fc0E22C13159884Cf9FD1d46e3C2Ad60C8F36
Arg [2] : _usdtAddress (address): 0xA0b86991c6218b36c1d19D4a2e9Eb0cE3606eB48
Arg [3] : _wbnbAddress (address): 0xC02aaA39b223FE8D0A0e5C4F27eAD9083C756Cc2
-----Encoded View---------------
4 Constructor Arguments found :
Arg [0] : 000000000000000000000000babffce575929ddd7ad29deeeb5b7a5f5dee4ab6
Arg [1] : 000000000000000000000000ad1fc0e22c13159884cf9fd1d46e3c2ad60c8f36
Arg [2] : 000000000000000000000000a0b86991c6218b36c1d19d4a2e9eb0ce3606eb48
Arg [3] : 000000000000000000000000c02aaa39b223fe8d0a0e5c4f27ead9083c756cc2
Loading...
Loading
Loading...
Loading
Multichain Portfolio | 26 Chains
Chain | Token | Portfolio % | Price | Amount | Value |
---|
Loading...
Loading
[ Download: CSV Export ]
[ 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.