ETH Price: $1,817.77 (+0.34%)

Transaction Decoder

Block:
15332752 at Aug-13-2022 10:00:52 AM +UTC
Transaction Fee:
0.001636495331968614 ETH $2.97
Gas Used:
135,727 Gas / 12.057257082 Gwei

Emitted Events:

109 izumiToken.Transfer( from=0xee264e74A2Fd2c7A55da705b80e092f05DaE5b5D, to=[Sender] 0xcffc183d6c445cbde55c1be49bac0b07b6f1618c, value=3140540106660318807 )
110 izumiToken.Approval( owner=0xee264e74A2Fd2c7A55da705b80e092f05DaE5b5D, spender=[Receiver] MiningFixRangeBoost, value=6491808216154363129226195 )
111 MiningFixRangeBoost.CollectReward( user=[Sender] 0xcffc183d6c445cbde55c1be49bac0b07b6f1618c, tokenId=289737, token=izumiToken, amount=3140540106660318807 )
112 izumiToken.Transfer( from=[Sender] 0xcffc183d6c445cbde55c1be49bac0b07b6f1618c, to=[Receiver] MiningFixRangeBoost, value=154000000000000000000 )
113 izumiToken.Approval( owner=[Sender] 0xcffc183d6c445cbde55c1be49bac0b07b6f1618c, spender=[Receiver] MiningFixRangeBoost, value=115792089237316195423570985008687907853269984665640564038903584007913129639935 )

Account State Difference:

  Address   Before After State Difference Code
(Crazy Pool)
179.764240478256127167 Eth179.764511932256127167 Eth0.000271454
0x8981c60f...F9cF68f62
0x9ad37205...4CEc5c200
0xCffC183D...7b6F1618C
0.055629859498624235 Eth
Nonce: 1263
0.053993364166655621 Eth
Nonce: 1264
0.001636495331968614

Execution Trace

MiningFixRangeBoost.depositIZI( tokenId=289737, deltaNIZI=154000000000000000000 )
  • izumiToken.transferFrom( sender=0xee264e74A2Fd2c7A55da705b80e092f05DaE5b5D, recipient=0xCffC183D6C445CBDE55c1BE49bAC0b07b6F1618C, amount=3140540106660318807 ) => ( True )
  • izumiToken.transferFrom( sender=0xCffC183D6C445CBDE55c1BE49bAC0b07b6F1618C, recipient=0x8981c60ff02CDBbF2A6AC1a9F150814F9cF68f62, amount=154000000000000000000 ) => ( True )
    File 1 of 2: MiningFixRangeBoost
    // SPDX-License-Identifier: BUSL-1.1
    pragma solidity 0.8.4;
    // Uncomment if needed.
    // import "hardhat/console.sol";
    import "@openzeppelin/contracts/token/ERC20/IERC20.sol";
    import "@openzeppelin/contracts/token/ERC20/utils/SafeERC20.sol";
    import "@openzeppelin/contracts/utils/structs/EnumerableSet.sol";
    import "@openzeppelin/contracts/access/Ownable.sol";
    import "@openzeppelin/contracts/security/ReentrancyGuard.sol";
    import "@openzeppelin/contracts/token/ERC721/IERC721Receiver.sol";
    import "../multicall.sol";
    /// @title Simple math library for Max and Min.
    library Math {
        function max(int24 a, int24 b) internal pure returns (int24) {
            return a >= b ? a : b;
        }
        function min(int24 a, int24 b) internal pure returns (int24) {
            return a < b ? a : b;
        }
        function max(uint256 a, uint256 b) internal pure returns (uint256) {
            return a >= b ? a : b;
        }
        function min(uint256 a, uint256 b) internal pure returns (uint256) {
            return a < b ? a : b;
        }
    }
    /// @title Uniswap V3 Nonfungible Position Manager Interface
    interface PositionManagerV3 {
        function positions(uint256 tokenId)
            external
            view
            returns (
                uint96 nonce,
                address operator,
                address token0,
                address token1,
                uint24 fee,
                int24 tickLower,
                int24 tickUpper,
                uint128 liquidity,
                uint256 feeGrowthInside0LastX128,
                uint256 feeGrowthInside1LastX128,
                uint128 tokensOwed0,
                uint128 tokensOwed1
            );
        function safeTransferFrom(
            address from,
            address to,
            uint256 tokenId
        ) external;
        function ownerOf(uint256 tokenId) external view returns (address);
    }
    /// @title Uniswap V3 Liquidity Mining Main Contract
    contract MiningFixRangeBoost is Ownable, Multicall, ReentrancyGuard, IERC721Receiver {
        using Math for int24;
        using SafeERC20 for IERC20;
        using EnumerableSet for EnumerableSet.UintSet;
        struct PoolInfo {
            address token0;
            address token1;
            uint24 fee;
        }
        /// @dev Contract of the uniV3 Nonfungible Position Manager.
        PositionManagerV3 uniV3NFTManager;
        PoolInfo public rewardPool;
        /// @dev The reward range of this mining contract.
        int24 rewardUpperTick;
        int24 rewardLowerTick;
        /// @dev Last block number that the accRewardRerShare is touched.
        uint256 lastTouchBlock;
        /// @dev The block number when NFT mining rewards starts/ends.
        uint256 startBlock;
        uint256 endBlock;
        struct RewardInfo {
            /// @dev Contract of the reward erc20 token.
            address rewardToken;
            /// @dev who provides reward
            address provider;
            /// @dev Accumulated Reward Tokens per share, times 1e128.
            uint256 accRewardPerShare;
            /// @dev Reward amount for each block.
            uint256 rewardPerBlock;
        }
        mapping(uint256 => RewardInfo) public rewardInfos;
        uint256 public rewardInfosLen;
        /// @dev Store the owner of the NFT token
        mapping(uint256 => address) public owners;
        /// @dev The inverse mapping of owners.
        mapping(address => EnumerableSet.UintSet) private tokenIds;
        /// @dev Record the status for a certain token for the last touched time.
        struct TokenStatus {
            uint256 vLiquidity;
            uint256 validVLiquidity;
            uint256 nIZI;
            uint256 lastTouchBlock;
            uint256[] lastTouchAccRewardPerShare;
        }
        mapping(uint256 => TokenStatus) public tokenStatus;
        function lastTouchAccRewardPerShare(uint256 tokenId)
            external
            view
            returns (uint256[] memory lta)
        {
            TokenStatus memory t = tokenStatus[tokenId];
            uint256 len = t.lastTouchAccRewardPerShare.length;
            lta = new uint256[](len);
            for (uint256 i = 0; i < len; i++) {
                lta[i] = t.lastTouchAccRewardPerShare[i];
            }
            return lta;
        }
        /// @dev token to lock, 0 for not boost
        IERC20 public iziToken;
        /// @dev current total nIZI.
        uint256 public totalNIZI;
        /// @dev Current total virtual liquidity.
        uint256 public totalVLiquidity;
        /// @dev 2 << 128
        uint256 internal constant Q128 = 0x100000000000000000000000000000000;
        // Events
        event Deposit(address indexed user, uint256 tokenId, uint256 nIZI);
        event Withdraw(address indexed user, uint256 tokenId);
        event CollectReward(address indexed user, uint256 tokenId, address token, uint256 amount);
        event ModifyEndBlock(uint256 endBlock);
        event ModifyRewardPerBlock(address indexed rewardToken, uint256 rewardPerBlock);
        event ModifyProvider(address indexed rewardToken, address provider);
        constructor(
            address _uniV3NFTManager,
            address token0,
            address token1,
            uint24 fee,
            RewardInfo[] memory _rewardInfos,
            address iziTokenAddr,
            int24 _rewardUpperTick,
            int24 _rewardLowerTick,
            uint256 _startBlock,
            uint256 _endBlock
        ) {
            uniV3NFTManager = PositionManagerV3(_uniV3NFTManager);
            require(_rewardLowerTick < _rewardUpperTick, "L<U");
            require(token0 < token1, "TOKEN0 < TOKEN1 NOT MATCH");
            rewardPool.token0 = token0;
            rewardPool.token1 = token1;
            rewardPool.fee = fee;
            rewardInfosLen = _rewardInfos.length;
            require(rewardInfosLen > 0, "NO REWARD");
            require(rewardInfosLen < 3, "AT MOST 2 REWARDS");
            for (uint256 i = 0; i < rewardInfosLen; i++) {
                rewardInfos[i] = _rewardInfos[i];
                rewardInfos[i].accRewardPerShare = 0;
            }
            // iziTokenAddr == 0 means not boost
            iziToken = IERC20(iziTokenAddr);
            rewardUpperTick = _rewardUpperTick;
            rewardLowerTick = _rewardLowerTick;
            startBlock = _startBlock;
            endBlock = _endBlock;
            lastTouchBlock = startBlock;
            totalVLiquidity = 0;
            totalNIZI = 0;
        }
        /// @notice Used for ERC721 safeTransferFrom
        function onERC721Received(address, address, uint256, bytes memory) 
            public 
            virtual 
            override 
            returns (bytes4) 
        {
            return this.onERC721Received.selector;
        }
        /// @notice Get the overall info for the mining contract.
        function getMiningContractInfo()
            external
            view
            returns (
                address token0_,
                address token1_,
                uint24 fee_,
                RewardInfo[] memory rewardInfos_,
                address iziTokenAddr_,
                int24 rewardUpperTick_,
                int24 rewardLowerTick_,
                uint256 lastTouchBlock_,
                uint256 totalVLiquidity_,
                uint256 startBlock_,
                uint256 endBlock_
            )
        {
            rewardInfos_ = new RewardInfo[](rewardInfosLen);
            for (uint256 i = 0; i < rewardInfosLen; i++) {
                rewardInfos_[i] = rewardInfos[i];
            }
            return (
                rewardPool.token0,
                rewardPool.token1,
                rewardPool.fee,
                rewardInfos_,
                address(iziToken),
                rewardUpperTick,
                rewardLowerTick,
                lastTouchBlock,
                totalVLiquidity,
                startBlock,
                endBlock
            );
        }
        /// @notice Compute the virtual liquidity from a position's parameters.
        /// @param tickLower The lower tick of a position.
        /// @param tickUpper The upper tick of a position.
        /// @param liquidity The liquidity of a a position.
        /// @dev vLiquidity = liquidity * validRange^2 / 1e6, where the validRange is the tick amount of the
        /// intersection between the position and the reward range.
        /// We divided it by 1e6 to keep vLiquidity smaller than Q128 in most cases. This is safe since liqudity is usually a large number.
        function _getVLiquidityForNFT(
            int24 tickLower,
            int24 tickUpper,
            uint128 liquidity
        ) internal view returns (uint256 vLiquidity) {
            // liquidity is roughly equals to sqrt(amountX*amountY)
            require(liquidity >= 1e6, "LIQUIDITY TOO SMALL");
            uint256 validRange = uint24(
                Math.max(
                    Math.min(rewardUpperTick, tickUpper) - Math.max(rewardLowerTick, tickLower),
                    0
                )
            );
            vLiquidity = (validRange * validRange * uint256(liquidity)) / 1e6;
            return vLiquidity;
        }
        /// @notice new a token status when touched.
        function _newTokenStatus(
            uint256 tokenId,
            uint256 vLiquidity,
            uint256 validVLiquidity,
            uint256 nIZI
        ) internal {
            TokenStatus storage t = tokenStatus[tokenId];
            t.vLiquidity = vLiquidity;
            t.validVLiquidity = validVLiquidity;
            t.nIZI = nIZI;
            t.lastTouchBlock = lastTouchBlock;
            t.lastTouchAccRewardPerShare = new uint256[](rewardInfosLen);
            for (uint256 i = 0; i < rewardInfosLen; i++) {
                t.lastTouchAccRewardPerShare[i] = rewardInfos[i].accRewardPerShare;
            }
        }
        /// @notice update a token status when touched
        function _updateTokenStatus(
            uint256 tokenId,
            uint256 vLiquidity,
            uint256 validVLiquidity,
            uint256 nIZI
        ) internal {
            TokenStatus storage t = tokenStatus[tokenId];
            t.vLiquidity = vLiquidity;
            // when not boost, validVL == vL
            t.validVLiquidity = validVLiquidity;
            t.nIZI = nIZI;
            t.lastTouchBlock = lastTouchBlock;
            for (uint256 i = 0; i < rewardInfosLen; i++) {
                t.lastTouchAccRewardPerShare[i] = rewardInfos[i].accRewardPerShare;
            }
        }
        /// @notice Update reward variables to be up-to-date.
        function _updateVLiquidity(uint256 vLiquidity, bool isAdd) internal {
            if (isAdd) {
                totalVLiquidity = totalVLiquidity + vLiquidity;
            } else {
                totalVLiquidity = totalVLiquidity - vLiquidity;
            }
            // Q128 is enough for 10^5 * 10^5 * 10^18 * 10^10
            require(totalVLiquidity <= Q128, "TOO MUCH LIQUIDITY STAKED");
        }
        function _updateNIZI(uint256 nIZI, bool isAdd) internal {
            if (isAdd) {
                totalNIZI = totalNIZI + nIZI;
            } else {
                totalNIZI = totalNIZI - nIZI;
            }
            // totalNIZI is always < Q96
        }
        /// @notice Update the global status.
        function _updateGlobalStatus() internal {
            if (block.number <= lastTouchBlock) {
                return;
            }
            if (lastTouchBlock >= endBlock) {
                return;
            }
            uint256 currBlockNumber = Math.min(block.number, endBlock);
            if (totalVLiquidity == 0) {
                lastTouchBlock = currBlockNumber;
                return;
            }
            for (uint256 i = 0; i < rewardInfosLen; i++) {
                uint256 tokenReward = (currBlockNumber - lastTouchBlock) * rewardInfos[i].rewardPerBlock;
                rewardInfos[i].accRewardPerShare = rewardInfos[i].accRewardPerShare + ((tokenReward * Q128) / totalVLiquidity);
            }
            lastTouchBlock = currBlockNumber;
        }
        function _computeValidVLiquidity(uint256 vLiquidity, uint256 nIZI)
            internal
            view
            returns (uint256)
        {
            if (totalNIZI == 0) {
                return vLiquidity;
            }
            uint256 iziVLiquidity = (vLiquidity * 4 + (totalVLiquidity * nIZI * 6) / totalNIZI) / 10;
            return Math.min(iziVLiquidity, vLiquidity);
        }
        /// @notice Deposit a single position.
        /// @param tokenId The related position id.
        /// @param nIZI the amount of izi to lock
        function deposit(uint256 tokenId, uint256 nIZI)
            external
            returns (uint256 vLiquidity)
        {
            address owner = uniV3NFTManager.ownerOf(tokenId);
            require(owner == msg.sender, "NOT OWNER");
            (
                ,
                ,
                address token0,
                address token1,
                uint24 fee,
                int24 tickLower,
                int24 tickUpper,
                uint128 liquidity,
                ,
                ,
                ,
            ) = uniV3NFTManager.positions(tokenId);
            // alternatively we can compute the pool address with tokens and fee and compare the address directly
            require(token0 == rewardPool.token0, "TOEKN0 NOT MATCH");
            require(token1 == rewardPool.token1, "TOKEN1 NOT MATCH");
            require(fee == rewardPool.fee, "FEE NOT MATCH");
            // require the NFT token has interaction with [rewardLowerTick, rewardUpperTick]
            vLiquidity = _getVLiquidityForNFT(tickLower, tickUpper, liquidity);
            require(vLiquidity > 0, "INVALID TOKEN");
            uniV3NFTManager.safeTransferFrom(msg.sender, address(this), tokenId);
            owners[tokenId] = msg.sender;
            bool res = tokenIds[msg.sender].add(tokenId);
            require(res);
            // the execution order for the next three lines is crutial
            _updateGlobalStatus();
            _updateVLiquidity(vLiquidity, true);
            if (address(iziToken) == address(0)) {
                // boost is not enabled
                nIZI = 0;
            }
            _updateNIZI(nIZI, true);
            uint256 validVLiquidity = _computeValidVLiquidity(vLiquidity, nIZI);
            require(nIZI < Q128 / 6, "NIZI O");
            _newTokenStatus(tokenId, vLiquidity, validVLiquidity, nIZI);
            if (nIZI > 0) {
                // lock izi in this contract
                iziToken.safeTransferFrom(msg.sender, address(this), nIZI);
            }
            emit Deposit(msg.sender, tokenId, nIZI);
            return vLiquidity;
        }
        /// @notice deposit iZi to an nft token
        /// @param tokenId nft already deposited
        /// @param deltaNIZI amount of izi to deposit
        function depositIZI(uint256 tokenId, uint256 deltaNIZI)
            external
            nonReentrant
        {
            require(owners[tokenId] == msg.sender, "NOT OWNER OR NOT EXIST");
            require(address(iziToken) != address(0), "NOT BOOST");
            require(deltaNIZI > 0, "DEPOSIT IZI MUST BE POSITIVE");
            _collectReward(tokenId);
            TokenStatus memory t = tokenStatus[tokenId];
            _updateNIZI(deltaNIZI, true);
            uint256 nIZI = t.nIZI + deltaNIZI;
            // update validVLiquidity
            uint256 validVLiquidity = _computeValidVLiquidity(t.vLiquidity, nIZI);
            _updateTokenStatus(tokenId, t.vLiquidity, validVLiquidity, nIZI);
            // transfer iZi from user
            iziToken.safeTransferFrom(msg.sender, address(this), deltaNIZI);
        }
        /// @notice withdraw a single position.
        /// @param tokenId The related position id.
        /// @param noReward true if donot collect reward
        function withdraw(uint256 tokenId, bool noReward) external nonReentrant {
            require(owners[tokenId] == msg.sender, "NOT OWNER OR NOT EXIST");
            if (noReward) {
                _updateGlobalStatus();
            } else {
                _collectReward(tokenId);
            }
            uint256 vLiquidity = tokenStatus[tokenId].vLiquidity;
            _updateVLiquidity(vLiquidity, false);
            uint256 nIZI = tokenStatus[tokenId].nIZI;
            if (nIZI > 0) {
                _updateNIZI(nIZI, false);
                // refund iZi to user
                iziToken.safeTransfer(msg.sender, nIZI);
            }
            uniV3NFTManager.safeTransferFrom(address(this), msg.sender, tokenId);
            owners[tokenId] = address(0);
            bool res = tokenIds[msg.sender].remove(tokenId);
            require(res);
            emit Withdraw(msg.sender, tokenId);
        }
        /// @notice Collect pending reward for a single position.
        /// @param tokenId The related position id.
        function _collectReward(uint256 tokenId) internal {
            TokenStatus memory t = tokenStatus[tokenId];
            _updateGlobalStatus();
            for (uint256 i = 0; i < rewardInfosLen; i++) {
                uint256 _reward = (t.validVLiquidity * (rewardInfos[i].accRewardPerShare - t.lastTouchAccRewardPerShare[i])) / Q128;
                if (_reward > 0) {
                    IERC20(rewardInfos[i].rewardToken).safeTransferFrom(
                        rewardInfos[i].provider,
                        msg.sender,
                        _reward
                    );
                }
                emit CollectReward(
                    msg.sender,
                    tokenId,
                    rewardInfos[i].rewardToken,
                    _reward
                );
            }
            uint256 nIZI = t.nIZI;
            // update validVLiquidity
            uint256 validVLiquidity = _computeValidVLiquidity(t.vLiquidity, nIZI);
            _updateTokenStatus(tokenId, t.vLiquidity, validVLiquidity, nIZI);
        }
        /// @notice Collect pending reward for a single position.
        /// @param tokenId The related position id.
        function collectReward(uint256 tokenId) external nonReentrant {
            require(owners[tokenId] == msg.sender, "NOT OWNER OR NOT EXIST");
            _collectReward(tokenId);
        }
        /// @notice Collect all pending rewards.
        function collectRewards() external nonReentrant {
            EnumerableSet.UintSet storage ids = tokenIds[msg.sender];
            for (uint256 i = 0; i < ids.length(); i++) {
                require(owners[ids.at(i)] == msg.sender, "NOT OWNER");
                _collectReward(ids.at(i));
            }
        }
        /// @notice View function to get position ids staked here for an user.
        /// @param _user The related address.
        function getTokenIds(address _user)
            external
            view
            returns (uint256[] memory)
        {
            EnumerableSet.UintSet storage ids = tokenIds[_user];
            // push could not be used in memory array
            // we set the tokenIdList into a fixed-length array rather than dynamic
            uint256[] memory tokenIdList = new uint256[](ids.length());
            for (uint256 i = 0; i < ids.length(); i++) {
                tokenIdList[i] = ids.at(i);
            }
            return tokenIdList;
        }
        /// @notice Return reward multiplier over the given _from to _to block.
        /// @param _from The start block.
        /// @param _to The end block.
        function _getMultiplier(uint256 _from, uint256 _to)
            internal
            view
            returns (uint256)
        {
            if (_from > _to) {
                return 0;
            }
            if (_to <= endBlock) {
                return _to - _from;
            } else if (_from >= endBlock) {
                return 0;
            } else {
                return endBlock - _from;
            }
        }
        /// @notice View function to see pending Reward for a single position.
        /// @param tokenId The related position id.
        function pendingReward(uint256 tokenId)
            public
            view
            returns (uint256[] memory)
        {
            TokenStatus memory t = tokenStatus[tokenId];
            uint256[] memory _reward = new uint256[](rewardInfosLen);
            for (uint256 i = 0; i < rewardInfosLen; i++) {
                uint256 multiplier = _getMultiplier(lastTouchBlock, block.number);
                uint256 tokenReward = multiplier * rewardInfos[i].rewardPerBlock;
                uint256 rewardPerShare = rewardInfos[i].accRewardPerShare + (tokenReward * Q128) / totalVLiquidity;
                // l * (currentAcc - lastAcc)
                _reward[i] = (t.validVLiquidity * (rewardPerShare - t.lastTouchAccRewardPerShare[i])) / Q128;
            }
            return _reward;
        }
        /// @notice View function to see pending Rewards for an address.
        /// @param _user The related address.
        function pendingRewards(address _user)
            external
            view
            returns (uint256[] memory)
        {
            uint256[] memory _reward = new uint256[](rewardInfosLen);
            for (uint256 j = 0; j < rewardInfosLen; j++) {
                _reward[j] = 0;
            }
            for (uint256 i = 0; i < tokenIds[_user].length(); i++) {
                uint256[] memory r = pendingReward(tokenIds[_user].at(i));
                for (uint256 j = 0; j < rewardInfosLen; j++) {
                    _reward[j] += r[j];
                }
            }
            return _reward;
        }
        // Control fuctions for the contract owner and operators.
        /// @notice If something goes wrong, we can send back user's nft and locked iZi
        /// @param tokenId The related position id.
        function emergenceWithdraw(uint256 tokenId) external onlyOwner {
            address owner = owners[tokenId];
            require(owner != address(0));
            uniV3NFTManager.safeTransferFrom(
                address(this),
                owners[tokenId],
                tokenId
            );
            uint256 nIZI = tokenStatus[tokenId].nIZI;
            if (nIZI > 0) {
                // we should ensure nft refund to user
                // omit the case when transfer() returns false unexpectedly
                iziToken.transfer(owner, nIZI);
            }
            // make sure user cannot withdraw/depositIZI or collect reward on this nft
            owners[tokenId] = address(0);
        }
        /// @notice Set new reward end block.
        /// @param _endBlock New end block.
        function modifyEndBlock(uint256 _endBlock) external onlyOwner {
            require(_endBlock > block.number, "OUT OF DATE");
            _updateGlobalStatus();
            // jump if origin endBlock < block.number
            lastTouchBlock = block.number;
            endBlock = _endBlock;
            emit ModifyEndBlock(endBlock);
        }
        /// @notice Set new reward per block.
        /// @param rewardIdx which rewardInfo to modify
        /// @param _rewardPerBlock new reward per block
        function modifyRewardPerBlock(uint256 rewardIdx, uint256 _rewardPerBlock)
            external
            onlyOwner
        {
            require(rewardIdx < rewardInfosLen, "OUT OF REWARD INFO RANGE");
            _updateGlobalStatus();
            rewardInfos[rewardIdx].rewardPerBlock = _rewardPerBlock;
            emit ModifyRewardPerBlock(
                rewardInfos[rewardIdx].rewardToken,
                _rewardPerBlock
            );
        }
        /// @notice Set new reward provider.
        /// @param rewardIdx which rewardInfo to modify
        /// @param provider New provider
        function modifyProvider(uint256 rewardIdx, address provider)
            external
            onlyOwner
        {
            require(rewardIdx < rewardInfosLen, "OUT OF REWARD INFO RANGE");
            rewardInfos[rewardIdx].provider = provider;
            emit ModifyProvider(rewardInfos[rewardIdx].rewardToken, provider);
        }
    }
    // SPDX-License-Identifier: MIT
    // OpenZeppelin Contracts v4.4.1 (token/ERC20/IERC20.sol)
    pragma solidity ^0.8.0;
    /**
     * @dev Interface of the ERC20 standard as defined in the EIP.
     */
    interface IERC20 {
        /**
         * @dev Returns the amount of tokens in existence.
         */
        function totalSupply() external view returns (uint256);
        /**
         * @dev Returns the amount of tokens owned by `account`.
         */
        function balanceOf(address account) external view returns (uint256);
        /**
         * @dev Moves `amount` tokens from the caller's account to `recipient`.
         *
         * Returns a boolean value indicating whether the operation succeeded.
         *
         * Emits a {Transfer} event.
         */
        function transfer(address recipient, uint256 amount) external returns (bool);
        /**
         * @dev Returns the remaining number of tokens that `spender` will be
         * allowed to spend on behalf of `owner` through {transferFrom}. This is
         * zero by default.
         *
         * This value changes when {approve} or {transferFrom} are called.
         */
        function allowance(address owner, address spender) external view returns (uint256);
        /**
         * @dev Sets `amount` as the allowance of `spender` over the caller's tokens.
         *
         * Returns a boolean value indicating whether the operation succeeded.
         *
         * IMPORTANT: Beware that changing an allowance with this method brings the risk
         * that someone may use both the old and the new allowance by unfortunate
         * transaction ordering. One possible solution to mitigate this race
         * condition is to first reduce the spender's allowance to 0 and set the
         * desired value afterwards:
         * https://github.com/ethereum/EIPs/issues/20#issuecomment-263524729
         *
         * Emits an {Approval} event.
         */
        function approve(address spender, uint256 amount) external returns (bool);
        /**
         * @dev Moves `amount` tokens from `sender` to `recipient` using the
         * allowance mechanism. `amount` is then deducted from the caller's
         * allowance.
         *
         * Returns a boolean value indicating whether the operation succeeded.
         *
         * Emits a {Transfer} event.
         */
        function transferFrom(
            address sender,
            address recipient,
            uint256 amount
        ) external returns (bool);
        /**
         * @dev Emitted when `value` tokens are moved from one account (`from`) to
         * another (`to`).
         *
         * Note that `value` may be zero.
         */
        event Transfer(address indexed from, address indexed to, uint256 value);
        /**
         * @dev Emitted when the allowance of a `spender` for an `owner` is set by
         * a call to {approve}. `value` is the new allowance.
         */
        event Approval(address indexed owner, address indexed spender, uint256 value);
    }
    // SPDX-License-Identifier: MIT
    // OpenZeppelin Contracts v4.4.1 (token/ERC20/utils/SafeERC20.sol)
    pragma solidity ^0.8.0;
    import "../IERC20.sol";
    import "../../../utils/Address.sol";
    /**
     * @title SafeERC20
     * @dev Wrappers around ERC20 operations that throw on failure (when the token
     * contract returns false). Tokens that return no value (and instead revert or
     * throw on failure) are also supported, non-reverting calls are assumed to be
     * successful.
     * To use this library you can add a `using SafeERC20 for IERC20;` statement to your contract,
     * which allows you to call the safe operations as `token.safeTransfer(...)`, etc.
     */
    library SafeERC20 {
        using Address for address;
        function safeTransfer(
            IERC20 token,
            address to,
            uint256 value
        ) internal {
            _callOptionalReturn(token, abi.encodeWithSelector(token.transfer.selector, to, value));
        }
        function safeTransferFrom(
            IERC20 token,
            address from,
            address to,
            uint256 value
        ) internal {
            _callOptionalReturn(token, abi.encodeWithSelector(token.transferFrom.selector, from, to, value));
        }
        /**
         * @dev Deprecated. This function has issues similar to the ones found in
         * {IERC20-approve}, and its usage is discouraged.
         *
         * Whenever possible, use {safeIncreaseAllowance} and
         * {safeDecreaseAllowance} instead.
         */
        function safeApprove(
            IERC20 token,
            address spender,
            uint256 value
        ) internal {
            // safeApprove should only be called when setting an initial allowance,
            // or when resetting it to zero. To increase and decrease it, use
            // 'safeIncreaseAllowance' and 'safeDecreaseAllowance'
            require(
                (value == 0) || (token.allowance(address(this), spender) == 0),
                "SafeERC20: approve from non-zero to non-zero allowance"
            );
            _callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, value));
        }
        function safeIncreaseAllowance(
            IERC20 token,
            address spender,
            uint256 value
        ) internal {
            uint256 newAllowance = token.allowance(address(this), spender) + value;
            _callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, newAllowance));
        }
        function safeDecreaseAllowance(
            IERC20 token,
            address spender,
            uint256 value
        ) internal {
            unchecked {
                uint256 oldAllowance = token.allowance(address(this), spender);
                require(oldAllowance >= value, "SafeERC20: decreased allowance below zero");
                uint256 newAllowance = oldAllowance - value;
                _callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, newAllowance));
            }
        }
        /**
         * @dev Imitates a Solidity high-level call (i.e. a regular function call to a contract), relaxing the requirement
         * on the return value: the return value is optional (but if data is returned, it must not be false).
         * @param token The token targeted by the call.
         * @param data The call data (encoded using abi.encode or one of its variants).
         */
        function _callOptionalReturn(IERC20 token, bytes memory data) private {
            // We need to perform a low level call here, to bypass Solidity's return data size checking mechanism, since
            // we're implementing it ourselves. We use {Address.functionCall} to perform this call, which verifies that
            // the target address contains contract code and also asserts for success in the low-level call.
            bytes memory returndata = address(token).functionCall(data, "SafeERC20: low-level call failed");
            if (returndata.length > 0) {
                // Return data is optional
                require(abi.decode(returndata, (bool)), "SafeERC20: ERC20 operation did not succeed");
            }
        }
    }
    // SPDX-License-Identifier: MIT
    // OpenZeppelin Contracts v4.4.1 (utils/structs/EnumerableSet.sol)
    pragma solidity ^0.8.0;
    /**
     * @dev Library for managing
     * https://en.wikipedia.org/wiki/Set_(abstract_data_type)[sets] of primitive
     * types.
     *
     * Sets have the following properties:
     *
     * - Elements are added, removed, and checked for existence in constant time
     * (O(1)).
     * - Elements are enumerated in O(n). No guarantees are made on the ordering.
     *
     * ```
     * contract Example {
     *     // Add the library methods
     *     using EnumerableSet for EnumerableSet.AddressSet;
     *
     *     // Declare a set state variable
     *     EnumerableSet.AddressSet private mySet;
     * }
     * ```
     *
     * As of v3.3.0, sets of type `bytes32` (`Bytes32Set`), `address` (`AddressSet`)
     * and `uint256` (`UintSet`) are supported.
     */
    library EnumerableSet {
        // To implement this library for multiple types with as little code
        // repetition as possible, we write it in terms of a generic Set type with
        // bytes32 values.
        // The Set implementation uses private functions, and user-facing
        // implementations (such as AddressSet) are just wrappers around the
        // underlying Set.
        // This means that we can only create new EnumerableSets for types that fit
        // in bytes32.
        struct Set {
            // Storage of set values
            bytes32[] _values;
            // Position of the value in the `values` array, plus 1 because index 0
            // means a value is not in the set.
            mapping(bytes32 => uint256) _indexes;
        }
        /**
         * @dev Add a value to a set. O(1).
         *
         * Returns true if the value was added to the set, that is if it was not
         * already present.
         */
        function _add(Set storage set, bytes32 value) private returns (bool) {
            if (!_contains(set, value)) {
                set._values.push(value);
                // The value is stored at length-1, but we add 1 to all indexes
                // and use 0 as a sentinel value
                set._indexes[value] = set._values.length;
                return true;
            } else {
                return false;
            }
        }
        /**
         * @dev Removes a value from a set. O(1).
         *
         * Returns true if the value was removed from the set, that is if it was
         * present.
         */
        function _remove(Set storage set, bytes32 value) private returns (bool) {
            // We read and store the value's index to prevent multiple reads from the same storage slot
            uint256 valueIndex = set._indexes[value];
            if (valueIndex != 0) {
                // Equivalent to contains(set, value)
                // To delete an element from the _values array in O(1), we swap the element to delete with the last one in
                // the array, and then remove the last element (sometimes called as 'swap and pop').
                // This modifies the order of the array, as noted in {at}.
                uint256 toDeleteIndex = valueIndex - 1;
                uint256 lastIndex = set._values.length - 1;
                if (lastIndex != toDeleteIndex) {
                    bytes32 lastvalue = set._values[lastIndex];
                    // Move the last value to the index where the value to delete is
                    set._values[toDeleteIndex] = lastvalue;
                    // Update the index for the moved value
                    set._indexes[lastvalue] = valueIndex; // Replace lastvalue's index to valueIndex
                }
                // Delete the slot where the moved value was stored
                set._values.pop();
                // Delete the index for the deleted slot
                delete set._indexes[value];
                return true;
            } else {
                return false;
            }
        }
        /**
         * @dev Returns true if the value is in the set. O(1).
         */
        function _contains(Set storage set, bytes32 value) private view returns (bool) {
            return set._indexes[value] != 0;
        }
        /**
         * @dev Returns the number of values on the set. O(1).
         */
        function _length(Set storage set) private view returns (uint256) {
            return set._values.length;
        }
        /**
         * @dev Returns the value stored at position `index` in the set. O(1).
         *
         * Note that there are no guarantees on the ordering of values inside the
         * array, and it may change when more values are added or removed.
         *
         * Requirements:
         *
         * - `index` must be strictly less than {length}.
         */
        function _at(Set storage set, uint256 index) private view returns (bytes32) {
            return set._values[index];
        }
        /**
         * @dev Return the entire set in an array
         *
         * WARNING: This operation will copy the entire storage to memory, which can be quite expensive. This is designed
         * to mostly be used by view accessors that are queried without any gas fees. Developers should keep in mind that
         * this function has an unbounded cost, and using it as part of a state-changing function may render the function
         * uncallable if the set grows to a point where copying to memory consumes too much gas to fit in a block.
         */
        function _values(Set storage set) private view returns (bytes32[] memory) {
            return set._values;
        }
        // Bytes32Set
        struct Bytes32Set {
            Set _inner;
        }
        /**
         * @dev Add a value to a set. O(1).
         *
         * Returns true if the value was added to the set, that is if it was not
         * already present.
         */
        function add(Bytes32Set storage set, bytes32 value) internal returns (bool) {
            return _add(set._inner, value);
        }
        /**
         * @dev Removes a value from a set. O(1).
         *
         * Returns true if the value was removed from the set, that is if it was
         * present.
         */
        function remove(Bytes32Set storage set, bytes32 value) internal returns (bool) {
            return _remove(set._inner, value);
        }
        /**
         * @dev Returns true if the value is in the set. O(1).
         */
        function contains(Bytes32Set storage set, bytes32 value) internal view returns (bool) {
            return _contains(set._inner, value);
        }
        /**
         * @dev Returns the number of values in the set. O(1).
         */
        function length(Bytes32Set storage set) internal view returns (uint256) {
            return _length(set._inner);
        }
        /**
         * @dev Returns the value stored at position `index` in the set. O(1).
         *
         * Note that there are no guarantees on the ordering of values inside the
         * array, and it may change when more values are added or removed.
         *
         * Requirements:
         *
         * - `index` must be strictly less than {length}.
         */
        function at(Bytes32Set storage set, uint256 index) internal view returns (bytes32) {
            return _at(set._inner, index);
        }
        /**
         * @dev Return the entire set in an array
         *
         * WARNING: This operation will copy the entire storage to memory, which can be quite expensive. This is designed
         * to mostly be used by view accessors that are queried without any gas fees. Developers should keep in mind that
         * this function has an unbounded cost, and using it as part of a state-changing function may render the function
         * uncallable if the set grows to a point where copying to memory consumes too much gas to fit in a block.
         */
        function values(Bytes32Set storage set) internal view returns (bytes32[] memory) {
            return _values(set._inner);
        }
        // AddressSet
        struct AddressSet {
            Set _inner;
        }
        /**
         * @dev Add a value to a set. O(1).
         *
         * Returns true if the value was added to the set, that is if it was not
         * already present.
         */
        function add(AddressSet storage set, address value) internal returns (bool) {
            return _add(set._inner, bytes32(uint256(uint160(value))));
        }
        /**
         * @dev Removes a value from a set. O(1).
         *
         * Returns true if the value was removed from the set, that is if it was
         * present.
         */
        function remove(AddressSet storage set, address value) internal returns (bool) {
            return _remove(set._inner, bytes32(uint256(uint160(value))));
        }
        /**
         * @dev Returns true if the value is in the set. O(1).
         */
        function contains(AddressSet storage set, address value) internal view returns (bool) {
            return _contains(set._inner, bytes32(uint256(uint160(value))));
        }
        /**
         * @dev Returns the number of values in the set. O(1).
         */
        function length(AddressSet storage set) internal view returns (uint256) {
            return _length(set._inner);
        }
        /**
         * @dev Returns the value stored at position `index` in the set. O(1).
         *
         * Note that there are no guarantees on the ordering of values inside the
         * array, and it may change when more values are added or removed.
         *
         * Requirements:
         *
         * - `index` must be strictly less than {length}.
         */
        function at(AddressSet storage set, uint256 index) internal view returns (address) {
            return address(uint160(uint256(_at(set._inner, index))));
        }
        /**
         * @dev Return the entire set in an array
         *
         * WARNING: This operation will copy the entire storage to memory, which can be quite expensive. This is designed
         * to mostly be used by view accessors that are queried without any gas fees. Developers should keep in mind that
         * this function has an unbounded cost, and using it as part of a state-changing function may render the function
         * uncallable if the set grows to a point where copying to memory consumes too much gas to fit in a block.
         */
        function values(AddressSet storage set) internal view returns (address[] memory) {
            bytes32[] memory store = _values(set._inner);
            address[] memory result;
            assembly {
                result := store
            }
            return result;
        }
        // UintSet
        struct UintSet {
            Set _inner;
        }
        /**
         * @dev Add a value to a set. O(1).
         *
         * Returns true if the value was added to the set, that is if it was not
         * already present.
         */
        function add(UintSet storage set, uint256 value) internal returns (bool) {
            return _add(set._inner, bytes32(value));
        }
        /**
         * @dev Removes a value from a set. O(1).
         *
         * Returns true if the value was removed from the set, that is if it was
         * present.
         */
        function remove(UintSet storage set, uint256 value) internal returns (bool) {
            return _remove(set._inner, bytes32(value));
        }
        /**
         * @dev Returns true if the value is in the set. O(1).
         */
        function contains(UintSet storage set, uint256 value) internal view returns (bool) {
            return _contains(set._inner, bytes32(value));
        }
        /**
         * @dev Returns the number of values on the set. O(1).
         */
        function length(UintSet storage set) internal view returns (uint256) {
            return _length(set._inner);
        }
        /**
         * @dev Returns the value stored at position `index` in the set. O(1).
         *
         * Note that there are no guarantees on the ordering of values inside the
         * array, and it may change when more values are added or removed.
         *
         * Requirements:
         *
         * - `index` must be strictly less than {length}.
         */
        function at(UintSet storage set, uint256 index) internal view returns (uint256) {
            return uint256(_at(set._inner, index));
        }
        /**
         * @dev Return the entire set in an array
         *
         * WARNING: This operation will copy the entire storage to memory, which can be quite expensive. This is designed
         * to mostly be used by view accessors that are queried without any gas fees. Developers should keep in mind that
         * this function has an unbounded cost, and using it as part of a state-changing function may render the function
         * uncallable if the set grows to a point where copying to memory consumes too much gas to fit in a block.
         */
        function values(UintSet storage set) internal view returns (uint256[] memory) {
            bytes32[] memory store = _values(set._inner);
            uint256[] memory result;
            assembly {
                result := store
            }
            return result;
        }
    }
    // SPDX-License-Identifier: MIT
    // OpenZeppelin Contracts v4.4.1 (access/Ownable.sol)
    pragma solidity ^0.8.0;
    import "../utils/Context.sol";
    /**
     * @dev Contract module which provides a basic access control mechanism, where
     * there is an account (an owner) that can be granted exclusive access to
     * specific functions.
     *
     * By default, the owner account will be the one that deploys the contract. This
     * can later be changed with {transferOwnership}.
     *
     * This module is used through inheritance. It will make available the modifier
     * `onlyOwner`, which can be applied to your functions to restrict their use to
     * the owner.
     */
    abstract contract Ownable is Context {
        address private _owner;
        event OwnershipTransferred(address indexed previousOwner, address indexed newOwner);
        /**
         * @dev Initializes the contract setting the deployer as the initial owner.
         */
        constructor() {
            _transferOwnership(_msgSender());
        }
        /**
         * @dev Returns the address of the current owner.
         */
        function owner() public view virtual returns (address) {
            return _owner;
        }
        /**
         * @dev Throws if called by any account other than the owner.
         */
        modifier onlyOwner() {
            require(owner() == _msgSender(), "Ownable: caller is not the owner");
            _;
        }
        /**
         * @dev Leaves the contract without owner. It will not be possible to call
         * `onlyOwner` functions anymore. Can only be called by the current owner.
         *
         * NOTE: Renouncing ownership will leave the contract without an owner,
         * thereby removing any functionality that is only available to the owner.
         */
        function renounceOwnership() public virtual onlyOwner {
            _transferOwnership(address(0));
        }
        /**
         * @dev Transfers ownership of the contract to a new account (`newOwner`).
         * Can only be called by the current owner.
         */
        function transferOwnership(address newOwner) public virtual onlyOwner {
            require(newOwner != address(0), "Ownable: new owner is the zero address");
            _transferOwnership(newOwner);
        }
        /**
         * @dev Transfers ownership of the contract to a new account (`newOwner`).
         * Internal function without access restriction.
         */
        function _transferOwnership(address newOwner) internal virtual {
            address oldOwner = _owner;
            _owner = newOwner;
            emit OwnershipTransferred(oldOwner, newOwner);
        }
    }
    // SPDX-License-Identifier: MIT
    // OpenZeppelin Contracts v4.4.1 (security/ReentrancyGuard.sol)
    pragma solidity ^0.8.0;
    /**
     * @dev Contract module that helps prevent reentrant calls to a function.
     *
     * Inheriting from `ReentrancyGuard` will make the {nonReentrant} modifier
     * available, which can be applied to functions to make sure there are no nested
     * (reentrant) calls to them.
     *
     * Note that because there is a single `nonReentrant` guard, functions marked as
     * `nonReentrant` may not call one another. This can be worked around by making
     * those functions `private`, and then adding `external` `nonReentrant` entry
     * points to them.
     *
     * TIP: If you would like to learn more about reentrancy and alternative ways
     * to protect against it, check out our blog post
     * https://blog.openzeppelin.com/reentrancy-after-istanbul/[Reentrancy After Istanbul].
     */
    abstract contract ReentrancyGuard {
        // Booleans are more expensive than uint256 or any type that takes up a full
        // word because each write operation emits an extra SLOAD to first read the
        // slot's contents, replace the bits taken up by the boolean, and then write
        // back. This is the compiler's defense against contract upgrades and
        // pointer aliasing, and it cannot be disabled.
        // The values being non-zero value makes deployment a bit more expensive,
        // but in exchange the refund on every call to nonReentrant will be lower in
        // amount. Since refunds are capped to a percentage of the total
        // transaction's gas, it is best to keep them low in cases like this one, to
        // increase the likelihood of the full refund coming into effect.
        uint256 private constant _NOT_ENTERED = 1;
        uint256 private constant _ENTERED = 2;
        uint256 private _status;
        constructor() {
            _status = _NOT_ENTERED;
        }
        /**
         * @dev Prevents a contract from calling itself, directly or indirectly.
         * Calling a `nonReentrant` function from another `nonReentrant`
         * function is not supported. It is possible to prevent this from happening
         * by making the `nonReentrant` function external, and making it call a
         * `private` function that does the actual work.
         */
        modifier nonReentrant() {
            // On the first call to nonReentrant, _notEntered will be true
            require(_status != _ENTERED, "ReentrancyGuard: reentrant call");
            // Any calls to nonReentrant after this point will fail
            _status = _ENTERED;
            _;
            // By storing the original value once again, a refund is triggered (see
            // https://eips.ethereum.org/EIPS/eip-2200)
            _status = _NOT_ENTERED;
        }
    }
    // SPDX-License-Identifier: MIT
    // OpenZeppelin Contracts v4.4.1 (token/ERC721/IERC721Receiver.sol)
    pragma solidity ^0.8.0;
    /**
     * @title ERC721 token receiver interface
     * @dev Interface for any contract that wants to support safeTransfers
     * from ERC721 asset contracts.
     */
    interface IERC721Receiver {
        /**
         * @dev Whenever an {IERC721} `tokenId` token is transferred to this contract via {IERC721-safeTransferFrom}
         * by `operator` from `from`, this function is called.
         *
         * It must return its Solidity selector to confirm the token transfer.
         * If any other value is returned or the interface is not implemented by the recipient, the transfer will be reverted.
         *
         * The selector can be obtained in Solidity with `IERC721.onERC721Received.selector`.
         */
        function onERC721Received(
            address operator,
            address from,
            uint256 tokenId,
            bytes calldata data
        ) external returns (bytes4);
    }
    //  SPDX-License-Identifier: MIT
    pragma solidity 0.8.4;
    /// @title Multicall
    /// @notice Enables calling multiple methods in a single call to the contract
    abstract contract Multicall {
        function multicall(bytes[] calldata data) external payable returns (bytes[] memory results) {
            results = new bytes[](data.length);
            for (uint256 i = 0; i < data.length; i++) {
                (bool success, bytes memory result) = address(this).delegatecall(data[i]);
                if (!success) {
                    // Next 5 lines from https://ethereum.stackexchange.com/a/83577
                    if (result.length < 68) revert();
                    assembly {
                        result := add(result, 0x04)
                    }
                    revert(abi.decode(result, (string)));
                }
                results[i] = result;
            }
        }
    }// SPDX-License-Identifier: MIT
    // OpenZeppelin Contracts v4.4.1 (utils/Address.sol)
    pragma solidity ^0.8.0;
    /**
     * @dev Collection of functions related to the address type
     */
    library Address {
        /**
         * @dev Returns true if `account` is a contract.
         *
         * [IMPORTANT]
         * ====
         * It is unsafe to assume that an address for which this function returns
         * false is an externally-owned account (EOA) and not a contract.
         *
         * Among others, `isContract` will return false for the following
         * types of addresses:
         *
         *  - an externally-owned account
         *  - a contract in construction
         *  - an address where a contract will be created
         *  - an address where a contract lived, but was destroyed
         * ====
         */
        function isContract(address account) internal view returns (bool) {
            // This method relies on extcodesize, which returns 0 for contracts in
            // construction, since the code is only stored at the end of the
            // constructor execution.
            uint256 size;
            assembly {
                size := extcodesize(account)
            }
            return size > 0;
        }
        /**
         * @dev Replacement for Solidity's `transfer`: sends `amount` wei to
         * `recipient`, forwarding all available gas and reverting on errors.
         *
         * https://eips.ethereum.org/EIPS/eip-1884[EIP1884] increases the gas cost
         * of certain opcodes, possibly making contracts go over the 2300 gas limit
         * imposed by `transfer`, making them unable to receive funds via
         * `transfer`. {sendValue} removes this limitation.
         *
         * https://diligence.consensys.net/posts/2019/09/stop-using-soliditys-transfer-now/[Learn more].
         *
         * IMPORTANT: because control is transferred to `recipient`, care must be
         * taken to not create reentrancy vulnerabilities. Consider using
         * {ReentrancyGuard} or the
         * https://solidity.readthedocs.io/en/v0.5.11/security-considerations.html#use-the-checks-effects-interactions-pattern[checks-effects-interactions pattern].
         */
        function sendValue(address payable recipient, uint256 amount) internal {
            require(address(this).balance >= amount, "Address: insufficient balance");
            (bool success, ) = recipient.call{value: amount}("");
            require(success, "Address: unable to send value, recipient may have reverted");
        }
        /**
         * @dev Performs a Solidity function call using a low level `call`. A
         * plain `call` is an unsafe replacement for a function call: use this
         * function instead.
         *
         * If `target` reverts with a revert reason, it is bubbled up by this
         * function (like regular Solidity function calls).
         *
         * Returns the raw returned data. To convert to the expected return value,
         * use https://solidity.readthedocs.io/en/latest/units-and-global-variables.html?highlight=abi.decode#abi-encoding-and-decoding-functions[`abi.decode`].
         *
         * Requirements:
         *
         * - `target` must be a contract.
         * - calling `target` with `data` must not revert.
         *
         * _Available since v3.1._
         */
        function functionCall(address target, bytes memory data) internal returns (bytes memory) {
            return functionCall(target, data, "Address: low-level call failed");
        }
        /**
         * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], but with
         * `errorMessage` as a fallback revert reason when `target` reverts.
         *
         * _Available since v3.1._
         */
        function functionCall(
            address target,
            bytes memory data,
            string memory errorMessage
        ) internal returns (bytes memory) {
            return functionCallWithValue(target, data, 0, errorMessage);
        }
        /**
         * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
         * but also transferring `value` wei to `target`.
         *
         * Requirements:
         *
         * - the calling contract must have an ETH balance of at least `value`.
         * - the called Solidity function must be `payable`.
         *
         * _Available since v3.1._
         */
        function functionCallWithValue(
            address target,
            bytes memory data,
            uint256 value
        ) internal returns (bytes memory) {
            return functionCallWithValue(target, data, value, "Address: low-level call with value failed");
        }
        /**
         * @dev Same as {xref-Address-functionCallWithValue-address-bytes-uint256-}[`functionCallWithValue`], but
         * with `errorMessage` as a fallback revert reason when `target` reverts.
         *
         * _Available since v3.1._
         */
        function functionCallWithValue(
            address target,
            bytes memory data,
            uint256 value,
            string memory errorMessage
        ) internal returns (bytes memory) {
            require(address(this).balance >= value, "Address: insufficient balance for call");
            require(isContract(target), "Address: call to non-contract");
            (bool success, bytes memory returndata) = target.call{value: value}(data);
            return verifyCallResult(success, returndata, errorMessage);
        }
        /**
         * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
         * but performing a static call.
         *
         * _Available since v3.3._
         */
        function functionStaticCall(address target, bytes memory data) internal view returns (bytes memory) {
            return functionStaticCall(target, data, "Address: low-level static call failed");
        }
        /**
         * @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`],
         * but performing a static call.
         *
         * _Available since v3.3._
         */
        function functionStaticCall(
            address target,
            bytes memory data,
            string memory errorMessage
        ) internal view returns (bytes memory) {
            require(isContract(target), "Address: static call to non-contract");
            (bool success, bytes memory returndata) = target.staticcall(data);
            return verifyCallResult(success, returndata, errorMessage);
        }
        /**
         * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
         * but performing a delegate call.
         *
         * _Available since v3.4._
         */
        function functionDelegateCall(address target, bytes memory data) internal returns (bytes memory) {
            return functionDelegateCall(target, data, "Address: low-level delegate call failed");
        }
        /**
         * @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`],
         * but performing a delegate call.
         *
         * _Available since v3.4._
         */
        function functionDelegateCall(
            address target,
            bytes memory data,
            string memory errorMessage
        ) internal returns (bytes memory) {
            require(isContract(target), "Address: delegate call to non-contract");
            (bool success, bytes memory returndata) = target.delegatecall(data);
            return verifyCallResult(success, returndata, errorMessage);
        }
        /**
         * @dev Tool to verifies that a low level call was successful, and revert if it wasn't, either by bubbling the
         * revert reason using the provided one.
         *
         * _Available since v4.3._
         */
        function verifyCallResult(
            bool success,
            bytes memory returndata,
            string memory errorMessage
        ) internal pure returns (bytes memory) {
            if (success) {
                return returndata;
            } else {
                // Look for revert reason and bubble it up if present
                if (returndata.length > 0) {
                    // The easiest way to bubble the revert reason is using memory via assembly
                    assembly {
                        let returndata_size := mload(returndata)
                        revert(add(32, returndata), returndata_size)
                    }
                } else {
                    revert(errorMessage);
                }
            }
        }
    }
    // SPDX-License-Identifier: MIT
    // OpenZeppelin Contracts v4.4.1 (utils/Context.sol)
    pragma solidity ^0.8.0;
    /**
     * @dev Provides information about the current execution context, including the
     * sender of the transaction and its data. While these are generally available
     * via msg.sender and msg.data, they should not be accessed in such a direct
     * manner, since when dealing with meta-transactions the account sending and
     * paying for execution may not be the actual sender (as far as an application
     * is concerned).
     *
     * This contract is only required for intermediate, library-like contracts.
     */
    abstract contract Context {
        function _msgSender() internal view virtual returns (address) {
            return msg.sender;
        }
        function _msgData() internal view virtual returns (bytes calldata) {
            return msg.data;
        }
    }
    

    File 2 of 2: izumiToken
    // SPDX-License-Identifier: MIT
    
    pragma solidity ^0.8.0;
    
    /*
     * @dev Provides information about the current execution context, including the
     * sender of the transaction and its data. While these are generally available
     * via msg.sender and msg.data, they should not be accessed in such a direct
     * manner, since when dealing with meta-transactions the account sending and
     * paying for execution may not be the actual sender (as far as an application
     * is concerned).
     *
     * This contract is only required for intermediate, library-like contracts.
     */
    abstract contract Context {
        function _msgSender() internal view virtual returns (address) {
            return msg.sender;
        }
    
        function _msgData() internal view virtual returns (bytes calldata) {
            this; // silence state mutability warning without generating bytecode - see https://github.com/ethereum/solidity/issues/2691
            return msg.data;
        }
    }
    
    
    /**
     * @dev Interface of the ERC20 standard as defined in the EIP.
     */
    interface IERC20 {
        /**
         * @dev Returns the amount of tokens in existence.
         */
        function totalSupply() external view returns (uint256);
    
        /**
         * @dev Returns the amount of tokens owned by `account`.
         */
        function balanceOf(address account) external view returns (uint256);
    
        /**
         * @dev Moves `amount` tokens from the caller's account to `recipient`.
         *
         * Returns a boolean value indicating whether the operation succeeded.
         *
         * Emits a {Transfer} event.
         */
        function transfer(address recipient, uint256 amount) external returns (bool);
    
        /**
         * @dev Returns the remaining number of tokens that `spender` will be
         * allowed to spend on behalf of `owner` through {transferFrom}. This is
         * zero by default.
         *
         * This value changes when {approve} or {transferFrom} are called.
         */
        function allowance(address owner, address spender) external view returns (uint256);
    
        /**
         * @dev Sets `amount` as the allowance of `spender` over the caller's tokens.
         *
         * Returns a boolean value indicating whether the operation succeeded.
         *
         * IMPORTANT: Beware that changing an allowance with this method brings the risk
         * that someone may use both the old and the new allowance by unfortunate
         * transaction ordering. One possible solution to mitigate this race
         * condition is to first reduce the spender's allowance to 0 and set the
         * desired value afterwards:
         * https://github.com/ethereum/EIPs/issues/20#issuecomment-263524729
         *
         * Emits an {Approval} event.
         */
        function approve(address spender, uint256 amount) external returns (bool);
    
        /**
         * @dev Moves `amount` tokens from `sender` to `recipient` using the
         * allowance mechanism. `amount` is then deducted from the caller's
         * allowance.
         *
         * Returns a boolean value indicating whether the operation succeeded.
         *
         * Emits a {Transfer} event.
         */
        function transferFrom(address sender, address recipient, uint256 amount) external returns (bool);
    
        /**
         * @dev Emitted when `value` tokens are moved from one account (`from`) to
         * another (`to`).
         *
         * Note that `value` may be zero.
         */
        event Transfer(address indexed from, address indexed to, uint256 value);
    
        /**
         * @dev Emitted when the allowance of a `spender` for an `owner` is set by
         * a call to {approve}. `value` is the new allowance.
         */
        event Approval(address indexed owner, address indexed spender, uint256 value);
    }
    
    
    /**
     * @dev Contract module which provides a basic access control mechanism, where
     * there is an account (an owner) that can be granted exclusive access to
     * specific functions.
     *
     * By default, the owner account will be the one that deploys the contract. This
     * can later be changed with {transferOwnership}.
     *
     * This module is used through inheritance. It will make available the modifier
     * `onlyOwner`, which can be applied to your functions to restrict their use to
     * the owner.
     */
    abstract contract Ownable is Context {
        address private _owner;
    
        event OwnershipTransferred(address indexed previousOwner, address indexed newOwner);
    
        /**
         * @dev Initializes the contract setting the deployer as the initial owner.
         */
        constructor () {
            address msgSender = _msgSender();
            _owner = msgSender;
            emit OwnershipTransferred(address(0), msgSender);
        }
    
        /**
         * @dev Returns the address of the current owner.
         */
        function owner() public view virtual returns (address) {
            return _owner;
        }
    
        /**
         * @dev Throws if called by any account other than the owner.
         */
        modifier onlyOwner() {
            require(owner() == _msgSender(), "Ownable: caller is not the owner");
            _;
        }
    
        /**
         * @dev Leaves the contract without owner. It will not be possible to call
         * `onlyOwner` functions anymore. Can only be called by the current owner.
         *
         * NOTE: Renouncing ownership will leave the contract without an owner,
         * thereby removing any functionality that is only available to the owner.
         */
        function renounceOwnership() public virtual onlyOwner {
            emit OwnershipTransferred(_owner, address(0));
            _owner = address(0);
        }
    
        /**
         * @dev Transfers ownership of the contract to a new account (`newOwner`).
         * Can only be called by the current owner.
         */
        function transferOwnership(address newOwner) public virtual onlyOwner {
            require(newOwner != address(0), "Ownable: new owner is the zero address");
            emit OwnershipTransferred(_owner, newOwner);
            _owner = newOwner;
        }
    }
    
    /**
     * @dev Implementation of the {IERC20} interface.
     *
     * This implementation is agnostic to the way tokens are created. This means
     * that a supply mechanism has to be added in a derived contract using {_mint}.
     * For a generic mechanism see {ERC20PresetMinterPauser}.
     *
     * TIP: For a detailed writeup see our guide
     * https://forum.zeppelin.solutions/t/how-to-implement-erc20-supply-mechanisms/226[How
     * to implement supply mechanisms].
     *
     * We have followed general OpenZeppelin guidelines: functions revert instead
     * of returning `false` on failure. This behavior is nonetheless conventional
     * and does not conflict with the expectations of ERC20 applications.
     *
     * Additionally, an {Approval} event is emitted on calls to {transferFrom}.
     * This allows applications to reconstruct the allowance for all accounts just
     * by listening to said events. Other implementations of the EIP may not emit
     * these events, as it isn't required by the specification.
     *
     * Finally, the non-standard {decreaseAllowance} and {increaseAllowance}
     * functions have been added to mitigate the well-known issues around setting
     * allowances. See {IERC20-approve}.
     */
    contract ERC20 is Context, IERC20 {
        mapping (address => uint256) private _balances;
    
        mapping (address => mapping (address => uint256)) private _allowances;
    
        uint256 private _totalSupply;
    
        string private _name;
        string private _symbol;
    
        /**
         * @dev Sets the values for {name} and {symbol}.
         *
         * The defaut value of {decimals} is 18. To select a different value for
         * {decimals} you should overload it.
         *
         * All three of these values are immutable: they can only be set once during
         * construction.
         */
        constructor (string memory name_, string memory symbol_) {
            _name = name_;
            _symbol = symbol_;
        }
    
        /**
         * @dev Returns the name of the token.
         */
        function name() public view virtual returns (string memory) {
            return _name;
        }
    
        /**
         * @dev Returns the symbol of the token, usually a shorter version of the
         * name.
         */
        function symbol() public view virtual returns (string memory) {
            return _symbol;
        }
    
        /**
         * @dev Returns the number of decimals used to get its user representation.
         * For example, if `decimals` equals `2`, a balance of `505` tokens should
         * be displayed to a user as `5,05` (`505 / 10 ** 2`).
         *
         * Tokens usually opt for a value of 18, imitating the relationship between
         * Ether and Wei. This is the value {ERC20} uses, unless this function is
         * overloaded;
         *
         * NOTE: This information is only used for _display_ purposes: it in
         * no way affects any of the arithmetic of the contract, including
         * {IERC20-balanceOf} and {IERC20-transfer}.
         */
        function decimals() public view virtual returns (uint8) {
            return 18;
        }
    
        /**
         * @dev See {IERC20-totalSupply}.
         */
        function totalSupply() public view virtual override returns (uint256) {
            return _totalSupply;
        }
    
        /**
         * @dev See {IERC20-balanceOf}.
         */
        function balanceOf(address account) public view virtual override returns (uint256) {
            return _balances[account];
        }
    
        /**
         * @dev See {IERC20-transfer}.
         *
         * Requirements:
         *
         * - `recipient` cannot be the zero address.
         * - the caller must have a balance of at least `amount`.
         */
        function transfer(address recipient, uint256 amount) public virtual override returns (bool) {
            _transfer(_msgSender(), recipient, amount);
            return true;
        }
    
        /**
         * @dev See {IERC20-allowance}.
         */
        function allowance(address owner, address spender) public view virtual override returns (uint256) {
            return _allowances[owner][spender];
        }
    
        /**
         * @dev See {IERC20-approve}.
         *
         * Requirements:
         *
         * - `spender` cannot be the zero address.
         */
        function approve(address spender, uint256 amount) public virtual override returns (bool) {
            _approve(_msgSender(), spender, amount);
            return true;
        }
    
        /**
         * @dev See {IERC20-transferFrom}.
         *
         * Emits an {Approval} event indicating the updated allowance. This is not
         * required by the EIP. See the note at the beginning of {ERC20}.
         *
         * Requirements:
         *
         * - `sender` and `recipient` cannot be the zero address.
         * - `sender` must have a balance of at least `amount`.
         * - the caller must have allowance for ``sender``'s tokens of at least
         * `amount`.
         */
        function transferFrom(address sender, address recipient, uint256 amount) public virtual override returns (bool) {
            _transfer(sender, recipient, amount);
    
            uint256 currentAllowance = _allowances[sender][_msgSender()];
            require(currentAllowance >= amount, "ERC20: transfer amount exceeds allowance");
            _approve(sender, _msgSender(), currentAllowance - amount);
    
            return true;
        }
    
        /**
         * @dev Atomically increases the allowance granted to `spender` by the caller.
         *
         * This is an alternative to {approve} that can be used as a mitigation for
         * problems described in {IERC20-approve}.
         *
         * Emits an {Approval} event indicating the updated allowance.
         *
         * Requirements:
         *
         * - `spender` cannot be the zero address.
         */
        function increaseAllowance(address spender, uint256 addedValue) public virtual returns (bool) {
            _approve(_msgSender(), spender, _allowances[_msgSender()][spender] + addedValue);
            return true;
        }
    
        /**
         * @dev Atomically decreases the allowance granted to `spender` by the caller.
         *
         * This is an alternative to {approve} that can be used as a mitigation for
         * problems described in {IERC20-approve}.
         *
         * Emits an {Approval} event indicating the updated allowance.
         *
         * Requirements:
         *
         * - `spender` cannot be the zero address.
         * - `spender` must have allowance for the caller of at least
         * `subtractedValue`.
         */
        function decreaseAllowance(address spender, uint256 subtractedValue) public virtual returns (bool) {
            uint256 currentAllowance = _allowances[_msgSender()][spender];
            require(currentAllowance >= subtractedValue, "ERC20: decreased allowance below zero");
            _approve(_msgSender(), spender, currentAllowance - subtractedValue);
    
            return true;
        }
    
        /**
         * @dev Moves tokens `amount` from `sender` to `recipient`.
         *
         * This is internal function is equivalent to {transfer}, and can be used to
         * e.g. implement automatic token fees, slashing mechanisms, etc.
         *
         * Emits a {Transfer} event.
         *
         * Requirements:
         *
         * - `sender` cannot be the zero address.
         * - `recipient` cannot be the zero address.
         * - `sender` must have a balance of at least `amount`.
         */
        function _transfer(address sender, address recipient, uint256 amount) internal virtual {
            require(sender != address(0), "ERC20: transfer from the zero address");
            require(recipient != address(0), "ERC20: transfer to the zero address");
    
            _beforeTokenTransfer(sender, recipient, amount);
    
            uint256 senderBalance = _balances[sender];
            require(senderBalance >= amount, "ERC20: transfer amount exceeds balance");
            _balances[sender] = senderBalance - amount;
            _balances[recipient] += amount;
    
            emit Transfer(sender, recipient, amount);
        }
    
        /** @dev Creates `amount` tokens and assigns them to `account`, increasing
         * the total supply.
         *
         * Emits a {Transfer} event with `from` set to the zero address.
         *
         * Requirements:
         *
         * - `to` cannot be the zero address.
         */
        function _mint(address account, uint256 amount) internal virtual {
            require(account != address(0), "ERC20: mint to the zero address");
    
            _beforeTokenTransfer(address(0), account, amount);
    
            _totalSupply += amount;
            _balances[account] += amount;
            emit Transfer(address(0), account, amount);
        }
    
        /**
         * @dev Destroys `amount` tokens from `account`, reducing the
         * total supply.
         *
         * Emits a {Transfer} event with `to` set to the zero address.
         *
         * Requirements:
         *
         * - `account` cannot be the zero address.
         * - `account` must have at least `amount` tokens.
         */
        function _burn(address account, uint256 amount) internal virtual {
            require(account != address(0), "ERC20: burn from the zero address");
    
            _beforeTokenTransfer(account, address(0), amount);
    
            uint256 accountBalance = _balances[account];
            require(accountBalance >= amount, "ERC20: burn amount exceeds balance");
            _balances[account] = accountBalance - amount;
            _totalSupply -= amount;
    
            emit Transfer(account, address(0), amount);
        }
    
        /**
         * @dev Sets `amount` as the allowance of `spender` over the `owner` s tokens.
         *
         * This internal function is equivalent to `approve`, and can be used to
         * e.g. set automatic allowances for certain subsystems, etc.
         *
         * Emits an {Approval} event.
         *
         * Requirements:
         *
         * - `owner` cannot be the zero address.
         * - `spender` cannot be the zero address.
         */
        function _approve(address owner, address spender, uint256 amount) internal virtual {
            require(owner != address(0), "ERC20: approve from the zero address");
            require(spender != address(0), "ERC20: approve to the zero address");
    
            _allowances[owner][spender] = amount;
            emit Approval(owner, spender, amount);
        }
    
        /**
         * @dev Hook that is called before any transfer of tokens. This includes
         * minting and burning.
         *
         * Calling conditions:
         *
         * - when `from` and `to` are both non-zero, `amount` of ``from``'s tokens
         * will be to transferred to `to`.
         * - when `from` is zero, `amount` tokens will be minted for `to`.
         * - when `to` is zero, `amount` of ``from``'s tokens will be burned.
         * - `from` and `to` are never both zero.
         *
         * To learn more about hooks, head to xref:ROOT:extending-contracts.adoc#using-hooks[Using Hooks].
         */
        function _beforeTokenTransfer(address from, address to, uint256 amount) internal virtual { }
    }
    
    contract OperableToken is ERC20, Ownable  {
      address public operator;
      mapping(address=>bool) public trusted;
    
      modifier onlyTrusted {
          require(trusted[msg.sender] || msg.sender == owner(), "not trusted");
          _;
      }
      modifier onlyOperator {
          require(msg.sender == operator, "operator only");
          _;
      }
    
      constructor(string memory name, string memory symbol) ERC20(name, symbol) {
        operator = msg.sender;
      }
    
      function transferOperator(address newOperator) public onlyOperator {
        require(newOperator != address(0), "zero operator");
        operator = newOperator;
      }
    
      function addTrusted(address user) public onlyOperator {
          trusted[user] = true;
      }
    
      function removeTrusted(address user) public onlyOperator {
          trusted[user] = false;
      }
    
      function mint(address account, uint amount) public onlyTrusted {
        _mint(account, amount);
      }
    
      function burn(address account, uint amount) public onlyTrusted {
        _burn(account, amount);
      }
    
    }
    
    contract izumiToken is OperableToken {
      constructor() OperableToken("izumi Token", "iZi") {}
    }