账户
0xef...e09d
0xEf...e09d

0xEf...e09d

$500
此合同的源代码已经过验证!
合同元数据
编译器
0.6.0+commit.26b70077
语言
Solidity
合同源代码
文件 1 的 1:MerkleRedeem.sol
// https://nhentai.net/g/177978/
//⠄⠄⠄⢰⣧⣼⣯⠄⣸⣠⣶⣶⣦⣾⠄⠄⠄⠄⡀⠄⢀⣿⣿⠄⠄⠄⢸⡇⠄⠄
//⠄⠄⠄⣾⣿⠿⠿⠶⠿⢿⣿⣿⣿⣿⣦⣤⣄⢀⡅⢠⣾⣛⡉⠄⠄⠄⠸⢀⣿⠄
//⠄⠄⢀⡋⣡⣴⣶⣶⡀⠄⠄⠙⢿⣿⣿⣿⣿⣿⣴⣿⣿⣿⢃⣤⣄⣀⣥⣿⣿⠄
//⠄⠄⢸⣇⠻⣿⣿⣿⣧⣀⢀⣠⡌⢻⣿⣿⣿⣿⣿⣿⣿⣿⣿⠿⠿⠿⣿⣿⣿⠄
//⠄⢀⢸⣿⣷⣤⣤⣤⣬⣙⣛⢿⣿⣿⣿⣿⣿⣿⡿⣿⣿⡍⠄⠄⢀⣤⣄⠉⠋⣰
//⠄⣼⣖⣿⣿⣿⣿⣿⣿⣿⣿⣿⢿⣿⣿⣿⣿⣿⢇⣿⣿⡷⠶⠶⢿⣿⣿⠇⢀⣤
//⠘⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣽⣿⣿⣿⡇⣿⣿⣿⣿⣿⣿⣷⣶⣥⣴⣿⡗
//⢀⠈⢿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⡟⠄
//⢸⣿⣦⣌⣛⣻⣿⣿⣧⠙⠛⠛⡭⠅⠒⠦⠭⣭⡻⣿⣿⣿⣿⣿⣿⣿⣿⡿⠃⠄
//⠘⣿⣿⣿⣿⣿⣿⣿⣿⡆⠄⠄⠄⠄⠄⠄⠄⠄⠹⠈⢋⣽⣿⣿⣿⣿⣵⣾⠃⠄
//⠄⠘⣿⣿⣿⣿⣿⣿⣿⣿⠄⣴⣿⣶⣄⠄⣴⣶⠄⢀⣾⣿⣿⣿⣿⣿⣿⠃⠄⠄
//⠄⠄⠈⠻⣿⣿⣿⣿⣿⣿⡄⢻⣿⣿⣿⠄⣿⣿⡀⣾⣿⣿⣿⣿⣛⠛⠁⠄⠄⠄
//⠄⠄⠄⠄⠈⠛⢿⣿⣿⣿⠁⠞⢿⣿⣿⡄⢿⣿⡇⣸⣿⣿⠿⠛⠁⠄⠄⠄⠄⠄
//⠄⠄⠄⠄⠄⠄⠄⠉⠻⣿⣿⣾⣦⡙⠻⣷⣾⣿⠃⠿⠋⠁⠄⠄⠄⠄⠄⢀⣠⣴
//⣿⣿⣿⣶⣶⣮⣥⣒⠲⢮⣝⡿⣿⣿⡆⣿⡿⠃⠄⠄⠄⠄⠄⠄⠄⣠⣴⣿⣿⣿

// File: @openzeppelin/contracts/cryptography/MerkleProof.sol

// SPDX-License-Identifier: MIT

pragma solidity ^0.6.0;

/**
 * @dev These functions deal with verification of Merkle trees (hash trees),
 */
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) {
        bytes32 computedHash = leaf;

        for (uint256 i = 0; i < proof.length; i++) {
            bytes32 proofElement = proof[i];

            if (computedHash <= proofElement) {
                // Hash(current computed hash + current element of the proof)
                computedHash = keccak256(abi.encodePacked(computedHash, proofElement));
            } else {
                // Hash(current element of the proof + current computed hash)
                computedHash = keccak256(abi.encodePacked(proofElement, computedHash));
            }
        }

        // Check if the computed hash (root) is equal to the provided root
        return computedHash == root;
    }
}

// File: contracts/redeem/IERC20.sol

pragma solidity 0.6.0;

interface IERC20 {
  event Approval(address indexed src, address indexed dst, uint amt);
  event Transfer(address indexed src, address indexed dst, uint amt);

  function totalSupply() external view returns (uint);
  function balanceOf(address whom) external view returns (uint);
  function allowance(address src, address dst) external view returns (uint);

  function approve(address dst, uint amt) external returns (bool);
  function transfer(address dst, uint amt) external returns (bool);
  function transferFrom(
    address src, address dst, uint amt
  ) external returns (bool);
}

// File: contracts/redeem/ISwapXToken.sol

pragma solidity >=0.5.0;

interface ISwapXToken {
    function initialize(string calldata name, string calldata sym, uint maxSupply) external;

    function transferOwnership(address newOwner) external;

    function verify(bool verified) external;

    function verified() external returns (bool);

    function addIssuer(address _addr) external returns (bool);

    function removeIssuer(address _addr) external returns (bool);

    function issue(address account, uint256 amount) external returns (bool);
}

// File: contracts/redeem/MerkleRedeem.sol

pragma solidity 0.6.0;
pragma experimental ABIEncoderV2;




contract MerkleRedeem {
    address public tokenAddress;
    address public owner;

    event Claimed(address _claimant, address _token, uint256 _balance);
    event VerifiedToken(address _token);

    // Recorded epochs
    uint256 latestEpoch;
    mapping(uint256 => bytes32) public epochMerkleRoots;
    mapping(uint256 => uint256) public epochTimestamps;
    mapping(uint256 => bytes32) public epochBlockHashes;
    mapping(uint256 => mapping(address => mapping(address => bool)))
        public claimed;

    address[] public _verifiedTokens;

    constructor() public {
        owner = msg.sender;
    }

    modifier onlyOwner() {
        require(msg.sender == owner, "Must be the contract owner");
        _;
    }

    modifier requireEpochInPast(uint256 epoch) {
        require(epoch <= latestEpoch, "Epoch cannot be in the future");
        _;
    }

    modifier requireEpochRecorded(uint256 _epoch) {
        require(epochTimestamps[_epoch] != 0);
        require(epochBlockHashes[_epoch] != 0);
        _;
    }

    modifier requireMerkleRootUnset(uint256 _epoch) {
        require(epochMerkleRoots[_epoch] == bytes32(0));
        _;
    }

    modifier requireUnverified(address _token) {
        require(verified(_token) == false);
        _;
    }

    function verify(address _token)
        external
        onlyOwner
        requireUnverified(_token)
    {
        ISwapXToken(_token).verify(true);
        _verifiedTokens.push(_token);
        emit VerifiedToken(_token);
    }

    function verified(address _token) public view returns (bool) {
        for (uint256 i = 0; i < _verifiedTokens.length; i++) {
            if (_token == _verifiedTokens[i]) {
                return true;
            }
        }
        return false;
    }

    function verifiedTokens() public view returns (address[] memory) {
        address[] memory result = new address[](1);
        if (0 == _verifiedTokens.length) {
            delete result;
            return result;
        }
        uint256 len = _verifiedTokens.length;
        address[] memory results = new address[](len);
        for (uint256 i = 0; i < _verifiedTokens.length; i++) {
            results[i] = _verifiedTokens[i];
        }

        return results;
    }

    function issue(address _token, uint256 amount) external onlyOwner {
        if (amount > 0) {
            ISwapXToken(_token).issue(address(this), amount);
        } else {
            revert("No amount would be minted - not gonna waste your gas");
        }
    }

    function disburse(
        address _liquidityProvider,
        address _token,
        uint256 _balance
    ) private {
        if (_balance > 0) {
            IERC20(_token).transfer(_liquidityProvider, _balance);
            emit Claimed(_liquidityProvider, _token, _balance);
        } else {
            revert("No balance would be transfered - not gonna waste your gas");
        }
    }

    function offsetRequirementMet(address user, uint256 _epoch)
        public
        view
        returns (bool)
    {
        bytes32 blockHash = epochBlockHashes[_epoch];
        uint256 timestamp = epochTimestamps[_epoch];
        uint256 offsetSeconds = userEpochOffset(user, blockHash);

        uint256 earliestClaimableTimestamp = timestamp + offsetSeconds;
        return earliestClaimableTimestamp < block.timestamp;
    }

    function claimEpoch(
        address _liquidityProvider,
        uint256 _epoch,
        address _token,
        uint256 _claimedBalance,
        bytes32[] memory _merkleProof
    ) public requireEpochInPast(_epoch) requireEpochRecorded(_epoch) {
        // if trying to claim for the current epoch
        if (_epoch == latestEpoch) {
            require(
                offsetRequirementMet(_liquidityProvider, latestEpoch),
                "It is too early to claim for the current epoch"
            );
        }

        require(!claimed[_epoch][_liquidityProvider][_token]);
        require(
            verifyClaim(
                _liquidityProvider,
                _epoch,
                _token,
                _claimedBalance,
                _merkleProof
            ),
            "Incorrect merkle proof"
        );

        claimed[_epoch][_liquidityProvider][_token] = true;
        disburse(_liquidityProvider, _token, _claimedBalance);
    }

    struct Claim {
        uint256 epoch;
        address token;
        uint256 balance;
        bytes32[] merkleProof;
    }

    mapping(address => uint256) tokenTotalBalances; //temp mapping

    function claimEpochs(address _liquidityProvider, Claim[] memory claims)
        public
    {
        Claim memory claim;
        address[] memory _tokens;
        for (uint256 i = 0; i < claims.length; i++) {
            claim = claims[i];
            require(
                claim.epoch <= latestEpoch,
                "Epoch cannot be in the future"
            );
            require(epochTimestamps[claim.epoch] != 0);
            require(epochBlockHashes[claim.epoch] != 0);

            // if trying to claim for the current epoch
            if (claim.epoch == latestEpoch) {
                require(
                    offsetRequirementMet(_liquidityProvider, latestEpoch),
                    "It is too early to claim for the current epoch"
                );
            }

            require(!claimed[claim.epoch][_liquidityProvider][claim.token]);
            require(
                verifyClaim(
                    _liquidityProvider,
                    claim.epoch,
                    claim.token,
                    claim.balance,
                    claim.merkleProof
                ),
                "Incorrect merkle proof"
            );

            if (tokenTotalBalances[claim.token] == uint256(0)) {
                _tokens[_tokens.length] = claim.token;
            }

            tokenTotalBalances[claim.token] += claim.balance;

            claimed[claim.epoch][_liquidityProvider][claim.token] = true;
        }

        for (uint256 i = 0; i < _tokens.length; i++) {
            disburse(
                _liquidityProvider,
                _tokens[i],
                tokenTotalBalances[_tokens[i]]
            );

            delete tokenTotalBalances[_tokens[i]];
        }
        delete _tokens;
    }

    function claimStatus(
        address _liquidityProvider,
        address _token,
        uint256 _begin,
        uint256 _end
    ) public view returns (bool[] memory) {
        uint256 size = 1 + _end - _begin;
        bool[] memory arr = new bool[](size);
        for (uint256 i = 0; i < size; i++) {
            arr[i] = claimed[_begin + i][_liquidityProvider][_token];
        }
        return arr;
    }

    function merkleRoots(uint256 _begin, uint256 _end)
        public
        view
        returns (bytes32[] memory)
    {
        uint256 size = 1 + _end - _begin;
        bytes32[] memory arr = new bytes32[](size);
        for (uint256 i = 0; i < size; i++) {
            arr[i] = epochMerkleRoots[_begin + i];
        }
        return arr;
    }

    function verifyClaim(
        address _liquidityProvider,
        uint256 _epoch,
        address _token,
        uint256 _claimedBalance,
        bytes32[] memory _merkleProof
    ) public view returns (bool valid) {
        bytes32 leaf = keccak256(
            abi.encodePacked(_liquidityProvider, _token, _claimedBalance)
        );
        return MerkleProof.verify(_merkleProof, epochMerkleRoots[_epoch], leaf);
    }

    function userEpochOffset(
        address _liquidityProvider,
        bytes32 _epochBlockHash
    ) public pure returns (uint256 offset) {
        bytes32 hash = keccak256(
            abi.encodePacked(_liquidityProvider, _epochBlockHash)
        );
        assembly {
            offset := mod(
                hash,
                86400 // seconds in a epoch
            )
        }
        return offset;
    }

    function finishEpoch(
        uint256 _epoch,
        uint256 _timestamp,
        bytes32 _blockHash
    ) public onlyOwner {
        epochTimestamps[_epoch] = _timestamp;
        epochBlockHashes[_epoch] = _blockHash;
        if (_epoch > latestEpoch) {
            // just in case we get these out of order
            latestEpoch = _epoch;
        }
    }

    function seedAllocations(uint256 _epoch, bytes32 _merkleRoot)
        external
        requireEpochRecorded(_epoch)
        requireMerkleRootUnset(_epoch)
        onlyOwner
    {
        require(
            epochMerkleRoots[_epoch] == bytes32(0),
            "cannot rewrite merkle root"
        );
        epochMerkleRoots[_epoch] = _merkleRoot;
    }
}
设置
{
  "compilationTarget": {
    "MerkleRedeem.sol": "MerkleRedeem"
  },
  "evmVersion": "istanbul",
  "libraries": {},
  "metadata": {
    "bytecodeHash": "ipfs"
  },
  "optimizer": {
    "enabled": true,
    "runs": 200
  },
  "remappings": []
}
ABI
[{"inputs":[],"stateMutability":"nonpayable","type":"constructor"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"_claimant","type":"address"},{"indexed":false,"internalType":"address","name":"_token","type":"address"},{"indexed":false,"internalType":"uint256","name":"_balance","type":"uint256"}],"name":"Claimed","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"_token","type":"address"}],"name":"VerifiedToken","type":"event"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"}],"name":"_verifiedTokens","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_liquidityProvider","type":"address"},{"internalType":"uint256","name":"_epoch","type":"uint256"},{"internalType":"address","name":"_token","type":"address"},{"internalType":"uint256","name":"_claimedBalance","type":"uint256"},{"internalType":"bytes32[]","name":"_merkleProof","type":"bytes32[]"}],"name":"claimEpoch","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_liquidityProvider","type":"address"},{"components":[{"internalType":"uint256","name":"epoch","type":"uint256"},{"internalType":"address","name":"token","type":"address"},{"internalType":"uint256","name":"balance","type":"uint256"},{"internalType":"bytes32[]","name":"merkleProof","type":"bytes32[]"}],"internalType":"struct MerkleRedeem.Claim[]","name":"claims","type":"tuple[]"}],"name":"claimEpochs","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_liquidityProvider","type":"address"},{"internalType":"address","name":"_token","type":"address"},{"internalType":"uint256","name":"_begin","type":"uint256"},{"internalType":"uint256","name":"_end","type":"uint256"}],"name":"claimStatus","outputs":[{"internalType":"bool[]","name":"","type":"bool[]"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"},{"internalType":"address","name":"","type":"address"},{"internalType":"address","name":"","type":"address"}],"name":"claimed","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"}],"name":"epochBlockHashes","outputs":[{"internalType":"bytes32","name":"","type":"bytes32"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"}],"name":"epochMerkleRoots","outputs":[{"internalType":"bytes32","name":"","type":"bytes32"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"}],"name":"epochTimestamps","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"_epoch","type":"uint256"},{"internalType":"uint256","name":"_timestamp","type":"uint256"},{"internalType":"bytes32","name":"_blockHash","type":"bytes32"}],"name":"finishEpoch","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_token","type":"address"},{"internalType":"uint256","name":"amount","type":"uint256"}],"name":"issue","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_begin","type":"uint256"},{"internalType":"uint256","name":"_end","type":"uint256"}],"name":"merkleRoots","outputs":[{"internalType":"bytes32[]","name":"","type":"bytes32[]"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"user","type":"address"},{"internalType":"uint256","name":"_epoch","type":"uint256"}],"name":"offsetRequirementMet","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"owner","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"_epoch","type":"uint256"},{"internalType":"bytes32","name":"_merkleRoot","type":"bytes32"}],"name":"seedAllocations","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"tokenAddress","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_liquidityProvider","type":"address"},{"internalType":"bytes32","name":"_epochBlockHash","type":"bytes32"}],"name":"userEpochOffset","outputs":[{"internalType":"uint256","name":"offset","type":"uint256"}],"stateMutability":"pure","type":"function"},{"inputs":[{"internalType":"address","name":"_token","type":"address"}],"name":"verified","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"verifiedTokens","outputs":[{"internalType":"address[]","name":"","type":"address[]"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_token","type":"address"}],"name":"verify","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_liquidityProvider","type":"address"},{"internalType":"uint256","name":"_epoch","type":"uint256"},{"internalType":"address","name":"_token","type":"address"},{"internalType":"uint256","name":"_claimedBalance","type":"uint256"},{"internalType":"bytes32[]","name":"_merkleProof","type":"bytes32[]"}],"name":"verifyClaim","outputs":[{"internalType":"bool","name":"valid","type":"bool"}],"stateMutability":"view","type":"function"}]