Feature Tip: Add private address tag to any address under My Name Tag !
Source Code
Latest 25 from a total of 636 transactions
| Transaction Hash |
Method
|
Block
|
From
|
|
To
|
||||
|---|---|---|---|---|---|---|---|---|---|
| Buy And Redeem | 24146931 | 17 days ago | IN | 0 ETH | 0.00000718 | ||||
| Buy And Redeem | 24143421 | 18 days ago | IN | 0 ETH | 0.00001111 | ||||
| Buy And Redeem | 24143395 | 18 days ago | IN | 0 ETH | 0.0000147 | ||||
| Mint And Sell721 | 24143333 | 18 days ago | IN | 0 ETH | 0.0000021 | ||||
| Mint And Sell721 | 24143297 | 18 days ago | IN | 0 ETH | 0.00000154 | ||||
| Mint And Sell721 | 24143296 | 18 days ago | IN | 0 ETH | 0.00000147 | ||||
| Mint And Sell721 | 16550849 | 1081 days ago | IN | 0 ETH | 0.00466002 | ||||
| Mint And Sell721 | 15528541 | 1224 days ago | IN | 0 ETH | 0.0055579 | ||||
| Mint And Sell721 | 15528520 | 1224 days ago | IN | 0 ETH | 0.00744527 | ||||
| Mint And Sell721 | 15528488 | 1224 days ago | IN | 0 ETH | 0.0067864 | ||||
| Mint And Sell721 | 15528446 | 1224 days ago | IN | 0 ETH | 0.01136715 | ||||
| Buy And Redeem | 15528230 | 1224 days ago | IN | 0.02631491 ETH | 0.00656488 | ||||
| Buy And Swap721 | 15528224 | 1224 days ago | IN | 0.0009791 ETH | 0.00590949 | ||||
| Buy And Redeem | 15528215 | 1224 days ago | IN | 0.13385413 ETH | 0.00649803 | ||||
| Mint And Sell721 | 15528211 | 1224 days ago | IN | 0 ETH | 0.00412174 | ||||
| Buy And Swap721 | 15528200 | 1224 days ago | IN | 0.02041185 ETH | 0.00704543 | ||||
| Buy And Swap721 | 15528105 | 1224 days ago | IN | 0.00316867 ETH | 0.00809697 | ||||
| Buy And Swap721 | 15528096 | 1224 days ago | IN | 0.00084557 ETH | 0.00690698 | ||||
| Buy And Redeem | 15528056 | 1224 days ago | IN | 0.06184664 ETH | 0.00822124 | ||||
| Mint And Sell721 | 15528012 | 1224 days ago | IN | 0 ETH | 0.01005985 | ||||
| Buy And Swap721 | 15527763 | 1224 days ago | IN | 0.00700654 ETH | 0.00750007 | ||||
| Mint And Sell721 | 15527624 | 1224 days ago | IN | 0 ETH | 0.01553444 | ||||
| Mint And Sell721 | 15527602 | 1224 days ago | IN | 0 ETH | 0.02781289 | ||||
| Mint And Sell721 | 15527196 | 1224 days ago | IN | 0 ETH | 0.01468692 | ||||
| Buy And Redeem | 15527148 | 1224 days ago | IN | 0.0924889 ETH | 0.02162883 |
Latest 25 internal transactions (View All)
Advanced mode:
| Parent Transaction Hash | Method | Block |
From
|
|
To
|
||
|---|---|---|---|---|---|---|---|
| Transfer | 15528541 | 1224 days ago | 1.44868738 ETH | ||||
| Transfer | 15528541 | 1224 days ago | 1.44868738 ETH | ||||
| Transfer | 15528520 | 1224 days ago | 0.02407981 ETH | ||||
| Transfer | 15528520 | 1224 days ago | 0.02407981 ETH | ||||
| Transfer | 15528488 | 1224 days ago | 0.01487294 ETH | ||||
| Transfer | 15528488 | 1224 days ago | 0.01487294 ETH | ||||
| Transfer | 15528446 | 1224 days ago | 0.08615096 ETH | ||||
| Transfer | 15528446 | 1224 days ago | 0.08615096 ETH | ||||
| Deposit | 15528230 | 1224 days ago | 0.02631491 ETH | ||||
| Deposit | 15528224 | 1224 days ago | 0.0009791 ETH | ||||
| Deposit | 15528215 | 1224 days ago | 0.13385413 ETH | ||||
| Transfer | 15528211 | 1224 days ago | 0.04374139 ETH | ||||
| Transfer | 15528211 | 1224 days ago | 0.04374139 ETH | ||||
| Deposit | 15528200 | 1224 days ago | 0.02041185 ETH | ||||
| Deposit | 15528105 | 1224 days ago | 0.00316867 ETH | ||||
| Deposit | 15528096 | 1224 days ago | 0.00084557 ETH | ||||
| Deposit | 15528056 | 1224 days ago | 0.06184664 ETH | ||||
| Transfer | 15528012 | 1224 days ago | 0.03716055 ETH | ||||
| Transfer | 15528012 | 1224 days ago | 0.03716055 ETH | ||||
| Deposit | 15527763 | 1224 days ago | 0.00700654 ETH | ||||
| Transfer | 15527624 | 1224 days ago | 0.08880498 ETH | ||||
| Transfer | 15527624 | 1224 days ago | 0.08880498 ETH | ||||
| Transfer | 15527602 | 1224 days ago | 0.93325034 ETH | ||||
| Transfer | 15527602 | 1224 days ago | 0.93325034 ETH | ||||
| Transfer | 15527196 | 1224 days ago | 0.75799449 ETH |
Loading...
Loading
Cross-Chain Transactions
Loading...
Loading
Contract Name:
NFTXMarketplace0xZap
Compiler Version
v0.8.4+commit.c7e474f2
Optimization Enabled:
Yes with 1000 runs
Other Settings:
default evmVersion
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 "./interface/INFTXFeeDistributor.sol";
import "./token/IERC1155Upgradeable.sol";
import "./token/IERC20Upgradeable.sol";
import "./token/ERC721HolderUpgradeable.sol";
import "./token/ERC1155HolderUpgradeable.sol";
import "./util/OwnableUpgradeable.sol";
import "./util/ReentrancyGuardUpgradeable.sol";
import "./util/SafeERC20Upgradeable.sol";
/**
* @notice A partial ERC20 interface.
*/
interface IERC20 {
function balanceOf(address owner) external view returns (uint256);
function approve(address spender, uint256 amount) external returns (bool);
function transfer(address to, uint256 amount) external returns (bool);
}
/**
* @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 OwnableUpgradeable, ReentrancyGuardUpgradeable, ERC721HolderUpgradeable, ERC1155HolderUpgradeable {
using SafeERC20Upgradeable for IERC20Upgradeable;
/// @notice An interface for the WETH contract
IWETH public immutable WETH;
/// @notice An interface for the NFTX Vault Factory contract
INFTXVaultFactory public immutable nftxFactory;
/// @notice A mapping of NFTX Vault IDs to their address corresponding vault contract address
mapping(uint256 => address) public nftxVaultAddresses;
/// @notice The decimal accuracy
uint256 constant BASE = 1e18;
// Set a constant address for specific contracts that need special logic
address constant CRYPTO_PUNKS = 0xb47e3cd837dDF8e4c57F05d70Ab865de6e193BBB;
/// @notice Emitted when ..
/// @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 when ..
/// @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 when ..
/// @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 Initialises our zap by setting contract addresses onto their
* respective interfaces.
*
* @param _nftxFactory NFTX Vault Factory contract address
* @param _WETH WETH contract address
*/
constructor(address _nftxFactory, address _WETH) {
nftxFactory = INFTXVaultFactory(_nftxFactory);
WETH = IWETH(_WETH);
}
/**
* @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 spender The `allowanceTarget` field from the API response
* @param swapTarget The `to` field from the API response
* @param swapCallData The `data` field from the API response
* @param to The recipient of the WETH from the tx
*/
function mintAndSell721(
uint256 vaultId,
uint256[] calldata ids,
address spender,
address payable swapTarget,
bytes calldata swapCallData,
address payable to
) external nonReentrant {
// 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), spender, swapTarget, swapCallData);
// Emit our sale event
emit Sell(ids.length, amount, to);
// Transfer the filled ETH to recipient
_transferEthOrWeth(to, amount, 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 spender The `allowanceTarget` field from the API response
* @param swapTarget The `to` field from the API response
* @param swapCallData The `data` field from the API response
* @param to The recipient of the WETH from the tx
*/
function buyAndSwap721(
uint256 vaultId,
uint256[] calldata idsIn,
uint256[] calldata specificIds,
address spender,
address payable swapTarget,
bytes calldata swapCallData,
address payable to
) external payable nonReentrant {
// 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');
// Wrap ETH into WETH for our contract from the sender
if (msg.value > 0) {
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, spender, swapTarget, swapCallData);
// Swap our tokens for the IDs requested
_swap721(vaultId, idsIn, specificIds, to);
emit Swap(idsIn.length, amount, to);
// Return any remaining WETH from the transaction
uint256 remaining = WETH.balanceOf(address(this));
if (remaining > 0) {
_transferEthOrWeth(spender, remaining, bool(msg.value > 0));
}
}
/**
* @notice TODO
*
* @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 spender The `allowanceTarget` field from the API response
* @param swapTarget The `to` field from the API response
* @param swapCallData The `data` field from the API response
* @param to The recipient of the WETH from the tx
*/
function buyAndRedeem(
uint256 vaultId,
uint256 amount,
uint256[] calldata specificIds,
address spender,
address payable swapTarget,
bytes calldata swapCallData,
address payable to
) external payable nonReentrant {
// 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
if (msg.value > 0) {
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, spender, swapTarget, swapCallData);
_redeem(vaultId, amount, specificIds, to);
emit Buy(amount, quoteAmount, to);
// Refund any remaining WETH
uint256 remaining = WETH.balanceOf(address(this));
if (remaining > 0) {
_transferEthOrWeth(spender, remaining, bool(msg.value > 0));
}
}
/**
* @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 spender The `allowanceTarget` field from the API response
* @param swapTarget The `to` field from the API response
* @param swapCallData The `data` field from the API response
* @param to The recipient of the WETH from the tx
*/
function mintAndSell1155(
uint256 vaultId,
uint256[] calldata ids,
uint256[] calldata amounts,
address spender,
address payable swapTarget,
bytes calldata swapCallData,
address payable to
) external nonReentrant {
// 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), spender, swapTarget, swapCallData);
// Emit our sale event
emit Sell(totalAmount, amount, to);
// Transfer the filled ETH to recipient
_transferEthOrWeth(to, amount, 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 spender The `allowanceTarget` field from the API response
* @param swapTarget The `to` field from the API response
* @param swapCallData The `data` field from the API response
* @param to The recipient of the WETH from the tx
*/
function buyAndSwap1155(
uint256 vaultId,
uint256[] calldata idsIn,
uint256[] calldata amounts,
uint256[] calldata specificIds,
address spender,
address payable swapTarget,
bytes calldata swapCallData,
address payable to
) external payable nonReentrant {
// 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(idsIn, amounts);
// Wrap ETH into WETH for our contract from the sender
if (msg.value > 0) {
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, spender, swapTarget, swapCallData);
// Swap our tokens for the IDs requested
_swap1155(vaultId, idsIn, amounts, specificIds, to);
emit Swap(totalAmount, amount, to);
// Return any remaining WETH from the transaction
uint256 remaining = WETH.balanceOf(address(this));
if (remaining > 0) {
_transferEthOrWeth(spender, remaining, bool(msg.value > 0));
}
}
/**
* @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();
IERC1155Upgradeable(assetAddress).safeBatchTransferFrom(msg.sender, address(this), ids, amounts, "");
IERC1155Upgradeable(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();
IERC1155Upgradeable(assetAddress).safeBatchTransferFrom(msg.sender, address(this), idsIn, amounts, "");
IERC1155Upgradeable(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 spender The `allowanceTarget` field from the API response
* @param swapTarget The `to` field from the API response
* @param swapCallData The `data` field from the API response
*/
function _fillQuote(
address sellToken,
address buyToken,
address spender,
address payable swapTarget,
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 ETH or WETH to a recipient, based on preference.
*
* @param to Recipient of the transfer
* @param amount Amount to be transferred
* @param isWeth If user prefers to receive WETH rather than ETH
*/
function _transferEthOrWeth(address to, uint amount, bool isWeth) internal {
if (isWeth) {
// Transfer the WETH to the recipient
WETH.transfer(to, amount);
} else {
// Unwrap our WETH into ETH and transfer it to the recipient
WETH.withdraw(amount);
(bool success, ) = payable(to).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 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 {
IERC20Upgradeable(token).safeTransfer(msg.sender, IERC20Upgradeable(token).balanceOf(address(this)));
}
}
/**
* @notice Allows our contract to receive any assets.
*/
receive() external payable {
//
}
}// SPDX-License-Identifier: MIT
pragma solidity ^0.8.0;
import "../interface/INFTXEligibility.sol";
import "../token/IERC20Upgradeable.sol";
import "../interface/INFTXVaultFactory.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 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 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) 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;
interface INFTXFeeDistributor {
struct FeeReceiver {
uint256 allocPoint;
address receiver;
bool isContract;
}
function nftxVaultFactory() external returns (address);
function lpStaking() external returns (address);
function treasury() external returns (address);
function defaultTreasuryAlloc() external returns (uint256);
function defaultLPAlloc() external returns (uint256);
function allocTotal(uint256 vaultId) external returns (uint256);
function specificTreasuryAlloc(uint256 vaultId) external returns (uint256);
// Write functions.
function __FeeDistributor__init__(address _lpStaking, address _treasury) external;
function rescueTokens(address token) external;
function distribute(uint256 vaultId) external;
function addReceiver(uint256 _vaultId, uint256 _allocPoint, address _receiver, bool _isContract) external;
function initializeVaultReceivers(uint256 _vaultId) external;
function changeMultipleReceiverAlloc(
uint256[] memory _vaultIds,
uint256[] memory _receiverIdxs,
uint256[] memory allocPoints
) external;
function changeMultipleReceiverAddress(
uint256[] memory _vaultIds,
uint256[] memory _receiverIdxs,
address[] memory addresses,
bool[] memory isContracts
) external;
function changeReceiverAlloc(uint256 _vaultId, uint256 _idx, uint256 _allocPoint) external;
function changeReceiverAddress(uint256 _vaultId, uint256 _idx, address _address, bool _isContract) external;
function removeReceiver(uint256 _vaultId, uint256 _receiverIdx) external;
// Configuration functions.
function setTreasuryAddress(address _treasury) external;
function setDefaultTreasuryAlloc(uint256 _allocPoint) external;
function setSpecificTreasuryAlloc(uint256 _vaultId, uint256 _allocPoint) external;
function setLPStakingAddress(address _lpStaking) external;
function setNFTXVaultFactory(address _factory) external;
function setDefaultLPAlloc(uint256 _allocPoint) external;
}// SPDX-License-Identifier: MIT
pragma solidity ^0.8.0;
import "../interface/IERC165Upgradeable.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 IERC1155Upgradeable is IERC165Upgradeable {
/**
* @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;
/**
* @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;
import "./IERC721ReceiverUpgradeable.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 ERC721HolderUpgradeable is IERC721ReceiverUpgradeable {
/**
* @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 "./ERC1155ReceiverUpgradeable.sol";
/**
* @dev _Available since v3.1._
*/
abstract contract ERC1155HolderUpgradeable is ERC1155ReceiverUpgradeable {
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 "./ContextUpgradeable.sol";
import "../proxy/Initializable.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 OwnableUpgradeable is Initializable, ContextUpgradeable {
address private _owner;
event OwnershipTransferred(address indexed previousOwner, address indexed newOwner);
/**
* @dev Initializes the contract setting the deployer as the initial owner.
*/
function __Ownable_init() internal initializer {
__Context_init_unchained();
__Ownable_init_unchained();
}
function __Ownable_init_unchained() internal initializer {
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;
}
uint256[49] private __gap;
}// SPDX-License-Identifier: MIT
pragma solidity ^0.8.0;
import "../proxy/Initializable.sol";
/**
* @dev Contract module that helps prevent reentrant calls to a function.
*
* Inheriting from `ReentrancyGuard` will make the {nonReentrant} modifier
* available, which can be applied to functions to make sure there are no nested
* (reentrant) calls to them.
*
* Note that because there is a single `nonReentrant` guard, functions marked as
* `nonReentrant` may not call one another. This can be worked around by making
* those functions `private`, and then adding `external` `nonReentrant` entry
* points to them.
*
* TIP: If you would like to learn more about reentrancy and alternative ways
* to protect against it, check out our blog post
* https://blog.openzeppelin.com/reentrancy-after-istanbul/[Reentrancy After Istanbul].
*/
abstract contract ReentrancyGuardUpgradeable is Initializable {
// 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;
function __ReentrancyGuard_init() internal initializer {
__ReentrancyGuard_init_unchained();
}
function __ReentrancyGuard_init_unchained() internal initializer {
_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;
}
uint256[49] private __gap;
}// SPDX-License-Identifier: MIT
pragma solidity ^0.8.0;
import "../token/IERC20Upgradeable.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 SafeERC20Upgradeable {
using Address for address;
function safeTransfer(IERC20Upgradeable token, address to, uint256 value) internal {
_callOptionalReturn(token, abi.encodeWithSelector(token.transfer.selector, to, value));
}
function safeTransferFrom(IERC20Upgradeable 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(IERC20Upgradeable token, address spender, uint256 value) internal {
// safeApprove should only be called when setting an initial allowance,
// or when resetting it to zero. To increase and decrease it, use
// 'safeIncreaseAllowance' and 'safeDecreaseAllowance'
// solhint-disable-next-line max-line-length
require((value == 0) || (token.allowance(address(this), spender) == 0),
"SafeERC20: approve from non-zero to non-zero allowance"
);
_callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, value));
}
function safeIncreaseAllowance(IERC20Upgradeable 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(IERC20Upgradeable 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));
}
}
/**
* @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(IERC20Upgradeable token, bytes memory data) private {
// We need to perform a low level call here, to bypass Solidity's return data size checking mechanism, since
// we're implementing it ourselves. We use {Address.functionCall} to perform this call, which verifies that
// the target address contains contract code and also asserts for success in the low-level call.
bytes memory returndata = address(token).functionCall(data, "SafeERC20: low-level call failed");
if (returndata.length > 0) { // Return data is optional
// solhint-disable-next-line max-line-length
require(abi.decode(returndata, (bool)), "SafeERC20: ERC20 operation did not succeed");
}
}
}// 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;
/**
* @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 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 IERC165Upgradeable {
/**
* @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
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 IERC721ReceiverUpgradeable {
/**
* @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;
import "./IERC1155ReceiverUpgradeable.sol";
import "../util/ERC165Upgradeable.sol";
/**
* @dev _Available since v3.1._
*/
abstract contract ERC1155ReceiverUpgradeable is ERC165Upgradeable, IERC1155ReceiverUpgradeable {
/**
* @dev See {IERC165-supportsInterface}.
*/
function supportsInterface(bytes4 interfaceId) public view virtual override(ERC165Upgradeable, IERC165Upgradeable) returns (bool) {
return interfaceId == type(IERC1155ReceiverUpgradeable).interfaceId
|| super.supportsInterface(interfaceId);
}
}// SPDX-License-Identifier: MIT
pragma solidity ^0.8.0;
import "../interface/IERC165Upgradeable.sol";
/**
* @dev _Available since v3.1._
*/
interface IERC1155ReceiverUpgradeable is IERC165Upgradeable {
/**
@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;
import "../interface/IERC165Upgradeable.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 ERC165Upgradeable is IERC165Upgradeable {
/**
* @dev See {IERC165-supportsInterface}.
*/
function supportsInterface(bytes4 interfaceId) public view virtual override returns (bool) {
return interfaceId == type(IERC165Upgradeable).interfaceId;
}
}// SPDX-License-Identifier: MIT
pragma solidity ^0.8.0;
import "../proxy/Initializable.sol";
/*
* @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 ContextUpgradeable is Initializable {
function __Context_init() internal initializer {
__Context_init_unchained();
}
function __Context_init_unchained() internal initializer {
}
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;
}
uint256[50] private __gap;
}// SPDX-License-Identifier: MIT
// solhint-disable-next-line compiler-version
pragma solidity ^0.8.0;
/**
* @dev This is a base contract to aid in writing upgradeable contracts, or any kind of contract that will be deployed
* behind a proxy. Since a proxied contract can't have a constructor, it's common to move constructor logic to an
* external initializer function, usually called `initialize`. It then becomes necessary to protect this initializer
* function so it can only be called once. The {initializer} modifier provided by this contract will have this effect.
*
* TIP: To avoid leaving the proxy in an uninitialized state, the initializer function should be called as early as
* possible by providing the encoded function call as the `_data` argument to {ERC1967Proxy-constructor}.
*
* CAUTION: When used with inheritance, manual care must be taken to not invoke a parent initializer twice, or to ensure
* that all initializers are idempotent. This is not verified automatically as constructors are by Solidity.
*/
abstract contract Initializable {
/**
* @dev Indicates that the contract has been initialized.
*/
bool private _initialized;
/**
* @dev Indicates that the contract is in the process of being initialized.
*/
bool private _initializing;
/**
* @dev Modifier to protect an initializer function from being invoked twice.
*/
modifier initializer() {
require(_initializing || !_initialized, "Initializable: contract is already initialized");
bool isTopLevelCall = !_initializing;
if (isTopLevelCall) {
_initializing = true;
_initialized = true;
}
_;
if (isTopLevelCall) {
_initializing = false;
}
}
}// 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);
}
}
}
}{
"optimizer": {
"enabled": true,
"runs": 1000
},
"outputSelection": {
"*": {
"*": [
"evm.bytecode",
"evm.deployedBytecode",
"devdoc",
"userdoc",
"metadata",
"abi"
]
}
},
"libraries": {}
}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"}],"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":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":"address","name":"spender","type":"address"},{"internalType":"address payable","name":"swapTarget","type":"address"},{"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":"address","name":"spender","type":"address"},{"internalType":"address payable","name":"swapTarget","type":"address"},{"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":"address","name":"spender","type":"address"},{"internalType":"address payable","name":"swapTarget","type":"address"},{"internalType":"bytes","name":"swapCallData","type":"bytes"},{"internalType":"address payable","name":"to","type":"address"}],"name":"buyAndSwap721","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[{"internalType":"uint256","name":"vaultId","type":"uint256"},{"internalType":"uint256[]","name":"ids","type":"uint256[]"},{"internalType":"uint256[]","name":"amounts","type":"uint256[]"},{"internalType":"address","name":"spender","type":"address"},{"internalType":"address payable","name":"swapTarget","type":"address"},{"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":"address","name":"spender","type":"address"},{"internalType":"address payable","name":"swapTarget","type":"address"},{"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":[],"name":"renounceOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"token","type":"address"}],"name":"rescue","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
60c06040523480156200001157600080fd5b506040516200358538038062003585833981016040819052620000349162000070565b6001600160601b0319606092831b811660a052911b16608052620000a7565b80516001600160a01b03811681146200006b57600080fd5b919050565b6000806040838503121562000083578182fd5b6200008e8362000053565b91506200009e6020840162000053565b90509250929050565b60805160601c60a05160601c61345062000135600039600081816102170152611d530152600081816102c40152818161058601528181610932015281816109b901528181610b0801528181610cc801528181610d4f01528181610e1a01528181611004015281816111930152818161121a0152818161132701528181611af90152611baa01526134506000f3fe6080604052600436106100f75760003560e01c80639f820d851161008a578063bca8b28211610059578063bca8b2821461032b578063e88898db1461034b578063f23a6e611461035e578063f2fde38b146103a357600080fd5b80639f820d851461028c578063aa0bee4c1461029f578063ad5c4648146102b2578063bc197c81146102e657600080fd5b80635ee50f88116100c65780635ee50f8814610205578063715018a614610239578063839006f21461024e5780638da5cb5b1461026e57600080fd5b806301ffc9a714610103578063150b7a02146101385780631e85677c1461019557806336fa8b65146101b757600080fd5b366100fe57005b600080fd5b34801561010f57600080fd5b5061012361011e366004612df2565b6103c3565b60405190151581526020015b60405180910390f35b34801561014457600080fd5b5061017c610153366004612c6e565b7f150b7a0200000000000000000000000000000000000000000000000000000000949350505050565b6040516001600160e01b0319909116815260200161012f565b3480156101a157600080fd5b506101b56101b0366004612efd565b61042c565b005b3480156101c357600080fd5b506101ed6101d2366004612e1a565b6097602052600090815260409020546001600160a01b031681565b6040516001600160a01b03909116815260200161012f565b34801561021157600080fd5b506101ed7f000000000000000000000000000000000000000000000000000000000000000081565b34801561024557600080fd5b506101b561061a565b34801561025a57600080fd5b506101b5610269366004612b8c565b6106cb565b34801561027a57600080fd5b506033546001600160a01b03166101ed565b6101b561029a366004612fd6565b610860565b6101b56102ad3660046130ea565b610bb7565b3480156102be57600080fd5b506101ed7f000000000000000000000000000000000000000000000000000000000000000081565b3480156102f257600080fd5b5061017c610301366004612bc4565b7fbc197c810000000000000000000000000000000000000000000000000000000095945050505050565b34801561033757600080fd5b506101b5610346366004612e4a565b610ec5565b6101b5610359366004612efd565b611095565b34801561036a57600080fd5b5061017c610379366004612cd8565b7ff23a6e610000000000000000000000000000000000000000000000000000000095945050505050565b3480156103af57600080fd5b506101b56103be366004612b8c565b6113bf565b60006001600160e01b031982167f4e2312e000000000000000000000000000000000000000000000000000000000148061042657507f01ffc9a7000000000000000000000000000000000000000000000000000000006001600160e01b03198316145b92915050565b600260655414156104845760405162461bcd60e51b815260206004820152601f60248201527f5265656e7472616e637947756172643a207265656e7472616e742063616c6c0060448201526064015b60405180910390fd5b60026065556001600160a01b038116158015906104aa57506001600160a01b0381163014155b6104ea5760405162461bcd60e51b8152602060048201526011602482015270125b9d985b1a59081c9958da5c1a595b9d607a1b604482015260640161047b565b60006104f88a8a8a8a6114fe565b915050600061057c8c8c8c80806020026020016040519081016040528093929190818152602001838360200280828437600081840152601f19601f820116905080830192505050505050508b8b8080602002602001604051908101604052809392919081815260200183836020028082843760009201919091525061164f92505050565b905060006105ae827f00000000000000000000000000000000000000000000000000000000000000008a8a8a8a611826565b60408051858152602081018390526001600160a01b0387168183015290519192507f1cdb5ee3c47e1a706ac452b89698e5e3f2ff4f835ca72dde8936d0f4fcf37d81919081900360600190a161060684826000611acd565b505060016065555050505050505050505050565b6033546001600160a01b031633146106745760405162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572604482015260640161047b565b6033546040516000916001600160a01b0316907f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e0908390a36033805473ffffffffffffffffffffffffffffffffffffffff19169055565b6033546001600160a01b031633146107255760405162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572604482015260640161047b565b6001600160a01b0381166107cf57604051600090339047908381818185875af1925050503d8060008114610775576040519150601f19603f3d011682016040523d82523d6000602084013e61077a565b606091505b50509050806107cb5760405162461bcd60e51b815260206004820152601d60248201527f416464726573733a20756e61626c6520746f2073656e642076616c7565000000604482015260640161047b565b5050565b6040516370a0823160e01b815230600482015261085d9033906001600160a01b038416906370a082319060240160206040518083038186803b15801561081457600080fd5b505afa158015610828573d6000803e3d6000fd5b505050506040513d601f19601f8201168201806040525081019061084c9190612e32565b6001600160a01b0384169190611cb6565b50565b600260655414156108b35760405162461bcd60e51b815260206004820152601f60248201527f5265656e7472616e637947756172643a207265656e7472616e742063616c6c00604482015260640161047b565b60026065556001600160a01b038116158015906108d957506001600160a01b0381163014155b6109195760405162461bcd60e51b8152602060048201526011602482015270125b9d985b1a59081c9958da5c1a595b9d607a1b604482015260640161047b565b60006109278c8c8c8c6114fe565b91505034156109a5577f00000000000000000000000000000000000000000000000000000000000000006001600160a01b031663d0e30db0346040518263ffffffff1660e01b81526004016000604051808303818588803b15801561098b57600080fd5b505af115801561099f573d6000803e3d6000fd5b50505050505b60006109b08e611d08565b905060006109e27f0000000000000000000000000000000000000000000000000000000000000000838a8a8a8a611826565b9050610aa78f8f8f80806020026020016040519081016040528093929190818152602001838360200280828437600081840152601f19601f820116905080830192505050505050508e8e80806020026020016040519081016040528093929190818152602001838360200280828437600081840152601f19601f820116905080830192505050505050508d8d808060200260200160405190810160405280939291908181526020018383602002808284376000920191909152508b9250611e8f915050565b5060408051848152602081018390526001600160a01b0386168183015290517f7af2bc3f8ec800c569b6555feaf16589d96a9d04a49d1645fd456d75fa0b372b9181900360600190a16040516370a0823160e01b81523060048201526000907f00000000000000000000000000000000000000000000000000000000000000006001600160a01b0316906370a082319060240160206040518083038186803b158015610b5257600080fd5b505afa158015610b66573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610b8a9190612e32565b90508015610ba057610ba0898260003411611acd565b505060016065555050505050505050505050505050565b60026065541415610c0a5760405162461bcd60e51b815260206004820152601f60248201527f5265656e7472616e637947756172643a207265656e7472616e742063616c6c00604482015260640161047b565b60026065556001600160a01b03811615801590610c3057506001600160a01b0381163014155b610c705760405162461bcd60e51b8152602060048201526011602482015270125b9d985b1a59081c9958da5c1a595b9d607a1b604482015260640161047b565b60008811610cc05760405162461bcd60e51b815260206004820152601060248201527f4d7573742073656e6420616d6f756e7400000000000000000000000000000000604482015260640161047b565b3415610d3b577f00000000000000000000000000000000000000000000000000000000000000006001600160a01b031663d0e30db0346040518263ffffffff1660e01b81526004016000604051808303818588803b158015610d2157600080fd5b505af1158015610d35573d6000803e3d6000fd5b50505050505b6000610d468a611d08565b90506000610d787f00000000000000000000000000000000000000000000000000000000000000008389898989611826565b9050610dba8b8b8b8b8080602002602001604051908101604052809392919081815260200183836020028082843760009201919091525089925061206e915050565b604080518b8152602081018390526001600160a01b0385168183015290517ff7735c8cb2a65788ca663fc8415b7c6a66cd6847d58346d8334e8d52a599d3df9181900360600190a16040516370a0823160e01b81523060048201526000907f00000000000000000000000000000000000000000000000000000000000000006001600160a01b0316906370a082319060240160206040518083038186803b158015610e6457600080fd5b505afa158015610e78573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610e9c9190612e32565b90508015610eb257610eb2888260003411611acd565b5050600160655550505050505050505050565b60026065541415610f185760405162461bcd60e51b815260206004820152601f60248201527f5265656e7472616e637947756172643a207265656e7472616e742063616c6c00604482015260640161047b565b60026065556001600160a01b03811615801590610f3e57506001600160a01b0381163014155b610f7e5760405162461bcd60e51b8152602060048201526011602482015270125b9d985b1a59081c9958da5c1a595b9d607a1b604482015260640161047b565b85610fbb5760405162461bcd60e51b815260206004820152600d60248201526c4d7573742073656e642049447360981b604482015260640161047b565b6000610ffa8989898080602002602001604051908101604052809392919081815260200183836020028082843760009201919091525061210392505050565b9050600061102c827f000000000000000000000000000000000000000000000000000000000000000089898989611826565b604080518a8152602081018390526001600160a01b0386168183015290519192507f1cdb5ee3c47e1a706ac452b89698e5e3f2ff4f835ca72dde8936d0f4fcf37d81919081900360600190a161108483826000611acd565b505060016065555050505050505050565b600260655414156110e85760405162461bcd60e51b815260206004820152601f60248201527f5265656e7472616e637947756172643a207265656e7472616e742063616c6c00604482015260640161047b565b60026065556001600160a01b0381161580159061110e57506001600160a01b0381163014155b61114e5760405162461bcd60e51b8152602060048201526011602482015270125b9d985b1a59081c9958da5c1a595b9d607a1b604482015260640161047b565b8761118b5760405162461bcd60e51b815260206004820152600d60248201526c4d7573742073656e642049447360981b604482015260640161047b565b3415611206577f00000000000000000000000000000000000000000000000000000000000000006001600160a01b031663d0e30db0346040518263ffffffff1660e01b81526004016000604051808303818588803b1580156111ec57600080fd5b505af1158015611200573d6000803e3d6000fd5b50505050505b60006112118b611d08565b905060006112437f00000000000000000000000000000000000000000000000000000000000000008389898989611826565b90506112c68c8c8c80806020026020016040519081016040528093929190818152602001838360200280828437600081840152601f19601f820116905080830192505050505050508b8b808060200260200160405190810160405280939291908181526020018383602002808284376000920191909152508992506122b5915050565b50604080518b8152602081018390526001600160a01b0385168183015290517f7af2bc3f8ec800c569b6555feaf16589d96a9d04a49d1645fd456d75fa0b372b9181900360600190a16040516370a0823160e01b81523060048201526000907f00000000000000000000000000000000000000000000000000000000000000006001600160a01b0316906370a082319060240160206040518083038186803b15801561137157600080fd5b505afa158015611385573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906113a99190612e32565b9050801561060657610606888260003411611acd565b6033546001600160a01b031633146114195760405162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572604482015260640161047b565b6001600160a01b0381166114955760405162461bcd60e51b815260206004820152602660248201527f4f776e61626c653a206e6577206f776e657220697320746865207a65726f206160448201527f6464726573730000000000000000000000000000000000000000000000000000606482015260840161047b565b6033546040516001600160a01b038084169216907f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e090600090a36033805473ffffffffffffffffffffffffffffffffffffffff19166001600160a01b0392909216919091179055565b8260008161153e5760405162461bcd60e51b815260206004820152600d60248201526c4d7573742073656e642049447360981b604482015260640161047b565b8282111561158e5760405162461bcd60e51b815260206004820152601f60248201527f4d75737420646566696e6520616d6f756e747320616761696e73742049447300604482015260640161047b565b60005b828110156116455760008585838181106115bb57634e487b7160e01b600052603260045260246000fd5b905060200201351161160f5760405162461bcd60e51b815260206004820152601360248201527f496e76616c6964203131353520616d6f756e7400000000000000000000000000604482015260640161047b565b84848281811061162f57634e487b7160e01b600052603260045260246000fd5b9050602002013582019150806001019050611591565b5094509492505050565b60008061165b85611d08565b90506000816001600160a01b0316631ba46cfd6040518163ffffffff1660e01b815260040160206040518083038186803b15801561169857600080fd5b505afa1580156116ac573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906116d09190612ba8565b604051631759616b60e11b81529091506001600160a01b03821690632eb2c2d69061170590339030908a908a9060040161320c565b600060405180830381600087803b15801561171f57600080fd5b505af1158015611733573d6000803e3d6000fd5b505060405163a22cb46560e01b81526001600160a01b038581166004830152600160248301528416925063a22cb4659150604401600060405180830381600087803b15801561178157600080fd5b505af1158015611795573d6000803e3d6000fd5b5050604051630f57464360e21b81526001600160a01b0385169250633d5d190c91506117c79088908890600401613264565b602060405180830381600087803b1580156117e157600080fd5b505af11580156117f5573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906118199190612e32565b50909150505b9392505050565b6040516370a0823160e01b815230600482015260009081906001600160a01b038816906370a082319060240160206040518083038186803b15801561186a57600080fd5b505afa15801561187e573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906118a29190612e32565b6040517f095ea7b30000000000000000000000000000000000000000000000000000000081526001600160a01b03878116600483015260001960248301529192509089169063095ea7b390604401602060405180830381600087803b15801561190a57600080fd5b505af115801561191e573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906119429190612dd2565b61198e5760405162461bcd60e51b815260206004820152601a60248201527f556e61626c6520746f20617070726f766520636f6e7472616374000000000000604482015260640161047b565b6000856001600160a01b031685856040516119aa9291906131e0565b6000604051808303816000865af19150503d80600081146119e7576040519150601f19603f3d011682016040523d82523d6000602084013e6119ec565b606091505b5050905080611a3d5760405162461bcd60e51b815260206004820152601060248201527f535741505f43414c4c5f4641494c454400000000000000000000000000000000604482015260640161047b565b6040516370a0823160e01b815230600482015282906001600160a01b038a16906370a082319060240160206040518083038186803b158015611a7e57600080fd5b505afa158015611a92573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190611ab69190612e32565b611ac091906133a0565b9998505050505050505050565b8015611b7b5760405163a9059cbb60e01b81526001600160a01b038481166004830152602482018490527f0000000000000000000000000000000000000000000000000000000000000000169063a9059cbb90604401602060405180830381600087803b158015611b3d57600080fd5b505af1158015611b51573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190611b759190612dd2565b50505050565b6040517f2e1a7d4d000000000000000000000000000000000000000000000000000000008152600481018390527f00000000000000000000000000000000000000000000000000000000000000006001600160a01b031690632e1a7d4d90602401600060405180830381600087803b158015611bf657600080fd5b505af1158015611c0a573d6000803e3d6000fd5b505050506000836001600160a01b03168360405160006040518083038185875af1925050503d8060008114611c5b576040519150601f19603f3d011682016040523d82523d6000602084013e611c60565b606091505b5050905080611b755760405162461bcd60e51b815260206004820152601d60248201527f556e61626c6520746f2073656e6420756e777261707065642057455448000000604482015260640161047b565b505050565b604080516001600160a01b038416602482015260448082018490528251808303909101815260649091019091526020810180516001600160e01b031663a9059cbb60e01b179052611cb1908490612455565b6000818152609760205260408120546001600160a01b0316611e0f576040517f81a36fb6000000000000000000000000000000000000000000000000000000008152600481018390527f00000000000000000000000000000000000000000000000000000000000000006001600160a01b0316906381a36fb69060240160206040518083038186803b158015611d9d57600080fd5b505afa158015611db1573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190611dd59190612ba8565b6000838152609760205260409020805473ffffffffffffffffffffffffffffffffffffffff19166001600160a01b03929092169190911790555b6000828152609760205260409020546001600160a01b0316611e735760405162461bcd60e51b815260206004820152601460248201527f5661756c7420646f6573206e6f74206578697374000000000000000000000000604482015260640161047b565b506000908152609760205260409020546001600160a01b031690565b600080611e9b87611d08565b90506000816001600160a01b0316631ba46cfd6040518163ffffffff1660e01b815260040160206040518083038186803b158015611ed857600080fd5b505afa158015611eec573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190611f109190612ba8565b604051631759616b60e11b81529091506001600160a01b03821690632eb2c2d690611f4590339030908c908c9060040161320c565b600060405180830381600087803b158015611f5f57600080fd5b505af1158015611f73573d6000803e3d6000fd5b505060405163a22cb46560e01b81526001600160a01b038581166004830152600160248301528416925063a22cb4659150604401600060405180830381600087803b158015611fc157600080fd5b505af1158015611fd5573d6000803e3d6000fd5b5050604051632321420560e21b81526001600160a01b0385169250638c850814915061200b908a908a908a908a90600401613292565b600060405180830381600087803b15801561202557600080fd5b505af1158015612039573d6000803e3d6000fd5b505050506040513d6000823e601f3d908101601f191682016040526120619190810190612d3f565b5090979650505050505050565b61207784611d08565b6001600160a01b0316639d54def68484846040518463ffffffff1660e01b81526004016120a693929190613319565b600060405180830381600087803b1580156120c057600080fd5b505af11580156120d4573d6000803e3d6000fd5b505050506040513d6000823e601f3d908101601f191682016040526120fc9190810190612d3f565b5050505050565b60008061210f84611d08565b90506000816001600160a01b0316631ba46cfd6040518163ffffffff1660e01b815260040160206040518083038186803b15801561214c57600080fd5b505afa158015612160573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906121849190612ba8565b845190915060005b81811015612225576121c6838783815181106121b857634e487b7160e01b600052603260045260246000fd5b60200260200101518661253a565b6001600160a01b03831673b47e3cd837ddf8e4c57f05d70ab865de6e193bbb141561221d5761221d8387838151811061220f57634e487b7160e01b600052603260045260246000fd5b6020026020010151866127e5565b60010161218c565b50604051630f57464360e21b81526060906001600160a01b03851690633d5d190c906122579089908590600401613264565b602060405180830381600087803b15801561227157600080fd5b505af1158015612285573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906122a99190612e32565b50929695505050505050565b6000806122c186611d08565b90506000816001600160a01b0316631ba46cfd6040518163ffffffff1660e01b815260040160206040518083038186803b1580156122fe57600080fd5b505afa158015612312573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906123369190612ba8565b865190915060005b818110156123bb5761236a838983815181106121b857634e487b7160e01b600052603260045260246000fd5b6001600160a01b03831673b47e3cd837ddf8e4c57f05d70ab865de6e193bbb14156123b3576123b38389838151811061220f57634e487b7160e01b600052603260045260246000fd5b60010161233e565b50604051632321420560e21b81526060906001600160a01b03851690638c850814906123f1908b9085908c908c90600401613292565b600060405180830381600087803b15801561240b57600080fd5b505af115801561241f573d6000803e3d6000fd5b505050506040513d6000823e601f3d908101601f191682016040526124479190810190612d3f565b509298975050505050505050565b60006124aa826040518060400160405280602081526020017f5361666545524332303a206c6f772d6c6576656c2063616c6c206661696c6564815250856001600160a01b03166128969092919063ffffffff16565b805190915015611cb157808060200190518101906124c89190612dd2565b611cb15760405162461bcd60e51b815260206004820152602a60248201527f5361666545524332303a204552433230206f7065726174696f6e20646964206e60448201527f6f74207375636365656400000000000000000000000000000000000000000000606482015260840161047b565b60606001600160a01b03841673b47e3cd837ddf8e4c57f05d70ab865de6e193bbb14156126f25760008360405160240161257691815260200190565b60408051601f198184030181529181526020820180516001600160e01b03167f58178168000000000000000000000000000000000000000000000000000000001790525190915060009081906001600160a01b038816906125d89085906131f0565b600060405180830381855afa9150503d8060008114612613576040519150601f19603f3d011682016040523d82523d6000602084013e612618565b606091505b50915091506000818060200190518101906126339190612ba8565b905082801561264a57506001600160a01b03811633145b6126965760405162461bcd60e51b815260206004820152601160248201527f4e6f7420746865204e4654206f776e6572000000000000000000000000000000604482015260640161047b565b6040516024810188905260440160408051601f198184030181529190526020810180516001600160e01b03167f8264fe9800000000000000000000000000000000000000000000000000000000179052945061275a9350505050565b6040513360248201526001600160a01b03831660448201526064810184905260840160408051601f198184030181529190526020810180516001600160e01b03167f42842e0e0000000000000000000000000000000000000000000000000000000017905290505b600080856001600160a01b03168360405161277591906131f0565b6000604051808303816000865af19150503d80600081146127b2576040519150601f19603f3d011682016040523d82523d6000602084013e6127b7565b606091505b50915091508181906127dc5760405162461bcd60e51b815260040161047b91906132e6565b50505050505050565b6001600160a01b03831673b47e3cd837ddf8e4c57f05d70ab865de6e193bbb1461280e57505050565b604051602481018390526000604482018190526001600160a01b03831660648301529060840160408051601f198184030181529181526020820180516001600160e01b03167fbf31196f000000000000000000000000000000000000000000000000000000001790525190915060009081906001600160a01b038716906127759085906131f0565b60606128a584846000856128ad565b949350505050565b6060824710156129255760405162461bcd60e51b815260206004820152602660248201527f416464726573733a20696e73756666696369656e742062616c616e636520666f60448201527f722063616c6c0000000000000000000000000000000000000000000000000000606482015260840161047b565b843b6129735760405162461bcd60e51b815260206004820152601d60248201527f416464726573733a2063616c6c20746f206e6f6e2d636f6e7472616374000000604482015260640161047b565b600080866001600160a01b0316858760405161298f91906131f0565b60006040518083038185875af1925050503d80600081146129cc576040519150601f19603f3d011682016040523d82523d6000602084013e6129d1565b606091505b50915091506129e18282866129ec565b979650505050505050565b606083156129fb57508161181f565b825115612a0b5782518084602001fd5b8160405162461bcd60e51b815260040161047b91906132e6565b8035612a3081613405565b919050565b60008083601f840112612a46578182fd5b50813567ffffffffffffffff811115612a5d578182fd5b6020830191508360208260051b8501011115612a7857600080fd5b9250929050565b600082601f830112612a8f578081fd5b81356020612aa4612a9f8361337c565b61334b565b80838252828201915082860187848660051b8901011115612ac3578586fd5b855b8581101561206157813584529284019290840190600101612ac5565b60008083601f840112612af2578182fd5b50813567ffffffffffffffff811115612b09578182fd5b602083019150836020828501011115612a7857600080fd5b600082601f830112612b31578081fd5b813567ffffffffffffffff811115612b4b57612b4b6133ef565b612b5e601f8201601f191660200161334b565b818152846020838601011115612b72578283fd5b816020850160208301379081016020019190915292915050565b600060208284031215612b9d578081fd5b813561181f81613405565b600060208284031215612bb9578081fd5b815161181f81613405565b600080600080600060a08688031215612bdb578081fd5b8535612be681613405565b94506020860135612bf681613405565b9350604086013567ffffffffffffffff80821115612c12578283fd5b612c1e89838a01612a7f565b94506060880135915080821115612c33578283fd5b612c3f89838a01612a7f565b93506080880135915080821115612c54578283fd5b50612c6188828901612b21565b9150509295509295909350565b60008060008060808587031215612c83578384fd5b8435612c8e81613405565b93506020850135612c9e81613405565b925060408501359150606085013567ffffffffffffffff811115612cc0578182fd5b612ccc87828801612b21565b91505092959194509250565b600080600080600060a08688031215612cef578081fd5b8535612cfa81613405565b94506020860135612d0a81613405565b93506040860135925060608601359150608086013567ffffffffffffffff811115612d33578182fd5b612c6188828901612b21565b60006020808385031215612d51578182fd5b825167ffffffffffffffff811115612d67578283fd5b8301601f81018513612d77578283fd5b8051612d85612a9f8261337c565b80828252848201915084840188868560051b8701011115612da4578687fd5b8694505b83851015612dc6578051835260019490940193918501918501612da8565b50979650505050505050565b600060208284031215612de3578081fd5b8151801515811461181f578182fd5b600060208284031215612e03578081fd5b81356001600160e01b03198116811461181f578182fd5b600060208284031215612e2b578081fd5b5035919050565b600060208284031215612e43578081fd5b5051919050565b60008060008060008060008060c0898b031215612e65578586fd5b88359750602089013567ffffffffffffffff80821115612e83578788fd5b612e8f8c838d01612a35565b909950975060408b01359150612ea482613405565b90955060608a013590612eb682613405565b90945060808a01359080821115612ecb578485fd5b50612ed88b828c01612ae1565b90945092505060a0890135612eec81613405565b809150509295985092959890939650565b60008060008060008060008060008060e08b8d031215612f1b578384fd5b8a35995060208b013567ffffffffffffffff80821115612f39578586fd5b612f458e838f01612a35565b909b50995060408d0135915080821115612f5d578586fd5b612f698e838f01612a35565b909950975060608d01359150612f7e82613405565b90955060808c013590612f9082613405565b90945060a08c01359080821115612fa5578384fd5b50612fb28d828e01612ae1565b9094509250612fc5905060c08c01612a25565b90509295989b9194979a5092959850565b6000806000806000806000806000806000806101008d8f031215612ff8578586fd5b8c359b5067ffffffffffffffff60208e01351115613014578586fd5b6130248e60208f01358f01612a35565b909b50995067ffffffffffffffff60408e01351115613041578586fd5b6130518e60408f01358f01612a35565b909950975067ffffffffffffffff60608e0135111561306e578586fd5b61307e8e60608f01358f01612a35565b909750955061308f60808e01612a25565b945061309d60a08e01612a25565b935067ffffffffffffffff60c08e013511156130b7578081fd5b6130c78e60c08f01358f01612ae1565b90935091506130d860e08e01612a25565b90509295989b509295989b509295989b565b600080600080600080600080600060e08a8c031215613107578283fd5b8935985060208a0135975060408a013567ffffffffffffffff8082111561312c578485fd5b6131388d838e01612a35565b909950975060608c0135915061314d82613405565b90955060808b01359061315f82613405565b90945060a08b01359080821115613174578485fd5b506131818c828d01612ae1565b90945092505060c08a013561319581613405565b809150509295985092959850929598565b6000815180845260208085019450808401835b838110156131d5578151875295820195908201906001016131b9565b509495945050505050565b8183823760009101908152919050565b600082516132028184602087016133c3565b9190910192915050565b60006001600160a01b03808716835280861660208401525060a0604083015261323860a08301856131a6565b828103606084015261324a81856131a6565b838103608090940193909352508152602001949350505050565b60408152600061327760408301856131a6565b828103602084015261328981856131a6565b95945050505050565b6080815260006132a560808301876131a6565b82810360208401526132b781876131a6565b905082810360408401526132cb81866131a6565b9150506001600160a01b038316606083015295945050505050565b60208152600082518060208401526133058160408501602087016133c3565b601f01601f19169190910160400192915050565b83815260606020820152600061333260608301856131a6565b90506001600160a01b0383166040830152949350505050565b604051601f8201601f1916810167ffffffffffffffff81118282101715613374576133746133ef565b604052919050565b600067ffffffffffffffff821115613396576133966133ef565b5060051b60200190565b6000828210156133be57634e487b7160e01b81526011600452602481fd5b500390565b60005b838110156133de5781810151838201526020016133c6565b83811115611b755750506000910152565b634e487b7160e01b600052604160045260246000fd5b6001600160a01b038116811461085d57600080fdfea264697066735822122000b66ab76fcfb21a06c18ca69ce46484f7327f73767f30619f90754bfd81600964736f6c63430008040033000000000000000000000000be86f647b167567525ccaafcd6f881f1ee558216000000000000000000000000c02aaa39b223fe8d0a0e5c4f27ead9083c756cc2
Deployed Bytecode
0x6080604052600436106100f75760003560e01c80639f820d851161008a578063bca8b28211610059578063bca8b2821461032b578063e88898db1461034b578063f23a6e611461035e578063f2fde38b146103a357600080fd5b80639f820d851461028c578063aa0bee4c1461029f578063ad5c4648146102b2578063bc197c81146102e657600080fd5b80635ee50f88116100c65780635ee50f8814610205578063715018a614610239578063839006f21461024e5780638da5cb5b1461026e57600080fd5b806301ffc9a714610103578063150b7a02146101385780631e85677c1461019557806336fa8b65146101b757600080fd5b366100fe57005b600080fd5b34801561010f57600080fd5b5061012361011e366004612df2565b6103c3565b60405190151581526020015b60405180910390f35b34801561014457600080fd5b5061017c610153366004612c6e565b7f150b7a0200000000000000000000000000000000000000000000000000000000949350505050565b6040516001600160e01b0319909116815260200161012f565b3480156101a157600080fd5b506101b56101b0366004612efd565b61042c565b005b3480156101c357600080fd5b506101ed6101d2366004612e1a565b6097602052600090815260409020546001600160a01b031681565b6040516001600160a01b03909116815260200161012f565b34801561021157600080fd5b506101ed7f000000000000000000000000be86f647b167567525ccaafcd6f881f1ee55821681565b34801561024557600080fd5b506101b561061a565b34801561025a57600080fd5b506101b5610269366004612b8c565b6106cb565b34801561027a57600080fd5b506033546001600160a01b03166101ed565b6101b561029a366004612fd6565b610860565b6101b56102ad3660046130ea565b610bb7565b3480156102be57600080fd5b506101ed7f000000000000000000000000c02aaa39b223fe8d0a0e5c4f27ead9083c756cc281565b3480156102f257600080fd5b5061017c610301366004612bc4565b7fbc197c810000000000000000000000000000000000000000000000000000000095945050505050565b34801561033757600080fd5b506101b5610346366004612e4a565b610ec5565b6101b5610359366004612efd565b611095565b34801561036a57600080fd5b5061017c610379366004612cd8565b7ff23a6e610000000000000000000000000000000000000000000000000000000095945050505050565b3480156103af57600080fd5b506101b56103be366004612b8c565b6113bf565b60006001600160e01b031982167f4e2312e000000000000000000000000000000000000000000000000000000000148061042657507f01ffc9a7000000000000000000000000000000000000000000000000000000006001600160e01b03198316145b92915050565b600260655414156104845760405162461bcd60e51b815260206004820152601f60248201527f5265656e7472616e637947756172643a207265656e7472616e742063616c6c0060448201526064015b60405180910390fd5b60026065556001600160a01b038116158015906104aa57506001600160a01b0381163014155b6104ea5760405162461bcd60e51b8152602060048201526011602482015270125b9d985b1a59081c9958da5c1a595b9d607a1b604482015260640161047b565b60006104f88a8a8a8a6114fe565b915050600061057c8c8c8c80806020026020016040519081016040528093929190818152602001838360200280828437600081840152601f19601f820116905080830192505050505050508b8b8080602002602001604051908101604052809392919081815260200183836020028082843760009201919091525061164f92505050565b905060006105ae827f000000000000000000000000c02aaa39b223fe8d0a0e5c4f27ead9083c756cc28a8a8a8a611826565b60408051858152602081018390526001600160a01b0387168183015290519192507f1cdb5ee3c47e1a706ac452b89698e5e3f2ff4f835ca72dde8936d0f4fcf37d81919081900360600190a161060684826000611acd565b505060016065555050505050505050505050565b6033546001600160a01b031633146106745760405162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572604482015260640161047b565b6033546040516000916001600160a01b0316907f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e0908390a36033805473ffffffffffffffffffffffffffffffffffffffff19169055565b6033546001600160a01b031633146107255760405162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572604482015260640161047b565b6001600160a01b0381166107cf57604051600090339047908381818185875af1925050503d8060008114610775576040519150601f19603f3d011682016040523d82523d6000602084013e61077a565b606091505b50509050806107cb5760405162461bcd60e51b815260206004820152601d60248201527f416464726573733a20756e61626c6520746f2073656e642076616c7565000000604482015260640161047b565b5050565b6040516370a0823160e01b815230600482015261085d9033906001600160a01b038416906370a082319060240160206040518083038186803b15801561081457600080fd5b505afa158015610828573d6000803e3d6000fd5b505050506040513d601f19601f8201168201806040525081019061084c9190612e32565b6001600160a01b0384169190611cb6565b50565b600260655414156108b35760405162461bcd60e51b815260206004820152601f60248201527f5265656e7472616e637947756172643a207265656e7472616e742063616c6c00604482015260640161047b565b60026065556001600160a01b038116158015906108d957506001600160a01b0381163014155b6109195760405162461bcd60e51b8152602060048201526011602482015270125b9d985b1a59081c9958da5c1a595b9d607a1b604482015260640161047b565b60006109278c8c8c8c6114fe565b91505034156109a5577f000000000000000000000000c02aaa39b223fe8d0a0e5c4f27ead9083c756cc26001600160a01b031663d0e30db0346040518263ffffffff1660e01b81526004016000604051808303818588803b15801561098b57600080fd5b505af115801561099f573d6000803e3d6000fd5b50505050505b60006109b08e611d08565b905060006109e27f000000000000000000000000c02aaa39b223fe8d0a0e5c4f27ead9083c756cc2838a8a8a8a611826565b9050610aa78f8f8f80806020026020016040519081016040528093929190818152602001838360200280828437600081840152601f19601f820116905080830192505050505050508e8e80806020026020016040519081016040528093929190818152602001838360200280828437600081840152601f19601f820116905080830192505050505050508d8d808060200260200160405190810160405280939291908181526020018383602002808284376000920191909152508b9250611e8f915050565b5060408051848152602081018390526001600160a01b0386168183015290517f7af2bc3f8ec800c569b6555feaf16589d96a9d04a49d1645fd456d75fa0b372b9181900360600190a16040516370a0823160e01b81523060048201526000907f000000000000000000000000c02aaa39b223fe8d0a0e5c4f27ead9083c756cc26001600160a01b0316906370a082319060240160206040518083038186803b158015610b5257600080fd5b505afa158015610b66573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610b8a9190612e32565b90508015610ba057610ba0898260003411611acd565b505060016065555050505050505050505050505050565b60026065541415610c0a5760405162461bcd60e51b815260206004820152601f60248201527f5265656e7472616e637947756172643a207265656e7472616e742063616c6c00604482015260640161047b565b60026065556001600160a01b03811615801590610c3057506001600160a01b0381163014155b610c705760405162461bcd60e51b8152602060048201526011602482015270125b9d985b1a59081c9958da5c1a595b9d607a1b604482015260640161047b565b60008811610cc05760405162461bcd60e51b815260206004820152601060248201527f4d7573742073656e6420616d6f756e7400000000000000000000000000000000604482015260640161047b565b3415610d3b577f000000000000000000000000c02aaa39b223fe8d0a0e5c4f27ead9083c756cc26001600160a01b031663d0e30db0346040518263ffffffff1660e01b81526004016000604051808303818588803b158015610d2157600080fd5b505af1158015610d35573d6000803e3d6000fd5b50505050505b6000610d468a611d08565b90506000610d787f000000000000000000000000c02aaa39b223fe8d0a0e5c4f27ead9083c756cc28389898989611826565b9050610dba8b8b8b8b8080602002602001604051908101604052809392919081815260200183836020028082843760009201919091525089925061206e915050565b604080518b8152602081018390526001600160a01b0385168183015290517ff7735c8cb2a65788ca663fc8415b7c6a66cd6847d58346d8334e8d52a599d3df9181900360600190a16040516370a0823160e01b81523060048201526000907f000000000000000000000000c02aaa39b223fe8d0a0e5c4f27ead9083c756cc26001600160a01b0316906370a082319060240160206040518083038186803b158015610e6457600080fd5b505afa158015610e78573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610e9c9190612e32565b90508015610eb257610eb2888260003411611acd565b5050600160655550505050505050505050565b60026065541415610f185760405162461bcd60e51b815260206004820152601f60248201527f5265656e7472616e637947756172643a207265656e7472616e742063616c6c00604482015260640161047b565b60026065556001600160a01b03811615801590610f3e57506001600160a01b0381163014155b610f7e5760405162461bcd60e51b8152602060048201526011602482015270125b9d985b1a59081c9958da5c1a595b9d607a1b604482015260640161047b565b85610fbb5760405162461bcd60e51b815260206004820152600d60248201526c4d7573742073656e642049447360981b604482015260640161047b565b6000610ffa8989898080602002602001604051908101604052809392919081815260200183836020028082843760009201919091525061210392505050565b9050600061102c827f000000000000000000000000c02aaa39b223fe8d0a0e5c4f27ead9083c756cc289898989611826565b604080518a8152602081018390526001600160a01b0386168183015290519192507f1cdb5ee3c47e1a706ac452b89698e5e3f2ff4f835ca72dde8936d0f4fcf37d81919081900360600190a161108483826000611acd565b505060016065555050505050505050565b600260655414156110e85760405162461bcd60e51b815260206004820152601f60248201527f5265656e7472616e637947756172643a207265656e7472616e742063616c6c00604482015260640161047b565b60026065556001600160a01b0381161580159061110e57506001600160a01b0381163014155b61114e5760405162461bcd60e51b8152602060048201526011602482015270125b9d985b1a59081c9958da5c1a595b9d607a1b604482015260640161047b565b8761118b5760405162461bcd60e51b815260206004820152600d60248201526c4d7573742073656e642049447360981b604482015260640161047b565b3415611206577f000000000000000000000000c02aaa39b223fe8d0a0e5c4f27ead9083c756cc26001600160a01b031663d0e30db0346040518263ffffffff1660e01b81526004016000604051808303818588803b1580156111ec57600080fd5b505af1158015611200573d6000803e3d6000fd5b50505050505b60006112118b611d08565b905060006112437f000000000000000000000000c02aaa39b223fe8d0a0e5c4f27ead9083c756cc28389898989611826565b90506112c68c8c8c80806020026020016040519081016040528093929190818152602001838360200280828437600081840152601f19601f820116905080830192505050505050508b8b808060200260200160405190810160405280939291908181526020018383602002808284376000920191909152508992506122b5915050565b50604080518b8152602081018390526001600160a01b0385168183015290517f7af2bc3f8ec800c569b6555feaf16589d96a9d04a49d1645fd456d75fa0b372b9181900360600190a16040516370a0823160e01b81523060048201526000907f000000000000000000000000c02aaa39b223fe8d0a0e5c4f27ead9083c756cc26001600160a01b0316906370a082319060240160206040518083038186803b15801561137157600080fd5b505afa158015611385573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906113a99190612e32565b9050801561060657610606888260003411611acd565b6033546001600160a01b031633146114195760405162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572604482015260640161047b565b6001600160a01b0381166114955760405162461bcd60e51b815260206004820152602660248201527f4f776e61626c653a206e6577206f776e657220697320746865207a65726f206160448201527f6464726573730000000000000000000000000000000000000000000000000000606482015260840161047b565b6033546040516001600160a01b038084169216907f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e090600090a36033805473ffffffffffffffffffffffffffffffffffffffff19166001600160a01b0392909216919091179055565b8260008161153e5760405162461bcd60e51b815260206004820152600d60248201526c4d7573742073656e642049447360981b604482015260640161047b565b8282111561158e5760405162461bcd60e51b815260206004820152601f60248201527f4d75737420646566696e6520616d6f756e747320616761696e73742049447300604482015260640161047b565b60005b828110156116455760008585838181106115bb57634e487b7160e01b600052603260045260246000fd5b905060200201351161160f5760405162461bcd60e51b815260206004820152601360248201527f496e76616c6964203131353520616d6f756e7400000000000000000000000000604482015260640161047b565b84848281811061162f57634e487b7160e01b600052603260045260246000fd5b9050602002013582019150806001019050611591565b5094509492505050565b60008061165b85611d08565b90506000816001600160a01b0316631ba46cfd6040518163ffffffff1660e01b815260040160206040518083038186803b15801561169857600080fd5b505afa1580156116ac573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906116d09190612ba8565b604051631759616b60e11b81529091506001600160a01b03821690632eb2c2d69061170590339030908a908a9060040161320c565b600060405180830381600087803b15801561171f57600080fd5b505af1158015611733573d6000803e3d6000fd5b505060405163a22cb46560e01b81526001600160a01b038581166004830152600160248301528416925063a22cb4659150604401600060405180830381600087803b15801561178157600080fd5b505af1158015611795573d6000803e3d6000fd5b5050604051630f57464360e21b81526001600160a01b0385169250633d5d190c91506117c79088908890600401613264565b602060405180830381600087803b1580156117e157600080fd5b505af11580156117f5573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906118199190612e32565b50909150505b9392505050565b6040516370a0823160e01b815230600482015260009081906001600160a01b038816906370a082319060240160206040518083038186803b15801561186a57600080fd5b505afa15801561187e573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906118a29190612e32565b6040517f095ea7b30000000000000000000000000000000000000000000000000000000081526001600160a01b03878116600483015260001960248301529192509089169063095ea7b390604401602060405180830381600087803b15801561190a57600080fd5b505af115801561191e573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906119429190612dd2565b61198e5760405162461bcd60e51b815260206004820152601a60248201527f556e61626c6520746f20617070726f766520636f6e7472616374000000000000604482015260640161047b565b6000856001600160a01b031685856040516119aa9291906131e0565b6000604051808303816000865af19150503d80600081146119e7576040519150601f19603f3d011682016040523d82523d6000602084013e6119ec565b606091505b5050905080611a3d5760405162461bcd60e51b815260206004820152601060248201527f535741505f43414c4c5f4641494c454400000000000000000000000000000000604482015260640161047b565b6040516370a0823160e01b815230600482015282906001600160a01b038a16906370a082319060240160206040518083038186803b158015611a7e57600080fd5b505afa158015611a92573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190611ab69190612e32565b611ac091906133a0565b9998505050505050505050565b8015611b7b5760405163a9059cbb60e01b81526001600160a01b038481166004830152602482018490527f000000000000000000000000c02aaa39b223fe8d0a0e5c4f27ead9083c756cc2169063a9059cbb90604401602060405180830381600087803b158015611b3d57600080fd5b505af1158015611b51573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190611b759190612dd2565b50505050565b6040517f2e1a7d4d000000000000000000000000000000000000000000000000000000008152600481018390527f000000000000000000000000c02aaa39b223fe8d0a0e5c4f27ead9083c756cc26001600160a01b031690632e1a7d4d90602401600060405180830381600087803b158015611bf657600080fd5b505af1158015611c0a573d6000803e3d6000fd5b505050506000836001600160a01b03168360405160006040518083038185875af1925050503d8060008114611c5b576040519150601f19603f3d011682016040523d82523d6000602084013e611c60565b606091505b5050905080611b755760405162461bcd60e51b815260206004820152601d60248201527f556e61626c6520746f2073656e6420756e777261707065642057455448000000604482015260640161047b565b505050565b604080516001600160a01b038416602482015260448082018490528251808303909101815260649091019091526020810180516001600160e01b031663a9059cbb60e01b179052611cb1908490612455565b6000818152609760205260408120546001600160a01b0316611e0f576040517f81a36fb6000000000000000000000000000000000000000000000000000000008152600481018390527f000000000000000000000000be86f647b167567525ccaafcd6f881f1ee5582166001600160a01b0316906381a36fb69060240160206040518083038186803b158015611d9d57600080fd5b505afa158015611db1573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190611dd59190612ba8565b6000838152609760205260409020805473ffffffffffffffffffffffffffffffffffffffff19166001600160a01b03929092169190911790555b6000828152609760205260409020546001600160a01b0316611e735760405162461bcd60e51b815260206004820152601460248201527f5661756c7420646f6573206e6f74206578697374000000000000000000000000604482015260640161047b565b506000908152609760205260409020546001600160a01b031690565b600080611e9b87611d08565b90506000816001600160a01b0316631ba46cfd6040518163ffffffff1660e01b815260040160206040518083038186803b158015611ed857600080fd5b505afa158015611eec573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190611f109190612ba8565b604051631759616b60e11b81529091506001600160a01b03821690632eb2c2d690611f4590339030908c908c9060040161320c565b600060405180830381600087803b158015611f5f57600080fd5b505af1158015611f73573d6000803e3d6000fd5b505060405163a22cb46560e01b81526001600160a01b038581166004830152600160248301528416925063a22cb4659150604401600060405180830381600087803b158015611fc157600080fd5b505af1158015611fd5573d6000803e3d6000fd5b5050604051632321420560e21b81526001600160a01b0385169250638c850814915061200b908a908a908a908a90600401613292565b600060405180830381600087803b15801561202557600080fd5b505af1158015612039573d6000803e3d6000fd5b505050506040513d6000823e601f3d908101601f191682016040526120619190810190612d3f565b5090979650505050505050565b61207784611d08565b6001600160a01b0316639d54def68484846040518463ffffffff1660e01b81526004016120a693929190613319565b600060405180830381600087803b1580156120c057600080fd5b505af11580156120d4573d6000803e3d6000fd5b505050506040513d6000823e601f3d908101601f191682016040526120fc9190810190612d3f565b5050505050565b60008061210f84611d08565b90506000816001600160a01b0316631ba46cfd6040518163ffffffff1660e01b815260040160206040518083038186803b15801561214c57600080fd5b505afa158015612160573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906121849190612ba8565b845190915060005b81811015612225576121c6838783815181106121b857634e487b7160e01b600052603260045260246000fd5b60200260200101518661253a565b6001600160a01b03831673b47e3cd837ddf8e4c57f05d70ab865de6e193bbb141561221d5761221d8387838151811061220f57634e487b7160e01b600052603260045260246000fd5b6020026020010151866127e5565b60010161218c565b50604051630f57464360e21b81526060906001600160a01b03851690633d5d190c906122579089908590600401613264565b602060405180830381600087803b15801561227157600080fd5b505af1158015612285573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906122a99190612e32565b50929695505050505050565b6000806122c186611d08565b90506000816001600160a01b0316631ba46cfd6040518163ffffffff1660e01b815260040160206040518083038186803b1580156122fe57600080fd5b505afa158015612312573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906123369190612ba8565b865190915060005b818110156123bb5761236a838983815181106121b857634e487b7160e01b600052603260045260246000fd5b6001600160a01b03831673b47e3cd837ddf8e4c57f05d70ab865de6e193bbb14156123b3576123b38389838151811061220f57634e487b7160e01b600052603260045260246000fd5b60010161233e565b50604051632321420560e21b81526060906001600160a01b03851690638c850814906123f1908b9085908c908c90600401613292565b600060405180830381600087803b15801561240b57600080fd5b505af115801561241f573d6000803e3d6000fd5b505050506040513d6000823e601f3d908101601f191682016040526124479190810190612d3f565b509298975050505050505050565b60006124aa826040518060400160405280602081526020017f5361666545524332303a206c6f772d6c6576656c2063616c6c206661696c6564815250856001600160a01b03166128969092919063ffffffff16565b805190915015611cb157808060200190518101906124c89190612dd2565b611cb15760405162461bcd60e51b815260206004820152602a60248201527f5361666545524332303a204552433230206f7065726174696f6e20646964206e60448201527f6f74207375636365656400000000000000000000000000000000000000000000606482015260840161047b565b60606001600160a01b03841673b47e3cd837ddf8e4c57f05d70ab865de6e193bbb14156126f25760008360405160240161257691815260200190565b60408051601f198184030181529181526020820180516001600160e01b03167f58178168000000000000000000000000000000000000000000000000000000001790525190915060009081906001600160a01b038816906125d89085906131f0565b600060405180830381855afa9150503d8060008114612613576040519150601f19603f3d011682016040523d82523d6000602084013e612618565b606091505b50915091506000818060200190518101906126339190612ba8565b905082801561264a57506001600160a01b03811633145b6126965760405162461bcd60e51b815260206004820152601160248201527f4e6f7420746865204e4654206f776e6572000000000000000000000000000000604482015260640161047b565b6040516024810188905260440160408051601f198184030181529190526020810180516001600160e01b03167f8264fe9800000000000000000000000000000000000000000000000000000000179052945061275a9350505050565b6040513360248201526001600160a01b03831660448201526064810184905260840160408051601f198184030181529190526020810180516001600160e01b03167f42842e0e0000000000000000000000000000000000000000000000000000000017905290505b600080856001600160a01b03168360405161277591906131f0565b6000604051808303816000865af19150503d80600081146127b2576040519150601f19603f3d011682016040523d82523d6000602084013e6127b7565b606091505b50915091508181906127dc5760405162461bcd60e51b815260040161047b91906132e6565b50505050505050565b6001600160a01b03831673b47e3cd837ddf8e4c57f05d70ab865de6e193bbb1461280e57505050565b604051602481018390526000604482018190526001600160a01b03831660648301529060840160408051601f198184030181529181526020820180516001600160e01b03167fbf31196f000000000000000000000000000000000000000000000000000000001790525190915060009081906001600160a01b038716906127759085906131f0565b60606128a584846000856128ad565b949350505050565b6060824710156129255760405162461bcd60e51b815260206004820152602660248201527f416464726573733a20696e73756666696369656e742062616c616e636520666f60448201527f722063616c6c0000000000000000000000000000000000000000000000000000606482015260840161047b565b843b6129735760405162461bcd60e51b815260206004820152601d60248201527f416464726573733a2063616c6c20746f206e6f6e2d636f6e7472616374000000604482015260640161047b565b600080866001600160a01b0316858760405161298f91906131f0565b60006040518083038185875af1925050503d80600081146129cc576040519150601f19603f3d011682016040523d82523d6000602084013e6129d1565b606091505b50915091506129e18282866129ec565b979650505050505050565b606083156129fb57508161181f565b825115612a0b5782518084602001fd5b8160405162461bcd60e51b815260040161047b91906132e6565b8035612a3081613405565b919050565b60008083601f840112612a46578182fd5b50813567ffffffffffffffff811115612a5d578182fd5b6020830191508360208260051b8501011115612a7857600080fd5b9250929050565b600082601f830112612a8f578081fd5b81356020612aa4612a9f8361337c565b61334b565b80838252828201915082860187848660051b8901011115612ac3578586fd5b855b8581101561206157813584529284019290840190600101612ac5565b60008083601f840112612af2578182fd5b50813567ffffffffffffffff811115612b09578182fd5b602083019150836020828501011115612a7857600080fd5b600082601f830112612b31578081fd5b813567ffffffffffffffff811115612b4b57612b4b6133ef565b612b5e601f8201601f191660200161334b565b818152846020838601011115612b72578283fd5b816020850160208301379081016020019190915292915050565b600060208284031215612b9d578081fd5b813561181f81613405565b600060208284031215612bb9578081fd5b815161181f81613405565b600080600080600060a08688031215612bdb578081fd5b8535612be681613405565b94506020860135612bf681613405565b9350604086013567ffffffffffffffff80821115612c12578283fd5b612c1e89838a01612a7f565b94506060880135915080821115612c33578283fd5b612c3f89838a01612a7f565b93506080880135915080821115612c54578283fd5b50612c6188828901612b21565b9150509295509295909350565b60008060008060808587031215612c83578384fd5b8435612c8e81613405565b93506020850135612c9e81613405565b925060408501359150606085013567ffffffffffffffff811115612cc0578182fd5b612ccc87828801612b21565b91505092959194509250565b600080600080600060a08688031215612cef578081fd5b8535612cfa81613405565b94506020860135612d0a81613405565b93506040860135925060608601359150608086013567ffffffffffffffff811115612d33578182fd5b612c6188828901612b21565b60006020808385031215612d51578182fd5b825167ffffffffffffffff811115612d67578283fd5b8301601f81018513612d77578283fd5b8051612d85612a9f8261337c565b80828252848201915084840188868560051b8701011115612da4578687fd5b8694505b83851015612dc6578051835260019490940193918501918501612da8565b50979650505050505050565b600060208284031215612de3578081fd5b8151801515811461181f578182fd5b600060208284031215612e03578081fd5b81356001600160e01b03198116811461181f578182fd5b600060208284031215612e2b578081fd5b5035919050565b600060208284031215612e43578081fd5b5051919050565b60008060008060008060008060c0898b031215612e65578586fd5b88359750602089013567ffffffffffffffff80821115612e83578788fd5b612e8f8c838d01612a35565b909950975060408b01359150612ea482613405565b90955060608a013590612eb682613405565b90945060808a01359080821115612ecb578485fd5b50612ed88b828c01612ae1565b90945092505060a0890135612eec81613405565b809150509295985092959890939650565b60008060008060008060008060008060e08b8d031215612f1b578384fd5b8a35995060208b013567ffffffffffffffff80821115612f39578586fd5b612f458e838f01612a35565b909b50995060408d0135915080821115612f5d578586fd5b612f698e838f01612a35565b909950975060608d01359150612f7e82613405565b90955060808c013590612f9082613405565b90945060a08c01359080821115612fa5578384fd5b50612fb28d828e01612ae1565b9094509250612fc5905060c08c01612a25565b90509295989b9194979a5092959850565b6000806000806000806000806000806000806101008d8f031215612ff8578586fd5b8c359b5067ffffffffffffffff60208e01351115613014578586fd5b6130248e60208f01358f01612a35565b909b50995067ffffffffffffffff60408e01351115613041578586fd5b6130518e60408f01358f01612a35565b909950975067ffffffffffffffff60608e0135111561306e578586fd5b61307e8e60608f01358f01612a35565b909750955061308f60808e01612a25565b945061309d60a08e01612a25565b935067ffffffffffffffff60c08e013511156130b7578081fd5b6130c78e60c08f01358f01612ae1565b90935091506130d860e08e01612a25565b90509295989b509295989b509295989b565b600080600080600080600080600060e08a8c031215613107578283fd5b8935985060208a0135975060408a013567ffffffffffffffff8082111561312c578485fd5b6131388d838e01612a35565b909950975060608c0135915061314d82613405565b90955060808b01359061315f82613405565b90945060a08b01359080821115613174578485fd5b506131818c828d01612ae1565b90945092505060c08a013561319581613405565b809150509295985092959850929598565b6000815180845260208085019450808401835b838110156131d5578151875295820195908201906001016131b9565b509495945050505050565b8183823760009101908152919050565b600082516132028184602087016133c3565b9190910192915050565b60006001600160a01b03808716835280861660208401525060a0604083015261323860a08301856131a6565b828103606084015261324a81856131a6565b838103608090940193909352508152602001949350505050565b60408152600061327760408301856131a6565b828103602084015261328981856131a6565b95945050505050565b6080815260006132a560808301876131a6565b82810360208401526132b781876131a6565b905082810360408401526132cb81866131a6565b9150506001600160a01b038316606083015295945050505050565b60208152600082518060208401526133058160408501602087016133c3565b601f01601f19169190910160400192915050565b83815260606020820152600061333260608301856131a6565b90506001600160a01b0383166040830152949350505050565b604051601f8201601f1916810167ffffffffffffffff81118282101715613374576133746133ef565b604052919050565b600067ffffffffffffffff821115613396576133966133ef565b5060051b60200190565b6000828210156133be57634e487b7160e01b81526011600452602481fd5b500390565b60005b838110156133de5781810151838201526020016133c6565b83811115611b755750506000910152565b634e487b7160e01b600052604160045260246000fd5b6001600160a01b038116811461085d57600080fdfea264697066735822122000b66ab76fcfb21a06c18ca69ce46484f7327f73767f30619f90754bfd81600964736f6c63430008040033
Constructor Arguments (ABI-Encoded and is the last bytes of the Contract Creation Code above)
000000000000000000000000be86f647b167567525ccaafcd6f881f1ee558216000000000000000000000000c02aaa39b223fe8d0a0e5c4f27ead9083c756cc2
-----Decoded View---------------
Arg [0] : _nftxFactory (address): 0xBE86f647b167567525cCAAfcd6f881F1Ee558216
Arg [1] : _WETH (address): 0xC02aaA39b223FE8D0A0e5C4F27eAD9083C756Cc2
-----Encoded View---------------
2 Constructor Arguments found :
Arg [0] : 000000000000000000000000be86f647b167567525ccaafcd6f881f1ee558216
Arg [1] : 000000000000000000000000c02aaa39b223fe8d0a0e5c4f27ead9083c756cc2
Loading...
Loading
Loading...
Loading
Net Worth in USD
$157.97
Net Worth in ETH
0.049826
Token Allocations
MILADY
98.14%
MEEB
0.94%
WIZARD
0.69%
Others
0.23%
Multichain Portfolio | 35 Chains
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.