Latest 25 from a total of 13,525 transactions
| Transaction Hash |
Method
|
Block
|
From
|
|
To
|
||||
|---|---|---|---|---|---|---|---|---|---|
| Buy And Redeem | 20918048 | 484 days ago | IN | 0.14119286 ETH | 0.0058934 | ||||
| Buy And Redeem | 20482787 | 545 days ago | IN | 0.11593788 ETH | 0.00099225 | ||||
| Mint And Sell721 | 20431717 | 552 days ago | IN | 0 ETH | 0.00031718 | ||||
| Mint And Sell721 | 20431715 | 552 days ago | IN | 0 ETH | 0.00220358 | ||||
| Buy And Redeem | 20431124 | 552 days ago | IN | 0.07043281 ETH | 0.00167503 | ||||
| Buy And Redeem | 20430377 | 552 days ago | IN | 0.04430835 ETH | 0.00370797 | ||||
| Mint And Sell721 | 20429886 | 552 days ago | IN | 0 ETH | 0.00233793 | ||||
| Mint And Sell721 | 20429726 | 552 days ago | IN | 0 ETH | 0.00218305 | ||||
| Mint And Sell115... | 20429463 | 552 days ago | IN | 0 ETH | 0.00419664 | ||||
| Buy And Redeem | 20425304 | 553 days ago | IN | 0.08078326 ETH | 0.00245222 | ||||
| Mint And Sell721 | 20424286 | 553 days ago | IN | 0 ETH | 0.00181372 | ||||
| Mint And Sell721 | 20424277 | 553 days ago | IN | 0 ETH | 0.00291422 | ||||
| Mint And Sell721 | 20423410 | 553 days ago | IN | 0 ETH | 0.00458937 | ||||
| Buy And Redeem | 20423351 | 553 days ago | IN | 24.31549613 ETH | 0.00200837 | ||||
| Mint And Sell721 | 20423256 | 553 days ago | IN | 0 ETH | 0.00512502 | ||||
| Mint And Sell721 | 20418761 | 553 days ago | IN | 0 ETH | 0.02598955 | ||||
| Mint And Sell721 | 20418753 | 553 days ago | IN | 0 ETH | 0.02493042 | ||||
| Mint And Sell721 | 20418638 | 554 days ago | IN | 0 ETH | 0.06887511 | ||||
| Mint And Sell115... | 20417864 | 554 days ago | IN | 0 ETH | 0.00260962 | ||||
| Mint And Sell721 | 20416041 | 554 days ago | IN | 0 ETH | 0.00201812 | ||||
| Buy And Redeem | 20415183 | 554 days ago | IN | 0.05084534 ETH | 0.00170925 | ||||
| Mint And Sell721 | 20415164 | 554 days ago | IN | 0 ETH | 0.00101147 | ||||
| Buy And Redeem | 20415065 | 554 days ago | IN | 0.00848387 ETH | 0.00211288 | ||||
| Buy And Redeem | 20414571 | 554 days ago | IN | 0.00971227 ETH | 0.00222103 | ||||
| Buy And Swap721 | 20412233 | 554 days ago | IN | 0.007544 ETH | 0.00209071 |
Latest 25 internal transactions (View All)
Advanced mode:
| Parent Transaction Hash | Method | Block |
From
|
|
To
|
||
|---|---|---|---|---|---|---|---|
| Transfer | 22244181 | 299 days ago | 0.00199579 ETH | ||||
| Transfer | 22244181 | 299 days ago | 0.00199579 ETH | ||||
| Deposit | 22244181 | 299 days ago | 0.40115468 ETH | ||||
| Buy And Redeem | 22244181 | 299 days ago | 0.40115468 ETH | ||||
| Transfer | 22243327 | 299 days ago | 0.0001515 ETH | ||||
| Transfer | 22243327 | 299 days ago | 0.0001515 ETH | ||||
| Deposit | 22243327 | 299 days ago | 0.03045267 ETH | ||||
| Buy And Redeem | 22243327 | 299 days ago | 0.03045267 ETH | ||||
| Transfer | 22236308 | 300 days ago | 0.0025507 ETH | ||||
| Transfer | 22236308 | 300 days ago | 0.0025507 ETH | ||||
| Deposit | 22236308 | 300 days ago | 0.51269202 ETH | ||||
| Buy And Redeem | 22236308 | 300 days ago | 0.51269202 ETH | ||||
| Deposit | 22234309 | 300 days ago | 0.11668468 ETH | ||||
| Buy And Redeem | 22234309 | 300 days ago | 0.11668468 ETH | ||||
| Deposit | 22232729 | 300 days ago | 0.12619542 ETH | ||||
| Buy And Redeem | 22232729 | 300 days ago | 0.12619542 ETH | ||||
| Deposit | 22232714 | 300 days ago | 0.11668468 ETH | ||||
| Buy And Redeem | 22232714 | 300 days ago | 0.11668468 ETH | ||||
| Transfer | 22232711 | 300 days ago | 0.00058052 ETH | ||||
| Transfer | 22232711 | 300 days ago | 0.00058052 ETH | ||||
| Deposit | 22232711 | 300 days ago | 0.11668468 ETH | ||||
| Buy And Redeem | 22232711 | 300 days ago | 0.11668468 ETH | ||||
| Transfer | 22230605 | 301 days ago | 0.0000409 ETH | ||||
| Transfer | 22230605 | 301 days ago | 0.0000409 ETH | ||||
| Deposit | 22230605 | 301 days ago | 0.00822175 ETH |
Loading...
Loading
Loading...
Loading
Cross-Chain Transactions
Loading...
Loading
Contract Name:
NFTXMarketplace0xZap
Compiler Version
v0.8.4+commit.c7e474f2
Contract Source Code (Solidity Standard Json-Input format)
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.0;
import "./interface/INFTXVault.sol";
import "./interface/INFTXVaultFactory.sol";
import "./testing/IERC1155.sol";
import "./testing/ERC721Holder.sol";
import "./testing/ERC1155Holder.sol";
import "./util/Ownable.sol";
import "./util/ReentrancyGuard.sol";
import "./util/SafeERC20.sol";
/**
* @notice A partial WETH interface.
*/
interface IWETH {
function deposit() external payable;
function transfer(address to, uint value) external returns (bool);
function withdraw(uint) external;
function balanceOf(address to) external view returns (uint256);
}
/**
* @notice Sets up a marketplace zap to interact with the 0x protocol. The 0x contract that
* is hit later on handles the token conversion based on parameters that are sent from the
* frontend.
*
* @author Twade
*/
contract NFTXMarketplace0xZap is Ownable, ReentrancyGuard, ERC721Holder, ERC1155Holder {
using SafeERC20 for IERC20;
/// @notice Allows zap to be paused
bool public paused = false;
/// @notice Sets our 0x swap target
address payable private immutable swapTarget;
/// @notice An interface for the WETH contract
IWETH public immutable WETH;
/// @notice An interface for the NFTX Vault Factory contract
INFTXVaultFactory public immutable nftxFactory;
address public immutable feeDistributor;
/// @notice The vToken threshold below which dust is sent to feeDistributor, else back to the user
uint256 public dustThreshold;
/// @notice A mapping of NFTX Vault IDs to their address corresponding vault contract address
mapping(uint256 => address) public nftxVaultAddresses;
// Set a constant address for specific contracts that need special logic
address constant CRYPTO_PUNKS = 0xb47e3cd837dDF8e4c57F05d70Ab865de6e193BBB;
/// @notice Emitted by the `buyAndRedeem` function.
/// @param count The number of tokens affected by the event
/// @param ethSpent The amount of ETH spent in the buy
/// @param to The user affected by the event
event Buy(uint256 count, uint256 ethSpent, address to);
/// @notice Emitted by the `mintAndSell` functions.
/// @param count The number of tokens affected by the event
/// @param ethReceived The amount of ETH received in the sell
/// @param to The user affected by the event
event Sell(uint256 count, uint256 ethReceived, address to);
/// @notice Emitted by the `buyAndSwap` functions.
/// @param count The number of tokens affected by the event
/// @param ethSpent The amount of ETH spent in the swap
/// @param to The user affected by the event
event Swap(uint256 count, uint256 ethSpent, address to);
/// @notice Emitted when dust is returned after a transaction.
/// @param ethAmount Amount of ETH returned to user
/// @param vTokenAmount Amount of vToken returned to user
/// @param to The user affected by the event
event DustReturned(uint256 ethAmount, uint256 vTokenAmount, address to);
/**
* @notice Initialises our zap by setting contract addresses onto their
* respective interfaces.
*
* @param _nftxFactory NFTX Vault Factory contract address
* @param _WETH WETH contract address
* @param _swapTarget The swap target specified by the 0x protocol
*/
constructor(address _nftxFactory, address _WETH, address payable _swapTarget, uint256 _dustThreshold) Ownable() ReentrancyGuard() {
nftxFactory = INFTXVaultFactory(_nftxFactory);
WETH = IWETH(_WETH);
swapTarget = _swapTarget;
feeDistributor = INFTXVaultFactory(_nftxFactory).feeDistributor();
dustThreshold = _dustThreshold;
}
/**
* @notice Mints tokens from our NFTX vault and sells them on 0x.
*
* @param vaultId The ID of the NFTX vault
* @param ids An array of token IDs to be minted
* @param swapCallData The `data` field from the API response
* @param to The recipient of ETH from the tx
*/
function mintAndSell721(
uint256 vaultId,
uint256[] calldata ids,
bytes calldata swapCallData,
address payable to
) external nonReentrant onlyOwnerIfPaused {
// Check that we aren't burning tokens or sending to ourselves
require(to != address(0) && to != address(this), 'Invalid recipient');
// Check that we have been provided IDs
require(ids.length != 0, 'Must send IDs');
// Mint our 721s against the vault
address vault = _mint721(vaultId, ids);
// Sell our vault token for WETH
uint256 amount = _fillQuote(vault, address(WETH), swapCallData);
// convert WETH to ETH and send to `to`
_transferAllWETH(to);
// Emit our sale event
emit Sell(ids.length, amount, to);
// Handle vault token dust
_transferDust(vault, false);
}
/**
* @notice Purchases vault tokens from 0x with WETH and then swaps the tokens for
* either random or specific token IDs from the vault. The specified recipient will
* receive the ERC721 tokens, as well as any WETH dust that is left over from the tx.
*
* @param vaultId The ID of the NFTX vault
* @param idsIn An array of random token IDs to be minted
* @param specificIds An array of any specific token IDs to be minted
* @param swapCallData The `data` field from the API response
* @param to The recipient of the token IDs from the tx
*/
function buyAndSwap721(
uint256 vaultId,
uint256[] calldata idsIn,
uint256[] calldata specificIds,
bytes calldata swapCallData,
address payable to
) external payable nonReentrant onlyOwnerIfPaused {
// Check that we aren't burning tokens or sending to ourselves
require(to != address(0) && to != address(this), 'Invalid recipient');
// Check that we have been provided IDs
require(idsIn.length != 0, 'Must send IDs');
// Check that we have a message value sent
require(msg.value > 0, 'Invalid amount');
// Wrap ETH into WETH for our contract from the sender
WETH.deposit{value: msg.value}();
// Get our NFTX vault
address vault = _vaultAddress(vaultId);
// Buy enough vault tokens to fuel our buy
uint256 amount = _fillQuote(address(WETH), vault, swapCallData);
// Swap our tokens for the IDs requested
_swap721(vaultId, idsIn, specificIds, to);
emit Swap(idsIn.length, amount, to);
// Transfer dust ETH to sender and handle vault token dust
_transferDust(vault, true);
}
/**
* @notice Purchases vault tokens from 0x with WETH and then redeems the tokens for
* either random or specific token IDs from the vault. The specified recipient will
* receive the ERC721 tokens, as well as any WETH dust that is left over from the tx.
*
* @param vaultId The ID of the NFTX vault
* @param amount The number of tokens to buy
* @param specificIds An array of any specific token IDs to be minted
* @param swapCallData The `data` field from the API response
* @param to The recipient of the token IDs from the tx
*/
function buyAndRedeem(
uint256 vaultId,
uint256 amount,
uint256[] calldata specificIds,
bytes calldata swapCallData,
address payable to
) external payable nonReentrant onlyOwnerIfPaused {
// Check that we aren't burning tokens or sending to ourselves
require(to != address(0) && to != address(this), 'Invalid recipient');
// Check that we have an amount specified
require(amount > 0, 'Must send amount');
// Wrap ETH into WETH for our contract from the sender
WETH.deposit{value: msg.value}();
// Get our vault address information
address vault = _vaultAddress(vaultId);
// Buy vault tokens that will cover our transaction
uint256 quoteAmount = _fillQuote(address(WETH), vault, swapCallData);
// check if received sufficient vault tokens
require(quoteAmount >= amount * 1e18, 'Insufficient vault tokens');
// Redeem token IDs from the vault
_redeem(vaultId, amount, specificIds, to);
emit Buy(amount, quoteAmount, to);
// Transfer dust ETH to sender and handle vault token dust
_transferDust(vault, true);
}
/**
* @notice Mints tokens from our NFTX vault and sells them on 0x.
*
* @param vaultId The ID of the NFTX vault
* @param ids An array of token IDs to be minted
* @param amounts The number of the corresponding ID to be minted
* @param swapCallData The `data` field from the API response
* @param to The recipient of ETH from the tx
*/
function mintAndSell1155(
uint256 vaultId,
uint256[] calldata ids,
uint256[] calldata amounts,
bytes calldata swapCallData,
address payable to
) external nonReentrant onlyOwnerIfPaused {
// Check that we aren't burning tokens or sending to ourselves
require(to != address(0) && to != address(this), 'Invalid recipient');
// Get a sum of the total number of IDs we have sent up, and validate that
// the data sent through is valid.
(, uint totalAmount) = _validate1155Ids(ids, amounts);
// Mint our 1155s against the vault
address vault = _mint1155(vaultId, ids, amounts);
// Sell our vault token for WETH
uint256 amount = _fillQuote(vault, address(WETH), swapCallData);
// convert WETH to ETH and send to `to`
_transferAllWETH(to);
// Emit our sale event
emit Sell(totalAmount, amount, to);
// Handle vault token dust
_transferDust(vault, false);
}
/**
* @notice Purchases vault tokens from 0x with WETH and then swaps the tokens for
* either random or specific token IDs from the vault. The specified recipient will
* receive the ERC1155 tokens, as well as any WETH dust that is left over from the tx.
*
* @param vaultId The ID of the NFTX vault
* @param idsIn An array of random token IDs to be minted
* @param specificIds An array of any specific token IDs to be minted
* @param swapCallData The `data` field from the API response
* @param to The recipient of token IDs from the tx
*/
function buyAndSwap1155(
uint256 vaultId,
uint256[] calldata idsIn,
uint256[] calldata amounts,
uint256[] calldata specificIds,
bytes calldata swapCallData,
address payable to
) external payable nonReentrant onlyOwnerIfPaused {
// Check that we aren't burning tokens or sending to ourselves
require(to != address(0) && to != address(this), 'Invalid recipient');
// Check that we have a message value sent
require(msg.value > 0, 'Invalid amount');
// Get a sum of the total number of IDs we have sent up, and validate that
// the data sent through is valid.
(, uint totalAmount) = _validate1155Ids(idsIn, amounts);
// Wrap ETH into WETH for our contract from the sender
WETH.deposit{value: msg.value}();
// Get our NFTX vault
address vault = _vaultAddress(vaultId);
// Buy enough vault tokens to fuel our buy
uint256 amount = _fillQuote(address(WETH), vault, swapCallData);
// Swap our tokens for the IDs requested
_swap1155(vaultId, idsIn, amounts, specificIds, to);
emit Swap(totalAmount, amount, to);
// Transfer dust ETH to sender and handle vault token dust
_transferDust(vault, true);
}
/**
* @param vaultId The ID of the NFTX vault
* @param ids An array of token IDs to be minted
*/
function _mint721(uint256 vaultId, uint256[] memory ids) internal returns (address) {
// Get our vault address information
address vault = _vaultAddress(vaultId);
// Transfer tokens from the message sender to the vault
address assetAddress = INFTXVault(vault).assetAddress();
uint256 length = ids.length;
for (uint256 i; i < length;) {
transferFromERC721(assetAddress, ids[i], vault);
if (assetAddress == CRYPTO_PUNKS) {
_approveERC721(assetAddress, ids[i], vault);
}
unchecked { ++i; }
}
// Mint our tokens from the vault to this contract
uint256[] memory emptyIds;
INFTXVault(vault).mint(ids, emptyIds);
return vault;
}
/**
* @param vaultId The ID of the NFTX vault
* @param ids An array of token IDs to be minted
* @param amounts An array of amounts whose indexes map to the ids array
*/
function _mint1155(uint256 vaultId, uint256[] memory ids, uint256[] memory amounts) internal returns (address) {
// Get our vault address information
address vault = _vaultAddress(vaultId);
// Transfer tokens from the message sender to the vault
address assetAddress = INFTXVault(vault).assetAddress();
IERC1155(assetAddress).safeBatchTransferFrom(msg.sender, address(this), ids, amounts, "");
IERC1155(assetAddress).setApprovalForAll(vault, true);
// Mint our tokens from the vault to this contract
INFTXVault(vault).mint(ids, amounts);
return vault;
}
/**
*
* @param vaultId The ID of the NFTX vault
* @param idsIn An array of token IDs to be minted
* @param idsOut An array of token IDs to be redeemed
* @param to The recipient of the idsOut from the tx
*/
function _swap721(
uint256 vaultId,
uint256[] memory idsIn,
uint256[] memory idsOut,
address to
) internal returns (address) {
// Get our vault address information
address vault = _vaultAddress(vaultId);
// Transfer tokens to zap
address assetAddress = INFTXVault(vault).assetAddress();
uint256 length = idsIn.length;
for (uint256 i; i < length;) {
transferFromERC721(assetAddress, idsIn[i], vault);
if (assetAddress == CRYPTO_PUNKS) {
_approveERC721(assetAddress, idsIn[i], vault);
}
unchecked { ++i; }
}
// Swap our tokens
uint256[] memory emptyIds;
INFTXVault(vault).swapTo(idsIn, emptyIds, idsOut, to);
return vault;
}
/**
* @notice Swaps 1155 tokens, transferring them from the recipient to this contract, and
* then sending them to the NFTX vault, that sends them to the recipient.
*
* @param vaultId The ID of the NFTX vault
* @param idsIn The IDs owned by the sender to be swapped
* @param amounts The number of each corresponding ID being swapped
* @param idsOut The requested IDs to be swapped for
* @param to The recipient of the swapped tokens
*
* @return address The address of the NFTX vault
*/
function _swap1155(
uint256 vaultId,
uint256[] memory idsIn,
uint256[] memory amounts,
uint256[] memory idsOut,
address to
) internal returns (address) {
// Get our vault address information
address vault = _vaultAddress(vaultId);
// Transfer tokens to zap and mint to NFTX.
address assetAddress = INFTXVault(vault).assetAddress();
IERC1155(assetAddress).safeBatchTransferFrom(msg.sender, address(this), idsIn, amounts, "");
IERC1155(assetAddress).setApprovalForAll(vault, true);
INFTXVault(vault).swapTo(idsIn, amounts, idsOut, to);
return vault;
}
/**
* @notice Redeems tokens from a vault to a recipient.
*
* @param vaultId The ID of the NFTX vault
* @param amount The number of tokens to be redeemed
* @param specificIds Specified token IDs if desired, otherwise will be _random_
* @param to The recipient of the token
*/
function _redeem(uint256 vaultId, uint256 amount, uint256[] memory specificIds, address to) internal {
INFTXVault(_vaultAddress(vaultId)).redeemTo(amount, specificIds, to);
}
/**
* @notice Transfers our ERC721 tokens to a specified recipient.
*
* @param assetAddr Address of the asset being transferred
* @param tokenId The ID of the token being transferred
* @param to The address the token is being transferred to
*/
function transferFromERC721(address assetAddr, uint256 tokenId, address to) internal virtual {
bytes memory data;
if (assetAddr == CRYPTO_PUNKS) {
// Fix here for frontrun attack.
bytes memory punkIndexToAddress = abi.encodeWithSignature("punkIndexToAddress(uint256)", tokenId);
(bool checkSuccess, bytes memory result) = address(assetAddr).staticcall(punkIndexToAddress);
(address nftOwner) = abi.decode(result, (address));
require(checkSuccess && nftOwner == msg.sender, "Not the NFT owner");
data = abi.encodeWithSignature("buyPunk(uint256)", tokenId);
} else {
// We push to the vault to avoid an unneeded transfer.
data = abi.encodeWithSignature("safeTransferFrom(address,address,uint256)", msg.sender, to, tokenId);
}
(bool success, bytes memory resultData) = address(assetAddr).call(data);
require(success, string(resultData));
}
/**
* @notice Approves our ERC721 tokens to be transferred.
*
* @dev This is only required to provide special logic for Cryptopunks.
*
* @param assetAddr Address of the asset being transferred
* @param tokenId The ID of the token being transferred
* @param to The address the token is being transferred to
*/
function _approveERC721(address assetAddr, uint256 tokenId, address to) internal virtual {
if (assetAddr != CRYPTO_PUNKS) {
return;
}
bytes memory data = abi.encodeWithSignature("offerPunkForSaleToAddress(uint256,uint256,address)", tokenId, 0, to);
(bool success, bytes memory resultData) = address(assetAddr).call(data);
require(success, string(resultData));
}
/**
* @notice Swaps ERC20->ERC20 tokens held by this contract using a 0x-API quote.
*
* @dev Must attach ETH equal to the `value` field from the API response.
*
* @param sellToken The `sellTokenAddress` field from the API response
* @param buyToken The `buyTokenAddress` field from the API response
* @param swapCallData The `data` field from the API response
*/
function _fillQuote(
address sellToken,
address buyToken,
bytes calldata swapCallData
) internal returns (uint256) {
// Track our balance of the buyToken to determine how much we've bought.
uint256 boughtAmount = IERC20(buyToken).balanceOf(address(this));
// Give `swapTarget` an infinite allowance to spend this contract's `sellToken`.
// Note that for some tokens (e.g., USDT, KNC), you must first reset any existing
// allowance to 0 before being able to update it.
require(IERC20(sellToken).approve(swapTarget, type(uint256).max), 'Unable to approve contract');
// Call the encoded swap function call on the contract at `swapTarget`
(bool success,) = swapTarget.call(swapCallData);
require(success, 'SWAP_CALL_FAILED');
// Use our current buyToken balance to determine how much we've bought.
return IERC20(buyToken).balanceOf(address(this)) - boughtAmount;
}
/**
* @notice Transfers remaining ETH to msg.sender.
* And transfers vault token dust to feeDistributor if below dustThreshold, else to msg.sender
*
* @param vault Address of the vault token
* @param isWETHDust Checks and transfers WETH dust if boolean is true
*/
function _transferDust(address vault, bool isWETHDust) internal {
uint256 remaining;
if(isWETHDust) {
remaining = _transferAllWETH(msg.sender);
}
uint256 dustBalance = IERC20(vault).balanceOf(address(this));
address dustRecipient;
if(dustBalance > 0) {
if (dustBalance > dustThreshold) {
dustRecipient = msg.sender;
} else {
dustRecipient = feeDistributor;
}
IERC20(vault).transfer(dustRecipient, dustBalance);
}
emit DustReturned(remaining, dustBalance, dustRecipient);
}
function _transferAllWETH(address recipient) internal returns(uint256 amount) {
amount = WETH.balanceOf(address(this));
if (amount > 0) {
// Unwrap our WETH into ETH and transfer it to the recipient
WETH.withdraw(amount);
(bool success, ) = payable(recipient).call{value: amount}("");
require(success, "Unable to send unwrapped WETH");
}
}
/**
* @notice Allows 1155 IDs and amounts to be validated.
*
* @param ids The IDs of the 1155 tokens.
* @param amounts The number of each corresponding token to process.
*
* @return totalIds The number of different IDs being sent.
* @return totalAmount The total number of IDs being processed.
*/
function _validate1155Ids(
uint[] calldata ids,
uint[] calldata amounts
) internal pure returns (
uint totalIds,
uint totalAmount
) {
totalIds = ids.length;
// Check that we have been provided IDs
require(totalIds != 0, 'Must send IDs');
require(totalIds <= amounts.length, 'Must define amounts against IDs');
// Sum the amounts for our emitted events
for (uint i; i < totalIds;) {
require(amounts[i] > 0, 'Invalid 1155 amount');
unchecked {
totalAmount += amounts[i];
++i;
}
}
}
/**
* @notice Maps a cached NFTX vault address against a vault ID.
*
* @param vaultId The ID of the NFTX vault
*/
function _vaultAddress(uint256 vaultId) internal returns (address) {
if (nftxVaultAddresses[vaultId] == address(0)) {
nftxVaultAddresses[vaultId] = nftxFactory.vault(vaultId);
}
require(nftxVaultAddresses[vaultId] != address(0), 'Vault does not exist');
return nftxVaultAddresses[vaultId];
}
/**
* @notice Allows our zap to be paused to prevent any processing.
*
* @param _paused New pause state
*/
function pause(bool _paused) external onlyOwner {
paused = _paused;
}
/**
* @notice Allows owner to modify dustThreshold value
*
* @param _dustThreshold New dustThreshold
*/
function setDustThreshold(uint256 _dustThreshold) external onlyOwner {
dustThreshold = _dustThreshold;
}
/**
* @notice Allows our owner to withdraw and tokens in the contract.
*
* @param token The address of the token to be rescued
*/
function rescue(address token) external onlyOwner {
if (token == address(0)) {
(bool success, ) = payable(msg.sender).call{value: address(this).balance}("");
require(success, "Address: unable to send value");
} else {
IERC20(token).safeTransfer(msg.sender, IERC20(token).balanceOf(address(this)));
}
}
/**
* @notice A modifier that only allows the owner to interact with the function
* if the contract is paused. If the contract is not paused then anyone can
* interact with the function.
*/
modifier onlyOwnerIfPaused() {
require(!paused || msg.sender == owner(), "Zap is paused");
_;
}
/**
* @notice Allows our contract to receive any assets.
*/
receive() external payable {}
}// SPDX-License-Identifier: MIT
pragma solidity ^0.8.0;
interface INFTXEligibility {
// Read functions.
function name() external pure returns (string memory);
function finalized() external view returns (bool);
function targetAsset() external pure returns (address);
function checkAllEligible(uint256[] calldata tokenIds)
external
view
returns (bool);
function checkEligible(uint256[] calldata tokenIds)
external
view
returns (bool[] memory);
function checkAllIneligible(uint256[] calldata tokenIds)
external
view
returns (bool);
function checkIsEligible(uint256 tokenId) external view returns (bool);
// Write functions.
function __NFTXEligibility_init_bytes(bytes calldata configData) external;
function beforeMintHook(uint256[] calldata tokenIds) external;
function afterMintHook(uint256[] calldata tokenIds) external;
function beforeRedeemHook(uint256[] calldata tokenIds) external;
function afterRedeemHook(uint256[] calldata tokenIds) external;
}// SPDX-License-Identifier: MIT
pragma solidity ^0.8.0;
import "../token/IERC20Upgradeable.sol";
import "./INFTXVaultFactory.sol";
import "./INFTXEligibility.sol";
interface INFTXVault is IERC20Upgradeable {
function manager() external view returns (address);
function assetAddress() external view returns (address);
function vaultFactory() external view returns (INFTXVaultFactory);
function eligibilityStorage() external view returns (INFTXEligibility);
function is1155() external view returns (bool);
function allowAllItems() external view returns (bool);
function enableMint() external view returns (bool);
function enableRandomRedeem() external view returns (bool);
function enableTargetRedeem() external view returns (bool);
function enableRandomSwap() external view returns (bool);
function enableTargetSwap() external view returns (bool);
function vaultId() external view returns (uint256);
function nftIdAt(uint256 holdingsIndex) external view returns (uint256);
function allHoldings() external view returns (uint256[] memory);
function totalHoldings() external view returns (uint256);
function mintFee() external view returns (uint256);
function randomRedeemFee() external view returns (uint256);
function targetRedeemFee() external view returns (uint256);
function randomSwapFee() external view returns (uint256);
function targetSwapFee() external view returns (uint256);
function vaultFees()
external
view
returns (
uint256,
uint256,
uint256,
uint256,
uint256
);
event VaultInit(
uint256 indexed vaultId,
address assetAddress,
bool is1155,
bool allowAllItems
);
event ManagerSet(address manager);
event EligibilityDeployed(uint256 moduleIndex, address eligibilityAddr);
// event CustomEligibilityDeployed(address eligibilityAddr);
event EnableMintUpdated(bool enabled);
event EnableRandomRedeemUpdated(bool enabled);
event EnableTargetRedeemUpdated(bool enabled);
event EnableRandomSwapUpdated(bool enabled);
event EnableTargetSwapUpdated(bool enabled);
event Minted(uint256[] nftIds, uint256[] amounts, address to);
event Redeemed(uint256[] nftIds, uint256[] specificIds, address to);
event Swapped(
uint256[] nftIds,
uint256[] amounts,
uint256[] specificIds,
uint256[] redeemedIds,
address to
);
function __NFTXVault_init(
string calldata _name,
string calldata _symbol,
address _assetAddress,
bool _is1155,
bool _allowAllItems
) external;
function finalizeVault() external;
function setVaultMetadata(string memory name_, string memory symbol_)
external;
function setVaultFeatures(
bool _enableMint,
bool _enableRandomRedeem,
bool _enableTargetRedeem,
bool _enableRandomSwap,
bool _enableTargetSwap
) external;
function setFees(
uint256 _mintFee,
uint256 _randomRedeemFee,
uint256 _targetRedeemFee,
uint256 _randomSwapFee,
uint256 _targetSwapFee
) external;
function disableVaultFees() external;
// This function allows for an easy setup of any eligibility module contract from the EligibilityManager.
// It takes in ABI encoded parameters for the desired module. This is to make sure they can all follow
// a similar interface.
function deployEligibilityStorage(
uint256 moduleIndex,
bytes calldata initData
) external returns (address);
// The manager has control over options like fees and features
function setManager(address _manager) external;
function mint(
uint256[] calldata tokenIds,
uint256[] calldata amounts /* ignored for ERC721 vaults */
) external returns (uint256);
function mintTo(
uint256[] calldata tokenIds,
uint256[] calldata amounts, /* ignored for ERC721 vaults */
address to
) external returns (uint256);
function redeem(uint256 amount, uint256[] calldata specificIds)
external
returns (uint256[] calldata);
function redeemTo(
uint256 amount,
uint256[] calldata specificIds,
address to
) external returns (uint256[] calldata);
function swap(
uint256[] calldata tokenIds,
uint256[] calldata amounts, /* ignored for ERC721 vaults */
uint256[] calldata specificIds
) external returns (uint256[] calldata);
function swapTo(
uint256[] calldata tokenIds,
uint256[] calldata amounts, /* ignored for ERC721 vaults */
uint256[] calldata specificIds,
address to
) external returns (uint256[] calldata);
function allValidNFTs(uint256[] calldata tokenIds)
external
view
returns (bool);
}// SPDX-License-Identifier: MIT
pragma solidity ^0.8.0;
import "../proxy/IBeacon.sol";
interface INFTXVaultFactory is IBeacon {
// Read functions.
function numVaults() external view returns (uint256);
function zapContract() external view returns (address);
function zapContracts(address addr) external view returns (bool);
function feeDistributor() external view returns (address);
function eligibilityManager() external view returns (address);
function vault(uint256 vaultId) external view returns (address);
function allVaults() external view returns (address[] memory);
function vaultsForAsset(address asset)
external
view
returns (address[] memory);
function isLocked(uint256 id) external view returns (bool);
function excludedFromFees(address addr) external view returns (bool);
function factoryMintFee() external view returns (uint64);
function factoryRandomRedeemFee() external view returns (uint64);
function factoryTargetRedeemFee() external view returns (uint64);
function factoryRandomSwapFee() external view returns (uint64);
function factoryTargetSwapFee() external view returns (uint64);
function vaultFees(uint256 vaultId)
external
view
returns (
uint256,
uint256,
uint256,
uint256,
uint256
);
event NewFeeDistributor(address oldDistributor, address newDistributor);
event NewZapContract(address oldZap, address newZap);
event UpdatedZapContract(address zap, bool excluded);
event FeeExclusion(address feeExcluded, bool excluded);
event NewEligibilityManager(address oldEligManager, address newEligManager);
event NewVault(
uint256 indexed vaultId,
address vaultAddress,
address assetAddress
);
event UpdateVaultFees(
uint256 vaultId,
uint256 mintFee,
uint256 randomRedeemFee,
uint256 targetRedeemFee,
uint256 randomSwapFee,
uint256 targetSwapFee
);
event DisableVaultFees(uint256 vaultId);
event UpdateFactoryFees(
uint256 mintFee,
uint256 randomRedeemFee,
uint256 targetRedeemFee,
uint256 randomSwapFee,
uint256 targetSwapFee
);
// Write functions.
function __NFTXVaultFactory_init(
address _vaultImpl,
address _feeDistributor
) external;
function createVault(
string calldata name,
string calldata symbol,
address _assetAddress,
bool is1155,
bool allowAllItems
) external returns (uint256);
function setFeeDistributor(address _feeDistributor) external;
function setEligibilityManager(address _eligibilityManager) external;
function setZapContract(address _zapContract, bool _excluded) external;
function setFeeExclusion(address _excludedAddr, bool excluded) external;
function setFactoryFees(
uint256 mintFee,
uint256 randomRedeemFee,
uint256 targetRedeemFee,
uint256 randomSwapFee,
uint256 targetSwapFee
) external;
function setVaultFees(
uint256 vaultId,
uint256 mintFee,
uint256 randomRedeemFee,
uint256 targetRedeemFee,
uint256 randomSwapFee,
uint256 targetSwapFee
) external;
function disableVaultFees(uint256 vaultId) external;
}// SPDX-License-Identifier: MIT
pragma solidity ^0.8.0;
/**
* @dev This is the interface that {BeaconProxy} expects of its beacon.
*/
interface IBeacon {
/**
* @dev Must return an address that can be used as a delegate call target.
*
* {BeaconProxy} will check that this address is a contract.
*/
function childImplementation() external view returns (address);
function upgradeChildTo(address newImplementation) external;
}// SPDX-License-Identifier: MIT
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) {
this; // silence state mutability warning without generating bytecode - see https://github.com/ethereum/solidity/issues/2691
return msg.data;
}
}// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts (last updated v4.5.0) (token/ERC1155/utils/ERC1155Holder.sol)
pragma solidity ^0.8.0;
import "./ERC1155Receiver.sol";
/**
* Simple implementation of `ERC1155Receiver` that will allow a contract to hold ERC1155 tokens.
*
* IMPORTANT: When inheriting this contract, you must include a way to use the received tokens, otherwise they will be
* stuck.
*
* @dev _Available since v3.1._
*/
contract ERC1155Holder is ERC1155Receiver {
function onERC1155Received(
address,
address,
uint256,
uint256,
bytes memory
) public virtual override returns (bytes4) {
return this.onERC1155Received.selector;
}
function onERC1155BatchReceived(
address,
address,
uint256[] memory,
uint256[] memory,
bytes memory
) public virtual override returns (bytes4) {
return this.onERC1155BatchReceived.selector;
}
}// SPDX-License-Identifier: MIT
pragma solidity ^0.8.0;
import "./IERC1155Receiver.sol";
import "./ERC165.sol";
/**
* @dev _Available since v3.1._
*/
abstract contract ERC1155Receiver is ERC165, IERC1155Receiver {
/**
* @dev See {IERC165-supportsInterface}.
*/
function supportsInterface(bytes4 interfaceId) public view virtual override(ERC165, IERC165) returns (bool) {
return interfaceId == type(IERC1155Receiver).interfaceId
|| super.supportsInterface(interfaceId);
}
}// SPDX-License-Identifier: MIT
pragma solidity ^0.8.0;
import "./IERC165.sol";
/**
* @dev Implementation of the {IERC165} interface.
*
* Contracts that want to implement ERC165 should inherit from this contract and override {supportsInterface} to check
* for the additional interface id that will be supported. For example:
*
* ```solidity
* function supportsInterface(bytes4 interfaceId) public view virtual override returns (bool) {
* return interfaceId == type(MyInterface).interfaceId || super.supportsInterface(interfaceId);
* }
* ```
*
* Alternatively, {ERC165Storage} provides an easier to use but more expensive implementation.
*/
abstract contract ERC165 is IERC165 {
/**
* @dev See {IERC165-supportsInterface}.
*/
function supportsInterface(bytes4 interfaceId) public view virtual override returns (bool) {
return interfaceId == type(IERC165).interfaceId;
}
}// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts v4.4.1 (token/ERC721/utils/ERC721Holder.sol)
pragma solidity ^0.8.0;
import "./IERC721Receiver.sol";
/**
* @dev Implementation of the {IERC721Receiver} interface.
*
* Accepts all token transfers.
* Make sure the contract is able to use its token with {IERC721-safeTransferFrom}, {IERC721-approve} or {IERC721-setApprovalForAll}.
*/
contract ERC721Holder is IERC721Receiver {
/**
* @dev See {IERC721Receiver-onERC721Received}.
*
* Always returns `IERC721Receiver.onERC721Received.selector`.
*/
function onERC721Received(
address,
address,
uint256,
bytes memory
) public virtual override returns (bytes4) {
return this.onERC721Received.selector;
}
}// SPDX-License-Identifier: MIT
pragma solidity ^0.8.0;
import "./IERC165.sol";
/**
* @dev Required interface of an ERC1155 compliant contract, as defined in the
* https://eips.ethereum.org/EIPS/eip-1155[EIP].
*
* _Available since v3.1._
*/
interface IERC1155 is IERC165 {
/**
* @dev Emitted when `value` tokens of token type `id` are transferred from `from` to `to` by `operator`.
*/
event TransferSingle(address indexed operator, address indexed from, address indexed to, uint256 id, uint256 value);
/**
* @dev Equivalent to multiple {TransferSingle} events, where `operator`, `from` and `to` are the same for all
* transfers.
*/
event TransferBatch(address indexed operator, address indexed from, address indexed to, uint256[] ids, uint256[] values);
/**
* @dev Emitted when `account` grants or revokes permission to `operator` to transfer their tokens, according to
* `approved`.
*/
event ApprovalForAll(address indexed account, address indexed operator, bool approved);
/**
* @dev Emitted when the URI for token type `id` changes to `value`, if it is a non-programmatic URI.
*
* If an {URI} event was emitted for `id`, the standard
* https://eips.ethereum.org/EIPS/eip-1155#metadata-extensions[guarantees] that `value` will equal the value
* returned by {IERC1155MetadataURI-uri}.
*/
event URI(string value, uint256 indexed id);
/**
* @dev Returns the amount of tokens of token type `id` owned by `account`.
*
* Requirements:
*
* - `account` cannot be the zero address.
*/
function balanceOf(address account, uint256 id) external view returns (uint256);
/**
* @dev xref:ROOT:erc1155.adoc#batch-operations[Batched] version of {balanceOf}.
*
* Requirements:
*
* - `accounts` and `ids` must have the same length.
*/
function balanceOfBatch(address[] calldata accounts, uint256[] calldata ids) external view returns (uint256[] memory);
/**
* @dev Grants or revokes permission to `operator` to transfer the caller's tokens, according to `approved`,
*
* Emits an {ApprovalForAll} event.
*
* Requirements:
*
* - `operator` cannot be the caller.
*/
function setApprovalForAll(address operator, bool approved) external;
/**
* @dev Returns true if `operator` is approved to transfer ``account``'s tokens.
*
* See {setApprovalForAll}.
*/
function isApprovedForAll(address account, address operator) external view returns (bool);
/**
* @dev Transfers `amount` tokens of token type `id` from `from` to `to`.
*
* Emits a {TransferSingle} event.
*
* Requirements:
*
* - `to` cannot be the zero address.
* - If the caller is not `from`, it must be have been approved to spend ``from``'s tokens via {setApprovalForAll}.
* - `from` must have a balance of tokens of type `id` of at least `amount`.
* - If `to` refers to a smart contract, it must implement {IERC1155Receiver-onERC1155Received} and return the
* acceptance magic value.
*/
function safeTransferFrom(address from, address to, uint256 id, uint256 amount, bytes calldata data) external;
/**
* @dev xref:ROOT:erc1155.adoc#batch-operations[Batched] version of {safeTransferFrom}.
*
* Emits a {TransferBatch} event.
*
* Requirements:
*
* - `ids` and `amounts` must have the same length.
* - If `to` refers to a smart contract, it must implement {IERC1155Receiver-onERC1155BatchReceived} and return the
* acceptance magic value.
*/
function safeBatchTransferFrom(address from, address to, uint256[] calldata ids, uint256[] calldata amounts, bytes calldata data) external;
}// SPDX-License-Identifier: MIT
pragma solidity ^0.8.0;
import "./IERC165.sol";
/**
* _Available since v3.1._
*/
interface IERC1155Receiver is IERC165 {
/**
@dev Handles the receipt of a single ERC1155 token type. This function is
called at the end of a `safeTransferFrom` after the balance has been updated.
To accept the transfer, this must return
`bytes4(keccak256("onERC1155Received(address,address,uint256,uint256,bytes)"))`
(i.e. 0xf23a6e61, or its own function selector).
@param operator The address which initiated the transfer (i.e. msg.sender)
@param from The address which previously owned the token
@param id The ID of the token being transferred
@param value The amount of tokens being transferred
@param data Additional data with no specified format
@return `bytes4(keccak256("onERC1155Received(address,address,uint256,uint256,bytes)"))` if transfer is allowed
*/
function onERC1155Received(
address operator,
address from,
uint256 id,
uint256 value,
bytes calldata data
) external returns (bytes4);
/**
@dev Handles the receipt of a multiple ERC1155 token types. This function
is called at the end of a `safeBatchTransferFrom` after the balances have
been updated. To accept the transfer(s), this must return
`bytes4(keccak256("onERC1155BatchReceived(address,address,uint256[],uint256[],bytes)"))`
(i.e. 0xbc197c81, or its own function selector).
@param operator The address which initiated the batch transfer (i.e. msg.sender)
@param from The address which previously owned the token
@param ids An array containing ids of each token being transferred (order and length must match values array)
@param values An array containing amounts of each token being transferred (order and length must match ids array)
@param data Additional data with no specified format
@return `bytes4(keccak256("onERC1155BatchReceived(address,address,uint256[],uint256[],bytes)"))` if transfer is allowed
*/
function onERC1155BatchReceived(
address operator,
address from,
uint256[] calldata ids,
uint256[] calldata values,
bytes calldata data
) external returns (bytes4);
}// SPDX-License-Identifier: MIT
pragma solidity ^0.8.0;
/**
* @dev Interface of the ERC165 standard, as defined in the
* https://eips.ethereum.org/EIPS/eip-165[EIP].
*
* Implementers can declare support of contract interfaces, which can then be
* queried by others ({ERC165Checker}).
*
* For an implementation, see {ERC165}.
*/
interface IERC165 {
/**
* @dev Returns true if this contract implements the interface defined by
* `interfaceId`. See the corresponding
* https://eips.ethereum.org/EIPS/eip-165#how-interfaces-are-identified[EIP section]
* to learn more about how these ids are created.
*
* This function call must use less than 30 000 gas.
*/
function supportsInterface(bytes4 interfaceId) external view returns (bool);
}// 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/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
pragma solidity ^0.8.0;
/**
* @title ERC721 token receiver interface
* @dev Interface for any contract that wants to support safeTransfers
* from ERC721 asset contracts.
*/
interface IERC721Receiver {
/**
* @dev Whenever an {IERC721} `tokenId` token is transferred to this contract via {IERC721-safeTransferFrom}
* by `operator` from `from`, this function is called.
*
* It must return its Solidity selector to confirm the token transfer.
* If any other value is returned or the interface is not implemented by the recipient, the transfer will be reverted.
*
* The selector can be obtained in Solidity with `IERC721.onERC721Received.selector`.
*/
function onERC721Received(address operator, address from, uint256 tokenId, bytes calldata data) external returns (bytes4);
}// SPDX-License-Identifier: MIT
pragma solidity ^0.8.0;
/**
* @dev Interface of the ERC20 standard as defined in the EIP.
*/
interface IERC20Upgradeable {
/**
* @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
);
}// SPDX-License-Identifier: MIT
pragma solidity ^0.8.0;
/**
* @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) {
// This method relies on extcodesize, which returns 0 for contracts in
// construction, since the code is only stored at the end of the
// constructor execution.
uint256 size;
// solhint-disable-next-line no-inline-assembly
assembly {
size := extcodesize(account)
}
return size > 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"
);
// 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"
);
require(isContract(target), "Address: call to non-contract");
// solhint-disable-next-line avoid-low-level-calls
(bool success, bytes memory returndata) = target.call{value: value}(
data
);
return _verifyCallResult(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) {
require(isContract(target), "Address: static call to non-contract");
// solhint-disable-next-line avoid-low-level-calls
(bool success, bytes memory returndata) = target.staticcall(data);
return _verifyCallResult(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) {
require(isContract(target), "Address: delegate call to non-contract");
// solhint-disable-next-line avoid-low-level-calls
(bool success, bytes memory returndata) = target.delegatecall(data);
return _verifyCallResult(success, returndata, errorMessage);
}
function _verifyCallResult(
bool success,
bytes memory returndata,
string memory errorMessage
) private pure returns (bytes memory) {
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);
}
}
}
}// SPDX-License-Identifier: MIT
pragma solidity ^0.8.0;
import "../testing/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() {
address msgSender = _msgSender();
_owner = msgSender;
emit OwnershipTransferred(address(0), msgSender);
}
/**
* @dev Returns the address of the current owner.
*/
function owner() public view virtual 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;
}
}// SPDX-License-Identifier: MIT
pragma solidity ^0.8.0;
abstract contract ReentrancyGuard {
// Booleans are more expensive than uint256 or any type that takes up a full
// word because each write operation emits an extra SLOAD to first read the
// slot's contents, replace the bits taken up by the boolean, and then write
// back. This is the compiler's defense against contract upgrades and
// pointer aliasing, and it cannot be disabled.
// The values being non-zero value makes deployment a bit more expensive,
// but in exchange the refund on every call to nonReentrant will be lower in
// amount. Since refunds are capped to a percentage of the total
// transaction's gas, it is best to keep them low in cases like this one, to
// increase the likelihood of the full refund coming into effect.
uint256 private constant _NOT_ENTERED = 1;
uint256 private constant _ENTERED = 2;
uint256 private _status;
constructor() {
_status = _NOT_ENTERED;
}
/**
* @dev Prevents a contract from calling itself, directly or indirectly.
* Calling a `nonReentrant` function from another `nonReentrant`
* function is not supported. It is possible to prevent this from happening
* by making the `nonReentrant` function external, and make it call a
* `private` function that does the actual work.
*/
modifier nonReentrant() {
// On the first call to nonReentrant, _notEntered will be true
require(_status != _ENTERED, "ReentrancyGuard: reentrant call");
// Any calls to nonReentrant after this point will fail
_status = _ENTERED;
_;
// By storing the original value once again, a refund is triggered (see
// https://eips.ethereum.org/EIPS/eip-2200)
_status = _NOT_ENTERED;
}
}// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts (last updated v4.8.0) (token/ERC20/utils/SafeERC20.sol)
pragma solidity ^0.8.0;
import "../testing/IERC20.sol";
import "../testing/IERC20Permit.sol";
import "./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"
);
}
}
}{
"evmVersion": "istanbul",
"libraries": {},
"metadata": {
"bytecodeHash": "ipfs",
"useLiteralContent": true
},
"optimizer": {
"enabled": true,
"runs": 1000
},
"remappings": [],
"outputSelection": {
"*": {
"*": [
"evm.bytecode",
"evm.deployedBytecode",
"devdoc",
"userdoc",
"metadata",
"abi"
]
}
}
}Contract Security Audit
- No Contract Security Audit Submitted- Submit Audit Here
Contract ABI
API[{"inputs":[{"internalType":"address","name":"_nftxFactory","type":"address"},{"internalType":"address","name":"_WETH","type":"address"},{"internalType":"address payable","name":"_swapTarget","type":"address"},{"internalType":"uint256","name":"_dustThreshold","type":"uint256"}],"stateMutability":"nonpayable","type":"constructor"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"count","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"ethSpent","type":"uint256"},{"indexed":false,"internalType":"address","name":"to","type":"address"}],"name":"Buy","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"ethAmount","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"vTokenAmount","type":"uint256"},{"indexed":false,"internalType":"address","name":"to","type":"address"}],"name":"DustReturned","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"previousOwner","type":"address"},{"indexed":true,"internalType":"address","name":"newOwner","type":"address"}],"name":"OwnershipTransferred","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"count","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"ethReceived","type":"uint256"},{"indexed":false,"internalType":"address","name":"to","type":"address"}],"name":"Sell","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"count","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"ethSpent","type":"uint256"},{"indexed":false,"internalType":"address","name":"to","type":"address"}],"name":"Swap","type":"event"},{"inputs":[],"name":"WETH","outputs":[{"internalType":"contract IWETH","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"vaultId","type":"uint256"},{"internalType":"uint256","name":"amount","type":"uint256"},{"internalType":"uint256[]","name":"specificIds","type":"uint256[]"},{"internalType":"bytes","name":"swapCallData","type":"bytes"},{"internalType":"address payable","name":"to","type":"address"}],"name":"buyAndRedeem","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[{"internalType":"uint256","name":"vaultId","type":"uint256"},{"internalType":"uint256[]","name":"idsIn","type":"uint256[]"},{"internalType":"uint256[]","name":"amounts","type":"uint256[]"},{"internalType":"uint256[]","name":"specificIds","type":"uint256[]"},{"internalType":"bytes","name":"swapCallData","type":"bytes"},{"internalType":"address payable","name":"to","type":"address"}],"name":"buyAndSwap1155","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[{"internalType":"uint256","name":"vaultId","type":"uint256"},{"internalType":"uint256[]","name":"idsIn","type":"uint256[]"},{"internalType":"uint256[]","name":"specificIds","type":"uint256[]"},{"internalType":"bytes","name":"swapCallData","type":"bytes"},{"internalType":"address payable","name":"to","type":"address"}],"name":"buyAndSwap721","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[],"name":"dustThreshold","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"feeDistributor","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"vaultId","type":"uint256"},{"internalType":"uint256[]","name":"ids","type":"uint256[]"},{"internalType":"uint256[]","name":"amounts","type":"uint256[]"},{"internalType":"bytes","name":"swapCallData","type":"bytes"},{"internalType":"address payable","name":"to","type":"address"}],"name":"mintAndSell1155","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"vaultId","type":"uint256"},{"internalType":"uint256[]","name":"ids","type":"uint256[]"},{"internalType":"bytes","name":"swapCallData","type":"bytes"},{"internalType":"address payable","name":"to","type":"address"}],"name":"mintAndSell721","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"nftxFactory","outputs":[{"internalType":"contract INFTXVaultFactory","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"}],"name":"nftxVaultAddresses","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"},{"internalType":"address","name":"","type":"address"},{"internalType":"uint256[]","name":"","type":"uint256[]"},{"internalType":"uint256[]","name":"","type":"uint256[]"},{"internalType":"bytes","name":"","type":"bytes"}],"name":"onERC1155BatchReceived","outputs":[{"internalType":"bytes4","name":"","type":"bytes4"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"},{"internalType":"address","name":"","type":"address"},{"internalType":"uint256","name":"","type":"uint256"},{"internalType":"uint256","name":"","type":"uint256"},{"internalType":"bytes","name":"","type":"bytes"}],"name":"onERC1155Received","outputs":[{"internalType":"bytes4","name":"","type":"bytes4"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"},{"internalType":"address","name":"","type":"address"},{"internalType":"uint256","name":"","type":"uint256"},{"internalType":"bytes","name":"","type":"bytes"}],"name":"onERC721Received","outputs":[{"internalType":"bytes4","name":"","type":"bytes4"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"owner","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"bool","name":"_paused","type":"bool"}],"name":"pause","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"paused","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"renounceOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"token","type":"address"}],"name":"rescue","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_dustThreshold","type":"uint256"}],"name":"setDustThreshold","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"bytes4","name":"interfaceId","type":"bytes4"}],"name":"supportsInterface","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"newOwner","type":"address"}],"name":"transferOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"stateMutability":"payable","type":"receive"}]Contract Creation Code
6101006040526002805460ff191690553480156200001c57600080fd5b5060405162003ab838038062003ab88339810160408190526200003f9162000165565b600080546001600160a01b031916339081178255604051909182917f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e0908290a350600180556001600160601b0319606085811b821660c05284811b821660a05283901b1660805260408051630d43e8ad60e01b815290516001600160a01b03861691630d43e8ad916004808301926020929190829003018186803b158015620000e757600080fd5b505afa158015620000fc573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906200012291906200013f565b60601b6001600160601b03191660e05260035550620001d7915050565b60006020828403121562000151578081fd5b81516200015e81620001be565b9392505050565b600080600080608085870312156200017b578283fd5b84516200018881620001be565b60208601519094506200019b81620001be565b6040860151909350620001ae81620001be565b6060959095015193969295505050565b6001600160a01b0381168114620001d457600080fd5b50565b60805160601c60a05160601c60c05160601c60e05160601c61384062000278600039600081816101e30152611d380152600081816102ef01526117d60152600081816103960152818161072a015281816107b001528181610cbe01528181610f4a0152818161128f01528181611315015281816116080152818161168e015281816121c2015261227b0152600081816119be0152611a9e01526138406000f3fe60806040526004361061016e5760003560e01c80638be3a0fb116100cb578063bc197c811161007f578063f2fde38b11610059578063f2fde38b146104a6578063f733dd7e146104c6578063fa457360146104d957600080fd5b8063bc197c81146103f8578063e8462e8f1461043d578063f23a6e611461046157600080fd5b8063ad5c4648116100b0578063ad5c464814610384578063ad7e55ba146103b8578063bb23db72146103d857600080fd5b80638be3a0fb146103465780638da5cb5b1461036657600080fd5b80634d3f078a116101225780635ee50f88116101075780635ee50f88146102dd578063715018a614610311578063839006f21461032657600080fd5b80634d3f078a146102b05780635c975abb146102c357600080fd5b80630d43e8ad116101535780630d43e8ad146101d1578063150b7a021461021d57806336fa8b651461027a57600080fd5b806301ffc9a71461017a57806302329a29146101af57600080fd5b3661017557005b600080fd5b34801561018657600080fd5b5061019a61019536600461325a565b6104ec565b60405190151581526020015b60405180910390f35b3480156101bb57600080fd5b506101cf6101ca366004613222565b610555565b005b3480156101dd57600080fd5b506102057f000000000000000000000000000000000000000000000000000000000000000081565b6040516001600160a01b0390911681526020016101a6565b34801561022957600080fd5b506102616102383660046130be565b7f150b7a0200000000000000000000000000000000000000000000000000000000949350505050565b6040516001600160e01b031990911681526020016101a6565b34801561028657600080fd5b50610205610295366004613282565b6004602052600090815260409020546001600160a01b031681565b6101cf6102be3660046134c7565b6105c7565b3480156102cf57600080fd5b5060025461019a9060ff1681565b3480156102e957600080fd5b506102057f000000000000000000000000000000000000000000000000000000000000000081565b34801561031d57600080fd5b506101cf6108dc565b34801561033257600080fd5b506101cf610341366004612fdc565b61098d565b34801561035257600080fd5b506101cf61036136600461338b565b610b22565b34801561037257600080fd5b506000546001600160a01b0316610205565b34801561039057600080fd5b506102057f000000000000000000000000000000000000000000000000000000000000000081565b3480156103c457600080fd5b506101cf6103d3366004613282565b610d54565b3480156103e457600080fd5b506101cf6103f336600461343e565b610db3565b34801561040457600080fd5b50610261610413366004613014565b7fbc197c810000000000000000000000000000000000000000000000000000000095945050505050565b34801561044957600080fd5b5061045360035481565b6040519081526020016101a6565b34801561046d57600080fd5b5061026161047c366004613128565b7ff23a6e610000000000000000000000000000000000000000000000000000000095945050505050565b3480156104b257600080fd5b506101cf6104c1366004612fdc565b610fdd565b6101cf6104d43660046132b2565b61111b565b6101cf6104e736600461338b565b611468565b60006001600160e01b031982167f4e2312e000000000000000000000000000000000000000000000000000000000148061054f57507f01ffc9a7000000000000000000000000000000000000000000000000000000006001600160e01b03198316145b92915050565b6000546001600160a01b031633146105b45760405162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e657260448201526064015b60405180910390fd5b6002805460ff1916911515919091179055565b6002600154141561061a5760405162461bcd60e51b815260206004820152601f60248201527f5265656e7472616e637947756172643a207265656e7472616e742063616c6c0060448201526064016105ab565b600260018190555460ff16158061063b57506000546001600160a01b031633145b6106775760405162461bcd60e51b815260206004820152600d60248201526c16985c081a5cc81c185d5cd959609a1b60448201526064016105ab565b6001600160a01b0381161580159061069857506001600160a01b0381163014155b6106d85760405162461bcd60e51b8152602060048201526011602482015270125b9d985b1a59081c9958da5c1a595b9d607a1b60448201526064016105ab565b600086116107285760405162461bcd60e51b815260206004820152601060248201527f4d7573742073656e6420616d6f756e740000000000000000000000000000000060448201526064016105ab565b7f00000000000000000000000000000000000000000000000000000000000000006001600160a01b031663d0e30db0346040518263ffffffff1660e01b81526004016000604051808303818588803b15801561078357600080fd5b505af1158015610797573d6000803e3d6000fd5b505050505060006107a78861178b565b905060006107d77f0000000000000000000000000000000000000000000000000000000000000000838787611912565b90506107eb88670de0b6b3a7640000613755565b81101561083a5760405162461bcd60e51b815260206004820152601960248201527f496e73756666696369656e74207661756c7420746f6b656e730000000000000060448201526064016105ab565b61087a8989898980806020026020016040519081016040528093929190818152602001838360200280828437600092019190915250899250611bf7915050565b60408051898152602081018390526001600160a01b0385168183015290517ff7735c8cb2a65788ca663fc8415b7c6a66cd6847d58346d8334e8d52a599d3df9181900360600190a16108cd826001611c8c565b50506001805550505050505050565b6000546001600160a01b031633146109365760405162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e657260448201526064016105ab565b600080546040516001600160a01b03909116907f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e0908390a36000805473ffffffffffffffffffffffffffffffffffffffff19169055565b6000546001600160a01b031633146109e75760405162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e657260448201526064016105ab565b6001600160a01b038116610a9157604051600090339047908381818185875af1925050503d8060008114610a37576040519150601f19603f3d011682016040523d82523d6000602084013e610a3c565b606091505b5050905080610a8d5760405162461bcd60e51b815260206004820152601d60248201527f416464726573733a20756e61626c6520746f2073656e642076616c756500000060448201526064016105ab565b5050565b6040516370a0823160e01b8152306004820152610b1f9033906001600160a01b038416906370a082319060240160206040518083038186803b158015610ad657600080fd5b505afa158015610aea573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610b0e919061329a565b6001600160a01b0384169190611e2b565b50565b60026001541415610b755760405162461bcd60e51b815260206004820152601f60248201527f5265656e7472616e637947756172643a207265656e7472616e742063616c6c0060448201526064016105ab565b600260018190555460ff161580610b9657506000546001600160a01b031633145b610bd25760405162461bcd60e51b815260206004820152600d60248201526c16985c081a5cc81c185d5cd959609a1b60448201526064016105ab565b6001600160a01b03811615801590610bf357506001600160a01b0381163014155b610c335760405162461bcd60e51b8152602060048201526011602482015270125b9d985b1a59081c9958da5c1a595b9d607a1b60448201526064016105ab565b6000610c4188888888611e82565b9150506000610cb48a8a8a8080602002602001604051908101604052809392919081815260200183836020028082843760009201919091525050604080516020808e0282810182019093528d82529093508d92508c918291850190849080828437600092019190915250611fd392505050565b90506000610ce4827f00000000000000000000000000000000000000000000000000000000000000008888611912565b9050610cef846121aa565b5060408051848152602081018390526001600160a01b0386168183015290517f1cdb5ee3c47e1a706ac452b89698e5e3f2ff4f835ca72dde8936d0f4fcf37d819181900360600190a1610d43826000611c8c565b505060018055505050505050505050565b6000546001600160a01b03163314610dae5760405162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e657260448201526064016105ab565b600355565b60026001541415610e065760405162461bcd60e51b815260206004820152601f60248201527f5265656e7472616e637947756172643a207265656e7472616e742063616c6c0060448201526064016105ab565b600260018190555460ff161580610e2757506000546001600160a01b031633145b610e635760405162461bcd60e51b815260206004820152600d60248201526c16985c081a5cc81c185d5cd959609a1b60448201526064016105ab565b6001600160a01b03811615801590610e8457506001600160a01b0381163014155b610ec45760405162461bcd60e51b8152602060048201526011602482015270125b9d985b1a59081c9958da5c1a595b9d607a1b60448201526064016105ab565b83610f015760405162461bcd60e51b815260206004820152600d60248201526c4d7573742073656e642049447360981b60448201526064016105ab565b6000610f408787878080602002602001604051908101604052809392919081815260200183836020028082843760009201919091525061238992505050565b90506000610f70827f00000000000000000000000000000000000000000000000000000000000000008787611912565b9050610f7b836121aa565b5060408051878152602081018390526001600160a01b0385168183015290517f1cdb5ee3c47e1a706ac452b89698e5e3f2ff4f835ca72dde8936d0f4fcf37d819181900360600190a1610fcf826000611c8c565b505060018055505050505050565b6000546001600160a01b031633146110375760405162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e657260448201526064016105ab565b6001600160a01b0381166110b35760405162461bcd60e51b815260206004820152602660248201527f4f776e61626c653a206e6577206f776e657220697320746865207a65726f206160448201527f646472657373000000000000000000000000000000000000000000000000000060648201526084016105ab565b600080546040516001600160a01b03808516939216917f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e091a36000805473ffffffffffffffffffffffffffffffffffffffff19166001600160a01b0392909216919091179055565b6002600154141561116e5760405162461bcd60e51b815260206004820152601f60248201527f5265656e7472616e637947756172643a207265656e7472616e742063616c6c0060448201526064016105ab565b600260018190555460ff16158061118f57506000546001600160a01b031633145b6111cb5760405162461bcd60e51b815260206004820152600d60248201526c16985c081a5cc81c185d5cd959609a1b60448201526064016105ab565b6001600160a01b038116158015906111ec57506001600160a01b0381163014155b61122c5760405162461bcd60e51b8152602060048201526011602482015270125b9d985b1a59081c9958da5c1a595b9d607a1b60448201526064016105ab565b6000341161127c5760405162461bcd60e51b815260206004820152600e60248201527f496e76616c696420616d6f756e7400000000000000000000000000000000000060448201526064016105ab565b600061128a8a8a8a8a611e82565b9150507f00000000000000000000000000000000000000000000000000000000000000006001600160a01b031663d0e30db0346040518263ffffffff1660e01b81526004016000604051808303818588803b1580156112e857600080fd5b505af11580156112fc573d6000803e3d6000fd5b5050505050600061130c8c61178b565b9050600061133c7f0000000000000000000000000000000000000000000000000000000000000000838888611912565b90506114018d8d8d80806020026020016040519081016040528093929190818152602001838360200280828437600081840152601f19601f820116905080830192505050505050508c8c80806020026020016040519081016040528093929190818152602001838360200280828437600081840152601f19601f820116905080830192505050505050508b8b808060200260200160405190810160405280939291908181526020018383602002808284376000920191909152508b925061253b915050565b5060408051848152602081018390526001600160a01b0386168183015290517f7af2bc3f8ec800c569b6555feaf16589d96a9d04a49d1645fd456d75fa0b372b9181900360600190a1611455826001611c8c565b5050600180555050505050505050505050565b600260015414156114bb5760405162461bcd60e51b815260206004820152601f60248201527f5265656e7472616e637947756172643a207265656e7472616e742063616c6c0060448201526064016105ab565b600260018190555460ff1615806114dc57506000546001600160a01b031633145b6115185760405162461bcd60e51b815260206004820152600d60248201526c16985c081a5cc81c185d5cd959609a1b60448201526064016105ab565b6001600160a01b0381161580159061153957506001600160a01b0381163014155b6115795760405162461bcd60e51b8152602060048201526011602482015270125b9d985b1a59081c9958da5c1a595b9d607a1b60448201526064016105ab565b856115b65760405162461bcd60e51b815260206004820152600d60248201526c4d7573742073656e642049447360981b60448201526064016105ab565b600034116116065760405162461bcd60e51b815260206004820152600e60248201527f496e76616c696420616d6f756e7400000000000000000000000000000000000060448201526064016105ab565b7f00000000000000000000000000000000000000000000000000000000000000006001600160a01b031663d0e30db0346040518263ffffffff1660e01b81526004016000604051808303818588803b15801561166157600080fd5b505af1158015611675573d6000803e3d6000fd5b505050505060006116858961178b565b905060006116b57f0000000000000000000000000000000000000000000000000000000000000000838787611912565b90506117278a8a8a8080602002602001604051908101604052809392919081815260200183836020028082843760009201919091525050604080516020808e0282810182019093528d82529093508d92508c91829185019084908082843760009201919091525089925061271a915050565b5060408051898152602081018390526001600160a01b0385168183015290517f7af2bc3f8ec800c569b6555feaf16589d96a9d04a49d1645fd456d75fa0b372b9181900360600190a161177b826001611c8c565b5050600180555050505050505050565b6000818152600460205260408120546001600160a01b0316611892576040517f81a36fb6000000000000000000000000000000000000000000000000000000008152600481018390527f00000000000000000000000000000000000000000000000000000000000000006001600160a01b0316906381a36fb69060240160206040518083038186803b15801561182057600080fd5b505afa158015611834573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906118589190612ff8565b6000838152600460205260409020805473ffffffffffffffffffffffffffffffffffffffff19166001600160a01b03929092169190911790555b6000828152600460205260409020546001600160a01b03166118f65760405162461bcd60e51b815260206004820152601460248201527f5661756c7420646f6573206e6f7420657869737400000000000000000000000060448201526064016105ab565b506000908152600460205260409020546001600160a01b031690565b6040516370a0823160e01b815230600482015260009081906001600160a01b038616906370a082319060240160206040518083038186803b15801561195657600080fd5b505afa15801561196a573d6000803e3d6000fd5b505050506040513d601f19601f8201168201806040525081019061198e919061329a565b6040517f095ea7b30000000000000000000000000000000000000000000000000000000081526001600160a01b037f00000000000000000000000000000000000000000000000000000000000000008116600483015260001960248301529192509087169063095ea7b390604401602060405180830381600087803b158015611a1657600080fd5b505af1158015611a2a573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190611a4e919061323e565b611a9a5760405162461bcd60e51b815260206004820152601a60248201527f556e61626c6520746f20617070726f766520636f6e747261637400000000000060448201526064016105ab565b60007f00000000000000000000000000000000000000000000000000000000000000006001600160a01b03168585604051611ad6929190613595565b6000604051808303816000865af19150503d8060008114611b13576040519150601f19603f3d011682016040523d82523d6000602084013e611b18565b606091505b5050905080611b695760405162461bcd60e51b815260206004820152601060248201527f535741505f43414c4c5f4641494c45440000000000000000000000000000000060448201526064016105ab565b6040516370a0823160e01b815230600482015282906001600160a01b038816906370a082319060240160206040518083038186803b158015611baa57600080fd5b505afa158015611bbe573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190611be2919061329a565b611bec9190613774565b979650505050505050565b611c008461178b565b6001600160a01b0316639d54def68484846040518463ffffffff1660e01b8152600401611c2f939291906136ce565b600060405180830381600087803b158015611c4957600080fd5b505af1158015611c5d573d6000803e3d6000fd5b505050506040513d6000823e601f3d908101601f19168201604052611c85919081019061318f565b5050505050565b60008115611ca057611c9d336121aa565b90505b6040516370a0823160e01b81523060048201526000906001600160a01b038516906370a082319060240160206040518083038186803b158015611ce257600080fd5b505afa158015611cf6573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190611d1a919061329a565b905060008115611ddc57600354821115611d35575033611d58565b507f00000000000000000000000000000000000000000000000000000000000000005b60405163a9059cbb60e01b81526001600160a01b0382811660048301526024820184905286169063a9059cbb90604401602060405180830381600087803b158015611da257600080fd5b505af1158015611db6573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190611dda919061323e565b505b60408051848152602081018490526001600160a01b0383168183015290517fd9b066a634638844b3b4701f7bfb9d8791265979e46666f7d7bc04cf4c06c9709181900360600190a15050505050565b604080516001600160a01b038416602482015260448082018490528251808303909101815260649091019091526020810180516001600160e01b031663a9059cbb60e01b179052611e7d9084906128ba565b505050565b82600081611ec25760405162461bcd60e51b815260206004820152600d60248201526c4d7573742073656e642049447360981b60448201526064016105ab565b82821115611f125760405162461bcd60e51b815260206004820152601f60248201527f4d75737420646566696e6520616d6f756e747320616761696e7374204944730060448201526064016105ab565b60005b82811015611fc9576000858583818110611f3f57634e487b7160e01b600052603260045260246000fd5b9050602002013511611f935760405162461bcd60e51b815260206004820152601360248201527f496e76616c6964203131353520616d6f756e740000000000000000000000000060448201526064016105ab565b848482818110611fb357634e487b7160e01b600052603260045260246000fd5b9050602002013582019150806001019050611f15565b5094509492505050565b600080611fdf8561178b565b90506000816001600160a01b0316631ba46cfd6040518163ffffffff1660e01b815260040160206040518083038186803b15801561201c57600080fd5b505afa158015612030573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906120549190612ff8565b604051631759616b60e11b81529091506001600160a01b03821690632eb2c2d69061208990339030908a908a906004016135c1565b600060405180830381600087803b1580156120a357600080fd5b505af11580156120b7573d6000803e3d6000fd5b505060405163a22cb46560e01b81526001600160a01b038581166004830152600160248301528416925063a22cb4659150604401600060405180830381600087803b15801561210557600080fd5b505af1158015612119573d6000803e3d6000fd5b5050604051630f57464360e21b81526001600160a01b0385169250633d5d190c915061214b9088908890600401613619565b602060405180830381600087803b15801561216557600080fd5b505af1158015612179573d6000803e3d6000fd5b505050506040513d601f19601f8201168201806040525081019061219d919061329a565b50909150505b9392505050565b6040516370a0823160e01b81523060048201526000907f00000000000000000000000000000000000000000000000000000000000000006001600160a01b0316906370a082319060240160206040518083038186803b15801561220c57600080fd5b505afa158015612220573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190612244919061329a565b90508015612384576040517f2e1a7d4d000000000000000000000000000000000000000000000000000000008152600481018290527f00000000000000000000000000000000000000000000000000000000000000006001600160a01b031690632e1a7d4d90602401600060405180830381600087803b1580156122c757600080fd5b505af11580156122db573d6000803e3d6000fd5b505050506000826001600160a01b03168260405160006040518083038185875af1925050503d806000811461232c576040519150601f19603f3d011682016040523d82523d6000602084013e612331565b606091505b50509050806123825760405162461bcd60e51b815260206004820152601d60248201527f556e61626c6520746f2073656e6420756e77726170706564205745544800000060448201526064016105ab565b505b919050565b6000806123958461178b565b90506000816001600160a01b0316631ba46cfd6040518163ffffffff1660e01b815260040160206040518083038186803b1580156123d257600080fd5b505afa1580156123e6573d6000803e3d6000fd5b505050506040513d601f19601f8201168201806040525081019061240a9190612ff8565b845190915060005b818110156124ab5761244c8387838151811061243e57634e487b7160e01b600052603260045260246000fd5b60200260200101518661299f565b6001600160a01b03831673b47e3cd837ddf8e4c57f05d70ab865de6e193bbb14156124a3576124a38387838151811061249557634e487b7160e01b600052603260045260246000fd5b602002602001015186612c4a565b600101612412565b50604051630f57464360e21b81526060906001600160a01b03851690633d5d190c906124dd9089908590600401613619565b602060405180830381600087803b1580156124f757600080fd5b505af115801561250b573d6000803e3d6000fd5b505050506040513d601f19601f8201168201806040525081019061252f919061329a565b50929695505050505050565b6000806125478761178b565b90506000816001600160a01b0316631ba46cfd6040518163ffffffff1660e01b815260040160206040518083038186803b15801561258457600080fd5b505afa158015612598573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906125bc9190612ff8565b604051631759616b60e11b81529091506001600160a01b03821690632eb2c2d6906125f190339030908c908c906004016135c1565b600060405180830381600087803b15801561260b57600080fd5b505af115801561261f573d6000803e3d6000fd5b505060405163a22cb46560e01b81526001600160a01b038581166004830152600160248301528416925063a22cb4659150604401600060405180830381600087803b15801561266d57600080fd5b505af1158015612681573d6000803e3d6000fd5b5050604051632321420560e21b81526001600160a01b0385169250638c85081491506126b7908a908a908a908a90600401613647565b600060405180830381600087803b1580156126d157600080fd5b505af11580156126e5573d6000803e3d6000fd5b505050506040513d6000823e601f3d908101601f1916820160405261270d919081019061318f565b5090979650505050505050565b6000806127268661178b565b90506000816001600160a01b0316631ba46cfd6040518163ffffffff1660e01b815260040160206040518083038186803b15801561276357600080fd5b505afa158015612777573d6000803e3d6000fd5b505050506040513d601f19601f8201168201806040525081019061279b9190612ff8565b865190915060005b81811015612820576127cf8389838151811061243e57634e487b7160e01b600052603260045260246000fd5b6001600160a01b03831673b47e3cd837ddf8e4c57f05d70ab865de6e193bbb1415612818576128188389838151811061249557634e487b7160e01b600052603260045260246000fd5b6001016127a3565b50604051632321420560e21b81526060906001600160a01b03851690638c85081490612856908b9085908c908c90600401613647565b600060405180830381600087803b15801561287057600080fd5b505af1158015612884573d6000803e3d6000fd5b505050506040513d6000823e601f3d908101601f191682016040526128ac919081019061318f565b509298975050505050505050565b600061290f826040518060400160405280602081526020017f5361666545524332303a206c6f772d6c6576656c2063616c6c206661696c6564815250856001600160a01b0316612cfb9092919063ffffffff16565b805190915015611e7d578080602001905181019061292d919061323e565b611e7d5760405162461bcd60e51b815260206004820152602a60248201527f5361666545524332303a204552433230206f7065726174696f6e20646964206e60448201527f6f7420737563636565640000000000000000000000000000000000000000000060648201526084016105ab565b60606001600160a01b03841673b47e3cd837ddf8e4c57f05d70ab865de6e193bbb1415612b57576000836040516024016129db91815260200190565b60408051601f198184030181529181526020820180516001600160e01b03167f58178168000000000000000000000000000000000000000000000000000000001790525190915060009081906001600160a01b03881690612a3d9085906135a5565b600060405180830381855afa9150503d8060008114612a78576040519150601f19603f3d011682016040523d82523d6000602084013e612a7d565b606091505b5091509150600081806020019051810190612a989190612ff8565b9050828015612aaf57506001600160a01b03811633145b612afb5760405162461bcd60e51b815260206004820152601160248201527f4e6f7420746865204e4654206f776e657200000000000000000000000000000060448201526064016105ab565b6040516024810188905260440160408051601f198184030181529190526020810180516001600160e01b03167f8264fe98000000000000000000000000000000000000000000000000000000001790529450612bbf9350505050565b6040513360248201526001600160a01b03831660448201526064810184905260840160408051601f198184030181529190526020810180516001600160e01b03167f42842e0e0000000000000000000000000000000000000000000000000000000017905290505b600080856001600160a01b031683604051612bda91906135a5565b6000604051808303816000865af19150503d8060008114612c17576040519150601f19603f3d011682016040523d82523d6000602084013e612c1c565b606091505b5091509150818190612c415760405162461bcd60e51b81526004016105ab919061369b565b50505050505050565b6001600160a01b03831673b47e3cd837ddf8e4c57f05d70ab865de6e193bbb14612c7357505050565b604051602481018390526000604482018190526001600160a01b03831660648301529060840160408051601f198184030181529181526020820180516001600160e01b03167fbf31196f000000000000000000000000000000000000000000000000000000001790525190915060009081906001600160a01b03871690612bda9085906135a5565b6060612d0a8484600085612d12565b949350505050565b606082471015612d8a5760405162461bcd60e51b815260206004820152602660248201527f416464726573733a20696e73756666696369656e742062616c616e636520666f60448201527f722063616c6c000000000000000000000000000000000000000000000000000060648201526084016105ab565b843b612dd85760405162461bcd60e51b815260206004820152601d60248201527f416464726573733a2063616c6c20746f206e6f6e2d636f6e747261637400000060448201526064016105ab565b600080866001600160a01b03168587604051612df491906135a5565b60006040518083038185875af1925050503d8060008114612e31576040519150601f19603f3d011682016040523d82523d6000602084013e612e36565b606091505b5091509150611bec82828660608315612e505750816121a3565b825115612e605782518084602001fd5b8160405162461bcd60e51b81526004016105ab919061369b565b8035612384816137e7565b60008083601f840112612e96578182fd5b50813567ffffffffffffffff811115612ead578182fd5b6020830191508360208260051b8501011115612ec857600080fd5b9250929050565b600082601f830112612edf578081fd5b81356020612ef4612eef83613731565b613700565b80838252828201915082860187848660051b8901011115612f13578586fd5b855b8581101561270d57813584529284019290840190600101612f15565b60008083601f840112612f42578182fd5b50813567ffffffffffffffff811115612f59578182fd5b602083019150836020828501011115612ec857600080fd5b600082601f830112612f81578081fd5b813567ffffffffffffffff811115612f9b57612f9b6137d1565b612fae601f8201601f1916602001613700565b818152846020838601011115612fc2578283fd5b816020850160208301379081016020019190915292915050565b600060208284031215612fed578081fd5b81356121a3816137e7565b600060208284031215613009578081fd5b81516121a3816137e7565b600080600080600060a0868803121561302b578081fd5b8535613036816137e7565b94506020860135613046816137e7565b9350604086013567ffffffffffffffff80821115613062578283fd5b61306e89838a01612ecf565b94506060880135915080821115613083578283fd5b61308f89838a01612ecf565b935060808801359150808211156130a4578283fd5b506130b188828901612f71565b9150509295509295909350565b600080600080608085870312156130d3578384fd5b84356130de816137e7565b935060208501356130ee816137e7565b925060408501359150606085013567ffffffffffffffff811115613110578182fd5b61311c87828801612f71565b91505092959194509250565b600080600080600060a0868803121561313f578081fd5b853561314a816137e7565b9450602086013561315a816137e7565b93506040860135925060608601359150608086013567ffffffffffffffff811115613183578182fd5b6130b188828901612f71565b600060208083850312156131a1578182fd5b825167ffffffffffffffff8111156131b7578283fd5b8301601f810185136131c7578283fd5b80516131d5612eef82613731565b80828252848201915084840188868560051b87010111156131f4578687fd5b8694505b838510156132165780518352600194909401939185019185016131f8565b50979650505050505050565b600060208284031215613233578081fd5b81356121a3816137fc565b60006020828403121561324f578081fd5b81516121a3816137fc565b60006020828403121561326b578081fd5b81356001600160e01b0319811681146121a3578182fd5b600060208284031215613293578081fd5b5035919050565b6000602082840312156132ab578081fd5b5051919050565b60008060008060008060008060008060c08b8d0312156132d0578788fd5b8a35995060208b013567ffffffffffffffff808211156132ee57898afd5b6132fa8e838f01612e85565b909b50995060408d0135915080821115613312578687fd5b61331e8e838f01612e85565b909950975060608d0135915080821115613336578687fd5b6133428e838f01612e85565b909750955060808d013591508082111561335a578485fd5b506133678d828e01612f31565b909450925061337a905060a08c01612e7a565b90509295989b9194979a5092959850565b60008060008060008060008060a0898b0312156133a6578182fd5b88359750602089013567ffffffffffffffff808211156133c4578384fd5b6133d08c838d01612e85565b909950975060408b01359150808211156133e8578384fd5b6133f48c838d01612e85565b909750955060608b013591508082111561340c578384fd5b506134198b828c01612f31565b909450925050608089013561342d816137e7565b809150509295985092959890939650565b60008060008060008060808789031215613456578384fd5b86359550602087013567ffffffffffffffff80821115613474578586fd5b6134808a838b01612e85565b90975095506040890135915080821115613498578384fd5b506134a589828a01612f31565b90945092505060608701356134b9816137e7565b809150509295509295509295565b600080600080600080600060a0888a0312156134e1578081fd5b8735965060208801359550604088013567ffffffffffffffff80821115613506578283fd5b6135128b838c01612e85565b909750955060608a013591508082111561352a578283fd5b506135378a828b01612f31565b909450925050608088013561354b816137e7565b8091505092959891949750929550565b6000815180845260208085019450808401835b8381101561358a5781518752958201959082019060010161356e565b509495945050505050565b8183823760009101908152919050565b600082516135b781846020870161378b565b9190910192915050565b60006001600160a01b03808716835280861660208401525060a060408301526135ed60a083018561355b565b82810360608401526135ff818561355b565b838103608090940193909352508152602001949350505050565b60408152600061362c604083018561355b565b828103602084015261363e818561355b565b95945050505050565b60808152600061365a608083018761355b565b828103602084015261366c818761355b565b90508281036040840152613680818661355b565b9150506001600160a01b038316606083015295945050505050565b60208152600082518060208401526136ba81604085016020870161378b565b601f01601f19169190910160400192915050565b8381526060602082015260006136e7606083018561355b565b90506001600160a01b0383166040830152949350505050565b604051601f8201601f1916810167ffffffffffffffff81118282101715613729576137296137d1565b604052919050565b600067ffffffffffffffff82111561374b5761374b6137d1565b5060051b60200190565b600081600019048311821515161561376f5761376f6137bb565b500290565b600082821015613786576137866137bb565b500390565b60005b838110156137a657818101518382015260200161378e565b838111156137b5576000848401525b50505050565b634e487b7160e01b600052601160045260246000fd5b634e487b7160e01b600052604160045260246000fd5b6001600160a01b0381168114610b1f57600080fd5b8015158114610b1f57600080fdfea2646970667358221220415f3bb0c24a66045ab2d81332621cf4f3d246e19637344f30dba84c78ff7c4e64736f6c63430008040033000000000000000000000000be86f647b167567525ccaafcd6f881f1ee558216000000000000000000000000c02aaa39b223fe8d0a0e5c4f27ead9083c756cc2000000000000000000000000def1c0ded9bec7f1a1670819833240f027b25eff000000000000000000000000000000000000000000000000000000003b9aca00
Deployed Bytecode
0x60806040526004361061016e5760003560e01c80638be3a0fb116100cb578063bc197c811161007f578063f2fde38b11610059578063f2fde38b146104a6578063f733dd7e146104c6578063fa457360146104d957600080fd5b8063bc197c81146103f8578063e8462e8f1461043d578063f23a6e611461046157600080fd5b8063ad5c4648116100b0578063ad5c464814610384578063ad7e55ba146103b8578063bb23db72146103d857600080fd5b80638be3a0fb146103465780638da5cb5b1461036657600080fd5b80634d3f078a116101225780635ee50f88116101075780635ee50f88146102dd578063715018a614610311578063839006f21461032657600080fd5b80634d3f078a146102b05780635c975abb146102c357600080fd5b80630d43e8ad116101535780630d43e8ad146101d1578063150b7a021461021d57806336fa8b651461027a57600080fd5b806301ffc9a71461017a57806302329a29146101af57600080fd5b3661017557005b600080fd5b34801561018657600080fd5b5061019a61019536600461325a565b6104ec565b60405190151581526020015b60405180910390f35b3480156101bb57600080fd5b506101cf6101ca366004613222565b610555565b005b3480156101dd57600080fd5b506102057f000000000000000000000000fd8a76dc204e461db5da4f38687adc9cc5ae4a8681565b6040516001600160a01b0390911681526020016101a6565b34801561022957600080fd5b506102616102383660046130be565b7f150b7a0200000000000000000000000000000000000000000000000000000000949350505050565b6040516001600160e01b031990911681526020016101a6565b34801561028657600080fd5b50610205610295366004613282565b6004602052600090815260409020546001600160a01b031681565b6101cf6102be3660046134c7565b6105c7565b3480156102cf57600080fd5b5060025461019a9060ff1681565b3480156102e957600080fd5b506102057f000000000000000000000000be86f647b167567525ccaafcd6f881f1ee55821681565b34801561031d57600080fd5b506101cf6108dc565b34801561033257600080fd5b506101cf610341366004612fdc565b61098d565b34801561035257600080fd5b506101cf61036136600461338b565b610b22565b34801561037257600080fd5b506000546001600160a01b0316610205565b34801561039057600080fd5b506102057f000000000000000000000000c02aaa39b223fe8d0a0e5c4f27ead9083c756cc281565b3480156103c457600080fd5b506101cf6103d3366004613282565b610d54565b3480156103e457600080fd5b506101cf6103f336600461343e565b610db3565b34801561040457600080fd5b50610261610413366004613014565b7fbc197c810000000000000000000000000000000000000000000000000000000095945050505050565b34801561044957600080fd5b5061045360035481565b6040519081526020016101a6565b34801561046d57600080fd5b5061026161047c366004613128565b7ff23a6e610000000000000000000000000000000000000000000000000000000095945050505050565b3480156104b257600080fd5b506101cf6104c1366004612fdc565b610fdd565b6101cf6104d43660046132b2565b61111b565b6101cf6104e736600461338b565b611468565b60006001600160e01b031982167f4e2312e000000000000000000000000000000000000000000000000000000000148061054f57507f01ffc9a7000000000000000000000000000000000000000000000000000000006001600160e01b03198316145b92915050565b6000546001600160a01b031633146105b45760405162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e657260448201526064015b60405180910390fd5b6002805460ff1916911515919091179055565b6002600154141561061a5760405162461bcd60e51b815260206004820152601f60248201527f5265656e7472616e637947756172643a207265656e7472616e742063616c6c0060448201526064016105ab565b600260018190555460ff16158061063b57506000546001600160a01b031633145b6106775760405162461bcd60e51b815260206004820152600d60248201526c16985c081a5cc81c185d5cd959609a1b60448201526064016105ab565b6001600160a01b0381161580159061069857506001600160a01b0381163014155b6106d85760405162461bcd60e51b8152602060048201526011602482015270125b9d985b1a59081c9958da5c1a595b9d607a1b60448201526064016105ab565b600086116107285760405162461bcd60e51b815260206004820152601060248201527f4d7573742073656e6420616d6f756e740000000000000000000000000000000060448201526064016105ab565b7f000000000000000000000000c02aaa39b223fe8d0a0e5c4f27ead9083c756cc26001600160a01b031663d0e30db0346040518263ffffffff1660e01b81526004016000604051808303818588803b15801561078357600080fd5b505af1158015610797573d6000803e3d6000fd5b505050505060006107a78861178b565b905060006107d77f000000000000000000000000c02aaa39b223fe8d0a0e5c4f27ead9083c756cc2838787611912565b90506107eb88670de0b6b3a7640000613755565b81101561083a5760405162461bcd60e51b815260206004820152601960248201527f496e73756666696369656e74207661756c7420746f6b656e730000000000000060448201526064016105ab565b61087a8989898980806020026020016040519081016040528093929190818152602001838360200280828437600092019190915250899250611bf7915050565b60408051898152602081018390526001600160a01b0385168183015290517ff7735c8cb2a65788ca663fc8415b7c6a66cd6847d58346d8334e8d52a599d3df9181900360600190a16108cd826001611c8c565b50506001805550505050505050565b6000546001600160a01b031633146109365760405162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e657260448201526064016105ab565b600080546040516001600160a01b03909116907f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e0908390a36000805473ffffffffffffffffffffffffffffffffffffffff19169055565b6000546001600160a01b031633146109e75760405162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e657260448201526064016105ab565b6001600160a01b038116610a9157604051600090339047908381818185875af1925050503d8060008114610a37576040519150601f19603f3d011682016040523d82523d6000602084013e610a3c565b606091505b5050905080610a8d5760405162461bcd60e51b815260206004820152601d60248201527f416464726573733a20756e61626c6520746f2073656e642076616c756500000060448201526064016105ab565b5050565b6040516370a0823160e01b8152306004820152610b1f9033906001600160a01b038416906370a082319060240160206040518083038186803b158015610ad657600080fd5b505afa158015610aea573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610b0e919061329a565b6001600160a01b0384169190611e2b565b50565b60026001541415610b755760405162461bcd60e51b815260206004820152601f60248201527f5265656e7472616e637947756172643a207265656e7472616e742063616c6c0060448201526064016105ab565b600260018190555460ff161580610b9657506000546001600160a01b031633145b610bd25760405162461bcd60e51b815260206004820152600d60248201526c16985c081a5cc81c185d5cd959609a1b60448201526064016105ab565b6001600160a01b03811615801590610bf357506001600160a01b0381163014155b610c335760405162461bcd60e51b8152602060048201526011602482015270125b9d985b1a59081c9958da5c1a595b9d607a1b60448201526064016105ab565b6000610c4188888888611e82565b9150506000610cb48a8a8a8080602002602001604051908101604052809392919081815260200183836020028082843760009201919091525050604080516020808e0282810182019093528d82529093508d92508c918291850190849080828437600092019190915250611fd392505050565b90506000610ce4827f000000000000000000000000c02aaa39b223fe8d0a0e5c4f27ead9083c756cc28888611912565b9050610cef846121aa565b5060408051848152602081018390526001600160a01b0386168183015290517f1cdb5ee3c47e1a706ac452b89698e5e3f2ff4f835ca72dde8936d0f4fcf37d819181900360600190a1610d43826000611c8c565b505060018055505050505050505050565b6000546001600160a01b03163314610dae5760405162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e657260448201526064016105ab565b600355565b60026001541415610e065760405162461bcd60e51b815260206004820152601f60248201527f5265656e7472616e637947756172643a207265656e7472616e742063616c6c0060448201526064016105ab565b600260018190555460ff161580610e2757506000546001600160a01b031633145b610e635760405162461bcd60e51b815260206004820152600d60248201526c16985c081a5cc81c185d5cd959609a1b60448201526064016105ab565b6001600160a01b03811615801590610e8457506001600160a01b0381163014155b610ec45760405162461bcd60e51b8152602060048201526011602482015270125b9d985b1a59081c9958da5c1a595b9d607a1b60448201526064016105ab565b83610f015760405162461bcd60e51b815260206004820152600d60248201526c4d7573742073656e642049447360981b60448201526064016105ab565b6000610f408787878080602002602001604051908101604052809392919081815260200183836020028082843760009201919091525061238992505050565b90506000610f70827f000000000000000000000000c02aaa39b223fe8d0a0e5c4f27ead9083c756cc28787611912565b9050610f7b836121aa565b5060408051878152602081018390526001600160a01b0385168183015290517f1cdb5ee3c47e1a706ac452b89698e5e3f2ff4f835ca72dde8936d0f4fcf37d819181900360600190a1610fcf826000611c8c565b505060018055505050505050565b6000546001600160a01b031633146110375760405162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e657260448201526064016105ab565b6001600160a01b0381166110b35760405162461bcd60e51b815260206004820152602660248201527f4f776e61626c653a206e6577206f776e657220697320746865207a65726f206160448201527f646472657373000000000000000000000000000000000000000000000000000060648201526084016105ab565b600080546040516001600160a01b03808516939216917f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e091a36000805473ffffffffffffffffffffffffffffffffffffffff19166001600160a01b0392909216919091179055565b6002600154141561116e5760405162461bcd60e51b815260206004820152601f60248201527f5265656e7472616e637947756172643a207265656e7472616e742063616c6c0060448201526064016105ab565b600260018190555460ff16158061118f57506000546001600160a01b031633145b6111cb5760405162461bcd60e51b815260206004820152600d60248201526c16985c081a5cc81c185d5cd959609a1b60448201526064016105ab565b6001600160a01b038116158015906111ec57506001600160a01b0381163014155b61122c5760405162461bcd60e51b8152602060048201526011602482015270125b9d985b1a59081c9958da5c1a595b9d607a1b60448201526064016105ab565b6000341161127c5760405162461bcd60e51b815260206004820152600e60248201527f496e76616c696420616d6f756e7400000000000000000000000000000000000060448201526064016105ab565b600061128a8a8a8a8a611e82565b9150507f000000000000000000000000c02aaa39b223fe8d0a0e5c4f27ead9083c756cc26001600160a01b031663d0e30db0346040518263ffffffff1660e01b81526004016000604051808303818588803b1580156112e857600080fd5b505af11580156112fc573d6000803e3d6000fd5b5050505050600061130c8c61178b565b9050600061133c7f000000000000000000000000c02aaa39b223fe8d0a0e5c4f27ead9083c756cc2838888611912565b90506114018d8d8d80806020026020016040519081016040528093929190818152602001838360200280828437600081840152601f19601f820116905080830192505050505050508c8c80806020026020016040519081016040528093929190818152602001838360200280828437600081840152601f19601f820116905080830192505050505050508b8b808060200260200160405190810160405280939291908181526020018383602002808284376000920191909152508b925061253b915050565b5060408051848152602081018390526001600160a01b0386168183015290517f7af2bc3f8ec800c569b6555feaf16589d96a9d04a49d1645fd456d75fa0b372b9181900360600190a1611455826001611c8c565b5050600180555050505050505050505050565b600260015414156114bb5760405162461bcd60e51b815260206004820152601f60248201527f5265656e7472616e637947756172643a207265656e7472616e742063616c6c0060448201526064016105ab565b600260018190555460ff1615806114dc57506000546001600160a01b031633145b6115185760405162461bcd60e51b815260206004820152600d60248201526c16985c081a5cc81c185d5cd959609a1b60448201526064016105ab565b6001600160a01b0381161580159061153957506001600160a01b0381163014155b6115795760405162461bcd60e51b8152602060048201526011602482015270125b9d985b1a59081c9958da5c1a595b9d607a1b60448201526064016105ab565b856115b65760405162461bcd60e51b815260206004820152600d60248201526c4d7573742073656e642049447360981b60448201526064016105ab565b600034116116065760405162461bcd60e51b815260206004820152600e60248201527f496e76616c696420616d6f756e7400000000000000000000000000000000000060448201526064016105ab565b7f000000000000000000000000c02aaa39b223fe8d0a0e5c4f27ead9083c756cc26001600160a01b031663d0e30db0346040518263ffffffff1660e01b81526004016000604051808303818588803b15801561166157600080fd5b505af1158015611675573d6000803e3d6000fd5b505050505060006116858961178b565b905060006116b57f000000000000000000000000c02aaa39b223fe8d0a0e5c4f27ead9083c756cc2838787611912565b90506117278a8a8a8080602002602001604051908101604052809392919081815260200183836020028082843760009201919091525050604080516020808e0282810182019093528d82529093508d92508c91829185019084908082843760009201919091525089925061271a915050565b5060408051898152602081018390526001600160a01b0385168183015290517f7af2bc3f8ec800c569b6555feaf16589d96a9d04a49d1645fd456d75fa0b372b9181900360600190a161177b826001611c8c565b5050600180555050505050505050565b6000818152600460205260408120546001600160a01b0316611892576040517f81a36fb6000000000000000000000000000000000000000000000000000000008152600481018390527f000000000000000000000000be86f647b167567525ccaafcd6f881f1ee5582166001600160a01b0316906381a36fb69060240160206040518083038186803b15801561182057600080fd5b505afa158015611834573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906118589190612ff8565b6000838152600460205260409020805473ffffffffffffffffffffffffffffffffffffffff19166001600160a01b03929092169190911790555b6000828152600460205260409020546001600160a01b03166118f65760405162461bcd60e51b815260206004820152601460248201527f5661756c7420646f6573206e6f7420657869737400000000000000000000000060448201526064016105ab565b506000908152600460205260409020546001600160a01b031690565b6040516370a0823160e01b815230600482015260009081906001600160a01b038616906370a082319060240160206040518083038186803b15801561195657600080fd5b505afa15801561196a573d6000803e3d6000fd5b505050506040513d601f19601f8201168201806040525081019061198e919061329a565b6040517f095ea7b30000000000000000000000000000000000000000000000000000000081526001600160a01b037f000000000000000000000000def1c0ded9bec7f1a1670819833240f027b25eff8116600483015260001960248301529192509087169063095ea7b390604401602060405180830381600087803b158015611a1657600080fd5b505af1158015611a2a573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190611a4e919061323e565b611a9a5760405162461bcd60e51b815260206004820152601a60248201527f556e61626c6520746f20617070726f766520636f6e747261637400000000000060448201526064016105ab565b60007f000000000000000000000000def1c0ded9bec7f1a1670819833240f027b25eff6001600160a01b03168585604051611ad6929190613595565b6000604051808303816000865af19150503d8060008114611b13576040519150601f19603f3d011682016040523d82523d6000602084013e611b18565b606091505b5050905080611b695760405162461bcd60e51b815260206004820152601060248201527f535741505f43414c4c5f4641494c45440000000000000000000000000000000060448201526064016105ab565b6040516370a0823160e01b815230600482015282906001600160a01b038816906370a082319060240160206040518083038186803b158015611baa57600080fd5b505afa158015611bbe573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190611be2919061329a565b611bec9190613774565b979650505050505050565b611c008461178b565b6001600160a01b0316639d54def68484846040518463ffffffff1660e01b8152600401611c2f939291906136ce565b600060405180830381600087803b158015611c4957600080fd5b505af1158015611c5d573d6000803e3d6000fd5b505050506040513d6000823e601f3d908101601f19168201604052611c85919081019061318f565b5050505050565b60008115611ca057611c9d336121aa565b90505b6040516370a0823160e01b81523060048201526000906001600160a01b038516906370a082319060240160206040518083038186803b158015611ce257600080fd5b505afa158015611cf6573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190611d1a919061329a565b905060008115611ddc57600354821115611d35575033611d58565b507f000000000000000000000000fd8a76dc204e461db5da4f38687adc9cc5ae4a865b60405163a9059cbb60e01b81526001600160a01b0382811660048301526024820184905286169063a9059cbb90604401602060405180830381600087803b158015611da257600080fd5b505af1158015611db6573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190611dda919061323e565b505b60408051848152602081018490526001600160a01b0383168183015290517fd9b066a634638844b3b4701f7bfb9d8791265979e46666f7d7bc04cf4c06c9709181900360600190a15050505050565b604080516001600160a01b038416602482015260448082018490528251808303909101815260649091019091526020810180516001600160e01b031663a9059cbb60e01b179052611e7d9084906128ba565b505050565b82600081611ec25760405162461bcd60e51b815260206004820152600d60248201526c4d7573742073656e642049447360981b60448201526064016105ab565b82821115611f125760405162461bcd60e51b815260206004820152601f60248201527f4d75737420646566696e6520616d6f756e747320616761696e7374204944730060448201526064016105ab565b60005b82811015611fc9576000858583818110611f3f57634e487b7160e01b600052603260045260246000fd5b9050602002013511611f935760405162461bcd60e51b815260206004820152601360248201527f496e76616c6964203131353520616d6f756e740000000000000000000000000060448201526064016105ab565b848482818110611fb357634e487b7160e01b600052603260045260246000fd5b9050602002013582019150806001019050611f15565b5094509492505050565b600080611fdf8561178b565b90506000816001600160a01b0316631ba46cfd6040518163ffffffff1660e01b815260040160206040518083038186803b15801561201c57600080fd5b505afa158015612030573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906120549190612ff8565b604051631759616b60e11b81529091506001600160a01b03821690632eb2c2d69061208990339030908a908a906004016135c1565b600060405180830381600087803b1580156120a357600080fd5b505af11580156120b7573d6000803e3d6000fd5b505060405163a22cb46560e01b81526001600160a01b038581166004830152600160248301528416925063a22cb4659150604401600060405180830381600087803b15801561210557600080fd5b505af1158015612119573d6000803e3d6000fd5b5050604051630f57464360e21b81526001600160a01b0385169250633d5d190c915061214b9088908890600401613619565b602060405180830381600087803b15801561216557600080fd5b505af1158015612179573d6000803e3d6000fd5b505050506040513d601f19601f8201168201806040525081019061219d919061329a565b50909150505b9392505050565b6040516370a0823160e01b81523060048201526000907f000000000000000000000000c02aaa39b223fe8d0a0e5c4f27ead9083c756cc26001600160a01b0316906370a082319060240160206040518083038186803b15801561220c57600080fd5b505afa158015612220573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190612244919061329a565b90508015612384576040517f2e1a7d4d000000000000000000000000000000000000000000000000000000008152600481018290527f000000000000000000000000c02aaa39b223fe8d0a0e5c4f27ead9083c756cc26001600160a01b031690632e1a7d4d90602401600060405180830381600087803b1580156122c757600080fd5b505af11580156122db573d6000803e3d6000fd5b505050506000826001600160a01b03168260405160006040518083038185875af1925050503d806000811461232c576040519150601f19603f3d011682016040523d82523d6000602084013e612331565b606091505b50509050806123825760405162461bcd60e51b815260206004820152601d60248201527f556e61626c6520746f2073656e6420756e77726170706564205745544800000060448201526064016105ab565b505b919050565b6000806123958461178b565b90506000816001600160a01b0316631ba46cfd6040518163ffffffff1660e01b815260040160206040518083038186803b1580156123d257600080fd5b505afa1580156123e6573d6000803e3d6000fd5b505050506040513d601f19601f8201168201806040525081019061240a9190612ff8565b845190915060005b818110156124ab5761244c8387838151811061243e57634e487b7160e01b600052603260045260246000fd5b60200260200101518661299f565b6001600160a01b03831673b47e3cd837ddf8e4c57f05d70ab865de6e193bbb14156124a3576124a38387838151811061249557634e487b7160e01b600052603260045260246000fd5b602002602001015186612c4a565b600101612412565b50604051630f57464360e21b81526060906001600160a01b03851690633d5d190c906124dd9089908590600401613619565b602060405180830381600087803b1580156124f757600080fd5b505af115801561250b573d6000803e3d6000fd5b505050506040513d601f19601f8201168201806040525081019061252f919061329a565b50929695505050505050565b6000806125478761178b565b90506000816001600160a01b0316631ba46cfd6040518163ffffffff1660e01b815260040160206040518083038186803b15801561258457600080fd5b505afa158015612598573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906125bc9190612ff8565b604051631759616b60e11b81529091506001600160a01b03821690632eb2c2d6906125f190339030908c908c906004016135c1565b600060405180830381600087803b15801561260b57600080fd5b505af115801561261f573d6000803e3d6000fd5b505060405163a22cb46560e01b81526001600160a01b038581166004830152600160248301528416925063a22cb4659150604401600060405180830381600087803b15801561266d57600080fd5b505af1158015612681573d6000803e3d6000fd5b5050604051632321420560e21b81526001600160a01b0385169250638c85081491506126b7908a908a908a908a90600401613647565b600060405180830381600087803b1580156126d157600080fd5b505af11580156126e5573d6000803e3d6000fd5b505050506040513d6000823e601f3d908101601f1916820160405261270d919081019061318f565b5090979650505050505050565b6000806127268661178b565b90506000816001600160a01b0316631ba46cfd6040518163ffffffff1660e01b815260040160206040518083038186803b15801561276357600080fd5b505afa158015612777573d6000803e3d6000fd5b505050506040513d601f19601f8201168201806040525081019061279b9190612ff8565b865190915060005b81811015612820576127cf8389838151811061243e57634e487b7160e01b600052603260045260246000fd5b6001600160a01b03831673b47e3cd837ddf8e4c57f05d70ab865de6e193bbb1415612818576128188389838151811061249557634e487b7160e01b600052603260045260246000fd5b6001016127a3565b50604051632321420560e21b81526060906001600160a01b03851690638c85081490612856908b9085908c908c90600401613647565b600060405180830381600087803b15801561287057600080fd5b505af1158015612884573d6000803e3d6000fd5b505050506040513d6000823e601f3d908101601f191682016040526128ac919081019061318f565b509298975050505050505050565b600061290f826040518060400160405280602081526020017f5361666545524332303a206c6f772d6c6576656c2063616c6c206661696c6564815250856001600160a01b0316612cfb9092919063ffffffff16565b805190915015611e7d578080602001905181019061292d919061323e565b611e7d5760405162461bcd60e51b815260206004820152602a60248201527f5361666545524332303a204552433230206f7065726174696f6e20646964206e60448201527f6f7420737563636565640000000000000000000000000000000000000000000060648201526084016105ab565b60606001600160a01b03841673b47e3cd837ddf8e4c57f05d70ab865de6e193bbb1415612b57576000836040516024016129db91815260200190565b60408051601f198184030181529181526020820180516001600160e01b03167f58178168000000000000000000000000000000000000000000000000000000001790525190915060009081906001600160a01b03881690612a3d9085906135a5565b600060405180830381855afa9150503d8060008114612a78576040519150601f19603f3d011682016040523d82523d6000602084013e612a7d565b606091505b5091509150600081806020019051810190612a989190612ff8565b9050828015612aaf57506001600160a01b03811633145b612afb5760405162461bcd60e51b815260206004820152601160248201527f4e6f7420746865204e4654206f776e657200000000000000000000000000000060448201526064016105ab565b6040516024810188905260440160408051601f198184030181529190526020810180516001600160e01b03167f8264fe98000000000000000000000000000000000000000000000000000000001790529450612bbf9350505050565b6040513360248201526001600160a01b03831660448201526064810184905260840160408051601f198184030181529190526020810180516001600160e01b03167f42842e0e0000000000000000000000000000000000000000000000000000000017905290505b600080856001600160a01b031683604051612bda91906135a5565b6000604051808303816000865af19150503d8060008114612c17576040519150601f19603f3d011682016040523d82523d6000602084013e612c1c565b606091505b5091509150818190612c415760405162461bcd60e51b81526004016105ab919061369b565b50505050505050565b6001600160a01b03831673b47e3cd837ddf8e4c57f05d70ab865de6e193bbb14612c7357505050565b604051602481018390526000604482018190526001600160a01b03831660648301529060840160408051601f198184030181529181526020820180516001600160e01b03167fbf31196f000000000000000000000000000000000000000000000000000000001790525190915060009081906001600160a01b03871690612bda9085906135a5565b6060612d0a8484600085612d12565b949350505050565b606082471015612d8a5760405162461bcd60e51b815260206004820152602660248201527f416464726573733a20696e73756666696369656e742062616c616e636520666f60448201527f722063616c6c000000000000000000000000000000000000000000000000000060648201526084016105ab565b843b612dd85760405162461bcd60e51b815260206004820152601d60248201527f416464726573733a2063616c6c20746f206e6f6e2d636f6e747261637400000060448201526064016105ab565b600080866001600160a01b03168587604051612df491906135a5565b60006040518083038185875af1925050503d8060008114612e31576040519150601f19603f3d011682016040523d82523d6000602084013e612e36565b606091505b5091509150611bec82828660608315612e505750816121a3565b825115612e605782518084602001fd5b8160405162461bcd60e51b81526004016105ab919061369b565b8035612384816137e7565b60008083601f840112612e96578182fd5b50813567ffffffffffffffff811115612ead578182fd5b6020830191508360208260051b8501011115612ec857600080fd5b9250929050565b600082601f830112612edf578081fd5b81356020612ef4612eef83613731565b613700565b80838252828201915082860187848660051b8901011115612f13578586fd5b855b8581101561270d57813584529284019290840190600101612f15565b60008083601f840112612f42578182fd5b50813567ffffffffffffffff811115612f59578182fd5b602083019150836020828501011115612ec857600080fd5b600082601f830112612f81578081fd5b813567ffffffffffffffff811115612f9b57612f9b6137d1565b612fae601f8201601f1916602001613700565b818152846020838601011115612fc2578283fd5b816020850160208301379081016020019190915292915050565b600060208284031215612fed578081fd5b81356121a3816137e7565b600060208284031215613009578081fd5b81516121a3816137e7565b600080600080600060a0868803121561302b578081fd5b8535613036816137e7565b94506020860135613046816137e7565b9350604086013567ffffffffffffffff80821115613062578283fd5b61306e89838a01612ecf565b94506060880135915080821115613083578283fd5b61308f89838a01612ecf565b935060808801359150808211156130a4578283fd5b506130b188828901612f71565b9150509295509295909350565b600080600080608085870312156130d3578384fd5b84356130de816137e7565b935060208501356130ee816137e7565b925060408501359150606085013567ffffffffffffffff811115613110578182fd5b61311c87828801612f71565b91505092959194509250565b600080600080600060a0868803121561313f578081fd5b853561314a816137e7565b9450602086013561315a816137e7565b93506040860135925060608601359150608086013567ffffffffffffffff811115613183578182fd5b6130b188828901612f71565b600060208083850312156131a1578182fd5b825167ffffffffffffffff8111156131b7578283fd5b8301601f810185136131c7578283fd5b80516131d5612eef82613731565b80828252848201915084840188868560051b87010111156131f4578687fd5b8694505b838510156132165780518352600194909401939185019185016131f8565b50979650505050505050565b600060208284031215613233578081fd5b81356121a3816137fc565b60006020828403121561324f578081fd5b81516121a3816137fc565b60006020828403121561326b578081fd5b81356001600160e01b0319811681146121a3578182fd5b600060208284031215613293578081fd5b5035919050565b6000602082840312156132ab578081fd5b5051919050565b60008060008060008060008060008060c08b8d0312156132d0578788fd5b8a35995060208b013567ffffffffffffffff808211156132ee57898afd5b6132fa8e838f01612e85565b909b50995060408d0135915080821115613312578687fd5b61331e8e838f01612e85565b909950975060608d0135915080821115613336578687fd5b6133428e838f01612e85565b909750955060808d013591508082111561335a578485fd5b506133678d828e01612f31565b909450925061337a905060a08c01612e7a565b90509295989b9194979a5092959850565b60008060008060008060008060a0898b0312156133a6578182fd5b88359750602089013567ffffffffffffffff808211156133c4578384fd5b6133d08c838d01612e85565b909950975060408b01359150808211156133e8578384fd5b6133f48c838d01612e85565b909750955060608b013591508082111561340c578384fd5b506134198b828c01612f31565b909450925050608089013561342d816137e7565b809150509295985092959890939650565b60008060008060008060808789031215613456578384fd5b86359550602087013567ffffffffffffffff80821115613474578586fd5b6134808a838b01612e85565b90975095506040890135915080821115613498578384fd5b506134a589828a01612f31565b90945092505060608701356134b9816137e7565b809150509295509295509295565b600080600080600080600060a0888a0312156134e1578081fd5b8735965060208801359550604088013567ffffffffffffffff80821115613506578283fd5b6135128b838c01612e85565b909750955060608a013591508082111561352a578283fd5b506135378a828b01612f31565b909450925050608088013561354b816137e7565b8091505092959891949750929550565b6000815180845260208085019450808401835b8381101561358a5781518752958201959082019060010161356e565b509495945050505050565b8183823760009101908152919050565b600082516135b781846020870161378b565b9190910192915050565b60006001600160a01b03808716835280861660208401525060a060408301526135ed60a083018561355b565b82810360608401526135ff818561355b565b838103608090940193909352508152602001949350505050565b60408152600061362c604083018561355b565b828103602084015261363e818561355b565b95945050505050565b60808152600061365a608083018761355b565b828103602084015261366c818761355b565b90508281036040840152613680818661355b565b9150506001600160a01b038316606083015295945050505050565b60208152600082518060208401526136ba81604085016020870161378b565b601f01601f19169190910160400192915050565b8381526060602082015260006136e7606083018561355b565b90506001600160a01b0383166040830152949350505050565b604051601f8201601f1916810167ffffffffffffffff81118282101715613729576137296137d1565b604052919050565b600067ffffffffffffffff82111561374b5761374b6137d1565b5060051b60200190565b600081600019048311821515161561376f5761376f6137bb565b500290565b600082821015613786576137866137bb565b500390565b60005b838110156137a657818101518382015260200161378e565b838111156137b5576000848401525b50505050565b634e487b7160e01b600052601160045260246000fd5b634e487b7160e01b600052604160045260246000fd5b6001600160a01b0381168114610b1f57600080fd5b8015158114610b1f57600080fdfea2646970667358221220415f3bb0c24a66045ab2d81332621cf4f3d246e19637344f30dba84c78ff7c4e64736f6c63430008040033
Constructor Arguments (ABI-Encoded and is the last bytes of the Contract Creation Code above)
000000000000000000000000be86f647b167567525ccaafcd6f881f1ee558216000000000000000000000000c02aaa39b223fe8d0a0e5c4f27ead9083c756cc2000000000000000000000000def1c0ded9bec7f1a1670819833240f027b25eff000000000000000000000000000000000000000000000000000000003b9aca00
-----Decoded View---------------
Arg [0] : _nftxFactory (address): 0xBE86f647b167567525cCAAfcd6f881F1Ee558216
Arg [1] : _WETH (address): 0xC02aaA39b223FE8D0A0e5C4F27eAD9083C756Cc2
Arg [2] : _swapTarget (address): 0xDef1C0ded9bec7F1a1670819833240f027b25EfF
Arg [3] : _dustThreshold (uint256): 1000000000
-----Encoded View---------------
4 Constructor Arguments found :
Arg [0] : 000000000000000000000000be86f647b167567525ccaafcd6f881f1ee558216
Arg [1] : 000000000000000000000000c02aaa39b223fe8d0a0e5c4f27ead9083c756cc2
Arg [2] : 000000000000000000000000def1c0ded9bec7f1a1670819833240f027b25eff
Arg [3] : 000000000000000000000000000000000000000000000000000000003b9aca00
Loading...
Loading
Loading...
Loading
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.