Transactions
Token Transfers
Tokens
Internal Transactions
Coin Balance History
Logs
Code
Read Contract
Write Contract
Contract is not verified. However, we found a verified contract with the same bytecode in Blockscout DB 0x69d7c0b81122b3f6964de6744f27a2a02e1840ba.
All metadata displayed below is from that contract. In order to verify current contract, click Verify & Publish button
Verify & Publish
All metadata displayed below is from that contract. In order to verify current contract, click Verify & Publish button
- Contract name:
- StrategyLP2
- Optimization enabled
- true
- Compiler version
- v0.6.12+commit.27d51765
- Optimization runs
- 200
- Verified at
- 2022-06-28T15:17:31.047882Z
Contract source code
// Sources flattened with hardhat v2.4.3 https://hardhat.org // File contracts/ISwapPathRegistry.sol pragma solidity ^0.6.0; interface ISwapPathRegistry { function getSwapRoute(address _router, address _fromToken, address _toToken) external view returns (address[] memory); } // File contracts/IPanwexPair.sol pragma solidity ^0.6.0; interface IPanwexPair { function token0() external view returns (address); function token1() external view returns (address); } // File @uniswap/v2-periphery/contracts/interfaces/[email protected] pragma solidity >=0.6.2; interface IUniswapV2Router01 { function factory() external pure returns (address); function WETH() external pure returns (address); function addLiquidity( address tokenA, address tokenB, uint amountADesired, uint amountBDesired, uint amountAMin, uint amountBMin, address to, uint deadline ) external returns (uint amountA, uint amountB, uint liquidity); function addLiquidityETH( address token, uint amountTokenDesired, uint amountTokenMin, uint amountETHMin, address to, uint deadline ) external payable returns (uint amountToken, uint amountETH, uint liquidity); function removeLiquidity( address tokenA, address tokenB, uint liquidity, uint amountAMin, uint amountBMin, address to, uint deadline ) external returns (uint amountA, uint amountB); function removeLiquidityETH( address token, uint liquidity, uint amountTokenMin, uint amountETHMin, address to, uint deadline ) external returns (uint amountToken, uint amountETH); function removeLiquidityWithPermit( address tokenA, address tokenB, uint liquidity, uint amountAMin, uint amountBMin, address to, uint deadline, bool approveMax, uint8 v, bytes32 r, bytes32 s ) external returns (uint amountA, uint amountB); function removeLiquidityETHWithPermit( address token, uint liquidity, uint amountTokenMin, uint amountETHMin, address to, uint deadline, bool approveMax, uint8 v, bytes32 r, bytes32 s ) external returns (uint amountToken, uint amountETH); function swapExactTokensForTokens( uint amountIn, uint amountOutMin, address[] calldata path, address to, uint deadline ) external returns (uint[] memory amounts); function swapTokensForExactTokens( uint amountOut, uint amountInMax, address[] calldata path, address to, uint deadline ) external returns (uint[] memory amounts); function swapExactETHForTokens(uint amountOutMin, address[] calldata path, address to, uint deadline) external payable returns (uint[] memory amounts); function swapTokensForExactETH(uint amountOut, uint amountInMax, address[] calldata path, address to, uint deadline) external returns (uint[] memory amounts); function swapExactTokensForETH(uint amountIn, uint amountOutMin, address[] calldata path, address to, uint deadline) external returns (uint[] memory amounts); function swapETHForExactTokens(uint amountOut, address[] calldata path, address to, uint deadline) external payable returns (uint[] memory amounts); function quote(uint amountA, uint reserveA, uint reserveB) external pure returns (uint amountB); function getAmountOut(uint amountIn, uint reserveIn, uint reserveOut) external pure returns (uint amountOut); function getAmountIn(uint amountOut, uint reserveIn, uint reserveOut) external pure returns (uint amountIn); function getAmountsOut(uint amountIn, address[] calldata path) external view returns (uint[] memory amounts); function getAmountsIn(uint amountOut, address[] calldata path) external view returns (uint[] memory amounts); } // File @uniswap/v2-periphery/contracts/interfaces/[email protected] pragma solidity >=0.6.2; interface IUniswapV2Router02 is IUniswapV2Router01 { function removeLiquidityETHSupportingFeeOnTransferTokens( address token, uint liquidity, uint amountTokenMin, uint amountETHMin, address to, uint deadline ) external returns (uint amountETH); function removeLiquidityETHWithPermitSupportingFeeOnTransferTokens( address token, uint liquidity, uint amountTokenMin, uint amountETHMin, address to, uint deadline, bool approveMax, uint8 v, bytes32 r, bytes32 s ) external returns (uint amountETH); function swapExactTokensForTokensSupportingFeeOnTransferTokens( uint amountIn, uint amountOutMin, address[] calldata path, address to, uint deadline ) external; function swapExactETHForTokensSupportingFeeOnTransferTokens( uint amountOutMin, address[] calldata path, address to, uint deadline ) external payable; function swapExactTokensForETHSupportingFeeOnTransferTokens( uint amountIn, uint amountOutMin, address[] calldata path, address to, uint deadline ) external; } // File contracts/KCSRouter2.sol pragma solidity >=0.6.2; interface KCSRouter2 is IUniswapV2Router02{ function addLiquidityKCS( address token, uint amountTokenDesired, uint amountTokenMin, uint amountETHMin, address to, uint deadline ) external payable returns (uint amountToken, uint amountETH, uint liquidity); // function addLiquidity( // address tokenA, // address tokenB, // uint amountADesired, // uint amountBDesired, // uint amountAMin, // uint amountBMin, // address to, // uint deadline // ) external returns (uint amountA, uint amountB, uint liquidity); // function getAmountsOut(uint amountIn, address[] calldata path) external view returns (uint[] memory amounts); // function swapExactTokensForTokensSupportingFeeOnTransferTokens( // uint amountIn, // uint amountOutMin, // address[] calldata path, // address to, // uint deadline // ) external; } // File @openzeppelin/contracts/GSN/[email protected] // SPDX-License-Identifier: MIT pragma solidity ^0.6.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 GSN 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 payable) { return msg.sender; } function _msgData() internal view virtual returns (bytes memory) { this; // silence state mutability warning without generating bytecode - see https://github.com/ethereum/solidity/issues/2691 return msg.data; } } // File contracts/Pausable.sol pragma solidity ^0.6.0; /** * @dev Contract module which allows children to implement an emergency stop * mechanism that can be triggered by an authorized account. * * This module is used through inheritance. It will make available the * modifiers `whenNotPaused` and `whenPaused`, which can be applied to * the functions of your contract. Note that they will not be pausable by * simply including this module, only once the modifiers are put in place. */ contract Pausable is Context { /** * @dev Emitted when the pause is triggered by `account`. */ event Paused(address account); /** * @dev Emitted when the pause is lifted by `account`. */ event Unpaused(address account); bool private _paused; /** * @dev Initializes the contract in unpaused state. */ constructor () internal { _paused = false; } /** * @dev Returns true if the contract is paused, and false otherwise. */ function paused() public view returns (bool) { return _paused; } /** * @dev Modifier to make a function callable only when the contract is not paused. * * Requirements: * * - The contract must not be paused. */ modifier whenNotPaused() { require(!_paused, "Pausable: paused"); _; } /** * @dev Modifier to make a function callable only when the contract is paused. * * Requirements: * * - The contract must be paused. */ modifier whenPaused() { require(_paused, "Pausable: not paused"); _; } /** * @dev Triggers stopped state. * * Requirements: * * - The contract must not be paused. */ function _pause() internal virtual whenNotPaused { _paused = true; emit Paused(_msgSender()); } /** * @dev Returns to normal state. * * Requirements: * * - The contract must be paused. */ function _unpause() internal virtual whenPaused { _paused = false; emit Unpaused(_msgSender()); } } // File contracts/Buybackstrat.sol pragma solidity ^0.6.0; interface Buybackstrat{ function performanceFeeBps() external view returns(uint); function withdrawalFeesBps() external view returns(uint); function setBuybackStrat() external; } // File @openzeppelin/contracts/token/ERC20/[email protected] pragma solidity ^0.6.0; /** * @dev Interface of the ERC20 standard as defined in the EIP. */ interface IERC20 { /** * @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 `recipient`. * * Returns a boolean value indicating whether the operation succeeded. * * Emits a {Transfer} event. */ function transfer(address recipient, 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 `sender` to `recipient` 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 sender, address recipient, uint256 amount) external returns (bool); /** * @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); } // File @openzeppelin/contracts/math/[email protected] pragma solidity ^0.6.0; /** * @dev Wrappers over Solidity's arithmetic operations with added overflow * checks. * * Arithmetic operations in Solidity wrap on overflow. This can easily result * in bugs, because programmers usually assume that an overflow raises an * error, which is the standard behavior in high level programming languages. * `SafeMath` restores this intuition by reverting the transaction when an * operation overflows. * * Using this library instead of the unchecked operations eliminates an entire * class of bugs, so it's recommended to use it always. */ library SafeMath { /** * @dev Returns the addition of two unsigned integers, reverting on * overflow. * * Counterpart to Solidity's `+` operator. * * Requirements: * * - Addition cannot overflow. */ function add(uint256 a, uint256 b) internal pure returns (uint256) { uint256 c = a + b; require(c >= a, "SafeMath: addition overflow"); return c; } /** * @dev Returns the subtraction of two unsigned integers, reverting on * overflow (when the result is negative). * * Counterpart to Solidity's `-` operator. * * Requirements: * * - Subtraction cannot overflow. */ function sub(uint256 a, uint256 b) internal pure returns (uint256) { return sub(a, b, "SafeMath: subtraction overflow"); } /** * @dev Returns the subtraction of two unsigned integers, reverting with custom message on * overflow (when the result is negative). * * Counterpart to Solidity's `-` operator. * * Requirements: * * - Subtraction cannot overflow. */ function sub(uint256 a, uint256 b, string memory errorMessage) internal pure returns (uint256) { require(b <= a, errorMessage); uint256 c = a - b; return c; } /** * @dev Returns the multiplication of two unsigned integers, reverting on * overflow. * * Counterpart to Solidity's `*` operator. * * Requirements: * * - Multiplication cannot overflow. */ function mul(uint256 a, uint256 b) internal pure returns (uint256) { // Gas optimization: this is cheaper than requiring 'a' not being zero, but the // benefit is lost if 'b' is also tested. // See: https://github.com/OpenZeppelin/openzeppelin-contracts/pull/522 if (a == 0) { return 0; } uint256 c = a * b; require(c / a == b, "SafeMath: multiplication overflow"); return c; } /** * @dev Returns the integer division of two unsigned integers. Reverts on * division by zero. The result is rounded towards zero. * * Counterpart to Solidity's `/` operator. Note: this function uses a * `revert` opcode (which leaves remaining gas untouched) while Solidity * uses an invalid opcode to revert (consuming all remaining gas). * * Requirements: * * - The divisor cannot be zero. */ function div(uint256 a, uint256 b) internal pure returns (uint256) { return div(a, b, "SafeMath: division by zero"); } /** * @dev Returns the integer division of two unsigned integers. Reverts with custom message on * division by zero. The result is rounded towards zero. * * Counterpart to Solidity's `/` operator. Note: this function uses a * `revert` opcode (which leaves remaining gas untouched) while Solidity * uses an invalid opcode to revert (consuming all remaining gas). * * Requirements: * * - The divisor cannot be zero. */ function div(uint256 a, uint256 b, string memory errorMessage) internal pure returns (uint256) { require(b > 0, errorMessage); uint256 c = a / b; // assert(a == b * c + a % b); // There is no case in which this doesn't hold return c; } /** * @dev Returns the remainder of dividing two unsigned integers. (unsigned integer modulo), * Reverts when dividing by zero. * * Counterpart to Solidity's `%` operator. This function uses a `revert` * opcode (which leaves remaining gas untouched) while Solidity uses an * invalid opcode to revert (consuming all remaining gas). * * Requirements: * * - The divisor cannot be zero. */ function mod(uint256 a, uint256 b) internal pure returns (uint256) { return mod(a, b, "SafeMath: modulo by zero"); } /** * @dev Returns the remainder of dividing two unsigned integers. (unsigned integer modulo), * Reverts with custom message when dividing by zero. * * Counterpart to Solidity's `%` operator. This function uses a `revert` * opcode (which leaves remaining gas untouched) while Solidity uses an * invalid opcode to revert (consuming all remaining gas). * * Requirements: * * - The divisor cannot be zero. */ function mod(uint256 a, uint256 b, string memory errorMessage) internal pure returns (uint256) { require(b != 0, errorMessage); return a % b; } } // File @openzeppelin/contracts/utils/[email protected] pragma solidity ^0.6.2; /** * @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 * ==== */ function isContract(address account) internal view returns (bool) { // According to EIP-1052, 0x0 is the value returned for not-yet created accounts // and 0xc5d2460186f7233c927e7db2dcc703c0e500b653ca82273b7bfad8045d85a470 is returned // for accounts without code, i.e. `keccak256('')` bytes32 codehash; bytes32 accountHash = 0xc5d2460186f7233c927e7db2dcc703c0e500b653ca82273b7bfad8045d85a470; // solhint-disable-next-line no-inline-assembly assembly { codehash := extcodehash(account) } return (codehash != accountHash && codehash != 0x0); } /** * @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"); // solhint-disable-next-line avoid-low-level-calls, avoid-call-value (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 functionCall(target, data, "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"); return _functionCallWithValue(target, data, value, errorMessage); } function _functionCallWithValue(address target, bytes memory data, uint256 weiValue, string memory errorMessage) private returns (bytes memory) { require(isContract(target), "Address: call to non-contract"); // solhint-disable-next-line avoid-low-level-calls (bool success, bytes memory returndata) = target.call{ value: weiValue }(data); if (success) { return returndata; } else { // 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 // solhint-disable-next-line no-inline-assembly assembly { let returndata_size := mload(returndata) revert(add(32, returndata), returndata_size) } } else { revert(errorMessage); } } } } // File @openzeppelin/contracts/token/ERC20/[email protected] pragma solidity ^0.6.0; /** * @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 SafeMath for uint256; 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' // solhint-disable-next-line max-line-length 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).add(value); _callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, newAllowance)); } function safeDecreaseAllowance(IERC20 token, address spender, uint256 value) internal { uint256 newAllowance = token.allowance(address(this), spender).sub(value, "SafeERC20: decreased allowance below zero"); _callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, newAllowance)); } /** * @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 // solhint-disable-next-line max-line-length require(abi.decode(returndata, (bool)), "SafeERC20: ERC20 operation did not succeed"); } } } // File @openzeppelin/contracts/access/[email protected] pragma solidity ^0.6.0; /** * @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. */ 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 () internal { address msgSender = _msgSender(); _owner = msgSender; emit OwnershipTransferred(address(0), msgSender); } /** * @dev Returns the address of the current owner. */ function owner() public view returns (address) { return _owner; } /** * @dev Throws if called by any account other than the owner. */ modifier onlyOwner() { require(_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 { emit OwnershipTransferred(_owner, address(0)); _owner = address(0); } /** * @dev Transfers ownership of the contract to a new account (`newOwner`). * Can only be called by the current owner. */ function transferOwnership(address newOwner) public virtual onlyOwner { require(newOwner != address(0), "Ownable: new owner is the zero address"); emit OwnershipTransferred(_owner, newOwner); _owner = newOwner; } } // File contracts/StrategyLP2.sol pragma solidity ^0.6.0; interface IMasterChefWStakingReferral { function deposit(uint256 poolId, uint256 amount) external; function withdraw(uint256 poolId, uint256 amount) external; function enterStaking(uint256 _amount) external; function leaveStaking(uint256 _amount) external; function deposit(uint256 poolId, uint256 amount, address referrer) external; function userInfo(uint256 poolId, address user) external view returns (uint256, uint256); } /** * @dev Implementation of a strategy to get yields from farming LP Pools in PanwexSwap. * PanwexSwap is an automated market maker (“AMM”) that allows two tokens to be exchanged on the Binance Smart Chain. * It is fast, cheap, and allows anyone to participate. PanwexSwap is aiming to be the #1 liquidity provider on BSC. * * This strategy simply deposits whatever funds it receives from the vault into the selected MasterChef pool. * CAKE rewards from providing liquidity are farmed every few minutes, sold and split 50/50. * The corresponding pair of assets are bought and more liquidity is added to the MasterChef pool. * * This strat is currently compatible with all LP pools. */ contract StrategyLP2 is Ownable, Pausable { using SafeERC20 for IERC20; using Address for address; using SafeMath for uint256; /** * @dev Tokens Used: * {wkcs} - Required for liquidity routing when doing swaps. * {wex} - Token generated by staking our funds. In this case it's the CAKEs token. * {eleven} - ElevenFinance token, used to send funds to the treasury. * {lpPair} - Token that the strategy maximizes. The same token that users deposit in the vault. * {lpToken0, lpToken1} - Tokens that the strategy maximizes. IPanwexPair tokens */ modifier onlyOwnerOrVault() { require(owner() == _msgSender() || vault == _msgSender() , "Ownable: caller is not the owner/vault"); _; } modifier onlyVault() { require(vault == _msgSender() , "Ownable: caller is not the vault"); _; } // these cant be changed after constructor address public lpPair; address public rewardToken; address public otherToken; address public baseToken; address public router; address public masterchef; uint256 public poolId; address public vault; // these can be changed after constructor, by the vault address public buybackstrat; address public swapPathRegistry; uint256 public MIN_TO_LIQUIFY = 100; bool public stakingMode = false; bool public referralMode = false; event Deposit(address indexed account, uint256 amt); event Withdraw(address indexed account, uint256 amt); event Compound(address indexed caller, uint256 lpAdded); event CompoundLoss(address indexed caller, uint256 change); // shouldnt happen since im comparing balanceOfPool function setBuybackStrat(address _address) external onlyOwner{ buybackstrat = _address; } function setStakingMode(bool _b) external onlyOwner{ stakingMode = _b; } function setReferralMode(bool _b) external onlyOwner{ referralMode = _b; } function setSwapPathRegistry(address _a) external onlyOwner{ swapPathRegistry = _a; } function setMinToLiquify(uint256 n) external onlyOwner{ MIN_TO_LIQUIFY = n; } function depositToFarm(uint256 amt) internal{ if (stakingMode){ IMasterChefWStakingReferral(masterchef).enterStaking(amt); } else if (referralMode){ IMasterChefWStakingReferral(masterchef).deposit(poolId,amt, address(0)); } else { IMasterChefWStakingReferral(masterchef).deposit(poolId,amt); } emit Deposit(msg.sender, amt); } function withdrawFromFarm(uint256 amt) internal { if (stakingMode){ IMasterChefWStakingReferral(masterchef).leaveStaking(amt); } else { IMasterChefWStakingReferral(masterchef).withdraw(poolId,amt); } emit Withdraw(msg.sender, amt); } /** * @dev Initializes the strategy with the token to maximize. */ constructor(address _lpPair, address _rewardToken, address _baseToken, // 0 when in token mode address _vault, address _masterchef, uint256 _poolId, address _router, address _swapPathRegistry, address _feeStrat) public { lpPair = _lpPair; masterchef = _masterchef; poolId = _poolId; router = _router; swapPathRegistry = _swapPathRegistry; buybackstrat = _feeStrat; vault = _vault; if (_baseToken == address(0)){ // token mode otherToken = address(0); baseToken = address(0); } else { address token0 = IPanwexPair(_lpPair).token0(); address token1 = IPanwexPair(_lpPair).token1(); (otherToken, baseToken) = (token0 == _baseToken) ? (token1,token0) : (token0,token1); IERC20(otherToken).safeApprove(router, 0); IERC20(otherToken).safeApprove(router, uint(-1)); IERC20(baseToken).safeApprove(router, 0); IERC20(baseToken).safeApprove(router, uint(-1)); } rewardToken = _rewardToken; IERC20(lpPair).safeApprove(masterchef, 0); IERC20(lpPair).safeApprove(masterchef, uint(-1)); IERC20(rewardToken).safeApprove(router, 0); IERC20(rewardToken).safeApprove(router, uint(-1)); } /** * @dev Function that puts the funds to work. * It gets called whenever someone deposits in the strategy's vault contract. * It deposits {lpPair} in the MasterChef to farm {wex} */ function deposit() public { uint256 pairBal = IERC20(lpPair).balanceOf(address(this)); if (pairBal > 0) { depositToFarm(pairBal); } } /** * @dev Withdraws funds and sents them back to the vault. * It withdraws {lpPair} from the MasterChef. * The available {lpPair} minus fees is returned to the vault. */ function withdraw(uint256 _amount) external onlyVault { uint256 pairBal = IERC20(lpPair).balanceOf(address(this)); if (pairBal < _amount) { withdrawFromFarm(_amount.sub(pairBal)); pairBal = IERC20(lpPair).balanceOf(address(this)); IERC20(lpPair).safeTransfer(vault, pairBal); } else { IERC20(lpPair).safeTransfer(vault, _amount); } } function addliquidity() internal{ uint amount = IERC20(rewardToken).balanceOf(address(this)); uint amtToSell = amount.div(2); address[] memory path = ISwapPathRegistry(swapPathRegistry).getSwapRoute(router, rewardToken, baseToken); require(path.length > 0, "check swapPathRegistry1"); KCSRouter2(router).swapExactTokensForTokensSupportingFeeOnTransferTokens(amtToSell, 0, path, address(this), now); if (otherToken != rewardToken){ address[] memory path2 = ISwapPathRegistry(swapPathRegistry).getSwapRoute(router, rewardToken, otherToken); require(path2.length > 0, "check swapPathRegistry2"); KCSRouter2(router).swapExactTokensForTokensSupportingFeeOnTransferTokens(amtToSell, 0, path2, address(this), now); } KCSRouter2(router).addLiquidity(otherToken, baseToken, IERC20(otherToken).balanceOf(address(this)), IERC20(baseToken).balanceOf(address(this)), 0, 0, address(this), now); } function swapToStakingToken() internal{ if (rewardToken != lpPair){ uint amount = IERC20(rewardToken).balanceOf(address(this)); address[] memory path = ISwapPathRegistry(swapPathRegistry).getSwapRoute(router, rewardToken, lpPair); require(path.length > 0, "check swapPathRegistry3"); KCSRouter2(router).swapExactTokensForTokensSupportingFeeOnTransferTokens(amount, 0, path, address(this), now); } } /** * @dev Core function of the strat, in charge of collecting and re-investing rewards. * 1. It claims rewards from the MasterChef. * 2. It charges the system fees to simplify the split. * 3. It swaps the {wex} token for {lpToken0} & {lpToken1} * 4. Adds more liquidity to the pool. * 5. It deposits the new LP tokens. */ function harvest() external { require(!Address.isContract(msg.sender), "!contract"); _harvest(); } function harvestFromVault() external onlyVault { _harvest(); } function _harvest() internal whenNotPaused { uint256 preBal = balanceOfPool(); depositToFarm(0); if (IERC20(rewardToken).balanceOf(address(this)) > MIN_TO_LIQUIFY){ chargeFees(); if (baseToken != address(0)){ addliquidity(); } else { swapToStakingToken(); } deposit(); } uint256 postBal = balanceOfPool(); if (postBal > preBal){ emit Compound(msg.sender, postBal.sub(preBal)); } else if (postBal < preBal){ emit CompoundLoss(msg.sender, preBal.sub(postBal)); } } function getBPSFee() public view returns (uint) { return Buybackstrat(buybackstrat).performanceFeeBps(); } function chargeFees() internal { if(buybackstrat!=address(0)){ uint toSell = IERC20(rewardToken).balanceOf(address(this)).mul(getBPSFee()).div(10000); IERC20(rewardToken).transfer(buybackstrat, toSell); } } /** * @dev Function to calculate the total underlaying {lpPair} held by the strat. * It takes into account both the funds in hand, as the funds allocated in the MasterChef. */ function balanceOf() public view returns (uint256) { return balanceOfLpPair().add(balanceOfPool()); } /** * @dev It calculates how much {lpPair} the contract holds. */ function balanceOfLpPair() public view returns (uint256) { return IERC20(lpPair).balanceOf(address(this)); } /** * @dev It calculates how much {lpPair} the strategy has allocated in the MasterChef */ function balanceOfPool() public view returns (uint256) { (uint256 _amount,) = IMasterChefWStakingReferral(masterchef).userInfo(poolId, address(this)); return _amount; } /** * @dev Function that has to be called as part of strat migration. It sends all the available funds back to the * vault, ready to be migrated to the new strat. */ function retireStrat() external onlyOwner { panic(); uint256 pairBal = IERC20(lpPair).balanceOf(address(this)); IERC20(lpPair).transfer(vault, pairBal); } /** * @dev Pauses deposits. Withdraws all funds from the MasterChef, leaving rewards behind */ function panic() public onlyOwner { pause(); withdrawFromFarm(balanceOfPool()); } /** * @dev Pauses the strat. */ function pause() public onlyOwner { _pause(); IERC20(lpPair).safeApprove(masterchef, 0); IERC20(rewardToken).safeApprove(router, 0); if (baseToken != address(0)){ IERC20(otherToken).safeApprove(router, 0); IERC20(baseToken).safeApprove(router, 0); } } /** * @dev Unpauses the strat. */ function unpause() external onlyOwner { _unpause(); IERC20(lpPair).safeApprove(masterchef, 0); IERC20(lpPair).safeApprove(masterchef, uint(-1)); IERC20(rewardToken).safeApprove(router, 0); IERC20(rewardToken).safeApprove(router, uint(-1)); if (baseToken != address(0)){ IERC20(otherToken).safeApprove(router, 0); IERC20(otherToken).safeApprove(router, uint(-1)); IERC20(baseToken).safeApprove(router, 0); IERC20(baseToken).safeApprove(router, uint(-1)); } } }
Contract ABI
[{"type":"constructor","stateMutability":"nonpayable","inputs":[{"type":"address","name":"_lpPair","internalType":"address"},{"type":"address","name":"_rewardToken","internalType":"address"},{"type":"address","name":"_baseToken","internalType":"address"},{"type":"address","name":"_vault","internalType":"address"},{"type":"address","name":"_masterchef","internalType":"address"},{"type":"uint256","name":"_poolId","internalType":"uint256"},{"type":"address","name":"_router","internalType":"address"},{"type":"address","name":"_swapPathRegistry","internalType":"address"},{"type":"address","name":"_feeStrat","internalType":"address"}]},{"type":"event","name":"Compound","inputs":[{"type":"address","name":"caller","internalType":"address","indexed":true},{"type":"uint256","name":"lpAdded","internalType":"uint256","indexed":false}],"anonymous":false},{"type":"event","name":"CompoundLoss","inputs":[{"type":"address","name":"caller","internalType":"address","indexed":true},{"type":"uint256","name":"change","internalType":"uint256","indexed":false}],"anonymous":false},{"type":"event","name":"Deposit","inputs":[{"type":"address","name":"account","internalType":"address","indexed":true},{"type":"uint256","name":"amt","internalType":"uint256","indexed":false}],"anonymous":false},{"type":"event","name":"OwnershipTransferred","inputs":[{"type":"address","name":"previousOwner","internalType":"address","indexed":true},{"type":"address","name":"newOwner","internalType":"address","indexed":true}],"anonymous":false},{"type":"event","name":"Paused","inputs":[{"type":"address","name":"account","internalType":"address","indexed":false}],"anonymous":false},{"type":"event","name":"Unpaused","inputs":[{"type":"address","name":"account","internalType":"address","indexed":false}],"anonymous":false},{"type":"event","name":"Withdraw","inputs":[{"type":"address","name":"account","internalType":"address","indexed":true},{"type":"uint256","name":"amt","internalType":"uint256","indexed":false}],"anonymous":false},{"type":"function","stateMutability":"view","outputs":[{"type":"uint256","name":"","internalType":"uint256"}],"name":"MIN_TO_LIQUIFY","inputs":[]},{"type":"function","stateMutability":"view","outputs":[{"type":"uint256","name":"","internalType":"uint256"}],"name":"balanceOf","inputs":[]},{"type":"function","stateMutability":"view","outputs":[{"type":"uint256","name":"","internalType":"uint256"}],"name":"balanceOfLpPair","inputs":[]},{"type":"function","stateMutability":"view","outputs":[{"type":"uint256","name":"","internalType":"uint256"}],"name":"balanceOfPool","inputs":[]},{"type":"function","stateMutability":"view","outputs":[{"type":"address","name":"","internalType":"address"}],"name":"baseToken","inputs":[]},{"type":"function","stateMutability":"view","outputs":[{"type":"address","name":"","internalType":"address"}],"name":"buybackstrat","inputs":[]},{"type":"function","stateMutability":"nonpayable","outputs":[],"name":"deposit","inputs":[]},{"type":"function","stateMutability":"view","outputs":[{"type":"uint256","name":"","internalType":"uint256"}],"name":"getBPSFee","inputs":[]},{"type":"function","stateMutability":"nonpayable","outputs":[],"name":"harvest","inputs":[]},{"type":"function","stateMutability":"nonpayable","outputs":[],"name":"harvestFromVault","inputs":[]},{"type":"function","stateMutability":"view","outputs":[{"type":"address","name":"","internalType":"address"}],"name":"lpPair","inputs":[]},{"type":"function","stateMutability":"view","outputs":[{"type":"address","name":"","internalType":"address"}],"name":"masterchef","inputs":[]},{"type":"function","stateMutability":"view","outputs":[{"type":"address","name":"","internalType":"address"}],"name":"otherToken","inputs":[]},{"type":"function","stateMutability":"view","outputs":[{"type":"address","name":"","internalType":"address"}],"name":"owner","inputs":[]},{"type":"function","stateMutability":"nonpayable","outputs":[],"name":"panic","inputs":[]},{"type":"function","stateMutability":"nonpayable","outputs":[],"name":"pause","inputs":[]},{"type":"function","stateMutability":"view","outputs":[{"type":"bool","name":"","internalType":"bool"}],"name":"paused","inputs":[]},{"type":"function","stateMutability":"view","outputs":[{"type":"uint256","name":"","internalType":"uint256"}],"name":"poolId","inputs":[]},{"type":"function","stateMutability":"view","outputs":[{"type":"bool","name":"","internalType":"bool"}],"name":"referralMode","inputs":[]},{"type":"function","stateMutability":"nonpayable","outputs":[],"name":"renounceOwnership","inputs":[]},{"type":"function","stateMutability":"nonpayable","outputs":[],"name":"retireStrat","inputs":[]},{"type":"function","stateMutability":"view","outputs":[{"type":"address","name":"","internalType":"address"}],"name":"rewardToken","inputs":[]},{"type":"function","stateMutability":"view","outputs":[{"type":"address","name":"","internalType":"address"}],"name":"router","inputs":[]},{"type":"function","stateMutability":"nonpayable","outputs":[],"name":"setBuybackStrat","inputs":[{"type":"address","name":"_address","internalType":"address"}]},{"type":"function","stateMutability":"nonpayable","outputs":[],"name":"setMinToLiquify","inputs":[{"type":"uint256","name":"n","internalType":"uint256"}]},{"type":"function","stateMutability":"nonpayable","outputs":[],"name":"setReferralMode","inputs":[{"type":"bool","name":"_b","internalType":"bool"}]},{"type":"function","stateMutability":"nonpayable","outputs":[],"name":"setStakingMode","inputs":[{"type":"bool","name":"_b","internalType":"bool"}]},{"type":"function","stateMutability":"nonpayable","outputs":[],"name":"setSwapPathRegistry","inputs":[{"type":"address","name":"_a","internalType":"address"}]},{"type":"function","stateMutability":"view","outputs":[{"type":"bool","name":"","internalType":"bool"}],"name":"stakingMode","inputs":[]},{"type":"function","stateMutability":"view","outputs":[{"type":"address","name":"","internalType":"address"}],"name":"swapPathRegistry","inputs":[]},{"type":"function","stateMutability":"nonpayable","outputs":[],"name":"transferOwnership","inputs":[{"type":"address","name":"newOwner","internalType":"address"}]},{"type":"function","stateMutability":"nonpayable","outputs":[],"name":"unpause","inputs":[]},{"type":"function","stateMutability":"view","outputs":[{"type":"address","name":"","internalType":"address"}],"name":"vault","inputs":[]},{"type":"function","stateMutability":"nonpayable","outputs":[],"name":"withdraw","inputs":[{"type":"uint256","name":"_amount","internalType":"uint256"}]}]
Contract Creation Code
0x60806040526064600b55600c805461ffff191690553480156200002157600080fd5b50604051620031343803806200313483398181016040526101208110156200004857600080fd5b508051602082015160408301516060840151608085015160a086015160c087015160e08801516101009098015196979596949593949293919290919060006200009062000427565b600080546001600160a01b0319166001600160a01b0383169081178255604051929350917f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e0908290a3506000805460ff60a01b19169055600180546001600160a01b03199081166001600160a01b038c8116919091179092556006805482168884161790556007869055600580548216868416179055600a805482168584161790556009805482168484161790556008805490911688831617905587166200017457600380546001600160a01b03199081169091556004805490911690556200034e565b6000896001600160a01b0316630dfe16816040518163ffffffff1660e01b815260040160206040518083038186803b158015620001b057600080fd5b505afa158015620001c5573d6000803e3d6000fd5b505050506040513d6020811015620001dc57600080fd5b50516040805163d21220a760e01b815290519192506000916001600160a01b038d169163d21220a7916004808301926020929190829003018186803b1580156200022557600080fd5b505afa1580156200023a573d6000803e3d6000fd5b505050506040513d60208110156200025157600080fd5b505190506001600160a01b03828116908a16146200027157818162000274565b80825b600480546001600160a01b03199081166001600160a01b0393841617909155600380549091169282169290921791829055600554620002c5928216911660006200042b602090811b6200112417901c565b600554600354620002f2916001600160a01b0391821691166000196200042b602090811b6200112417901c565b6005546004546200031e916001600160a01b03918216911660006200042b602090811b6200112417901c565b6005546004546200034b916001600160a01b0391821691166000196200042b602090811b6200112417901c565b50505b600280546001600160a01b0319166001600160a01b038a8116919091179091556006546001546200039292908116911660006200042b602090811b6200112417901c565b600654600154620003bf916001600160a01b0391821691166000196200042b602090811b6200112417901c565b600554600254620003eb916001600160a01b03918216911660006200042b602090811b6200112417901c565b60055460025462000418916001600160a01b0391821691166000196200042b602090811b6200112417901c565b50505050505050505062000816565b3390565b801580620004b5575060408051636eb1769f60e11b81523060048201526001600160a01b03848116602483015291519185169163dd62ed3e91604480820192602092909190829003018186803b1580156200048557600080fd5b505afa1580156200049a573d6000803e3d6000fd5b505050506040513d6020811015620004b157600080fd5b5051155b620004f25760405162461bcd60e51b8152600401808060200182810382526036815260200180620030fe6036913960400191505060405180910390fd5b604080516001600160a01b038416602482015260448082018490528251808303909101815260649091019091526020810180516001600160e01b0390811663095ea7b360e01b179091526200054a9185916200054f16565b505050565b6060620005ab826040518060400160405280602081526020017f5361666545524332303a206c6f772d6c6576656c2063616c6c206661696c6564815250856001600160a01b03166200060b60201b62001237179092919060201c565b8051909150156200054a57808060200190516020811015620005cc57600080fd5b50516200054a5760405162461bcd60e51b815260040180806020018281038252602a815260200180620030d4602a913960400191505060405180910390fd5b60606200061c848460008562000624565b949350505050565b60606200063185620007dc565b62000683576040805162461bcd60e51b815260206004820152601d60248201527f416464726573733a2063616c6c20746f206e6f6e2d636f6e7472616374000000604482015290519081900360640190fd5b60006060866001600160a01b031685876040518082805190602001908083835b60208310620006c45780518252601f199092019160209182019101620006a3565b6001836020036101000a03801982511681845116808217855250505050505090500191505060006040518083038185875af1925050503d806000811462000728576040519150601f19603f3d011682016040523d82523d6000602084013e6200072d565b606091505b50915091508115620007435791506200061c9050565b805115620007545780518082602001fd5b8360405162461bcd60e51b81526004018080602001828103825283818151815260200191508051906020019080838360005b83811015620007a057818101518382015260200162000786565b50505050905090810190601f168015620007ce5780820380516001836020036101000a031916815260200191505b509250505060405180910390fd5b6000813f7fc5d2460186f7233c927e7db2dcc703c0e500b653ca82273b7bfad8045d85a4708181148015906200061c575050151592915050565b6128ae80620008266000396000f3fe608060405234801561001057600080fd5b50600436106102065760003560e01c80637c5126221161011a578063d0e30db0116100ad578063f7c618c11161007c578063f7c618c1146103e9578063f887ea40146103f1578063fb1db278146103f9578063fb61778714610401578063fbfa77cf1461040957610206565b8063d0e30db014610376578063d2cd94441461037e578063d8ad1e60146103a4578063f2fde38b146103c357610206565b80638da5cb5b116100e95780638da5cb5b14610356578063a6057fa71461035e578063c2351cdd14610366578063c55dae631461036e57610206565b80637c512622146103015780637d034b13146103275780638456cb591461034657806386e655ec1461034e57610206565b8063452ed4f11161019d57806367c1fe3d1161016c57806367c1fe3d146102c45780636a0ddbd5146102cc5780636fa9deaf146102d4578063715018a6146102f1578063722713f7146102f957610206565b8063452ed4f1146102905780634641257d146102985780634700d305146102a05780635c975abb146102a857610206565b80633e0dc34e116101d95780633e0dc34e146102545780633f4ba83a1461025c57806340b8622414610264578063433b3c051461028857610206565b806304f5606c1461020b5780631158808614610215578063118118581461022f5780632e1a7d4d14610237575b600080fd5b610213610411565b005b61021d610485565b60408051918252519081900360200190f35b61021d61050c565b6102136004803603602081101561024d57600080fd5b5035610582565b61021d61073c565b610213610742565b61026c6108a7565b604080516001600160a01b039092168252519081900360200190f35b61026c6108b6565b61026c6108c5565b6102136108d4565b61021361091b565b6102b0610986565b604080519115158252519081900360200190f35b61021d610996565b61026c61099c565b610213600480360360208110156102ea57600080fd5b50356109ab565b610213610a08565b61021d610aaa565b6102136004803603602081101561031757600080fd5b50356001600160a01b0316610aca565b6102136004803603602081101561033d57600080fd5b50351515610b44565b610213610bb6565b6102b0610c9f565b61026c610ca8565b6102b0610cb7565b61021d610cc5565b61026c610d10565b610213610d1f565b6102136004803603602081101561039457600080fd5b50356001600160a01b0316610daa565b610213600480360360208110156103ba57600080fd5b50351515610e24565b610213600480360360208110156103d957600080fd5b50356001600160a01b0316610e8f565b61026c610f87565b61026c610f96565b61026c610fa5565b610213610fb4565b61026c611115565b61041961124e565b6008546001600160a01b0390811691161461047b576040805162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865207661756c74604482015290519081900360640190fd5b610483611252565b565b600654600754604080516393f1a40b60e01b81526004810192909252306024830152805160009384936001600160a01b03909116926393f1a40b92604480840193829003018186803b1580156104da57600080fd5b505afa1580156104ee573d6000803e3d6000fd5b505050506040513d604081101561050457600080fd5b505191505090565b6009546040805163b9d4e87960e01b815290516000926001600160a01b03169163b9d4e879916004808301926020929190829003018186803b15801561055157600080fd5b505afa158015610565573d6000803e3d6000fd5b505050506040513d602081101561057b57600080fd5b5051905090565b61058a61124e565b6008546001600160a01b039081169116146105ec576040805162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865207661756c74604482015290519081900360640190fd5b600154604080516370a0823160e01b815230600482015290516000926001600160a01b0316916370a08231916024808301926020929190829003018186803b15801561063757600080fd5b505afa15801561064b573d6000803e3d6000fd5b505050506040513d602081101561066157600080fd5b505190508181101561071b5761067f61067a838361140f565b61145a565b600154604080516370a0823160e01b815230600482015290516001600160a01b03909216916370a0823191602480820192602092909190829003018186803b1580156106ca57600080fd5b505afa1580156106de573d6000803e3d6000fd5b505050506040513d60208110156106f457600080fd5b5051600854600154919250610716916001600160a01b03908116911683611578565b610738565b600854600154610738916001600160a01b03918216911684611578565b5050565b60075481565b61074a61124e565b6000546001600160a01b0390811691161461079a576040805162461bcd60e51b815260206004820181905260248201526000805160206127f9833981519152604482015290519081900360640190fd5b6107a26115ca565b6006546001546107c0916001600160a01b0391821691166000611124565b6006546001546107df916001600160a01b039182169116600019611124565b6005546002546107fd916001600160a01b0391821691166000611124565b60055460025461081c916001600160a01b039182169116600019611124565b6004546001600160a01b0316156104835760055460035461084b916001600160a01b0391821691166000611124565b60055460035461086a916001600160a01b039182169116600019611124565b600554600454610888916001600160a01b0391821691166000611124565b600554600454610483916001600160a01b039182169116600019611124565b6009546001600160a01b031681565b6003546001600160a01b031681565b6001546001600160a01b031681565b6108dd33611672565b1561047b576040805162461bcd60e51b81526020600482015260096024820152680858dbdb9d1c9858dd60ba1b604482015290519081900360640190fd5b61092361124e565b6000546001600160a01b03908116911614610973576040805162461bcd60e51b815260206004820181905260248201526000805160206127f9833981519152604482015290519081900360640190fd5b61097b610bb6565b61048361067a610485565b600054600160a01b900460ff1690565b600b5481565b600a546001600160a01b031681565b6109b361124e565b6000546001600160a01b03908116911614610a03576040805162461bcd60e51b815260206004820181905260248201526000805160206127f9833981519152604482015290519081900360640190fd5b600b55565b610a1061124e565b6000546001600160a01b03908116911614610a60576040805162461bcd60e51b815260206004820181905260248201526000805160206127f9833981519152604482015290519081900360640190fd5b600080546040516001600160a01b03909116907f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e0908390a3600080546001600160a01b0319169055565b6000610ac5610ab7610485565b610abf610cc5565b906116ab565b905090565b610ad261124e565b6000546001600160a01b03908116911614610b22576040805162461bcd60e51b815260206004820181905260248201526000805160206127f9833981519152604482015290519081900360640190fd5b600980546001600160a01b0319166001600160a01b0392909216919091179055565b610b4c61124e565b6000546001600160a01b03908116911614610b9c576040805162461bcd60e51b815260206004820181905260248201526000805160206127f9833981519152604482015290519081900360640190fd5b600c80549115156101000261ff0019909216919091179055565b610bbe61124e565b6000546001600160a01b03908116911614610c0e576040805162461bcd60e51b815260206004820181905260248201526000805160206127f9833981519152604482015290519081900360640190fd5b610c16611705565b600654600154610c34916001600160a01b0391821691166000611124565b600554600254610c52916001600160a01b0391821691166000611124565b6004546001600160a01b03161561048357600554600354610c81916001600160a01b0391821691166000611124565b600554600454610483916001600160a01b0391821691166000611124565b600c5460ff1681565b6000546001600160a01b031690565b600c54610100900460ff1681565b600154604080516370a0823160e01b815230600482015290516000926001600160a01b0316916370a08231916024808301926020929190829003018186803b15801561055157600080fd5b6004546001600160a01b031681565b600154604080516370a0823160e01b815230600482015290516000926001600160a01b0316916370a08231916024808301926020929190829003018186803b158015610d6a57600080fd5b505afa158015610d7e573d6000803e3d6000fd5b505050506040513d6020811015610d9457600080fd5b505190508015610da757610da781611793565b50565b610db261124e565b6000546001600160a01b03908116911614610e02576040805162461bcd60e51b815260206004820181905260248201526000805160206127f9833981519152604482015290519081900360640190fd5b600a80546001600160a01b0319166001600160a01b0392909216919091179055565b610e2c61124e565b6000546001600160a01b03908116911614610e7c576040805162461bcd60e51b815260206004820181905260248201526000805160206127f9833981519152604482015290519081900360640190fd5b600c805460ff1916911515919091179055565b610e9761124e565b6000546001600160a01b03908116911614610ee7576040805162461bcd60e51b815260206004820181905260248201526000805160206127f9833981519152604482015290519081900360640190fd5b6001600160a01b038116610f2c5760405162461bcd60e51b81526004018080602001828103825260268152602001806127b26026913960400191505060405180910390fd5b600080546040516001600160a01b03808516939216917f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e091a3600080546001600160a01b0319166001600160a01b0392909216919091179055565b6002546001600160a01b031681565b6005546001600160a01b031681565b6006546001600160a01b031681565b610fbc61124e565b6000546001600160a01b0390811691161461100c576040805162461bcd60e51b815260206004820181905260248201526000805160206127f9833981519152604482015290519081900360640190fd5b61101461091b565b600154604080516370a0823160e01b815230600482015290516000926001600160a01b0316916370a08231916024808301926020929190829003018186803b15801561105f57600080fd5b505afa158015611073573d6000803e3d6000fd5b505050506040513d602081101561108957600080fd5b50516001546008546040805163a9059cbb60e01b81526001600160a01b03928316600482015260248101859052905193945091169163a9059cbb916044808201926020929091908290030181600087803b1580156110e657600080fd5b505af11580156110fa573d6000803e3d6000fd5b505050506040513d602081101561111057600080fd5b505050565b6008546001600160a01b031681565b8015806111aa575060408051636eb1769f60e11b81523060048201526001600160a01b03848116602483015291519185169163dd62ed3e91604480820192602092909190829003018186803b15801561117c57600080fd5b505afa158015611190573d6000803e3d6000fd5b505050506040513d60208110156111a657600080fd5b5051155b6111e55760405162461bcd60e51b81526004018080602001828103825260368152602001806128436036913960400191505060405180910390fd5b604080516001600160a01b038416602482015260448082018490528251808303909101815260649091019091526020810180516001600160e01b031663095ea7b360e01b17905261111090849061191e565b606061124684846000856119cf565b949350505050565b3390565b600054600160a01b900460ff16156112a4576040805162461bcd60e51b815260206004820152601060248201526f14185d5cd8589b194e881c185d5cd95960821b604482015290519081900360640190fd5b60006112ae610485565b90506112ba6000611793565b600b54600254604080516370a0823160e01b815230600482015290516001600160a01b03909216916370a0823191602480820192602092909190829003018186803b15801561130857600080fd5b505afa15801561131c573d6000803e3d6000fd5b505050506040513d602081101561133257600080fd5b5051111561137057611342611b7a565b6004546001600160a01b0316156113605761135b611c7c565b611368565b611368612376565b611370610d1f565b600061137a610485565b9050818111156113c657337f169f1815ebdea059aac3bb00ec9a9594c7a5ffcb64a17e8392b5d84909a145566113b0838561140f565b60408051918252519081900360200190a2610738565b8181101561073857337fdf5ee50bba1a120f14b1b64d8ed407af28c5baa9ddb2cef0b98a75e241ab71116113fa848461140f565b60408051918252519081900360200190a25050565b600061145183836040518060400160405280601e81526020017f536166654d6174683a207375627472616374696f6e206f766572666c6f77000081525061265c565b90505b92915050565b600c5460ff16156114cf5760065460408051631058d28160e01b81526004810184905290516001600160a01b0390921691631058d2819160248082019260009290919082900301818387803b1580156114b257600080fd5b505af11580156114c6573d6000803e3d6000fd5b5050505061153f565b60065460075460408051630441a3e760e41b8152600481019290925260248201849052516001600160a01b039092169163441a3e709160448082019260009290919082900301818387803b15801561152657600080fd5b505af115801561153a573d6000803e3d6000fd5b505050505b60408051828152905133917f884edad9ce6fa2440d8a54cc123490eb96d2768479d49ff9c7366125a9424364919081900360200190a250565b604080516001600160a01b038416602482015260448082018490528251808303909101815260649091019091526020810180516001600160e01b031663a9059cbb60e01b17905261111090849061191e565b600054600160a01b900460ff1661161f576040805162461bcd60e51b815260206004820152601460248201527314185d5cd8589b194e881b9bdd081c185d5cd95960621b604482015290519081900360640190fd5b6000805460ff60a01b191690557f5db9ee0a495bf2e6ff9c91a7834c1ba4fdd244a5e8aa4e537bd38aeae4b073aa61165561124e565b604080516001600160a01b039092168252519081900360200190a1565b6000813f7fc5d2460186f7233c927e7db2dcc703c0e500b653ca82273b7bfad8045d85a470818114801590611246575050151592915050565b600082820183811015611451576040805162461bcd60e51b815260206004820152601b60248201527f536166654d6174683a206164646974696f6e206f766572666c6f770000000000604482015290519081900360640190fd5b600054600160a01b900460ff1615611757576040805162461bcd60e51b815260206004820152601060248201526f14185d5cd8589b194e881c185d5cd95960821b604482015290519081900360640190fd5b6000805460ff60a01b1916600160a01b1790557f62e78cea01bee320cd4e420270b5ea74000d11b0c9f74754ebdbfc544b05a25861165561124e565b600c5460ff161561180857600654604080516341441d3b60e01b81526004810184905290516001600160a01b03909216916341441d3b9160248082019260009290919082900301818387803b1580156117eb57600080fd5b505af11580156117ff573d6000803e3d6000fd5b505050506118e5565b600c54610100900460ff16156118755760065460075460408051638dbdbe6d60e01b815260048101929092526024820184905260006044830181905290516001600160a01b0390931692638dbdbe6d92606480820193929182900301818387803b1580156117eb57600080fd5b60065460075460408051631c57762b60e31b8152600481019290925260248201849052516001600160a01b039092169163e2bbb1589160448082019260009290919082900301818387803b1580156118cc57600080fd5b505af11580156118e0573d6000803e3d6000fd5b505050505b60408051828152905133917fe1fffcc4923d04b559f4d29a8bfc6cda04eb5b0d3c460751c2402c5c5cc9109c919081900360200190a250565b6060611973826040518060400160405280602081526020017f5361666545524332303a206c6f772d6c6576656c2063616c6c206661696c6564815250856001600160a01b03166112379092919063ffffffff16565b8051909150156111105780806020019051602081101561199257600080fd5b50516111105760405162461bcd60e51b815260040180806020018281038252602a815260200180612819602a913960400191505060405180910390fd5b60606119da85611672565b611a2b576040805162461bcd60e51b815260206004820152601d60248201527f416464726573733a2063616c6c20746f206e6f6e2d636f6e7472616374000000604482015290519081900360640190fd5b60006060866001600160a01b031685876040518082805190602001908083835b60208310611a6a5780518252601f199092019160209182019101611a4b565b6001836020036101000a03801982511681845116808217855250505050505090500191505060006040518083038185875af1925050503d8060008114611acc576040519150601f19603f3d011682016040523d82523d6000602084013e611ad1565b606091505b50915091508115611ae55791506112469050565b805115611af55780518082602001fd5b8360405162461bcd60e51b81526004018080602001828103825283818151815260200191508051906020019080838360005b83811015611b3f578181015183820152602001611b27565b50505050905090810190601f168015611b6c5780820380516001836020036101000a031916815260200191505b509250505060405180910390fd5b6009546001600160a01b031615610483576000611c21612710611c1b611b9e61050c565b600254604080516370a0823160e01b815230600482015290516001600160a01b03909216916370a0823191602480820192602092909190829003018186803b158015611be957600080fd5b505afa158015611bfd573d6000803e3d6000fd5b505050506040513d6020811015611c1357600080fd5b5051906126b6565b9061270f565b6002546009546040805163a9059cbb60e01b81526001600160a01b03928316600482015260248101859052905193945091169163a9059cbb916044808201926020929091908290030181600087803b1580156110e657600080fd5b600254604080516370a0823160e01b815230600482015290516000926001600160a01b0316916370a08231916024808301926020929190829003018186803b158015611cc757600080fd5b505afa158015611cdb573d6000803e3d6000fd5b505050506040513d6020811015611cf157600080fd5b505190506000611d0282600261270f565b600a546005546002546004805460408051631206e48360e21b81526001600160a01b03958616938101939093529284166024830152831660448201529051939450606093919092169163481b920c916064808301926000929190829003018186803b158015611d7057600080fd5b505afa158015611d84573d6000803e3d6000fd5b505050506040513d6000823e601f3d908101601f191682016040526020811015611dad57600080fd5b8101908080516040519392919084640100000000821115611dcd57600080fd5b908301906020820185811115611de257600080fd5b8251866020820283011164010000000082111715611dff57600080fd5b82525081516020918201928201910280838360005b83811015611e2c578181015183820152602001611e14565b5050505090500160405250505090506000815111611e91576040805162461bcd60e51b815260206004820152601760248201527f636865636b207377617050617468526567697374727931000000000000000000604482015290519081900360640190fd5b600554604051635c11d79560e01b8152600481018481526000602483018190523060648401819052426084850181905260a060448601908152875160a487015287516001600160a01b0390971696635c11d795968a968a9594939092909160c40190602080880191028083838b5b83811015611f17578181015183820152602001611eff565b505050509050019650505050505050600060405180830381600087803b158015611f4057600080fd5b505af1158015611f54573d6000803e3d6000fd5b50506002546003546001600160a01b0390811691161491506121c5905057600a5460055460025460035460408051631206e48360e21b81526001600160a01b0394851660048201529284166024840152908316604483015251606093929092169163481b920c91606480820192600092909190829003018186803b158015611fdb57600080fd5b505afa158015611fef573d6000803e3d6000fd5b505050506040513d6000823e601f3d908101601f19168201604052602081101561201857600080fd5b810190808051604051939291908464010000000082111561203857600080fd5b90830190602082018581111561204d57600080fd5b825186602082028301116401000000008211171561206a57600080fd5b82525081516020918201928201910280838360005b8381101561209757818101518382015260200161207f565b50505050905001604052505050905060008151116120fc576040805162461bcd60e51b815260206004820152601760248201527f636865636b207377617050617468526567697374727932000000000000000000604482015290519081900360640190fd5b600554604051635c11d79560e01b8152600481018581526000602483018190523060648401819052426084850181905260a060448601908152875160a487015287516001600160a01b0390971696635c11d795968b968a9594939092909160c40190602080880191028083838b5b8381101561218257818101518382015260200161216a565b505050509050019650505050505050600060405180830381600087803b1580156121ab57600080fd5b505af11580156121bf573d6000803e3d6000fd5b50505050505b60055460035460048054604080516370a0823160e01b81523093810193909352516001600160a01b039485169463e8e337009481169392169183916370a0823191602480820192602092909190829003018186803b15801561222657600080fd5b505afa15801561223a573d6000803e3d6000fd5b505050506040513d602081101561225057600080fd5b505160048054604080516370a0823160e01b81523093810193909352516001600160a01b03909116916370a08231916024808301926020929190829003018186803b15801561229e57600080fd5b505afa1580156122b2573d6000803e3d6000fd5b505050506040513d60208110156122c857600080fd5b5051604080516001600160e01b031960e088901b1681526001600160a01b0395861660048201529390941660248401526044830191909152606482015260006084820181905260a482018190523060c48301524260e483015291516101048083019360609383900390910190829087803b15801561234557600080fd5b505af1158015612359573d6000803e3d6000fd5b505050506040513d606081101561236f57600080fd5b5050505050565b6001546002546001600160a01b0390811691161461048357600254604080516370a0823160e01b815230600482015290516000926001600160a01b0316916370a08231916024808301926020929190829003018186803b1580156123d957600080fd5b505afa1580156123ed573d6000803e3d6000fd5b505050506040513d602081101561240357600080fd5b5051600a5460055460025460015460408051631206e48360e21b81526001600160a01b0394851660048201529284166024840152908316604483015251939450606093919092169163481b920c916064808301926000929190829003018186803b15801561247057600080fd5b505afa158015612484573d6000803e3d6000fd5b505050506040513d6000823e601f3d908101601f1916820160405260208110156124ad57600080fd5b81019080805160405193929190846401000000008211156124cd57600080fd5b9083019060208201858111156124e257600080fd5b82518660208202830111640100000000821117156124ff57600080fd5b82525081516020918201928201910280838360005b8381101561252c578181015183820152602001612514565b5050505090500160405250505090506000815111612591576040805162461bcd60e51b815260206004820152601760248201527f636865636b207377617050617468526567697374727933000000000000000000604482015290519081900360640190fd5b600554604051635c11d79560e01b8152600481018481526000602483018190523060648401819052426084850181905260a060448601908152875160a487015287516001600160a01b0390971696635c11d795968a968a9594939092909160c40190602080880191028083838b5b838110156126175781810151838201526020016125ff565b505050509050019650505050505050600060405180830381600087803b15801561264057600080fd5b505af1158015612654573d6000803e3d6000fd5b505050505050565b600081848411156126ae5760405162461bcd60e51b8152602060048201818152835160248401528351909283926044909101919085019080838360008315611b3f578181015183820152602001611b27565b505050900390565b6000826126c557506000611454565b828202828482816126d257fe5b04146114515760405162461bcd60e51b81526004018080602001828103825260218152602001806127d86021913960400191505060405180910390fd5b600061145183836040518060400160405280601a81526020017f536166654d6174683a206469766973696f6e206279207a65726f0000000000008152506000818361279b5760405162461bcd60e51b8152602060048201818152835160248401528351909283926044909101919085019080838360008315611b3f578181015183820152602001611b27565b5060008385816127a757fe5b049594505050505056fe4f776e61626c653a206e6577206f776e657220697320746865207a65726f2061646472657373536166654d6174683a206d756c7469706c69636174696f6e206f766572666c6f774f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e65725361666545524332303a204552433230206f7065726174696f6e20646964206e6f7420737563636565645361666545524332303a20617070726f76652066726f6d206e6f6e2d7a65726f20746f206e6f6e2d7a65726f20616c6c6f77616e6365a26469706673582212204e31d5a0f233e862b3e6bc9f6c32820a15e73b6655fd53e3f31d6d8bcad9ad2564736f6c634300060c00335361666545524332303a204552433230206f7065726174696f6e20646964206e6f7420737563636565645361666545524332303a20617070726f76652066726f6d206e6f6e2d7a65726f20746f206e6f6e2d7a65726f20616c6c6f77616e6365000000000000000000000000001724d7d4341f3318e57baee2829bb93d51dc8b0000000000000000000000006c1b568a1d7fb33de6707238803f8821e9472539000000000000000000000000980a5afef3d17ad98635f6c5aebcbaeded3c34300000000000000000000000001d911824caf39861257552578abe3c28a96ff321000000000000000000000000723f61a9bcd6c390474d0d2b3d5e65e1f9ada8240000000000000000000000000000000000000000000000000000000000000005000000000000000000000000f742609f4cafeef624816e309d770222aa8a55cc0000000000000000000000007dcc096adf6b80f106accf9133a095fa277d6714000000000000000000000000d5f08f5c0cf28d3da911a2bf3172068f08293657
Deployed ByteCode
0x608060405234801561001057600080fd5b50600436106102065760003560e01c80637c5126221161011a578063d0e30db0116100ad578063f7c618c11161007c578063f7c618c1146103e9578063f887ea40146103f1578063fb1db278146103f9578063fb61778714610401578063fbfa77cf1461040957610206565b8063d0e30db014610376578063d2cd94441461037e578063d8ad1e60146103a4578063f2fde38b146103c357610206565b80638da5cb5b116100e95780638da5cb5b14610356578063a6057fa71461035e578063c2351cdd14610366578063c55dae631461036e57610206565b80637c512622146103015780637d034b13146103275780638456cb591461034657806386e655ec1461034e57610206565b8063452ed4f11161019d57806367c1fe3d1161016c57806367c1fe3d146102c45780636a0ddbd5146102cc5780636fa9deaf146102d4578063715018a6146102f1578063722713f7146102f957610206565b8063452ed4f1146102905780634641257d146102985780634700d305146102a05780635c975abb146102a857610206565b80633e0dc34e116101d95780633e0dc34e146102545780633f4ba83a1461025c57806340b8622414610264578063433b3c051461028857610206565b806304f5606c1461020b5780631158808614610215578063118118581461022f5780632e1a7d4d14610237575b600080fd5b610213610411565b005b61021d610485565b60408051918252519081900360200190f35b61021d61050c565b6102136004803603602081101561024d57600080fd5b5035610582565b61021d61073c565b610213610742565b61026c6108a7565b604080516001600160a01b039092168252519081900360200190f35b61026c6108b6565b61026c6108c5565b6102136108d4565b61021361091b565b6102b0610986565b604080519115158252519081900360200190f35b61021d610996565b61026c61099c565b610213600480360360208110156102ea57600080fd5b50356109ab565b610213610a08565b61021d610aaa565b6102136004803603602081101561031757600080fd5b50356001600160a01b0316610aca565b6102136004803603602081101561033d57600080fd5b50351515610b44565b610213610bb6565b6102b0610c9f565b61026c610ca8565b6102b0610cb7565b61021d610cc5565b61026c610d10565b610213610d1f565b6102136004803603602081101561039457600080fd5b50356001600160a01b0316610daa565b610213600480360360208110156103ba57600080fd5b50351515610e24565b610213600480360360208110156103d957600080fd5b50356001600160a01b0316610e8f565b61026c610f87565b61026c610f96565b61026c610fa5565b610213610fb4565b61026c611115565b61041961124e565b6008546001600160a01b0390811691161461047b576040805162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865207661756c74604482015290519081900360640190fd5b610483611252565b565b600654600754604080516393f1a40b60e01b81526004810192909252306024830152805160009384936001600160a01b03909116926393f1a40b92604480840193829003018186803b1580156104da57600080fd5b505afa1580156104ee573d6000803e3d6000fd5b505050506040513d604081101561050457600080fd5b505191505090565b6009546040805163b9d4e87960e01b815290516000926001600160a01b03169163b9d4e879916004808301926020929190829003018186803b15801561055157600080fd5b505afa158015610565573d6000803e3d6000fd5b505050506040513d602081101561057b57600080fd5b5051905090565b61058a61124e565b6008546001600160a01b039081169116146105ec576040805162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865207661756c74604482015290519081900360640190fd5b600154604080516370a0823160e01b815230600482015290516000926001600160a01b0316916370a08231916024808301926020929190829003018186803b15801561063757600080fd5b505afa15801561064b573d6000803e3d6000fd5b505050506040513d602081101561066157600080fd5b505190508181101561071b5761067f61067a838361140f565b61145a565b600154604080516370a0823160e01b815230600482015290516001600160a01b03909216916370a0823191602480820192602092909190829003018186803b1580156106ca57600080fd5b505afa1580156106de573d6000803e3d6000fd5b505050506040513d60208110156106f457600080fd5b5051600854600154919250610716916001600160a01b03908116911683611578565b610738565b600854600154610738916001600160a01b03918216911684611578565b5050565b60075481565b61074a61124e565b6000546001600160a01b0390811691161461079a576040805162461bcd60e51b815260206004820181905260248201526000805160206127f9833981519152604482015290519081900360640190fd5b6107a26115ca565b6006546001546107c0916001600160a01b0391821691166000611124565b6006546001546107df916001600160a01b039182169116600019611124565b6005546002546107fd916001600160a01b0391821691166000611124565b60055460025461081c916001600160a01b039182169116600019611124565b6004546001600160a01b0316156104835760055460035461084b916001600160a01b0391821691166000611124565b60055460035461086a916001600160a01b039182169116600019611124565b600554600454610888916001600160a01b0391821691166000611124565b600554600454610483916001600160a01b039182169116600019611124565b6009546001600160a01b031681565b6003546001600160a01b031681565b6001546001600160a01b031681565b6108dd33611672565b1561047b576040805162461bcd60e51b81526020600482015260096024820152680858dbdb9d1c9858dd60ba1b604482015290519081900360640190fd5b61092361124e565b6000546001600160a01b03908116911614610973576040805162461bcd60e51b815260206004820181905260248201526000805160206127f9833981519152604482015290519081900360640190fd5b61097b610bb6565b61048361067a610485565b600054600160a01b900460ff1690565b600b5481565b600a546001600160a01b031681565b6109b361124e565b6000546001600160a01b03908116911614610a03576040805162461bcd60e51b815260206004820181905260248201526000805160206127f9833981519152604482015290519081900360640190fd5b600b55565b610a1061124e565b6000546001600160a01b03908116911614610a60576040805162461bcd60e51b815260206004820181905260248201526000805160206127f9833981519152604482015290519081900360640190fd5b600080546040516001600160a01b03909116907f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e0908390a3600080546001600160a01b0319169055565b6000610ac5610ab7610485565b610abf610cc5565b906116ab565b905090565b610ad261124e565b6000546001600160a01b03908116911614610b22576040805162461bcd60e51b815260206004820181905260248201526000805160206127f9833981519152604482015290519081900360640190fd5b600980546001600160a01b0319166001600160a01b0392909216919091179055565b610b4c61124e565b6000546001600160a01b03908116911614610b9c576040805162461bcd60e51b815260206004820181905260248201526000805160206127f9833981519152604482015290519081900360640190fd5b600c80549115156101000261ff0019909216919091179055565b610bbe61124e565b6000546001600160a01b03908116911614610c0e576040805162461bcd60e51b815260206004820181905260248201526000805160206127f9833981519152604482015290519081900360640190fd5b610c16611705565b600654600154610c34916001600160a01b0391821691166000611124565b600554600254610c52916001600160a01b0391821691166000611124565b6004546001600160a01b03161561048357600554600354610c81916001600160a01b0391821691166000611124565b600554600454610483916001600160a01b0391821691166000611124565b600c5460ff1681565b6000546001600160a01b031690565b600c54610100900460ff1681565b600154604080516370a0823160e01b815230600482015290516000926001600160a01b0316916370a08231916024808301926020929190829003018186803b15801561055157600080fd5b6004546001600160a01b031681565b600154604080516370a0823160e01b815230600482015290516000926001600160a01b0316916370a08231916024808301926020929190829003018186803b158015610d6a57600080fd5b505afa158015610d7e573d6000803e3d6000fd5b505050506040513d6020811015610d9457600080fd5b505190508015610da757610da781611793565b50565b610db261124e565b6000546001600160a01b03908116911614610e02576040805162461bcd60e51b815260206004820181905260248201526000805160206127f9833981519152604482015290519081900360640190fd5b600a80546001600160a01b0319166001600160a01b0392909216919091179055565b610e2c61124e565b6000546001600160a01b03908116911614610e7c576040805162461bcd60e51b815260206004820181905260248201526000805160206127f9833981519152604482015290519081900360640190fd5b600c805460ff1916911515919091179055565b610e9761124e565b6000546001600160a01b03908116911614610ee7576040805162461bcd60e51b815260206004820181905260248201526000805160206127f9833981519152604482015290519081900360640190fd5b6001600160a01b038116610f2c5760405162461bcd60e51b81526004018080602001828103825260268152602001806127b26026913960400191505060405180910390fd5b600080546040516001600160a01b03808516939216917f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e091a3600080546001600160a01b0319166001600160a01b0392909216919091179055565b6002546001600160a01b031681565b6005546001600160a01b031681565b6006546001600160a01b031681565b610fbc61124e565b6000546001600160a01b0390811691161461100c576040805162461bcd60e51b815260206004820181905260248201526000805160206127f9833981519152604482015290519081900360640190fd5b61101461091b565b600154604080516370a0823160e01b815230600482015290516000926001600160a01b0316916370a08231916024808301926020929190829003018186803b15801561105f57600080fd5b505afa158015611073573d6000803e3d6000fd5b505050506040513d602081101561108957600080fd5b50516001546008546040805163a9059cbb60e01b81526001600160a01b03928316600482015260248101859052905193945091169163a9059cbb916044808201926020929091908290030181600087803b1580156110e657600080fd5b505af11580156110fa573d6000803e3d6000fd5b505050506040513d602081101561111057600080fd5b505050565b6008546001600160a01b031681565b8015806111aa575060408051636eb1769f60e11b81523060048201526001600160a01b03848116602483015291519185169163dd62ed3e91604480820192602092909190829003018186803b15801561117c57600080fd5b505afa158015611190573d6000803e3d6000fd5b505050506040513d60208110156111a657600080fd5b5051155b6111e55760405162461bcd60e51b81526004018080602001828103825260368152602001806128436036913960400191505060405180910390fd5b604080516001600160a01b038416602482015260448082018490528251808303909101815260649091019091526020810180516001600160e01b031663095ea7b360e01b17905261111090849061191e565b606061124684846000856119cf565b949350505050565b3390565b600054600160a01b900460ff16156112a4576040805162461bcd60e51b815260206004820152601060248201526f14185d5cd8589b194e881c185d5cd95960821b604482015290519081900360640190fd5b60006112ae610485565b90506112ba6000611793565b600b54600254604080516370a0823160e01b815230600482015290516001600160a01b03909216916370a0823191602480820192602092909190829003018186803b15801561130857600080fd5b505afa15801561131c573d6000803e3d6000fd5b505050506040513d602081101561133257600080fd5b5051111561137057611342611b7a565b6004546001600160a01b0316156113605761135b611c7c565b611368565b611368612376565b611370610d1f565b600061137a610485565b9050818111156113c657337f169f1815ebdea059aac3bb00ec9a9594c7a5ffcb64a17e8392b5d84909a145566113b0838561140f565b60408051918252519081900360200190a2610738565b8181101561073857337fdf5ee50bba1a120f14b1b64d8ed407af28c5baa9ddb2cef0b98a75e241ab71116113fa848461140f565b60408051918252519081900360200190a25050565b600061145183836040518060400160405280601e81526020017f536166654d6174683a207375627472616374696f6e206f766572666c6f77000081525061265c565b90505b92915050565b600c5460ff16156114cf5760065460408051631058d28160e01b81526004810184905290516001600160a01b0390921691631058d2819160248082019260009290919082900301818387803b1580156114b257600080fd5b505af11580156114c6573d6000803e3d6000fd5b5050505061153f565b60065460075460408051630441a3e760e41b8152600481019290925260248201849052516001600160a01b039092169163441a3e709160448082019260009290919082900301818387803b15801561152657600080fd5b505af115801561153a573d6000803e3d6000fd5b505050505b60408051828152905133917f884edad9ce6fa2440d8a54cc123490eb96d2768479d49ff9c7366125a9424364919081900360200190a250565b604080516001600160a01b038416602482015260448082018490528251808303909101815260649091019091526020810180516001600160e01b031663a9059cbb60e01b17905261111090849061191e565b600054600160a01b900460ff1661161f576040805162461bcd60e51b815260206004820152601460248201527314185d5cd8589b194e881b9bdd081c185d5cd95960621b604482015290519081900360640190fd5b6000805460ff60a01b191690557f5db9ee0a495bf2e6ff9c91a7834c1ba4fdd244a5e8aa4e537bd38aeae4b073aa61165561124e565b604080516001600160a01b039092168252519081900360200190a1565b6000813f7fc5d2460186f7233c927e7db2dcc703c0e500b653ca82273b7bfad8045d85a470818114801590611246575050151592915050565b600082820183811015611451576040805162461bcd60e51b815260206004820152601b60248201527f536166654d6174683a206164646974696f6e206f766572666c6f770000000000604482015290519081900360640190fd5b600054600160a01b900460ff1615611757576040805162461bcd60e51b815260206004820152601060248201526f14185d5cd8589b194e881c185d5cd95960821b604482015290519081900360640190fd5b6000805460ff60a01b1916600160a01b1790557f62e78cea01bee320cd4e420270b5ea74000d11b0c9f74754ebdbfc544b05a25861165561124e565b600c5460ff161561180857600654604080516341441d3b60e01b81526004810184905290516001600160a01b03909216916341441d3b9160248082019260009290919082900301818387803b1580156117eb57600080fd5b505af11580156117ff573d6000803e3d6000fd5b505050506118e5565b600c54610100900460ff16156118755760065460075460408051638dbdbe6d60e01b815260048101929092526024820184905260006044830181905290516001600160a01b0390931692638dbdbe6d92606480820193929182900301818387803b1580156117eb57600080fd5b60065460075460408051631c57762b60e31b8152600481019290925260248201849052516001600160a01b039092169163e2bbb1589160448082019260009290919082900301818387803b1580156118cc57600080fd5b505af11580156118e0573d6000803e3d6000fd5b505050505b60408051828152905133917fe1fffcc4923d04b559f4d29a8bfc6cda04eb5b0d3c460751c2402c5c5cc9109c919081900360200190a250565b6060611973826040518060400160405280602081526020017f5361666545524332303a206c6f772d6c6576656c2063616c6c206661696c6564815250856001600160a01b03166112379092919063ffffffff16565b8051909150156111105780806020019051602081101561199257600080fd5b50516111105760405162461bcd60e51b815260040180806020018281038252602a815260200180612819602a913960400191505060405180910390fd5b60606119da85611672565b611a2b576040805162461bcd60e51b815260206004820152601d60248201527f416464726573733a2063616c6c20746f206e6f6e2d636f6e7472616374000000604482015290519081900360640190fd5b60006060866001600160a01b031685876040518082805190602001908083835b60208310611a6a5780518252601f199092019160209182019101611a4b565b6001836020036101000a03801982511681845116808217855250505050505090500191505060006040518083038185875af1925050503d8060008114611acc576040519150601f19603f3d011682016040523d82523d6000602084013e611ad1565b606091505b50915091508115611ae55791506112469050565b805115611af55780518082602001fd5b8360405162461bcd60e51b81526004018080602001828103825283818151815260200191508051906020019080838360005b83811015611b3f578181015183820152602001611b27565b50505050905090810190601f168015611b6c5780820380516001836020036101000a031916815260200191505b509250505060405180910390fd5b6009546001600160a01b031615610483576000611c21612710611c1b611b9e61050c565b600254604080516370a0823160e01b815230600482015290516001600160a01b03909216916370a0823191602480820192602092909190829003018186803b158015611be957600080fd5b505afa158015611bfd573d6000803e3d6000fd5b505050506040513d6020811015611c1357600080fd5b5051906126b6565b9061270f565b6002546009546040805163a9059cbb60e01b81526001600160a01b03928316600482015260248101859052905193945091169163a9059cbb916044808201926020929091908290030181600087803b1580156110e657600080fd5b600254604080516370a0823160e01b815230600482015290516000926001600160a01b0316916370a08231916024808301926020929190829003018186803b158015611cc757600080fd5b505afa158015611cdb573d6000803e3d6000fd5b505050506040513d6020811015611cf157600080fd5b505190506000611d0282600261270f565b600a546005546002546004805460408051631206e48360e21b81526001600160a01b03958616938101939093529284166024830152831660448201529051939450606093919092169163481b920c916064808301926000929190829003018186803b158015611d7057600080fd5b505afa158015611d84573d6000803e3d6000fd5b505050506040513d6000823e601f3d908101601f191682016040526020811015611dad57600080fd5b8101908080516040519392919084640100000000821115611dcd57600080fd5b908301906020820185811115611de257600080fd5b8251866020820283011164010000000082111715611dff57600080fd5b82525081516020918201928201910280838360005b83811015611e2c578181015183820152602001611e14565b5050505090500160405250505090506000815111611e91576040805162461bcd60e51b815260206004820152601760248201527f636865636b207377617050617468526567697374727931000000000000000000604482015290519081900360640190fd5b600554604051635c11d79560e01b8152600481018481526000602483018190523060648401819052426084850181905260a060448601908152875160a487015287516001600160a01b0390971696635c11d795968a968a9594939092909160c40190602080880191028083838b5b83811015611f17578181015183820152602001611eff565b505050509050019650505050505050600060405180830381600087803b158015611f4057600080fd5b505af1158015611f54573d6000803e3d6000fd5b50506002546003546001600160a01b0390811691161491506121c5905057600a5460055460025460035460408051631206e48360e21b81526001600160a01b0394851660048201529284166024840152908316604483015251606093929092169163481b920c91606480820192600092909190829003018186803b158015611fdb57600080fd5b505afa158015611fef573d6000803e3d6000fd5b505050506040513d6000823e601f3d908101601f19168201604052602081101561201857600080fd5b810190808051604051939291908464010000000082111561203857600080fd5b90830190602082018581111561204d57600080fd5b825186602082028301116401000000008211171561206a57600080fd5b82525081516020918201928201910280838360005b8381101561209757818101518382015260200161207f565b50505050905001604052505050905060008151116120fc576040805162461bcd60e51b815260206004820152601760248201527f636865636b207377617050617468526567697374727932000000000000000000604482015290519081900360640190fd5b600554604051635c11d79560e01b8152600481018581526000602483018190523060648401819052426084850181905260a060448601908152875160a487015287516001600160a01b0390971696635c11d795968b968a9594939092909160c40190602080880191028083838b5b8381101561218257818101518382015260200161216a565b505050509050019650505050505050600060405180830381600087803b1580156121ab57600080fd5b505af11580156121bf573d6000803e3d6000fd5b50505050505b60055460035460048054604080516370a0823160e01b81523093810193909352516001600160a01b039485169463e8e337009481169392169183916370a0823191602480820192602092909190829003018186803b15801561222657600080fd5b505afa15801561223a573d6000803e3d6000fd5b505050506040513d602081101561225057600080fd5b505160048054604080516370a0823160e01b81523093810193909352516001600160a01b03909116916370a08231916024808301926020929190829003018186803b15801561229e57600080fd5b505afa1580156122b2573d6000803e3d6000fd5b505050506040513d60208110156122c857600080fd5b5051604080516001600160e01b031960e088901b1681526001600160a01b0395861660048201529390941660248401526044830191909152606482015260006084820181905260a482018190523060c48301524260e483015291516101048083019360609383900390910190829087803b15801561234557600080fd5b505af1158015612359573d6000803e3d6000fd5b505050506040513d606081101561236f57600080fd5b5050505050565b6001546002546001600160a01b0390811691161461048357600254604080516370a0823160e01b815230600482015290516000926001600160a01b0316916370a08231916024808301926020929190829003018186803b1580156123d957600080fd5b505afa1580156123ed573d6000803e3d6000fd5b505050506040513d602081101561240357600080fd5b5051600a5460055460025460015460408051631206e48360e21b81526001600160a01b0394851660048201529284166024840152908316604483015251939450606093919092169163481b920c916064808301926000929190829003018186803b15801561247057600080fd5b505afa158015612484573d6000803e3d6000fd5b505050506040513d6000823e601f3d908101601f1916820160405260208110156124ad57600080fd5b81019080805160405193929190846401000000008211156124cd57600080fd5b9083019060208201858111156124e257600080fd5b82518660208202830111640100000000821117156124ff57600080fd5b82525081516020918201928201910280838360005b8381101561252c578181015183820152602001612514565b5050505090500160405250505090506000815111612591576040805162461bcd60e51b815260206004820152601760248201527f636865636b207377617050617468526567697374727933000000000000000000604482015290519081900360640190fd5b600554604051635c11d79560e01b8152600481018481526000602483018190523060648401819052426084850181905260a060448601908152875160a487015287516001600160a01b0390971696635c11d795968a968a9594939092909160c40190602080880191028083838b5b838110156126175781810151838201526020016125ff565b505050509050019650505050505050600060405180830381600087803b15801561264057600080fd5b505af1158015612654573d6000803e3d6000fd5b505050505050565b600081848411156126ae5760405162461bcd60e51b8152602060048201818152835160248401528351909283926044909101919085019080838360008315611b3f578181015183820152602001611b27565b505050900390565b6000826126c557506000611454565b828202828482816126d257fe5b04146114515760405162461bcd60e51b81526004018080602001828103825260218152602001806127d86021913960400191505060405180910390fd5b600061145183836040518060400160405280601a81526020017f536166654d6174683a206469766973696f6e206279207a65726f0000000000008152506000818361279b5760405162461bcd60e51b8152602060048201818152835160248401528351909283926044909101919085019080838360008315611b3f578181015183820152602001611b27565b5060008385816127a757fe5b049594505050505056fe4f776e61626c653a206e6577206f776e657220697320746865207a65726f2061646472657373536166654d6174683a206d756c7469706c69636174696f6e206f766572666c6f774f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e65725361666545524332303a204552433230206f7065726174696f6e20646964206e6f7420737563636565645361666545524332303a20617070726f76652066726f6d206e6f6e2d7a65726f20746f206e6f6e2d7a65726f20616c6c6f77616e6365a26469706673582212204e31d5a0f233e862b3e6bc9f6c32820a15e73b6655fd53e3f31d6d8bcad9ad2564736f6c634300060c0033