ETH Price: $3,355.59 (+0.27%)
 

Overview

Max Total Supply

487 CA

Holders

179

Market

Volume (24H)

N/A

Min Price (24H)

N/A

Max Price (24H)

N/A
Filtered by Token Holder
woodificould.eth
Balance
1 CA
0xb0396C7b1102a978a87A863Fbca7FD095b933D58
Loading...
Loading
Loading...
Loading
Loading...
Loading

Click here to update the token information / general information
# Exchange Pair Price  24H Volume % Volume

Contract Source Code Verified (Exact Match)

Contract Name:
ChainWaves

Compiler Version
v0.8.13+commit.abaa5c0e

Optimization Enabled:
Yes with 200 runs

Other Settings:
default evmVersion
File 1 of 7 : ChainWaves.sol
// SPDX-License-Identifier: UNLICENSED
pragma solidity ^0.8.13;

import "solmate/tokens/ERC721.sol";
import "solmate/auth/Owned.sol";
import "openzeppelin-contracts/utils/cryptography/MerkleProof.sol";
import "./AnonymiceLibrary.sol";
import "./ChainWavesGenerator.sol";
import "./ChainWavesErrors.sol";

contract ChainWaves is ChainWavesErrors, ERC721, Owned {
    using AnonymiceLibrary for uint8;

    struct Trait {
        string traitName;
        string traitType;
    }

    uint256 public constant MAX_SUPPLY = 512;
    uint256 public constant MINT_PRICE = 0.0256 ether;
    uint256 public constant MINT_START = 1674156600;
    uint256 public constant MAX_MINT = 3;
    uint256 public snowcrashReserve = 120;
    bool public MINTING_LIVE;

    uint256 public totalSupply;

    // TODO: generate actual root (this is folded faces)
    bytes32 constant snowcrashRoot =
        0xea35e50958ff75fe96e04a6dd792de75a26dd0c2a2d12e8a4c485d938961eb39;

    bool private freeMinted;

    mapping(address => uint256) mintInfo;
    mapping(uint256 => uint256) tokenIdToHash;
    mapping(uint256 => Trait[]) public traitTypes;

    //Mappings

    ChainWavesGenerator chainWavesGenerator;

    //uint arrays
    uint16[][6] private TIERS;

    constructor()
        ERC721("ChainWaves", "CA")
        Owned(0xB6eE8B1899e4cad7e28015995B82969e44BD0bb0)
    {
        chainWavesGenerator = new ChainWavesGenerator();

        //Palette
        TIERS[0] = [1000, 1500, 1400, 1700, 1200, 400, 400, 1600, 800];
        //Noise
        TIERS[1] = [1000, 4000, 4000, 1000];
        //Speed
        TIERS[2] = [1000, 4000, 4000, 1000];
        //Char set
        TIERS[3] = [2250, 2250, 2250, 2250, 600, 400];
        //Detail
        TIERS[4] = [1000, 6000, 3000];
        //NumCols
        TIERS[5] = [800, 6200, 2600, 400];
    }

    //prevents someone calling read functions the same block they mint
    modifier disallowIfStateIsChanging() {
        if ((mintInfo[msg.sender] >> 8) == block.number) revert Stap();
        _;
    }

    /**
     * @dev Converts a digit from 0 - 10000 into its corresponding rarity based on the given rarity tier.
     * @param _randinput The input from 0 - 10000 to use for rarity gen.
     * @param _rarityTier The tier to use.
     */
    function rarityGen(uint256 _randinput, uint8 _rarityTier)
        internal
        view
        returns (uint8)
    {
        uint16 currentLowerBound;
        uint256 tiersLength = TIERS[_rarityTier].length;
        for (uint8 i; i < tiersLength; ++i) {
            uint16 thisPercentage = TIERS[_rarityTier][i];
            if (
                _randinput >= currentLowerBound &&
                _randinput < currentLowerBound + thisPercentage
            ) return i;
            currentLowerBound = currentLowerBound + thisPercentage;
        }

        revert();
    }

    /**
     * @param _a The address to be used within the hash.
     */
    function hash(address _a, uint256 _tokenId)
        internal
        view
        returns (uint256)
    {
        return
            uint256(
                keccak256(
                    abi.encodePacked(
                        block.timestamp,
                        block.difficulty,
                        _a,
                        _tokenId
                    )
                )
            );
    }

    function normieMint(uint256 _amount) external payable {
        if (_amount > MAX_MINT) revert MaxThree();
        if (msg.value != MINT_PRICE * _amount) revert MintPrice();

        uint256 minterInfo = mintInfo[msg.sender];
        if ((minterInfo & 0xF) != 0) revert PublicMinted();

        minterInfo |= 1;
        minterInfo = (minterInfo & 0xFF) + (block.number << 8);
        mintInfo[msg.sender] = minterInfo;

        mintInternal(msg.sender, _amount);
    }

    // TODO: add merkle root,
    function snowcrashMint(bytes32[] calldata merkleProof) external payable {
        bytes32 node = keccak256(abi.encodePacked(msg.sender));
        require(
            MerkleProof.verify(merkleProof, snowcrashRoot, node),
            "Not on WL"
        );
        if (msg.value != MINT_PRICE) revert MintPrice();

        uint256 minterInfo = mintInfo[msg.sender];
        if (((minterInfo & 0xF0) >> 4) != 0) revert SnowcrashMinted();
        if (snowcrashReserve == 0) revert ReserveClosed();
        --snowcrashReserve;

        minterInfo |= (1 << 4);
        minterInfo = (minterInfo & 0xFF) + (block.number << 8);
        mintInfo[msg.sender] = minterInfo;

        mintInternal(msg.sender, 1);
    }

    function freeMints(
        address[] calldata _addresses,
        uint256[] calldata _amount
    ) external payable onlyOwner {
        if (freeMinted) revert FreeMintDone();
        uint256 addressesLength = _addresses.length;
        if (addressesLength != _amount.length) revert ArrayLengths();
        for (uint256 i; i < addressesLength; ++i) {
            mintInternal(_addresses[i], _amount[i]);
        }

        freeMinted = true;
    }

    function mintInternal(address _to, uint256 _amount) internal {
        if (!MINTING_LIVE || block.timestamp < MINT_START) revert NotLive();
        if (_amount == 0) revert MintZero();
        if (totalSupply + _amount + snowcrashReserve > MAX_SUPPLY)
            revert SoldOut();
        uint256 nextTokenId = totalSupply;
        uint256 newTotalSupply = totalSupply + _amount;

        for (; nextTokenId < newTotalSupply; ++nextTokenId) {
            tokenIdToHash[nextTokenId] = hash(_to, nextTokenId);
            _mint(_to, nextTokenId);
        }
        totalSupply = newTotalSupply;
    }

    // hash stuff

    /**
     * @dev Converts a `uint256` to its ASCII `string` decimal representation.
     * From anonymice
     */

    function buildHash(uint256 _t) internal view returns (string memory) {
        // This will generate a 4 character string.
        string memory currentHash = "";
        uint256 tokenHash = tokenIdToHash[_t];

        for (uint8 i; i < 6; ++i) {
            uint16 _randinput = uint16(
                uint256(keccak256(abi.encodePacked(tokenHash, i))) % 10000
            );
            currentHash = string(
                abi.encodePacked(
                    currentHash,
                    rarityGen(_randinput, i).toString()
                )
            );
        }
        return currentHash;
    }

    // Views

    function hashToMetadata(string memory _hash)
        public
        view
        disallowIfStateIsChanging
        returns (string memory)
    {
        string memory metadataString;

        for (uint8 i; i < 6; ++i) {
            uint8 thisTraitIndex = AnonymiceLibrary.parseInt(
                AnonymiceLibrary.substring(_hash, i, i + 1)
            );

            metadataString = string(
                abi.encodePacked(
                    metadataString,
                    '{"trait_type":"',
                    traitTypes[i][thisTraitIndex].traitType,
                    '","value":"',
                    traitTypes[i][thisTraitIndex].traitName,
                    '"}'
                )
            );

            if (i != 5)
                metadataString = string(abi.encodePacked(metadataString, ","));
        }

        return string(abi.encodePacked("[", metadataString, "]"));
    }

    function _tokenIdToHash(uint256 _tokenId)
        public
        view
        disallowIfStateIsChanging
        returns (string memory tokenHash)
    {
        if (_tokenId >= totalSupply) revert NonExistantId();
        tokenHash = buildHash(_tokenId);
    }

    function tokenURI(uint256 _tokenId)
        public
        view
        override
        returns (string memory _URI)
    {
        if (_tokenId >= totalSupply) revert NonExistantId();
        string memory _hash = _tokenIdToHash(_tokenId);
        _URI = string(
            abi.encodePacked(
                "data:application/json;base64,",
                AnonymiceLibrary.encode(
                    bytes(
                        string(
                            abi.encodePacked(
                                '{"name": "CH41NW4V35 #',
                                AnonymiceLibrary.toString(_tokenId),
                                '","description": "Fully onchain generative art SVG collection. Created by McToady & Circolors."',
                                ',"image": "data:image/svg+xml;base64,',
                                AnonymiceLibrary.encode(
                                    bytes(
                                        abi.encodePacked(
                                            "<svg viewBox='0 0 20 20' width='600' height='600' xmlns='http://www.w3.org/2000/svg' preserveAspectRatio='xMidYMin'><rect width='20' height='20' fill='#",
                                            chainWavesGenerator.buildSVG(
                                                _tokenId,
                                                _hash
                                            ),
                                            "</svg>"
                                        )
                                    )
                                ),
                                '","attributes":',
                                hashToMetadata(_hash),
                                "}"
                            )
                        )
                    )
                )
            )
        );
    }

    // Owner Functions
    /**
     * @dev Add a trait type
     * @param _traitTypeIndex The trait type index
     * @param traits Array of traits to add
     */

    function addTraitType(uint256 _traitTypeIndex, Trait[] memory traits)
        external
        payable
        onlyOwner
    {
        for (uint256 i; i < traits.length; ++i) {
            traitTypes[_traitTypeIndex].push(
                Trait(traits[i].traitName, traits[i].traitType)
            );
        }

        return;
    }

    function flipMint() external payable onlyOwner {
        MINTING_LIVE = !MINTING_LIVE;
    }

    function withdraw() external payable onlyOwner {
        uint256 twelve = (address(this).balance / 100) * 12;
        uint256 eightythree = (address(this).balance / 100) * 83;
        uint256 five = (address(this).balance / 100) * 5;
        (bool sentI, ) = payable(
            address(0x4533d1F65906368ebfd61259dAee561DF3f3559D)
        ).call{value: twelve}("");
        if (!sentI) revert WithdrawFail();
        (bool sentC, ) = payable(
            address(0x888f8AA938dbb18b28bdD111fa4A0D3B8e10C871)
        ).call{value: five}("");
        if (!sentC) revert WithdrawFail();
        (bool sentT, ) = payable(
            address(0xE4260Df86f5261A41D19c2066f1Eb2Eb4F009e84)
        ).call{value: eightythree}("");
        if (!sentT) revert WithdrawFail();
    }

    function wipeSnowcrashReserve() external payable onlyOwner {
        snowcrashReserve = 0;
    }
}

File 2 of 7 : ERC721.sol
// SPDX-License-Identifier: AGPL-3.0-only
pragma solidity >=0.8.0;

/// @notice Modern, minimalist, and gas efficient ERC-721 implementation.
/// @author Solmate (https://github.com/transmissions11/solmate/blob/main/src/tokens/ERC721.sol)
abstract contract ERC721 {
    /*//////////////////////////////////////////////////////////////
                                 EVENTS
    //////////////////////////////////////////////////////////////*/

    event Transfer(address indexed from, address indexed to, uint256 indexed id);

    event Approval(address indexed owner, address indexed spender, uint256 indexed id);

    event ApprovalForAll(address indexed owner, address indexed operator, bool approved);

    /*//////////////////////////////////////////////////////////////
                         METADATA STORAGE/LOGIC
    //////////////////////////////////////////////////////////////*/

    string public name;

    string public symbol;

    function tokenURI(uint256 id) public view virtual returns (string memory);

    /*//////////////////////////////////////////////////////////////
                      ERC721 BALANCE/OWNER STORAGE
    //////////////////////////////////////////////////////////////*/

    mapping(uint256 => address) internal _ownerOf;

    mapping(address => uint256) internal _balanceOf;

    function ownerOf(uint256 id) public view virtual returns (address owner) {
        require((owner = _ownerOf[id]) != address(0), "NOT_MINTED");
    }

    function balanceOf(address owner) public view virtual returns (uint256) {
        require(owner != address(0), "ZERO_ADDRESS");

        return _balanceOf[owner];
    }

    /*//////////////////////////////////////////////////////////////
                         ERC721 APPROVAL STORAGE
    //////////////////////////////////////////////////////////////*/

    mapping(uint256 => address) public getApproved;

    mapping(address => mapping(address => bool)) public isApprovedForAll;

    /*//////////////////////////////////////////////////////////////
                               CONSTRUCTOR
    //////////////////////////////////////////////////////////////*/

    constructor(string memory _name, string memory _symbol) {
        name = _name;
        symbol = _symbol;
    }

    /*//////////////////////////////////////////////////////////////
                              ERC721 LOGIC
    //////////////////////////////////////////////////////////////*/

    function approve(address spender, uint256 id) public virtual {
        address owner = _ownerOf[id];

        require(msg.sender == owner || isApprovedForAll[owner][msg.sender], "NOT_AUTHORIZED");

        getApproved[id] = spender;

        emit Approval(owner, spender, id);
    }

    function setApprovalForAll(address operator, bool approved) public virtual {
        isApprovedForAll[msg.sender][operator] = approved;

        emit ApprovalForAll(msg.sender, operator, approved);
    }

    function transferFrom(
        address from,
        address to,
        uint256 id
    ) public virtual {
        require(from == _ownerOf[id], "WRONG_FROM");

        require(to != address(0), "INVALID_RECIPIENT");

        require(
            msg.sender == from || isApprovedForAll[from][msg.sender] || msg.sender == getApproved[id],
            "NOT_AUTHORIZED"
        );

        // Underflow of the sender's balance is impossible because we check for
        // ownership above and the recipient's balance can't realistically overflow.
        unchecked {
            _balanceOf[from]--;

            _balanceOf[to]++;
        }

        _ownerOf[id] = to;

        delete getApproved[id];

        emit Transfer(from, to, id);
    }

    function safeTransferFrom(
        address from,
        address to,
        uint256 id
    ) public virtual {
        transferFrom(from, to, id);

        require(
            to.code.length == 0 ||
                ERC721TokenReceiver(to).onERC721Received(msg.sender, from, id, "") ==
                ERC721TokenReceiver.onERC721Received.selector,
            "UNSAFE_RECIPIENT"
        );
    }

    function safeTransferFrom(
        address from,
        address to,
        uint256 id,
        bytes calldata data
    ) public virtual {
        transferFrom(from, to, id);

        require(
            to.code.length == 0 ||
                ERC721TokenReceiver(to).onERC721Received(msg.sender, from, id, data) ==
                ERC721TokenReceiver.onERC721Received.selector,
            "UNSAFE_RECIPIENT"
        );
    }

    /*//////////////////////////////////////////////////////////////
                              ERC165 LOGIC
    //////////////////////////////////////////////////////////////*/

    function supportsInterface(bytes4 interfaceId) public view virtual returns (bool) {
        return
            interfaceId == 0x01ffc9a7 || // ERC165 Interface ID for ERC165
            interfaceId == 0x80ac58cd || // ERC165 Interface ID for ERC721
            interfaceId == 0x5b5e139f; // ERC165 Interface ID for ERC721Metadata
    }

    /*//////////////////////////////////////////////////////////////
                        INTERNAL MINT/BURN LOGIC
    //////////////////////////////////////////////////////////////*/

    function _mint(address to, uint256 id) internal virtual {
        require(to != address(0), "INVALID_RECIPIENT");

        require(_ownerOf[id] == address(0), "ALREADY_MINTED");

        // Counter overflow is incredibly unrealistic.
        unchecked {
            _balanceOf[to]++;
        }

        _ownerOf[id] = to;

        emit Transfer(address(0), to, id);
    }

    function _burn(uint256 id) internal virtual {
        address owner = _ownerOf[id];

        require(owner != address(0), "NOT_MINTED");

        // Ownership check above ensures no underflow.
        unchecked {
            _balanceOf[owner]--;
        }

        delete _ownerOf[id];

        delete getApproved[id];

        emit Transfer(owner, address(0), id);
    }

    /*//////////////////////////////////////////////////////////////
                        INTERNAL SAFE MINT LOGIC
    //////////////////////////////////////////////////////////////*/

    function _safeMint(address to, uint256 id) internal virtual {
        _mint(to, id);

        require(
            to.code.length == 0 ||
                ERC721TokenReceiver(to).onERC721Received(msg.sender, address(0), id, "") ==
                ERC721TokenReceiver.onERC721Received.selector,
            "UNSAFE_RECIPIENT"
        );
    }

    function _safeMint(
        address to,
        uint256 id,
        bytes memory data
    ) internal virtual {
        _mint(to, id);

        require(
            to.code.length == 0 ||
                ERC721TokenReceiver(to).onERC721Received(msg.sender, address(0), id, data) ==
                ERC721TokenReceiver.onERC721Received.selector,
            "UNSAFE_RECIPIENT"
        );
    }
}

/// @notice A generic interface for a contract which properly accepts ERC721 tokens.
/// @author Solmate (https://github.com/transmissions11/solmate/blob/main/src/tokens/ERC721.sol)
abstract contract ERC721TokenReceiver {
    function onERC721Received(
        address,
        address,
        uint256,
        bytes calldata
    ) external virtual returns (bytes4) {
        return ERC721TokenReceiver.onERC721Received.selector;
    }
}

File 3 of 7 : Owned.sol
// SPDX-License-Identifier: AGPL-3.0-only
pragma solidity >=0.8.0;

/// @notice Simple single owner authorization mixin.
/// @author Solmate (https://github.com/transmissions11/solmate/blob/main/src/auth/Owned.sol)
abstract contract Owned {
    /*//////////////////////////////////////////////////////////////
                                 EVENTS
    //////////////////////////////////////////////////////////////*/

    event OwnershipTransferred(address indexed user, address indexed newOwner);

    /*//////////////////////////////////////////////////////////////
                            OWNERSHIP STORAGE
    //////////////////////////////////////////////////////////////*/

    address public owner;

    modifier onlyOwner() virtual {
        require(msg.sender == owner, "UNAUTHORIZED");

        _;
    }

    /*//////////////////////////////////////////////////////////////
                               CONSTRUCTOR
    //////////////////////////////////////////////////////////////*/

    constructor(address _owner) {
        owner = _owner;

        emit OwnershipTransferred(address(0), _owner);
    }

    /*//////////////////////////////////////////////////////////////
                             OWNERSHIP LOGIC
    //////////////////////////////////////////////////////////////*/

    function transferOwnership(address newOwner) public virtual onlyOwner {
        owner = newOwner;

        emit OwnershipTransferred(msg.sender, newOwner);
    }
}

File 4 of 7 : MerkleProof.sol
// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts (last updated v4.8.0) (utils/cryptography/MerkleProof.sol)

pragma solidity ^0.8.0;

/**
 * @dev These functions deal with verification of Merkle Tree proofs.
 *
 * The tree and the proofs can be generated using our
 * https://github.com/OpenZeppelin/merkle-tree[JavaScript library].
 * You will find a quickstart guide in the readme.
 *
 * WARNING: You should avoid using leaf values that are 64 bytes long prior to
 * hashing, or use a hash function other than keccak256 for hashing leaves.
 * This is because the concatenation of a sorted pair of internal nodes in
 * the merkle tree could be reinterpreted as a leaf value.
 * OpenZeppelin's JavaScript library generates merkle trees that are safe
 * against this attack out of the box.
 */
library MerkleProof {
    /**
     * @dev Returns true if a `leaf` can be proved to be a part of a Merkle tree
     * defined by `root`. For this, a `proof` must be provided, containing
     * sibling hashes on the branch from the leaf to the root of the tree. Each
     * pair of leaves and each pair of pre-images are assumed to be sorted.
     */
    function verify(bytes32[] memory proof, bytes32 root, bytes32 leaf) internal pure returns (bool) {
        return processProof(proof, leaf) == root;
    }

    /**
     * @dev Calldata version of {verify}
     *
     * _Available since v4.7._
     */
    function verifyCalldata(bytes32[] calldata proof, bytes32 root, bytes32 leaf) internal pure returns (bool) {
        return processProofCalldata(proof, leaf) == root;
    }

    /**
     * @dev Returns the rebuilt hash obtained by traversing a Merkle tree up
     * from `leaf` using `proof`. A `proof` is valid if and only if the rebuilt
     * hash matches the root of the tree. When processing the proof, the pairs
     * of leafs & pre-images are assumed to be sorted.
     *
     * _Available since v4.4._
     */
    function processProof(bytes32[] memory proof, bytes32 leaf) internal pure returns (bytes32) {
        bytes32 computedHash = leaf;
        for (uint256 i = 0; i < proof.length; i++) {
            computedHash = _hashPair(computedHash, proof[i]);
        }
        return computedHash;
    }

    /**
     * @dev Calldata version of {processProof}
     *
     * _Available since v4.7._
     */
    function processProofCalldata(bytes32[] calldata proof, bytes32 leaf) internal pure returns (bytes32) {
        bytes32 computedHash = leaf;
        for (uint256 i = 0; i < proof.length; i++) {
            computedHash = _hashPair(computedHash, proof[i]);
        }
        return computedHash;
    }

    /**
     * @dev Returns true if the `leaves` can be simultaneously proven to be a part of a merkle tree defined by
     * `root`, according to `proof` and `proofFlags` as described in {processMultiProof}.
     *
     * CAUTION: Not all merkle trees admit multiproofs. See {processMultiProof} for details.
     *
     * _Available since v4.7._
     */
    function multiProofVerify(
        bytes32[] memory proof,
        bool[] memory proofFlags,
        bytes32 root,
        bytes32[] memory leaves
    ) internal pure returns (bool) {
        return processMultiProof(proof, proofFlags, leaves) == root;
    }

    /**
     * @dev Calldata version of {multiProofVerify}
     *
     * CAUTION: Not all merkle trees admit multiproofs. See {processMultiProof} for details.
     *
     * _Available since v4.7._
     */
    function multiProofVerifyCalldata(
        bytes32[] calldata proof,
        bool[] calldata proofFlags,
        bytes32 root,
        bytes32[] memory leaves
    ) internal pure returns (bool) {
        return processMultiProofCalldata(proof, proofFlags, leaves) == root;
    }

    /**
     * @dev Returns the root of a tree reconstructed from `leaves` and sibling nodes in `proof`. The reconstruction
     * proceeds by incrementally reconstructing all inner nodes by combining a leaf/inner node with either another
     * leaf/inner node or a proof sibling node, depending on whether each `proofFlags` item is true or false
     * respectively.
     *
     * CAUTION: Not all merkle trees admit multiproofs. To use multiproofs, it is sufficient to ensure that: 1) the tree
     * is complete (but not necessarily perfect), 2) the leaves to be proven are in the opposite order they are in the
     * tree (i.e., as seen from right to left starting at the deepest layer and continuing at the next layer).
     *
     * _Available since v4.7._
     */
    function processMultiProof(
        bytes32[] memory proof,
        bool[] memory proofFlags,
        bytes32[] memory leaves
    ) internal pure returns (bytes32 merkleRoot) {
        // This function rebuilds the root hash by traversing the tree up from the leaves. The root is rebuilt by
        // consuming and producing values on a queue. The queue starts with the `leaves` array, then goes onto the
        // `hashes` array. At the end of the process, the last hash in the `hashes` array should contain the root of
        // the merkle tree.
        uint256 leavesLen = leaves.length;
        uint256 totalHashes = proofFlags.length;

        // Check proof validity.
        require(leavesLen + proof.length - 1 == totalHashes, "MerkleProof: invalid multiproof");

        // The xxxPos values are "pointers" to the next value to consume in each array. All accesses are done using
        // `xxx[xxxPos++]`, which return the current value and increment the pointer, thus mimicking a queue's "pop".
        bytes32[] memory hashes = new bytes32[](totalHashes);
        uint256 leafPos = 0;
        uint256 hashPos = 0;
        uint256 proofPos = 0;
        // At each step, we compute the next hash using two values:
        // - a value from the "main queue". If not all leaves have been consumed, we get the next leaf, otherwise we
        //   get the next hash.
        // - depending on the flag, either another value from the "main queue" (merging branches) or an element from the
        //   `proof` array.
        for (uint256 i = 0; i < totalHashes; i++) {
            bytes32 a = leafPos < leavesLen ? leaves[leafPos++] : hashes[hashPos++];
            bytes32 b = proofFlags[i]
                ? (leafPos < leavesLen ? leaves[leafPos++] : hashes[hashPos++])
                : proof[proofPos++];
            hashes[i] = _hashPair(a, b);
        }

        if (totalHashes > 0) {
            unchecked {
                return hashes[totalHashes - 1];
            }
        } else if (leavesLen > 0) {
            return leaves[0];
        } else {
            return proof[0];
        }
    }

    /**
     * @dev Calldata version of {processMultiProof}.
     *
     * CAUTION: Not all merkle trees admit multiproofs. See {processMultiProof} for details.
     *
     * _Available since v4.7._
     */
    function processMultiProofCalldata(
        bytes32[] calldata proof,
        bool[] calldata proofFlags,
        bytes32[] memory leaves
    ) internal pure returns (bytes32 merkleRoot) {
        // This function rebuilds the root hash by traversing the tree up from the leaves. The root is rebuilt by
        // consuming and producing values on a queue. The queue starts with the `leaves` array, then goes onto the
        // `hashes` array. At the end of the process, the last hash in the `hashes` array should contain the root of
        // the merkle tree.
        uint256 leavesLen = leaves.length;
        uint256 totalHashes = proofFlags.length;

        // Check proof validity.
        require(leavesLen + proof.length - 1 == totalHashes, "MerkleProof: invalid multiproof");

        // The xxxPos values are "pointers" to the next value to consume in each array. All accesses are done using
        // `xxx[xxxPos++]`, which return the current value and increment the pointer, thus mimicking a queue's "pop".
        bytes32[] memory hashes = new bytes32[](totalHashes);
        uint256 leafPos = 0;
        uint256 hashPos = 0;
        uint256 proofPos = 0;
        // At each step, we compute the next hash using two values:
        // - a value from the "main queue". If not all leaves have been consumed, we get the next leaf, otherwise we
        //   get the next hash.
        // - depending on the flag, either another value from the "main queue" (merging branches) or an element from the
        //   `proof` array.
        for (uint256 i = 0; i < totalHashes; i++) {
            bytes32 a = leafPos < leavesLen ? leaves[leafPos++] : hashes[hashPos++];
            bytes32 b = proofFlags[i]
                ? (leafPos < leavesLen ? leaves[leafPos++] : hashes[hashPos++])
                : proof[proofPos++];
            hashes[i] = _hashPair(a, b);
        }

        if (totalHashes > 0) {
            unchecked {
                return hashes[totalHashes - 1];
            }
        } else if (leavesLen > 0) {
            return leaves[0];
        } else {
            return proof[0];
        }
    }

    function _hashPair(bytes32 a, bytes32 b) private pure returns (bytes32) {
        return a < b ? _efficientHash(a, b) : _efficientHash(b, a);
    }

    function _efficientHash(bytes32 a, bytes32 b) private pure returns (bytes32 value) {
        /// @solidity memory-safe-assembly
        assembly {
            mstore(0x00, a)
            mstore(0x20, b)
            value := keccak256(0x00, 0x40)
        }
    }
}

File 5 of 7 : AnonymiceLibrary.sol
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.0;

library AnonymiceLibrary {
    string internal constant TABLE =
        "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/";

    function encode(bytes memory data) internal pure returns (string memory) {
        if (data.length == 0) return "";

        // load the table into memory
        string memory table = TABLE;

        // multiply by 4/3 rounded up
        uint256 encodedLen = 4 * ((data.length + 2) / 3);

        // add some extra buffer at the end required for the writing
        string memory result = new string(encodedLen + 32);

        assembly {
            // set the actual output length
            mstore(result, encodedLen)

            // prepare the lookup table
            let tablePtr := add(table, 1)

            // input ptr
            let dataPtr := data
            let endPtr := add(dataPtr, mload(data))

            // result ptr, jump over length
            let resultPtr := add(result, 32)

            // run over the input, 3 bytes at a time
            for {

            } lt(dataPtr, endPtr) {

            } {
                dataPtr := add(dataPtr, 3)

                // read 3 bytes
                let input := mload(dataPtr)

                // write 4 characters
                mstore(
                    resultPtr,
                    shl(248, mload(add(tablePtr, and(shr(18, input), 0x3F))))
                )
                resultPtr := add(resultPtr, 1)
                mstore(
                    resultPtr,
                    shl(248, mload(add(tablePtr, and(shr(12, input), 0x3F))))
                )
                resultPtr := add(resultPtr, 1)
                mstore(
                    resultPtr,
                    shl(248, mload(add(tablePtr, and(shr(6, input), 0x3F))))
                )
                resultPtr := add(resultPtr, 1)
                mstore(
                    resultPtr,
                    shl(248, mload(add(tablePtr, and(input, 0x3F))))
                )
                resultPtr := add(resultPtr, 1)
            }

            // padding with '='
            switch mod(mload(data), 3)
            case 1 {
                mstore(sub(resultPtr, 2), shl(240, 0x3d3d))
            }
            case 2 {
                mstore(sub(resultPtr, 1), shl(248, 0x3d))
            }
        }

        return result;
    }

    /**
     * @dev Converts a `uint256` to its ASCII `string` decimal representation.
     */
    function toString(uint256 value) internal pure returns (string memory) {
        if (value == 0) {
            return "0";
        }
        uint256 temp = value;
        uint256 digits;
        while (temp != 0) {
            digits++;
            temp /= 10;
        }
        bytes memory buffer = new bytes(digits);
        while (value != 0) {
            digits -= 1;
            buffer[digits] = bytes1(uint8(48 + uint256(value % 10)));
            value /= 10;
        }
        return string(buffer);
    }

    function parseInt(string memory _a)
        internal
        pure
        returns (uint8 _parsedInt)
    {
        bytes memory bresult = bytes(_a);
        uint8 mint = 0;
        for (uint8 i = 0; i < bresult.length; i++) {
            if (
                (uint8(uint8(bresult[i])) >= 48) &&
                (uint8(uint8(bresult[i])) <= 57)
            ) {
                mint *= 10;
                mint += uint8(bresult[i]) - 48;
            }
        }
        return mint;
    }

    function substring(
        string memory str,
        uint256 startIndex,
        uint256 endIndex
    ) internal pure returns (string memory) {
        bytes memory strBytes = bytes(str);
        bytes memory result = new bytes(endIndex - startIndex);
        for (uint256 i = startIndex; i < endIndex; i++) {
            result[i - startIndex] = strBytes[i];
        }
        return string(result);
    }

    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;
        assembly {
            size := extcodesize(account)
        }
        return size > 0;
    }
}

File 6 of 7 : ChainWavesGenerator.sol
// SPDX-License-Identifier: UNLICENSED
pragma solidity ^0.8.13;

import "./AnonymiceLibrary.sol";

contract ChainWavesGenerator {
    using AnonymiceLibrary for uint8;

    string[][9] private PALETTES;
    uint256[] private NOISE;
    uint256[] private SPEED;
    string[] private CHARS;
    uint256[] private TIGHTNESS;

    struct Traits {
        string[] palette;
        uint256 noise;
        uint256 speed;
        string charSet;
        uint256 tightness;
        uint256 numCols;
    }

    constructor() {
        //lava
        PALETTES[0] = ["d00000", "370617", "faa307", "e85d04", "03071e"];
        //flamingo
        PALETTES[1] = ["3a0ca3", "f72585", "4cc9f0", "7209b7", "4cc9f0"];
        //rioja
        PALETTES[2] = ["250902", "38040e", "640d14", "800e13", "ad2831"];
        //forest
        PALETTES[3] = ["013026", "a1ce3f", "107e57", "014760", "cbe58e"];
        //samba
        PALETTES[4] = ["009638", "F6D800", "002672", "fff", "f8961e"];
        //pepewaves
        PALETTES[5] = ["23B024", "F02423", "294AF6", "fff", "000"];
        //cow
        PALETTES[6] = ["aabf98", "1f1f1f", "f2f2f2", "b5caa3", "20251e"];
        //pastelize
        PALETTES[7] = ["7067cf", "b7c0ee", "cbf3d2", "f87575", "ef626c"];
        //dank
        PALETTES[8] = ["414Cb3", "06061a", "e80663", "fff", "ff0066"];

        NOISE = [20, 35, 55, 85];

        SPEED = [95, 75, 50, 25];

        CHARS = ["#83!:", "@94?;", "W72a+", "N$50c", "0101/", "gm;)'"];

        TIGHTNESS = [2, 3, 5];
    }

    struct Palette {
        bytes3 bg;
        bytes3 colOne;
        bytes3 colTwo;
    }

    function buildLine(
        string memory _chars,
        uint256 _modJump,
        uint8 _x,
        uint8 _y
    ) public pure returns (string memory lineOut) {
        bytes memory byteChars = bytes(_chars);

        uint256 randomModulo = 1;
        lineOut = string(
            abi.encodePacked(
                "<text x ='-",
                _x.toString(),
                "' y='",
                _y.toString(),
                "'>"
            )
        );
        for (uint256 i; i < 12; ++i) {
            string memory charChoice = string(
                abi.encodePacked(byteChars[randomModulo % 4])
            );
            lineOut = string(abi.encodePacked(lineOut, charChoice));
            randomModulo += _modJump;
        }
        lineOut = string(abi.encodePacked(lineOut, "</text>"));
    }

    function buildXLines(
        string memory _chars,
        uint256 _modStart,
        uint256 numLines
    ) public pure returns (string memory lineOut) {
        uint8 x = 1;
        uint8 y;
        for (uint256 i; i < numLines; ++i) {
            lineOut = string(
                abi.encodePacked(lineOut, buildLine(_chars, _modStart, x, y))
            );
            _modStart += 7;
            y += 4;
            if (x == 1) {
                x = 3;
            } else {
                x = 1;
            }
        }
    }

    function buildSVG(uint256 _tokenId, string memory _hash)
        public
        view
        returns (string memory _svg)
    {
        // get traits from id
        Traits memory tokenTraits = buildTraits(_hash);

        uint256 modStart = tokenTraits.noise + tokenTraits.tightness;
        _svg = string(
            abi.encodePacked(
                tokenTraits.palette[0],
                "'/><defs><g id='chars' font-family='monospace'>",
                buildXLines(
                    tokenTraits.charSet,
                    modStart,
                    10 - tokenTraits.numCols
                ),
                "<animate attributeName='font-size' attributeType='XML' values='100%;",
                AnonymiceLibrary.toString(tokenTraits.speed),
                "%;100%' begin='0s' dur='15s' repeatCount='indefinite'/></g><filter id='turbulence'><feTurbulence type='turbulence' baseFrequency='0.",
                AnonymiceLibrary.toString(tokenTraits.noise),
                "' numOctaves='",
                AnonymiceLibrary.toString(tokenTraits.tightness),
                "' result='noise' seed='",
                AnonymiceLibrary.toString(_tokenId),
                buildUseLines(tokenTraits.palette, tokenTraits.numCols)
            )
        );
    }

    function buildTraits(string memory _hash)
        public
        view
        returns (Traits memory tokenTraits)
    {
        uint256[] memory traitArray = new uint256[](6);

        for (uint256 i; i < 6; ++i) {
            traitArray[i] = AnonymiceLibrary.parseInt(
                AnonymiceLibrary.substring(_hash, i, i + 1)
            );
        }
        tokenTraits = Traits(
            PALETTES[traitArray[0]],
            NOISE[traitArray[1]],
            SPEED[traitArray[2]],
            CHARS[traitArray[3]],
            TIGHTNESS[traitArray[4]],
            traitArray[5] + 1
        );
        // Go palettes array and return this palette
    }

    function buildUseLines(string[] memory _pal, uint256 _numCols)
        internal
        pure
        returns (string memory output)
    {
        output = "'/><feDisplacementMap in='SourceGraphic' in2='noise' scale='3' /></filter></defs>";
        uint256 y;

        for (uint256 i; i < _numCols; ++i) {
            output = string(
                abi.encodePacked(
                    output,
                    "<use href='#chars' y='",
                    AnonymiceLibrary.toString(y),
                    "' x='0' filter='url(#turbulence)' width='20' height='20' fill='#",
                    _pal[i + 1],
                    "'/>"
                )
            );

            y += 3;
        }
    }
}

File 7 of 7 : ChainWavesErrors.sol
// SPDX-License-Identifier: UNLICENSED
pragma solidity ^0.8.13;

interface ChainWavesErrors {
    error SoldOut();
    error NotLive();
    error MintPrice();
    error MaxThree();
    error PublicMinted();
    error SnowcrashMinted();
    error NotToad();
    error FreeMintDone();
    error NotSnowcrashList();
    error ReserveClosed();
    error SelfMintOnly();
    error ArrayLengths();
    error NonExistantId();
    error Stap();
    error WithdrawFail();
    error MintZero();
}

Settings
{
  "remappings": [
    "ds-test/=lib/forge-std/lib/ds-test/src/",
    "erc4626-tests/=lib/openzeppelin-contracts/lib/erc4626-tests/",
    "forge-std/=lib/forge-std/src/",
    "openzeppelin-contracts/=lib/openzeppelin-contracts/contracts/",
    "solmate/=lib/solmate/src/",
    "src/=src/",
    "test/=test/",
    "script/=script/"
  ],
  "optimizer": {
    "enabled": true,
    "runs": 200
  },
  "metadata": {
    "bytecodeHash": "ipfs"
  },
  "outputSelection": {
    "*": {
      "*": [
        "evm.bytecode",
        "evm.deployedBytecode",
        "devdoc",
        "userdoc",
        "metadata",
        "abi"
      ]
    }
  },
  "evmVersion": "london",
  "libraries": {}
}

Contract Security Audit

Contract ABI

[{"inputs":[],"stateMutability":"nonpayable","type":"constructor"},{"inputs":[],"name":"ArrayLengths","type":"error"},{"inputs":[],"name":"FreeMintDone","type":"error"},{"inputs":[],"name":"MaxThree","type":"error"},{"inputs":[],"name":"MintPrice","type":"error"},{"inputs":[],"name":"MintZero","type":"error"},{"inputs":[],"name":"NonExistantId","type":"error"},{"inputs":[],"name":"NotLive","type":"error"},{"inputs":[],"name":"NotSnowcrashList","type":"error"},{"inputs":[],"name":"NotToad","type":"error"},{"inputs":[],"name":"PublicMinted","type":"error"},{"inputs":[],"name":"ReserveClosed","type":"error"},{"inputs":[],"name":"SelfMintOnly","type":"error"},{"inputs":[],"name":"SnowcrashMinted","type":"error"},{"inputs":[],"name":"SoldOut","type":"error"},{"inputs":[],"name":"Stap","type":"error"},{"inputs":[],"name":"WithdrawFail","type":"error"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"owner","type":"address"},{"indexed":true,"internalType":"address","name":"spender","type":"address"},{"indexed":true,"internalType":"uint256","name":"id","type":"uint256"}],"name":"Approval","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"owner","type":"address"},{"indexed":true,"internalType":"address","name":"operator","type":"address"},{"indexed":false,"internalType":"bool","name":"approved","type":"bool"}],"name":"ApprovalForAll","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"user","type":"address"},{"indexed":true,"internalType":"address","name":"newOwner","type":"address"}],"name":"OwnershipTransferred","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"from","type":"address"},{"indexed":true,"internalType":"address","name":"to","type":"address"},{"indexed":true,"internalType":"uint256","name":"id","type":"uint256"}],"name":"Transfer","type":"event"},{"inputs":[],"name":"MAX_MINT","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"MAX_SUPPLY","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"MINTING_LIVE","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"MINT_PRICE","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"MINT_START","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"_tokenId","type":"uint256"}],"name":"_tokenIdToHash","outputs":[{"internalType":"string","name":"tokenHash","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"_traitTypeIndex","type":"uint256"},{"components":[{"internalType":"string","name":"traitName","type":"string"},{"internalType":"string","name":"traitType","type":"string"}],"internalType":"struct ChainWaves.Trait[]","name":"traits","type":"tuple[]"}],"name":"addTraitType","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[{"internalType":"address","name":"spender","type":"address"},{"internalType":"uint256","name":"id","type":"uint256"}],"name":"approve","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"owner","type":"address"}],"name":"balanceOf","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"flipMint","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[{"internalType":"address[]","name":"_addresses","type":"address[]"},{"internalType":"uint256[]","name":"_amount","type":"uint256[]"}],"name":"freeMints","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"}],"name":"getApproved","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"string","name":"_hash","type":"string"}],"name":"hashToMetadata","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"},{"internalType":"address","name":"","type":"address"}],"name":"isApprovedForAll","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"name","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"_amount","type":"uint256"}],"name":"normieMint","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[],"name":"owner","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"id","type":"uint256"}],"name":"ownerOf","outputs":[{"internalType":"address","name":"owner","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"from","type":"address"},{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"id","type":"uint256"}],"name":"safeTransferFrom","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"from","type":"address"},{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"id","type":"uint256"},{"internalType":"bytes","name":"data","type":"bytes"}],"name":"safeTransferFrom","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"operator","type":"address"},{"internalType":"bool","name":"approved","type":"bool"}],"name":"setApprovalForAll","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"bytes32[]","name":"merkleProof","type":"bytes32[]"}],"name":"snowcrashMint","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[],"name":"snowcrashReserve","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"bytes4","name":"interfaceId","type":"bytes4"}],"name":"supportsInterface","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"symbol","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"_tokenId","type":"uint256"}],"name":"tokenURI","outputs":[{"internalType":"string","name":"_URI","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"totalSupply","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"},{"internalType":"uint256","name":"","type":"uint256"}],"name":"traitTypes","outputs":[{"internalType":"string","name":"traitName","type":"string"},{"internalType":"string","name":"traitType","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"from","type":"address"},{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"id","type":"uint256"}],"name":"transferFrom","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"newOwner","type":"address"}],"name":"transferOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"wipeSnowcrashReserve","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[],"name":"withdraw","outputs":[],"stateMutability":"payable","type":"function"}]

608060405260786007553480156200001657600080fd5b50604080518082018252600a815269436861696e576176657360b01b602080830191825283518085019094526002845261434160f01b90840152815173b6ee8b1899e4cad7e28015995b82969e44bd0bb09391620000789160009190620002b8565b5080516200008e906001906020840190620002b8565b5050600680546001600160a01b0319166001600160a01b0384169081179091556040519091506000907f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e0908290a350604051620000eb9062000347565b604051809103906000f08015801562000108573d6000803e3d6000fd5b50600e80546001600160a01b0319166001600160a01b039290921691909117905560408051610120810182526103e881526105dc6020820152610578918101919091526106a460608201526104b0608082015261019060a0820181905260c082015261064060e08201526103206101008201526200018b90600f90600962000355565b50604080516080810182526103e8808252610fa060208301819052928201929092526060810191909152620001c590601090600462000355565b50604080516080810182526103e8808252610fa060208301819052928201929092526060810191909152620001ff90601190600462000355565b506040805160c0810182526108ca808252602082018190529181018290526060810191909152610258608082015261019060a08201526200024590601290600662000355565b50604080516060810182526103e881526117706020820152610bb8918101919091526200027790601390600362000355565b506040805160808101825261032081526118386020820152610a28918101919091526101906060820152620002b190601490600462000355565b506200044e565b828054620002c69062000412565b90600052602060002090601f016020900481019282620002ea576000855562000335565b82601f106200030557805160ff191683800117855562000335565b8280016001018555821562000335579182015b828111156200033557825182559160200191906001019062000318565b5062000343929150620003fb565b5090565b611ccf806200323783390190565b82805482825590600052602060002090600f01601090048101928215620003355791602002820160005b83821115620003c157835183826101000a81548161ffff021916908361ffff16021790555092602001926002016020816001010492830192600103026200037f565b8015620003f15782816101000a81549061ffff0219169055600201602081600101049283019260010302620003c1565b5050620003439291505b5b80821115620003435760008155600101620003fc565b600181811c908216806200042757607f821691505b6020821081036200044857634e487b7160e01b600052602260045260246000fd5b50919050565b612dd9806200045e6000396000f3fe6080604052600436106101e25760003560e01c806366e3387011610102578063c6374d0c11610095578063e985e9c511610064578063e985e9c51461052e578063f0292a0314610569578063f2fde38b1461057e578063f756b53e1461059e57600080fd5b8063c6374d0c146104db578063c87b56dd146104f3578063d06aec2114610513578063d2ed5c591461052657600080fd5b806395d89b41116100d157806395d89b411461046b578063a22cb46514610480578063b88d4fde146104a0578063c002d23d146104c057600080fd5b806366e33870146103f157806370a082311461041157806372f90ac1146104315780638da5cb5b1461044b57600080fd5b806323b872dd1161017a5780633ccfd60b116101495780633ccfd60b1461039657806342842e0e1461039e57806352f0dc9b146103be5780636352211e146103d157600080fd5b806323b872dd1461031f5780632fb098d21461033f57806332cb6b0c1461036d578063349d27481461038357600080fd5b806308338e4b116101b657806308338e4b146102b057806308871ceb146102d4578063095ea7b3146102e957806318160ddd1461030957600080fd5b80625ea307146101e757806301ffc9a71461021d57806306fdde031461024d578063081812fc14610262575b600080fd5b3480156101f357600080fd5b50610207610202366004612094565b6105a6565b6040516102149190612105565b60405180910390f35b34801561022957600080fd5b5061023d610238366004612131565b61060f565b6040519015158152602001610214565b34801561025957600080fd5b5061020761065d565b34801561026e57600080fd5b5061029861027d366004612094565b6004602052600090815260409020546001600160a01b031681565b6040516001600160a01b039091168152602001610214565b3480156102bc57600080fd5b506102c660075481565b604051908152602001610214565b6102e76102e2366004612199565b6106eb565b005b3480156102f557600080fd5b506102e76103043660046121f1565b61088c565b34801561031557600080fd5b506102c660095481565b34801561032b57600080fd5b506102e761033a36600461221b565b61096e565b34801561034b57600080fd5b5061035f61035a366004612257565b610b35565b604051610214929190612279565b34801561037957600080fd5b506102c661020081565b6102e7610391366004612392565b610c86565b6102e7610d7f565b3480156103aa57600080fd5b506102e76103b936600461221b565b610f74565b6102e76103cc366004612094565b611067565b3480156103dd57600080fd5b506102986103ec366004612094565b61111d565b3480156103fd57600080fd5b5061020761040c3660046124ba565b611174565b34801561041d57600080fd5b506102c661042c3660046124ee565b6112e6565b34801561043d57600080fd5b5060085461023d9060ff1681565b34801561045757600080fd5b50600654610298906001600160a01b031681565b34801561047757600080fd5b50610207611349565b34801561048c57600080fd5b506102e761049b366004612509565b611356565b3480156104ac57600080fd5b506102e76104bb366004612545565b6113c2565b3480156104cc57600080fd5b506102c6665af3107a40000081565b3480156104e757600080fd5b506102c66363c99a3881565b3480156104ff57600080fd5b5061020761050e366004612094565b6114aa565b6102e76105213660046125df565b6115af565b6102e7611696565b34801561053a57600080fd5b5061023d61054936600461264a565b600560209081526000928352604080842090915290825290205460ff1681565b34801561057557600080fd5b506102c6600381565b34801561058a57600080fd5b506102e76105993660046124ee565b6116d4565b6102e761174a565b336000908152600b60205260409020546060904360089190911c036105de5760405163a5e0079160e01b815260040160405180910390fd5b60095482106106005760405163732f483160e11b815260040160405180910390fd5b6106098261177b565b92915050565b60006301ffc9a760e01b6001600160e01b03198316148061064057506380ac58cd60e01b6001600160e01b03198316145b806106095750506001600160e01b031916635b5e139f60e01b1490565b6000805461066a9061267d565b80601f01602080910402602001604051908101604052809291908181526020018280546106969061267d565b80156106e35780601f106106b8576101008083540402835291602001916106e3565b820191906000526020600020905b8154815290600101906020018083116106c657829003601f168201915b505050505081565b6040516bffffffffffffffffffffffff193360601b1660208201526000906034016040516020818303038152906040528051906020012090506107848383808060200260200160405190810160405280939291908181526020018383602002808284376000920191909152507fea35e50958ff75fe96e04a6dd792de75a26dd0c2a2d12e8a4c485d938961eb3992508591506118519050565b6107c15760405162461bcd60e51b8152602060048201526009602482015268139bdd081bdb8815d360ba1b60448201526064015b60405180910390fd5b665af3107a40000034146107e857604051637edfcf7960e11b815260040160405180910390fd5b336000908152600b6020526040902054600f600482901c161561081e57604051633eae439d60e01b815260040160405180910390fd5b600754600003610840576040516298598d60e61b815260040160405180910390fd5b60076000815461084f906126cd565b909155506010176108664360081b60ff83166126e4565b336000818152600b60205260409020829055909150610886906001611867565b50505050565b6000818152600260205260409020546001600160a01b0316338114806108d557506001600160a01b038116600090815260056020908152604080832033845290915290205460ff165b6109125760405162461bcd60e51b815260206004820152600e60248201526d1393d517d055551213d49256915160921b60448201526064016107b8565b60008281526004602052604080822080546001600160a01b0319166001600160a01b0387811691821790925591518593918516917f8c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b92591a4505050565b6000818152600260205260409020546001600160a01b038481169116146109c45760405162461bcd60e51b815260206004820152600a60248201526957524f4e475f46524f4d60b01b60448201526064016107b8565b6001600160a01b038216610a0e5760405162461bcd60e51b81526020600482015260116024820152701253959053125117d49150d25412515395607a1b60448201526064016107b8565b336001600160a01b0384161480610a4857506001600160a01b038316600090815260056020908152604080832033845290915290205460ff165b80610a6957506000818152600460205260409020546001600160a01b031633145b610aa65760405162461bcd60e51b815260206004820152600e60248201526d1393d517d055551213d49256915160921b60448201526064016107b8565b6001600160a01b0380841660008181526003602090815260408083208054600019019055938616808352848320805460010190558583526002825284832080546001600160a01b03199081168317909155600490925284832080549092169091559251849392917fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef91a4505050565b600d6020528160005260406000208181548110610b5157600080fd5b906000526020600020906002020160009150915050806000018054610b759061267d565b80601f0160208091040260200160405190810160405280929190818152602001828054610ba19061267d565b8015610bee5780601f10610bc357610100808354040283529160200191610bee565b820191906000526020600020905b815481529060010190602001808311610bd157829003601f168201915b505050505090806001018054610c039061267d565b80601f0160208091040260200160405190810160405280929190818152602001828054610c2f9061267d565b8015610c7c5780601f10610c5157610100808354040283529160200191610c7c565b820191906000526020600020905b815481529060010190602001808311610c5f57829003601f168201915b5050505050905082565b6006546001600160a01b03163314610cb05760405162461bcd60e51b81526004016107b8906126fc565b60005b8151811015610d7957600d60008481526020019081526020016000206040518060400160405280848481518110610cec57610cec612722565b6020026020010151600001518152602001848481518110610d0f57610d0f612722565b602090810291909101810151810151909152825460018101845560009384529281902082518051939460020290910192610d4c9284920190611ffb565b506020828101518051610d659260018501920190611ffb565b50505080610d7290612738565b9050610cb3565b505b5050565b6006546001600160a01b03163314610da95760405162461bcd60e51b81526004016107b8906126fc565b6000610db6606447612767565b610dc190600c61277b565b90506000610dd0606447612767565b610ddb90605361277b565b90506000610dea606447612767565b610df590600561277b565b604051909150600090734533d1f65906368ebfd61259daee561df3f3559d9085908381818185875af1925050503d8060008114610e4e576040519150601f19603f3d011682016040523d82523d6000602084013e610e53565b606091505b5050905080610e745760405162c0f29960e01b815260040160405180910390fd5b60405160009073888f8aa938dbb18b28bdd111fa4a0d3b8e10c8719084908381818185875af1925050503d8060008114610eca576040519150601f19603f3d011682016040523d82523d6000602084013e610ecf565b606091505b5050905080610ef05760405162c0f29960e01b815260040160405180910390fd5b60405160009073e4260df86f5261a41d19c2066f1eb2eb4f009e849086908381818185875af1925050503d8060008114610f46576040519150601f19603f3d011682016040523d82523d6000602084013e610f4b565b606091505b5050905080610f6c5760405162c0f29960e01b815260040160405180910390fd5b505050505050565b610f7f83838361096e565b6001600160a01b0382163b15806110285750604051630a85bd0160e11b8082523360048301526001600160a01b03858116602484015260448301849052608060648401526000608484015290919084169063150b7a029060a4016020604051808303816000875af1158015610ff8573d6000803e3d6000fd5b505050506040513d601f19601f8201168201806040525081019061101c919061279a565b6001600160e01b031916145b610d795760405162461bcd60e51b815260206004820152601060248201526f155394d0519157d49150d2541251539560821b60448201526064016107b8565b60038111156110895760405163355ad2e760e01b815260040160405180910390fd5b61109a81665af3107a40000061277b565b34146110b957604051637edfcf7960e11b815260040160405180910390fd5b336000908152600b6020526040902054600f8116156110eb5760405163317f85f760e21b815260040160405180910390fd5b6001176110fe4360081b60ff83166126e4565b336000818152600b60205260409020829055909150610d7b9083611867565b6000818152600260205260409020546001600160a01b03168061116f5760405162461bcd60e51b815260206004820152600a6024820152691393d517d3525395115160b21b60448201526064016107b8565b919050565b336000908152600b60205260409020546060904360089190911c036111ac5760405163a5e0079160e01b815260040160405180910390fd5b606060005b60068160ff1610156112bd5760006111e26111dd8660ff85166111d58660016127b7565b60ff1661194d565b611a19565b905082600d60008460ff1681526020019081526020016000208260ff168154811061120f5761120f612722565b9060005260206000209060020201600101600d60008560ff1681526020019081526020016000208360ff168154811061124a5761124a612722565b906000526020600020906002020160000160405160200161126d93929190612875565b60405160208183030381529060405292508160ff166005146112ac578260405160200161129a91906128e2565b60405160208183030381529060405292505b506112b681612907565b90506111b1565b50806040516020016112cf9190612926565b604051602081830303815290604052915050919050565b60006001600160a01b03821661132d5760405162461bcd60e51b815260206004820152600c60248201526b5a45524f5f4144445245535360a01b60448201526064016107b8565b506001600160a01b031660009081526003602052604090205490565b6001805461066a9061267d565b3360008181526005602090815260408083206001600160a01b03871680855290835292819020805460ff191686151590811790915590519081529192917f17307eab39ab6107e8899845ad3d59bd9653f200f220920489ca2b5937696c31910160405180910390a35050565b6113cd85858561096e565b6001600160a01b0384163b15806114645750604051630a85bd0160e11b808252906001600160a01b0386169063150b7a02906114159033908a9089908990899060040161295a565b6020604051808303816000875af1158015611434573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190611458919061279a565b6001600160e01b031916145b6114a35760405162461bcd60e51b815260206004820152601060248201526f155394d0519157d49150d2541251539560821b60448201526064016107b8565b5050505050565b606060095482106114ce5760405163732f483160e11b815260040160405180910390fd5b60006114d9836105a6565b905061159f6114e784611ad7565b600e54604051630e4b241160e11b8152611584916001600160a01b031690631c9648229061151b90899088906004016129ae565b600060405180830381865afa158015611538573d6000803e3d6000fd5b505050506040513d6000823e601f3d908101601f1916820160405261156091908101906129c7565b6040516020016115709190612a34565b604051602081830303815290604052611bdf565b61158d84611174565b60405160200161157093929190612b21565b6040516020016112cf9190612c5b565b6006546001600160a01b031633146115d95760405162461bcd60e51b81526004016107b8906126fc565b600a5460ff16156115fd5760405163eddaeeb360e01b815260040160405180910390fd5b8281811461161e57604051637faa19c760e01b815260040160405180910390fd5b60005b818110156116815761167186868381811061163e5761163e612722565b905060200201602081019061165391906124ee565b85858481811061166557611665612722565b90506020020135611867565b61167a81612738565b9050611621565b5050600a805460ff1916600117905550505050565b6006546001600160a01b031633146116c05760405162461bcd60e51b81526004016107b8906126fc565b6008805460ff19811660ff90911615179055565b6006546001600160a01b031633146116fe5760405162461bcd60e51b81526004016107b8906126fc565b600680546001600160a01b0319166001600160a01b03831690811790915560405133907f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e090600090a350565b6006546001600160a01b031633146117745760405162461bcd60e51b81526004016107b8906126fc565b6000600755565b60408051602080820183526000808352848152600c909152918220546060925b60068160ff16101561184857600061271083836040516020016117d592919091825260f81b6001600160f81b031916602082015260210190565b6040516020818303038152906040528051906020012060001c6117f89190612ca0565b90508361181461180c8361ffff1685611d45565b60ff16611ad7565b604051602001611825929190612cb4565b6040516020818303038152906040529350508061184190612907565b905061179b565b50909392505050565b60008261185e8584611e19565b14949350505050565b60085460ff16158061187c57506363c99a3842105b1561189a5760405163baf13b3f60e01b815260040160405180910390fd5b806000036118bb5760405163220328c760e11b815260040160405180910390fd5b610200600754826009546118cf91906126e4565b6118d991906126e4565b11156118f8576040516352df9fe560e01b815260040160405180910390fd5b600954600061190783836126e4565b90505b808210156119455761191c8483611e5e565b6000838152600c60205260409020556119358483611ebe565b61193e82612738565b915061190a565b600955505050565b606083600061195c8585612ce3565b6001600160401b03811115611973576119736122a7565b6040519080825280601f01601f19166020018201604052801561199d576020820181803683370190505b509050845b84811015611a0f578281815181106119bc576119bc612722565b01602001516001600160f81b031916826119d68884612ce3565b815181106119e6576119e6612722565b60200101906001600160f81b031916908160001a90535080611a0781612738565b9150506119a2565b5095945050505050565b60008181805b82518160ff161015611acf576030838260ff1681518110611a4257611a42612722565b016020015160f81c10801590611a7557506039838260ff1681518110611a6a57611a6a612722565b016020015160f81c11155b15611abd57611a85600a83612cfa565b91506030838260ff1681518110611a9e57611a9e612722565b0160200151611ab0919060f81c612d23565b611aba90836127b7565b91505b80611ac781612907565b915050611a1f565b509392505050565b606081600003611afe5750506040805180820190915260018152600360fc1b602082015290565b8160005b8115611b285780611b1281612738565b9150611b219050600a83612767565b9150611b02565b6000816001600160401b03811115611b4257611b426122a7565b6040519080825280601f01601f191660200182016040528015611b6c576020820181803683370190505b5090505b8415611bd757611b81600183612ce3565b9150611b8e600a86612ca0565b611b999060306126e4565b60f81b818381518110611bae57611bae612722565b60200101906001600160f81b031916908160001a905350611bd0600a86612767565b9450611b70565b949350505050565b60608151600003611bfe57505060408051602081019091526000815290565b6000604051806060016040528060408152602001612d646040913990506000600384516002611c2d91906126e4565b611c379190612767565b611c4290600461277b565b90506000611c518260206126e4565b6001600160401b03811115611c6857611c686122a7565b6040519080825280601f01601f191660200182016040528015611c92576020820181803683370190505b509050818152600183018586518101602084015b81831015611d005760039283018051603f601282901c811687015160f890811b8552600c83901c8216880151811b6001860152600683901c8216880151811b60028601529116860151901b93820193909352600401611ca6565b600389510660018114611d1a5760028114611d2b57611d37565b613d3d60f01b600119830152611d37565b603d60f81b6000198301525b509398975050505050505050565b6000806000600f8460ff1660068110611d6057611d60612722565b0154905060005b818160ff161015611e13576000600f8660ff1660068110611d8a57611d8a612722565b018260ff1681548110611d9f57611d9f612722565b90600052602060002090601091828204019190066002029054906101000a900461ffff1690508361ffff168710158015611de55750611dde8185612d46565b61ffff1687105b15611df557509250610609915050565b611dff8185612d46565b93505080611e0c90612907565b9050611d67565b50600080fd5b600081815b8451811015611acf57611e4a82868381518110611e3d57611e3d612722565b6020026020010151611fc9565b915080611e5681612738565b915050611e1e565b600042448484604051602001611e9f9493929190938452602084019290925260601b6bffffffffffffffffffffffff19166040830152605482015260740190565b60408051601f1981840301815291905280516020909101209392505050565b6001600160a01b038216611f085760405162461bcd60e51b81526020600482015260116024820152701253959053125117d49150d25412515395607a1b60448201526064016107b8565b6000818152600260205260409020546001600160a01b031615611f5e5760405162461bcd60e51b815260206004820152600e60248201526d1053149150511657d3525395115160921b60448201526064016107b8565b6001600160a01b038216600081815260036020908152604080832080546001019055848352600290915280822080546001600160a01b0319168417905551839291907fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef908290a45050565b6000818310611fe5576000828152602084905260409020611ff4565b60008381526020839052604090205b9392505050565b8280546120079061267d565b90600052602060002090601f016020900481019282612029576000855561206f565b82601f1061204257805160ff191683800117855561206f565b8280016001018555821561206f579182015b8281111561206f578251825591602001919060010190612054565b5061207b92915061207f565b5090565b5b8082111561207b5760008155600101612080565b6000602082840312156120a657600080fd5b5035919050565b60005b838110156120c85781810151838201526020016120b0565b838111156108865750506000910152565b600081518084526120f18160208601602086016120ad565b601f01601f19169290920160200192915050565b602081526000611ff460208301846120d9565b6001600160e01b03198116811461212e57600080fd5b50565b60006020828403121561214357600080fd5b8135611ff481612118565b60008083601f84011261216057600080fd5b5081356001600160401b0381111561217757600080fd5b6020830191508360208260051b850101111561219257600080fd5b9250929050565b600080602083850312156121ac57600080fd5b82356001600160401b038111156121c257600080fd5b6121ce8582860161214e565b90969095509350505050565b80356001600160a01b038116811461116f57600080fd5b6000806040838503121561220457600080fd5b61220d836121da565b946020939093013593505050565b60008060006060848603121561223057600080fd5b612239846121da565b9250612247602085016121da565b9150604084013590509250925092565b6000806040838503121561226a57600080fd5b50508035926020909101359150565b60408152600061228c60408301856120d9565b828103602084015261229e81856120d9565b95945050505050565b634e487b7160e01b600052604160045260246000fd5b604080519081016001600160401b03811182821017156122df576122df6122a7565b60405290565b604051601f8201601f191681016001600160401b038111828210171561230d5761230d6122a7565b604052919050565b60006001600160401b0382111561232e5761232e6122a7565b50601f01601f191660200190565b600082601f83011261234d57600080fd5b813561236061235b82612315565b6122e5565b81815284602083860101111561237557600080fd5b816020850160208301376000918101602001919091529392505050565b60008060408084860312156123a657600080fd5b833592506020808501356001600160401b03808211156123c557600080fd5b818701915087601f8301126123d957600080fd5b8135818111156123eb576123eb6122a7565b8060051b6123fa8582016122e5565b918252838101850191858101908b84111561241457600080fd5b86860192505b838310156124a8578235858111156124325760008081fd5b8601808d03601f19018913156124485760008081fd5b6124506122bd565b88820135878111156124625760008081fd5b6124708f8b8386010161233c565b82525089820135878111156124855760008081fd5b6124938f8b8386010161233c565b828b015250835250918601919086019061241a565b80985050505050505050509250929050565b6000602082840312156124cc57600080fd5b81356001600160401b038111156124e257600080fd5b611bd78482850161233c565b60006020828403121561250057600080fd5b611ff4826121da565b6000806040838503121561251c57600080fd5b612525836121da565b91506020830135801515811461253a57600080fd5b809150509250929050565b60008060008060006080868803121561255d57600080fd5b612566866121da565b9450612574602087016121da565b93506040860135925060608601356001600160401b038082111561259757600080fd5b818801915088601f8301126125ab57600080fd5b8135818111156125ba57600080fd5b8960208285010111156125cc57600080fd5b9699959850939650602001949392505050565b600080600080604085870312156125f557600080fd5b84356001600160401b038082111561260c57600080fd5b6126188883890161214e565b9096509450602087013591508082111561263157600080fd5b5061263e8782880161214e565b95989497509550505050565b6000806040838503121561265d57600080fd5b612666836121da565b9150612674602084016121da565b90509250929050565b600181811c9082168061269157607f821691505b6020821081036126b157634e487b7160e01b600052602260045260246000fd5b50919050565b634e487b7160e01b600052601160045260246000fd5b6000816126dc576126dc6126b7565b506000190190565b600082198211156126f7576126f76126b7565b500190565b6020808252600c908201526b15539055551213d49256915160a21b604082015260600190565b634e487b7160e01b600052603260045260246000fd5b60006001820161274a5761274a6126b7565b5060010190565b634e487b7160e01b600052601260045260246000fd5b60008261277657612776612751565b500490565b6000816000190483118215151615612795576127956126b7565b500290565b6000602082840312156127ac57600080fd5b8151611ff481612118565b600060ff821660ff84168060ff038211156127d4576127d46126b7565b019392505050565b8054600090600181811c90808316806127f657607f831692505b6020808410820361281757634e487b7160e01b600052602260045260246000fd5b81801561282b576001811461283c57612869565b60ff19861689528489019650612869565b60008881526020902060005b868110156128615781548b820152908501908301612848565b505084890196505b50505050505092915050565b600084516128878184602089016120ad565b6e3d913a3930b4ba2fba3cb832911d1160891b9083019081526128ad600f8201866127dc565b6a1116113b30b63ab2911d1160a91b815290506128cd600b8201856127dc565b61227d60f01b81526002019695505050505050565b600082516128f48184602087016120ad565b600b60fa1b920191825250600101919050565b600060ff821660ff810361291d5761291d6126b7565b60010192915050565b605b60f81b8152600082516129428160018501602087016120ad565b605d60f81b6001939091019283015250600201919050565b6001600160a01b038681168252851660208201526040810184905260806060820181905281018290526000828460a0840137600060a0848401015260a0601f19601f85011683010190509695505050505050565b828152604060208201526000611bd760408301846120d9565b6000602082840312156129d957600080fd5b81516001600160401b038111156129ef57600080fd5b8201601f81018413612a0057600080fd5b8051612a0e61235b82612315565b818152856020838501011115612a2357600080fd5b61229e8260208301602086016120ad565b7f3c7376672076696577426f783d27302030203230203230272077696474683d2781527f36303027206865696768743d273630302720786d6c6e733d27687474703a2f2f60208201527f7777772e77332e6f72672f323030302f7376672720707265736572766541737060408201527f656374526174696f3d27784d6964594d696e273e3c726563742077696474683d60608201527f27323027206865696768743d273230272066696c6c3d27230000000000000000608082015260008251612b048160988501602087016120ad565b651e17b9bb339f60d11b6098939091019283015250609e01919050565b757b226e616d65223a2022434834314e5734563335202360501b81528351600090612b538160168501602089016120ad565b7f222c226465736372697074696f6e223a202246756c6c79206f6e636861696e206016918401918201527f67656e65726174697665206172742053564720636f6c6c656374696f6e2e204360368201527f726561746564206279204d63546f616479202620436972636f6c6f72732e220060568201527f2c22696d616765223a2022646174613a696d6167652f7376672b786d6c3b62616075820152641cd94d8d0b60da1b60958201528451612c1081609a8401602089016120ad565b6e11161130ba3a3934b13aba32b9911d60891b609a92909101918201528351612c408160a98401602088016120ad565b607d60f81b60a9929091019182015260aa0195945050505050565b7f646174613a6170706c69636174696f6e2f6a736f6e3b6261736536342c000000815260008251612c9381601d8501602087016120ad565b91909101601d0192915050565b600082612caf57612caf612751565b500690565b60008351612cc68184602088016120ad565b835190830190612cda8183602088016120ad565b01949350505050565b600082821015612cf557612cf56126b7565b500390565b600060ff821660ff84168160ff0481118215151615612d1b57612d1b6126b7565b029392505050565b600060ff821660ff841680821015612d3d57612d3d6126b7565b90039392505050565b600061ffff808316818516808303821115612cda57612cda6126b756fe4142434445464748494a4b4c4d4e4f505152535455565758595a6162636465666768696a6b6c6d6e6f707172737475767778797a303132333435363738392b2fa2646970667358221220a16ad100558bec1aa07c2dbb585111b82addb6abfed4f6b7bf30663d18999e8b64736f6c634300080d003360806040523480156200001157600080fd5b506040805160e081018252600660a082018181526506430303030360d41b60c08401528252825180840184528181526533373036313760d01b60208083019190915280840191909152835180850185528281526566616133303760d01b81830152838501528351808501855282815265194e0d590c0d60d21b81830152606084015283518085019094529083526530333037316560d01b908301526080810191909152620000c4906000906005620007ea565b506040805160e081018252600660a082018181526533613063613360d01b60c08401528252825180840184528181526566373235383560d01b60208083019190915280840191909152835180850185528281526503463633966360d41b81830181905284860191909152845180860186528381526537323039623760d01b8184015260608501528451808601909552918452830152608081019190915262000171906001906005620007ea565b506040805160e081018252600660a0820181815265191a981c981960d11b60c08401528252825180840184528181526533383034306560d01b6020808301919091528084019190915283518085018552828152650d8d0c190c4d60d21b8183015283850152835180850185528281526538303065313360d01b81830152606084015283518085019094529083526561643238333160d01b90830152608081019190915262000224906002906005620007ea565b506040805160e081018252600660a082018181526518189998191b60d11b60c0840152825282518084018452818152653098b1b299b360d11b60208083019190915280840191909152835180850185528281526531303765353760d01b8183015283850152835180850185528281526503031343736360d41b81830152606084015283518085019094529083526563626535386560d01b908301526080810191909152620002d7906003906005620007ea565b506040805160e081018252600660a082018181526506060726c66760d31b60c08401528252825180840184528181526504636443830360d41b6020808301919091528084019190915283518085018552828152651818191b1b9960d11b818301528385015283518085018552600381526233333360e91b81830152606084015283518085019094529083526566383936316560d01b90830152608081019190915262000388906004906005620007ea565b506040518060a00160405280604051806040016040528060068152602001650c8cd08c0c8d60d21b81525081526020016040518060400160405280600681526020016546303234323360d01b815250815260200160405180604001604052806006815260200165191c9a20a31b60d11b81525081526020016040518060400160405280600381526020016233333360e91b81525081526020016040518060400160405280600381526020016203030360ec1b8152508152506000600560098110620004575762000457620009e8565b62000467929101906005620007ea565b506040805160e081018252600660a08201818152650c2c2c4cc72760d31b60c08401528252825180840184528181526518b318b318b360d11b60208083019190915280840191909152835180850185528281526533193319331960d11b8183015283850152835180850185528281526562356361613360d01b81830152606084015283518085019094528184526532303235316560d01b9084015260808201929092526200051891906005620007ea565b506040805160e081018252600660a08201818152651b981b1bb1b360d11b60c08401528252825180840184528181526562376330656560d01b60208083019190915280840191909152835180850185528281526531b13319b21960d11b8183015283850152835180850185528281526566383735373560d01b81830152606084015283518085019094529083526565663632366360d01b908301526080810191909152620005cb906007906005620007ea565b506040805160e081018252600660a082018181526534313443623360d01b60c08401528252825180840184528181526530363036316160d01b60208083019190915280840191909152835180850185528281526565383036363360d01b818301528385015283518085018552600381526233333360e91b818301526060840152835180850190945290835265333318181b1b60d11b9083015260808101919091526200067c906008906005620007ea565b5060408051608081018252601481526023602082015260379181019190915260556060820152620006b29060099060046200084e565b5060408051608081018252605f8152604b602082015260329181019190915260196060820152620006e890600a9060046200084e565b506040805161010081018252600560c0820181815264119c19909d60d91b60e0840152825282518084018452818152644039343f3b60d81b602082810191909152808401919091528351808501855282815264573732612b60d81b818301528385015283518085018552828152644e2435306360d81b8183015260608401528351808501855282815264303130312f60d81b818301526080840152835180850190945290835264676d3b292760d81b9083015260a0810191909152620007b390600b9060066200089f565b506040805160608101825260028152600360208201819052600592820192909252620007e391600c91906200084e565b5062000a3a565b8280548282559060005260206000209081019282156200083c579160200282015b828111156200083c57825180516200082b918491602090910190620008f1565b50916020019190600101906200080b565b506200084a9291506200096e565b5090565b82805482825590600052602060002090810192821562000891579160200282015b8281111562000891578251829060ff169055916020019190600101906200086f565b506200084a9291506200098f565b8280548282559060005260206000209081019282156200083c579160200282015b828111156200083c5782518051620008e0918491602090910190620008f1565b5091602001919060010190620008c0565b828054620008ff90620009fe565b90600052602060002090601f01602090048101928262000923576000855562000891565b82601f106200093e57805160ff191683800117855562000891565b8280016001018555821562000891579182015b828111156200089157825182559160200191906001019062000951565b808211156200084a576000620009858282620009a6565b506001016200096e565b5b808211156200084a576000815560010162000990565b508054620009b490620009fe565b6000825580601f10620009c5575050565b601f016020900490600052602060002090810190620009e591906200098f565b50565b634e487b7160e01b600052603260045260246000fd5b600181811c9082168062000a1357607f821691505b60208210810362000a3457634e487b7160e01b600052602260045260246000fd5b50919050565b6112858062000a4a6000396000f3fe608060405234801561001057600080fd5b506004361061004c5760003560e01c80631c9648221461005157806362c6313e1461007a57806378c120541461008d578063e9a90acc146100a0575b600080fd5b61006461005f366004610a6f565b6100c0565b6040516100719190610b12565b60405180910390f35b610064610088366004610b42565b610191565b61006461009b366004610baa565b61029b565b6100b36100ae366004610bf8565b610328565b6040516100719190610c2d565b606060006100cd83610328565b90506000816080015182602001516100e59190610cf7565b905081600001516000815181106100fe576100fe610d0f565b60200260200101516101208360600151838560a00151600a61009b9190610d25565b61012d8460400151610698565b61013a8560200151610698565b6101478660800151610698565b6101508a610698565b61016288600001518960a001516107a1565b6040516020016101789796959493929190610d58565b6040516020818303038152906040529250505092915050565b60608460016101a260ff8616610698565b6101ae8560ff16610698565b6040516020016101bf929190610f57565b604051602081830303815290604052925060005b600c81101561026e576000836101ea600485610fd3565b815181106101fa576101fa610d0f565b016020908101516040516001600160f81b0319909116918101919091526021016040516020818303038152906040529050848160405160200161023e929190610fe7565b60408051601f19818403018152919052945061025a8884610cf7565b9250508061026790611016565b90506101d3565b5082604051602001610280919061102f565b60405160208183030381529060405292505050949350505050565b606060016000805b8481101561031e57836102b888888686610191565b6040516020016102c9929190610fe7565b60408051601f1981840301815291905293506102e6600787610cf7565b95506102f360048361105a565b91508260ff16600103610309576003925061030e565b600192505b61031781611016565b90506102a3565b5050509392505050565b6103616040518060c001604052806060815260200160008152602001600081526020016060815260200160008152602001600081525090565b60408051600680825260e082019092526000916020820160c08036833701905050905060005b60068110156103dd576103ad6103a885836103a3816001610cf7565b610841565b61090e565b60ff168282815181106103c2576103c2610d0f565b60209081029190910101526103d681611016565b9050610387565b506040518060c001604052806000836000815181106103fe576103fe610d0f565b60200260200101516009811061041657610416610d0f565b01805480602002602001604051908101604052809291908181526020016000905b828210156104e35783829060005260206000200180546104569061107f565b80601f01602080910402602001604051908101604052809291908181526020018280546104829061107f565b80156104cf5780601f106104a4576101008083540402835291602001916104cf565b820191906000526020600020905b8154815290600101906020018083116104b257829003601f168201915b505050505081526020019060010190610437565b50505050815260200160098360018151811061050157610501610d0f565b60200260200101518154811061051957610519610d0f565b90600052602060002001548152602001600a8360028151811061053e5761053e610d0f565b60200260200101518154811061055657610556610d0f565b90600052602060002001548152602001600b8360038151811061057b5761057b610d0f565b60200260200101518154811061059357610593610d0f565b9060005260206000200180546105a89061107f565b80601f01602080910402602001604051908101604052809291908181526020018280546105d49061107f565b80156106215780601f106105f657610100808354040283529160200191610621565b820191906000526020600020905b81548152906001019060200180831161060457829003601f168201915b50505050508152602001600c8360048151811061064057610640610d0f565b60200260200101518154811061065857610658610d0f565b906000526020600020015481526020018260058151811061067b5761067b610d0f565b6020026020010151600161068f9190610cf7565b90529392505050565b6060816000036106bf5750506040805180820190915260018152600360fc1b602082015290565b8160005b81156106e957806106d381611016565b91506106e29050600a836110b9565b91506106c3565b60008167ffffffffffffffff811115610704576107046109cc565b6040519080825280601f01601f19166020018201604052801561072e576020820181803683370190505b5090505b841561079957610743600183610d25565b9150610750600a86610fd3565b61075b906030610cf7565b60f81b81838151811061077057610770610d0f565b60200101906001600160f81b031916908160001a905350610792600a866110b9565b9450610732565b949350505050565b60606040518060800160405280605181526020016111ff6051913990506000805b8381101561083957826107d483610698565b866107e0846001610cf7565b815181106107f0576107f0610d0f565b602002602001015160405160200161080a939291906110cd565b60408051601f198184030181529190529250610827600383610cf7565b915061083281611016565b90506107c2565b505092915050565b60608360006108508585610d25565b67ffffffffffffffff811115610868576108686109cc565b6040519080825280601f01601f191660200182016040528015610892576020820181803683370190505b509050845b84811015610904578281815181106108b1576108b1610d0f565b01602001516001600160f81b031916826108cb8884610d25565b815181106108db576108db610d0f565b60200101906001600160f81b031916908160001a905350806108fc81611016565b915050610897565b5095945050505050565b60008181805b82518160ff1610156109c4576030838260ff168151811061093757610937610d0f565b016020015160f81c1080159061096a57506039838260ff168151811061095f5761095f610d0f565b016020015160f81c11155b156109b25761097a600a83611193565b91506030838260ff168151811061099357610993610d0f565b01602001516109a5919060f81c6111bc565b6109af908361105a565b91505b806109bc816111df565b915050610914565b509392505050565b634e487b7160e01b600052604160045260246000fd5b600082601f8301126109f357600080fd5b813567ffffffffffffffff80821115610a0e57610a0e6109cc565b604051601f8301601f19908116603f01168101908282118183101715610a3657610a366109cc565b81604052838152866020858801011115610a4f57600080fd5b836020870160208301376000602085830101528094505050505092915050565b60008060408385031215610a8257600080fd5b82359150602083013567ffffffffffffffff811115610aa057600080fd5b610aac858286016109e2565b9150509250929050565b60005b83811015610ad1578181015183820152602001610ab9565b83811115610ae0576000848401525b50505050565b60008151808452610afe816020860160208601610ab6565b601f01601f19169290920160200192915050565b602081526000610b256020830184610ae6565b9392505050565b803560ff81168114610b3d57600080fd5b919050565b60008060008060808587031215610b5857600080fd5b843567ffffffffffffffff811115610b6f57600080fd5b610b7b878288016109e2565b94505060208501359250610b9160408601610b2c565b9150610b9f60608601610b2c565b905092959194509250565b600080600060608486031215610bbf57600080fd5b833567ffffffffffffffff811115610bd657600080fd5b610be2868287016109e2565b9660208601359650604090950135949350505050565b600060208284031215610c0a57600080fd5b813567ffffffffffffffff811115610c2157600080fd5b610799848285016109e2565b6020808252825160c083830152805160e0840181905260009291610100600583901b86018101929184019190860190855b81811015610c8c5760ff19888603018352610c7a858551610ae6565b94509285019291850191600101610c5e565b505050508185015160408501526040850151606085015260608501519150601f19848203016080850152610cc08183610ae6565b915050608084015160a084015260a084015160c08401528091505092915050565b634e487b7160e01b600052601160045260246000fd5b60008219821115610d0a57610d0a610ce1565b500190565b634e487b7160e01b600052603260045260246000fd5b600082821015610d3757610d37610ce1565b500390565b60008151610d4e818560208601610ab6565b9290920192915050565b60008851610d6a818460208d01610ab6565b80830190507f272f3e3c646566733e3c672069643d2763686172732720666f6e742d66616d6981526e363c9e93b6b7b737b9b830b1b2939f60891b60208201528851610dbd81602f840160208d01610ab6565b7f3c616e696d617465206174747269627574654e616d653d27666f6e742d73697a602f92909101918201527f652720617474726962757465547970653d27584d4c272076616c7565733d2731604f820152633030253b60e01b606f8201528751610e2e816073840160208c01610ab6565b7f253b313030252720626567696e3d27307327206475723d273135732720726570607392909101918201527f656174436f756e743d27696e646566696e697465272f3e3c2f673e3c66696c7460938201527f65722069643d2774757262756c656e6365273e3c666554757262756c656e636560b38201527f20747970653d2774757262756c656e63652720626173654672657175656e637960d3820152631e93981760e11b60f3820152610f49610f43610f3d610f14610f0e610ef460f787018d610d3c565b6d27206e756d4f6374617665733d2760901b8152600e0190565b8a610d3c565b7f2720726573756c743d276e6f6973652720736565643d27000000000000000000815260170190565b87610d3c565b85610d3c565b9a9950505050505050505050565b6a3c746578742078203d272d60a81b815260008351610f7d81600b850160208801610ab6565b642720793d2760d81b600b918401918201528351610fa2816010840160208801610ab6565b61139f60f11b60109290910191820152601201949350505050565b634e487b7160e01b600052601260045260246000fd5b600082610fe257610fe2610fbd565b500690565b60008351610ff9818460208801610ab6565b83519083019061100d818360208801610ab6565b01949350505050565b60006001820161102857611028610ce1565b5060010190565b60008251611041818460208701610ab6565b661e17ba32bc3a1f60c91b920191825250600701919050565b600060ff821660ff84168060ff0382111561107757611077610ce1565b019392505050565b600181811c9082168061109357607f821691505b6020821081036110b357634e487b7160e01b600052602260045260246000fd5b50919050565b6000826110c8576110c8610fbd565b500490565b600084516110df818460208901610ab6565b753c75736520687265663d272363686172732720793d2760501b9083019081528451611112816016840160208901610ab6565b7f2720783d2730272066696c7465723d2775726c282374757262756c656e636529601692909101918201527f272077696474683d27323027206865696768743d273230272066696c6c3d272360368201528351611176816056840160208801610ab6565b6213979f60e91b6056929091019182015260590195945050505050565b600060ff821660ff84168160ff04811182151516156111b4576111b4610ce1565b029392505050565b600060ff821660ff8416808210156111d6576111d6610ce1565b90039392505050565b600060ff821660ff81036111f5576111f5610ce1565b6001019291505056fe272f3e3c6665446973706c6163656d656e744d617020696e3d27536f75726365477261706869632720696e323d276e6f69736527207363616c653d273327202f3e3c2f66696c7465723e3c2f646566733ea26469706673582212208a4030ad5bc31661300283ed655cdb2ad84abcb9dd89c4a2b3fe1b7a845729c564736f6c634300080d0033

Deployed Bytecode

0x6080604052600436106101e25760003560e01c806366e3387011610102578063c6374d0c11610095578063e985e9c511610064578063e985e9c51461052e578063f0292a0314610569578063f2fde38b1461057e578063f756b53e1461059e57600080fd5b8063c6374d0c146104db578063c87b56dd146104f3578063d06aec2114610513578063d2ed5c591461052657600080fd5b806395d89b41116100d157806395d89b411461046b578063a22cb46514610480578063b88d4fde146104a0578063c002d23d146104c057600080fd5b806366e33870146103f157806370a082311461041157806372f90ac1146104315780638da5cb5b1461044b57600080fd5b806323b872dd1161017a5780633ccfd60b116101495780633ccfd60b1461039657806342842e0e1461039e57806352f0dc9b146103be5780636352211e146103d157600080fd5b806323b872dd1461031f5780632fb098d21461033f57806332cb6b0c1461036d578063349d27481461038357600080fd5b806308338e4b116101b657806308338e4b146102b057806308871ceb146102d4578063095ea7b3146102e957806318160ddd1461030957600080fd5b80625ea307146101e757806301ffc9a71461021d57806306fdde031461024d578063081812fc14610262575b600080fd5b3480156101f357600080fd5b50610207610202366004612094565b6105a6565b6040516102149190612105565b60405180910390f35b34801561022957600080fd5b5061023d610238366004612131565b61060f565b6040519015158152602001610214565b34801561025957600080fd5b5061020761065d565b34801561026e57600080fd5b5061029861027d366004612094565b6004602052600090815260409020546001600160a01b031681565b6040516001600160a01b039091168152602001610214565b3480156102bc57600080fd5b506102c660075481565b604051908152602001610214565b6102e76102e2366004612199565b6106eb565b005b3480156102f557600080fd5b506102e76103043660046121f1565b61088c565b34801561031557600080fd5b506102c660095481565b34801561032b57600080fd5b506102e761033a36600461221b565b61096e565b34801561034b57600080fd5b5061035f61035a366004612257565b610b35565b604051610214929190612279565b34801561037957600080fd5b506102c661020081565b6102e7610391366004612392565b610c86565b6102e7610d7f565b3480156103aa57600080fd5b506102e76103b936600461221b565b610f74565b6102e76103cc366004612094565b611067565b3480156103dd57600080fd5b506102986103ec366004612094565b61111d565b3480156103fd57600080fd5b5061020761040c3660046124ba565b611174565b34801561041d57600080fd5b506102c661042c3660046124ee565b6112e6565b34801561043d57600080fd5b5060085461023d9060ff1681565b34801561045757600080fd5b50600654610298906001600160a01b031681565b34801561047757600080fd5b50610207611349565b34801561048c57600080fd5b506102e761049b366004612509565b611356565b3480156104ac57600080fd5b506102e76104bb366004612545565b6113c2565b3480156104cc57600080fd5b506102c6665af3107a40000081565b3480156104e757600080fd5b506102c66363c99a3881565b3480156104ff57600080fd5b5061020761050e366004612094565b6114aa565b6102e76105213660046125df565b6115af565b6102e7611696565b34801561053a57600080fd5b5061023d61054936600461264a565b600560209081526000928352604080842090915290825290205460ff1681565b34801561057557600080fd5b506102c6600381565b34801561058a57600080fd5b506102e76105993660046124ee565b6116d4565b6102e761174a565b336000908152600b60205260409020546060904360089190911c036105de5760405163a5e0079160e01b815260040160405180910390fd5b60095482106106005760405163732f483160e11b815260040160405180910390fd5b6106098261177b565b92915050565b60006301ffc9a760e01b6001600160e01b03198316148061064057506380ac58cd60e01b6001600160e01b03198316145b806106095750506001600160e01b031916635b5e139f60e01b1490565b6000805461066a9061267d565b80601f01602080910402602001604051908101604052809291908181526020018280546106969061267d565b80156106e35780601f106106b8576101008083540402835291602001916106e3565b820191906000526020600020905b8154815290600101906020018083116106c657829003601f168201915b505050505081565b6040516bffffffffffffffffffffffff193360601b1660208201526000906034016040516020818303038152906040528051906020012090506107848383808060200260200160405190810160405280939291908181526020018383602002808284376000920191909152507fea35e50958ff75fe96e04a6dd792de75a26dd0c2a2d12e8a4c485d938961eb3992508591506118519050565b6107c15760405162461bcd60e51b8152602060048201526009602482015268139bdd081bdb8815d360ba1b60448201526064015b60405180910390fd5b665af3107a40000034146107e857604051637edfcf7960e11b815260040160405180910390fd5b336000908152600b6020526040902054600f600482901c161561081e57604051633eae439d60e01b815260040160405180910390fd5b600754600003610840576040516298598d60e61b815260040160405180910390fd5b60076000815461084f906126cd565b909155506010176108664360081b60ff83166126e4565b336000818152600b60205260409020829055909150610886906001611867565b50505050565b6000818152600260205260409020546001600160a01b0316338114806108d557506001600160a01b038116600090815260056020908152604080832033845290915290205460ff165b6109125760405162461bcd60e51b815260206004820152600e60248201526d1393d517d055551213d49256915160921b60448201526064016107b8565b60008281526004602052604080822080546001600160a01b0319166001600160a01b0387811691821790925591518593918516917f8c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b92591a4505050565b6000818152600260205260409020546001600160a01b038481169116146109c45760405162461bcd60e51b815260206004820152600a60248201526957524f4e475f46524f4d60b01b60448201526064016107b8565b6001600160a01b038216610a0e5760405162461bcd60e51b81526020600482015260116024820152701253959053125117d49150d25412515395607a1b60448201526064016107b8565b336001600160a01b0384161480610a4857506001600160a01b038316600090815260056020908152604080832033845290915290205460ff165b80610a6957506000818152600460205260409020546001600160a01b031633145b610aa65760405162461bcd60e51b815260206004820152600e60248201526d1393d517d055551213d49256915160921b60448201526064016107b8565b6001600160a01b0380841660008181526003602090815260408083208054600019019055938616808352848320805460010190558583526002825284832080546001600160a01b03199081168317909155600490925284832080549092169091559251849392917fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef91a4505050565b600d6020528160005260406000208181548110610b5157600080fd5b906000526020600020906002020160009150915050806000018054610b759061267d565b80601f0160208091040260200160405190810160405280929190818152602001828054610ba19061267d565b8015610bee5780601f10610bc357610100808354040283529160200191610bee565b820191906000526020600020905b815481529060010190602001808311610bd157829003601f168201915b505050505090806001018054610c039061267d565b80601f0160208091040260200160405190810160405280929190818152602001828054610c2f9061267d565b8015610c7c5780601f10610c5157610100808354040283529160200191610c7c565b820191906000526020600020905b815481529060010190602001808311610c5f57829003601f168201915b5050505050905082565b6006546001600160a01b03163314610cb05760405162461bcd60e51b81526004016107b8906126fc565b60005b8151811015610d7957600d60008481526020019081526020016000206040518060400160405280848481518110610cec57610cec612722565b6020026020010151600001518152602001848481518110610d0f57610d0f612722565b602090810291909101810151810151909152825460018101845560009384529281902082518051939460020290910192610d4c9284920190611ffb565b506020828101518051610d659260018501920190611ffb565b50505080610d7290612738565b9050610cb3565b505b5050565b6006546001600160a01b03163314610da95760405162461bcd60e51b81526004016107b8906126fc565b6000610db6606447612767565b610dc190600c61277b565b90506000610dd0606447612767565b610ddb90605361277b565b90506000610dea606447612767565b610df590600561277b565b604051909150600090734533d1f65906368ebfd61259daee561df3f3559d9085908381818185875af1925050503d8060008114610e4e576040519150601f19603f3d011682016040523d82523d6000602084013e610e53565b606091505b5050905080610e745760405162c0f29960e01b815260040160405180910390fd5b60405160009073888f8aa938dbb18b28bdd111fa4a0d3b8e10c8719084908381818185875af1925050503d8060008114610eca576040519150601f19603f3d011682016040523d82523d6000602084013e610ecf565b606091505b5050905080610ef05760405162c0f29960e01b815260040160405180910390fd5b60405160009073e4260df86f5261a41d19c2066f1eb2eb4f009e849086908381818185875af1925050503d8060008114610f46576040519150601f19603f3d011682016040523d82523d6000602084013e610f4b565b606091505b5050905080610f6c5760405162c0f29960e01b815260040160405180910390fd5b505050505050565b610f7f83838361096e565b6001600160a01b0382163b15806110285750604051630a85bd0160e11b8082523360048301526001600160a01b03858116602484015260448301849052608060648401526000608484015290919084169063150b7a029060a4016020604051808303816000875af1158015610ff8573d6000803e3d6000fd5b505050506040513d601f19601f8201168201806040525081019061101c919061279a565b6001600160e01b031916145b610d795760405162461bcd60e51b815260206004820152601060248201526f155394d0519157d49150d2541251539560821b60448201526064016107b8565b60038111156110895760405163355ad2e760e01b815260040160405180910390fd5b61109a81665af3107a40000061277b565b34146110b957604051637edfcf7960e11b815260040160405180910390fd5b336000908152600b6020526040902054600f8116156110eb5760405163317f85f760e21b815260040160405180910390fd5b6001176110fe4360081b60ff83166126e4565b336000818152600b60205260409020829055909150610d7b9083611867565b6000818152600260205260409020546001600160a01b03168061116f5760405162461bcd60e51b815260206004820152600a6024820152691393d517d3525395115160b21b60448201526064016107b8565b919050565b336000908152600b60205260409020546060904360089190911c036111ac5760405163a5e0079160e01b815260040160405180910390fd5b606060005b60068160ff1610156112bd5760006111e26111dd8660ff85166111d58660016127b7565b60ff1661194d565b611a19565b905082600d60008460ff1681526020019081526020016000208260ff168154811061120f5761120f612722565b9060005260206000209060020201600101600d60008560ff1681526020019081526020016000208360ff168154811061124a5761124a612722565b906000526020600020906002020160000160405160200161126d93929190612875565b60405160208183030381529060405292508160ff166005146112ac578260405160200161129a91906128e2565b60405160208183030381529060405292505b506112b681612907565b90506111b1565b50806040516020016112cf9190612926565b604051602081830303815290604052915050919050565b60006001600160a01b03821661132d5760405162461bcd60e51b815260206004820152600c60248201526b5a45524f5f4144445245535360a01b60448201526064016107b8565b506001600160a01b031660009081526003602052604090205490565b6001805461066a9061267d565b3360008181526005602090815260408083206001600160a01b03871680855290835292819020805460ff191686151590811790915590519081529192917f17307eab39ab6107e8899845ad3d59bd9653f200f220920489ca2b5937696c31910160405180910390a35050565b6113cd85858561096e565b6001600160a01b0384163b15806114645750604051630a85bd0160e11b808252906001600160a01b0386169063150b7a02906114159033908a9089908990899060040161295a565b6020604051808303816000875af1158015611434573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190611458919061279a565b6001600160e01b031916145b6114a35760405162461bcd60e51b815260206004820152601060248201526f155394d0519157d49150d2541251539560821b60448201526064016107b8565b5050505050565b606060095482106114ce5760405163732f483160e11b815260040160405180910390fd5b60006114d9836105a6565b905061159f6114e784611ad7565b600e54604051630e4b241160e11b8152611584916001600160a01b031690631c9648229061151b90899088906004016129ae565b600060405180830381865afa158015611538573d6000803e3d6000fd5b505050506040513d6000823e601f3d908101601f1916820160405261156091908101906129c7565b6040516020016115709190612a34565b604051602081830303815290604052611bdf565b61158d84611174565b60405160200161157093929190612b21565b6040516020016112cf9190612c5b565b6006546001600160a01b031633146115d95760405162461bcd60e51b81526004016107b8906126fc565b600a5460ff16156115fd5760405163eddaeeb360e01b815260040160405180910390fd5b8281811461161e57604051637faa19c760e01b815260040160405180910390fd5b60005b818110156116815761167186868381811061163e5761163e612722565b905060200201602081019061165391906124ee565b85858481811061166557611665612722565b90506020020135611867565b61167a81612738565b9050611621565b5050600a805460ff1916600117905550505050565b6006546001600160a01b031633146116c05760405162461bcd60e51b81526004016107b8906126fc565b6008805460ff19811660ff90911615179055565b6006546001600160a01b031633146116fe5760405162461bcd60e51b81526004016107b8906126fc565b600680546001600160a01b0319166001600160a01b03831690811790915560405133907f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e090600090a350565b6006546001600160a01b031633146117745760405162461bcd60e51b81526004016107b8906126fc565b6000600755565b60408051602080820183526000808352848152600c909152918220546060925b60068160ff16101561184857600061271083836040516020016117d592919091825260f81b6001600160f81b031916602082015260210190565b6040516020818303038152906040528051906020012060001c6117f89190612ca0565b90508361181461180c8361ffff1685611d45565b60ff16611ad7565b604051602001611825929190612cb4565b6040516020818303038152906040529350508061184190612907565b905061179b565b50909392505050565b60008261185e8584611e19565b14949350505050565b60085460ff16158061187c57506363c99a3842105b1561189a5760405163baf13b3f60e01b815260040160405180910390fd5b806000036118bb5760405163220328c760e11b815260040160405180910390fd5b610200600754826009546118cf91906126e4565b6118d991906126e4565b11156118f8576040516352df9fe560e01b815260040160405180910390fd5b600954600061190783836126e4565b90505b808210156119455761191c8483611e5e565b6000838152600c60205260409020556119358483611ebe565b61193e82612738565b915061190a565b600955505050565b606083600061195c8585612ce3565b6001600160401b03811115611973576119736122a7565b6040519080825280601f01601f19166020018201604052801561199d576020820181803683370190505b509050845b84811015611a0f578281815181106119bc576119bc612722565b01602001516001600160f81b031916826119d68884612ce3565b815181106119e6576119e6612722565b60200101906001600160f81b031916908160001a90535080611a0781612738565b9150506119a2565b5095945050505050565b60008181805b82518160ff161015611acf576030838260ff1681518110611a4257611a42612722565b016020015160f81c10801590611a7557506039838260ff1681518110611a6a57611a6a612722565b016020015160f81c11155b15611abd57611a85600a83612cfa565b91506030838260ff1681518110611a9e57611a9e612722565b0160200151611ab0919060f81c612d23565b611aba90836127b7565b91505b80611ac781612907565b915050611a1f565b509392505050565b606081600003611afe5750506040805180820190915260018152600360fc1b602082015290565b8160005b8115611b285780611b1281612738565b9150611b219050600a83612767565b9150611b02565b6000816001600160401b03811115611b4257611b426122a7565b6040519080825280601f01601f191660200182016040528015611b6c576020820181803683370190505b5090505b8415611bd757611b81600183612ce3565b9150611b8e600a86612ca0565b611b999060306126e4565b60f81b818381518110611bae57611bae612722565b60200101906001600160f81b031916908160001a905350611bd0600a86612767565b9450611b70565b949350505050565b60608151600003611bfe57505060408051602081019091526000815290565b6000604051806060016040528060408152602001612d646040913990506000600384516002611c2d91906126e4565b611c379190612767565b611c4290600461277b565b90506000611c518260206126e4565b6001600160401b03811115611c6857611c686122a7565b6040519080825280601f01601f191660200182016040528015611c92576020820181803683370190505b509050818152600183018586518101602084015b81831015611d005760039283018051603f601282901c811687015160f890811b8552600c83901c8216880151811b6001860152600683901c8216880151811b60028601529116860151901b93820193909352600401611ca6565b600389510660018114611d1a5760028114611d2b57611d37565b613d3d60f01b600119830152611d37565b603d60f81b6000198301525b509398975050505050505050565b6000806000600f8460ff1660068110611d6057611d60612722565b0154905060005b818160ff161015611e13576000600f8660ff1660068110611d8a57611d8a612722565b018260ff1681548110611d9f57611d9f612722565b90600052602060002090601091828204019190066002029054906101000a900461ffff1690508361ffff168710158015611de55750611dde8185612d46565b61ffff1687105b15611df557509250610609915050565b611dff8185612d46565b93505080611e0c90612907565b9050611d67565b50600080fd5b600081815b8451811015611acf57611e4a82868381518110611e3d57611e3d612722565b6020026020010151611fc9565b915080611e5681612738565b915050611e1e565b600042448484604051602001611e9f9493929190938452602084019290925260601b6bffffffffffffffffffffffff19166040830152605482015260740190565b60408051601f1981840301815291905280516020909101209392505050565b6001600160a01b038216611f085760405162461bcd60e51b81526020600482015260116024820152701253959053125117d49150d25412515395607a1b60448201526064016107b8565b6000818152600260205260409020546001600160a01b031615611f5e5760405162461bcd60e51b815260206004820152600e60248201526d1053149150511657d3525395115160921b60448201526064016107b8565b6001600160a01b038216600081815260036020908152604080832080546001019055848352600290915280822080546001600160a01b0319168417905551839291907fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef908290a45050565b6000818310611fe5576000828152602084905260409020611ff4565b60008381526020839052604090205b9392505050565b8280546120079061267d565b90600052602060002090601f016020900481019282612029576000855561206f565b82601f1061204257805160ff191683800117855561206f565b8280016001018555821561206f579182015b8281111561206f578251825591602001919060010190612054565b5061207b92915061207f565b5090565b5b8082111561207b5760008155600101612080565b6000602082840312156120a657600080fd5b5035919050565b60005b838110156120c85781810151838201526020016120b0565b838111156108865750506000910152565b600081518084526120f18160208601602086016120ad565b601f01601f19169290920160200192915050565b602081526000611ff460208301846120d9565b6001600160e01b03198116811461212e57600080fd5b50565b60006020828403121561214357600080fd5b8135611ff481612118565b60008083601f84011261216057600080fd5b5081356001600160401b0381111561217757600080fd5b6020830191508360208260051b850101111561219257600080fd5b9250929050565b600080602083850312156121ac57600080fd5b82356001600160401b038111156121c257600080fd5b6121ce8582860161214e565b90969095509350505050565b80356001600160a01b038116811461116f57600080fd5b6000806040838503121561220457600080fd5b61220d836121da565b946020939093013593505050565b60008060006060848603121561223057600080fd5b612239846121da565b9250612247602085016121da565b9150604084013590509250925092565b6000806040838503121561226a57600080fd5b50508035926020909101359150565b60408152600061228c60408301856120d9565b828103602084015261229e81856120d9565b95945050505050565b634e487b7160e01b600052604160045260246000fd5b604080519081016001600160401b03811182821017156122df576122df6122a7565b60405290565b604051601f8201601f191681016001600160401b038111828210171561230d5761230d6122a7565b604052919050565b60006001600160401b0382111561232e5761232e6122a7565b50601f01601f191660200190565b600082601f83011261234d57600080fd5b813561236061235b82612315565b6122e5565b81815284602083860101111561237557600080fd5b816020850160208301376000918101602001919091529392505050565b60008060408084860312156123a657600080fd5b833592506020808501356001600160401b03808211156123c557600080fd5b818701915087601f8301126123d957600080fd5b8135818111156123eb576123eb6122a7565b8060051b6123fa8582016122e5565b918252838101850191858101908b84111561241457600080fd5b86860192505b838310156124a8578235858111156124325760008081fd5b8601808d03601f19018913156124485760008081fd5b6124506122bd565b88820135878111156124625760008081fd5b6124708f8b8386010161233c565b82525089820135878111156124855760008081fd5b6124938f8b8386010161233c565b828b015250835250918601919086019061241a565b80985050505050505050509250929050565b6000602082840312156124cc57600080fd5b81356001600160401b038111156124e257600080fd5b611bd78482850161233c565b60006020828403121561250057600080fd5b611ff4826121da565b6000806040838503121561251c57600080fd5b612525836121da565b91506020830135801515811461253a57600080fd5b809150509250929050565b60008060008060006080868803121561255d57600080fd5b612566866121da565b9450612574602087016121da565b93506040860135925060608601356001600160401b038082111561259757600080fd5b818801915088601f8301126125ab57600080fd5b8135818111156125ba57600080fd5b8960208285010111156125cc57600080fd5b9699959850939650602001949392505050565b600080600080604085870312156125f557600080fd5b84356001600160401b038082111561260c57600080fd5b6126188883890161214e565b9096509450602087013591508082111561263157600080fd5b5061263e8782880161214e565b95989497509550505050565b6000806040838503121561265d57600080fd5b612666836121da565b9150612674602084016121da565b90509250929050565b600181811c9082168061269157607f821691505b6020821081036126b157634e487b7160e01b600052602260045260246000fd5b50919050565b634e487b7160e01b600052601160045260246000fd5b6000816126dc576126dc6126b7565b506000190190565b600082198211156126f7576126f76126b7565b500190565b6020808252600c908201526b15539055551213d49256915160a21b604082015260600190565b634e487b7160e01b600052603260045260246000fd5b60006001820161274a5761274a6126b7565b5060010190565b634e487b7160e01b600052601260045260246000fd5b60008261277657612776612751565b500490565b6000816000190483118215151615612795576127956126b7565b500290565b6000602082840312156127ac57600080fd5b8151611ff481612118565b600060ff821660ff84168060ff038211156127d4576127d46126b7565b019392505050565b8054600090600181811c90808316806127f657607f831692505b6020808410820361281757634e487b7160e01b600052602260045260246000fd5b81801561282b576001811461283c57612869565b60ff19861689528489019650612869565b60008881526020902060005b868110156128615781548b820152908501908301612848565b505084890196505b50505050505092915050565b600084516128878184602089016120ad565b6e3d913a3930b4ba2fba3cb832911d1160891b9083019081526128ad600f8201866127dc565b6a1116113b30b63ab2911d1160a91b815290506128cd600b8201856127dc565b61227d60f01b81526002019695505050505050565b600082516128f48184602087016120ad565b600b60fa1b920191825250600101919050565b600060ff821660ff810361291d5761291d6126b7565b60010192915050565b605b60f81b8152600082516129428160018501602087016120ad565b605d60f81b6001939091019283015250600201919050565b6001600160a01b038681168252851660208201526040810184905260806060820181905281018290526000828460a0840137600060a0848401015260a0601f19601f85011683010190509695505050505050565b828152604060208201526000611bd760408301846120d9565b6000602082840312156129d957600080fd5b81516001600160401b038111156129ef57600080fd5b8201601f81018413612a0057600080fd5b8051612a0e61235b82612315565b818152856020838501011115612a2357600080fd5b61229e8260208301602086016120ad565b7f3c7376672076696577426f783d27302030203230203230272077696474683d2781527f36303027206865696768743d273630302720786d6c6e733d27687474703a2f2f60208201527f7777772e77332e6f72672f323030302f7376672720707265736572766541737060408201527f656374526174696f3d27784d6964594d696e273e3c726563742077696474683d60608201527f27323027206865696768743d273230272066696c6c3d27230000000000000000608082015260008251612b048160988501602087016120ad565b651e17b9bb339f60d11b6098939091019283015250609e01919050565b757b226e616d65223a2022434834314e5734563335202360501b81528351600090612b538160168501602089016120ad565b7f222c226465736372697074696f6e223a202246756c6c79206f6e636861696e206016918401918201527f67656e65726174697665206172742053564720636f6c6c656374696f6e2e204360368201527f726561746564206279204d63546f616479202620436972636f6c6f72732e220060568201527f2c22696d616765223a2022646174613a696d6167652f7376672b786d6c3b62616075820152641cd94d8d0b60da1b60958201528451612c1081609a8401602089016120ad565b6e11161130ba3a3934b13aba32b9911d60891b609a92909101918201528351612c408160a98401602088016120ad565b607d60f81b60a9929091019182015260aa0195945050505050565b7f646174613a6170706c69636174696f6e2f6a736f6e3b6261736536342c000000815260008251612c9381601d8501602087016120ad565b91909101601d0192915050565b600082612caf57612caf612751565b500690565b60008351612cc68184602088016120ad565b835190830190612cda8183602088016120ad565b01949350505050565b600082821015612cf557612cf56126b7565b500390565b600060ff821660ff84168160ff0481118215151615612d1b57612d1b6126b7565b029392505050565b600060ff821660ff841680821015612d3d57612d3d6126b7565b90039392505050565b600061ffff808316818516808303821115612cda57612cda6126b756fe4142434445464748494a4b4c4d4e4f505152535455565758595a6162636465666768696a6b6c6d6e6f707172737475767778797a303132333435363738392b2fa2646970667358221220a16ad100558bec1aa07c2dbb585111b82addb6abfed4f6b7bf30663d18999e8b64736f6c634300080d0033

Loading...
Loading
Loading...
Loading
[ Download: CSV Export  ]
[ Download: CSV Export  ]

A token is a representation of an on-chain or off-chain asset. The token page shows information such as price, total supply, holders, transfers and social links. Learn more about this page in our Knowledge Base.