S Price: $0.43495 (-12.62%)

Token

Wind 💨 (WIND)

Overview

Max Total Supply

10,000,000 WIND

Holders

47

Market

Price

$0.00 @ 0.000000 S

Onchain Market Cap

$0.00

Circulating Supply Market Cap

-

Other Info

Token Contract (WITH 18 Decimals)

Balance
107.593860724146912758 WIND

Value
$0.00
0xa7ff0bc6f74fc6c49ccba87f3df51b1adf98543d
Loading...
Loading
Loading...
Loading
Loading...
Loading

Click here to update the token information / general information

Contract Source Code Verified (Exact Match)

Contract Name:
WindToken

Compiler Version
v0.8.20+commit.a1b79de6

Optimization Enabled:
No with 200 runs

Other Settings:
paris EvmVersion
File 1 of 9 : Wind.sol
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.20;

import { ERC20 } from "@openzeppelin/contracts/token/ERC20/ERC20.sol";
import { Ownable } from "@openzeppelin/contracts/access/Ownable.sol";
import { IWind } from "interfaces/IWind.sol";
import {ReentrancyGuard} from "@openzeppelin/contracts/utils/ReentrancyGuard.sol";

/**
 * @title WindToken
 * @author bravenoob21
 * @dev Implements a multi-phase token distribution with random allocation amounts
 */
contract WindToken is ERC20, Ownable, IWind, ReentrancyGuard {
    
	uint256 public constant MAX_SUPPLY = 10_000_000 * 10 ** 18;
	uint256 public constant LP_ALLOCATION = 1_000_000 * 10 ** 18;
	uint256 public constant FUTURE_GROWTH_ALLOCATION = 2_000_000 * 10 ** 18;

	// Phase allocations
	uint256 public constant PHASE_1_ALLOCATION = 1_000_000 * 10 ** 18;
	uint256 public constant PHASE_2_ALLOCATION = 2_000_000 * 10 ** 18;
	uint256 public constant PHASE_3_ALLOCATION = 4_000_000 * 10 ** 18;

	// Constants for logarithmic reduction
	uint256 private constant REDUCTION_BASE = 1000; //100%
	uint256 private constant MIN_REDUCTION = 100; // 10%

	// Current phase tracking
	uint8 public currentPhase = 0;
	bool public activePhase;
	uint256 public phaseStartTime;
	uint256 public claimActivationTime;
	uint256 public currentPhaseRemainingTokens;
	bool public claimDropCompleted;

	uint256 public firstTimeThreshold = 5;
	uint256 public secondTimeThreshold = 10;

	mapping(address => uint256) public claimCount;
	mapping(address => uint256) public pendingClaims;
	mapping(address => uint256) public transferAmountUsed;
	mapping(address => bool) public excludedDexes;

	// against bots.
	uint256 private salt;

	event WindClaimed(address indexed receiver, string indexed affiliate, uint256 amount);
	event TokensTransfered(address indexed owner, uint256 amount);

	
	constructor() ERC20(unicode"Wind 💨", "WIND") Ownable(msg.sender) {
		// Mint all tokens at start
		_mint(address(this), MAX_SUPPLY);

		// Transfer lp allocation
		_transfer(address(this), msg.sender, LP_ALLOCATION);
	}

	/**
	 * @notice Allows users to claim WIND tokens during an active phase
	 * @dev The amount claimed depends on elapsed time since phase start and user's current balance
	 *      Emits a WindClaimed event on successful claim
	 */
	function claimWind(string memory affiliate) external nonReentrant {
		require(msg.sender == tx.origin, "No contracts");
		require(currentPhase > 0, "Claimdrop not active");
		require(activePhase, "Claimdrop not active");
		require(block.timestamp >= phaseStartTime, "Claimdrop not active");

		uint256 elapsedTime = block.timestamp - phaseStartTime;
		uint256 baseAllocation;

		if (elapsedTime <= firstTimeThreshold) {
			baseAllocation = 1 * 10 ** 18;
		} else if (elapsedTime <= secondTimeThreshold) {
			baseAllocation = _randomNumber(1, 10) * 10 ** 18;
		} else {
			uint256 currentBalance = balanceOf(msg.sender);
			uint256 maxAllocation;
			if (currentBalance == 0){
				maxAllocation = 10;
			} else if (currentBalance >= 1000 * 10 ** 18){
				maxAllocation = 1000;
			}else{
				maxAllocation = currentBalance / 10 ** 18;
			}
			
			uint256 multiplier = getReductionMultiplier(claimCount[msg.sender]);
			maxAllocation = (maxAllocation * multiplier) / REDUCTION_BASE;
			maxAllocation = maxAllocation > 10 ? maxAllocation : 10;
			maxAllocation = maxAllocation;

			baseAllocation = _randomNumber(10, maxAllocation) * 10 ** 18;
			claimCount[msg.sender]++;
		}

		if (baseAllocation > currentPhaseRemainingTokens) {
			baseAllocation = currentPhaseRemainingTokens;
			_endPhase();
		}

		pendingClaims[msg.sender] += baseAllocation;
		currentPhaseRemainingTokens -= baseAllocation;

		emit WindClaimed(msg.sender, affiliate, baseAllocation);
	}

	/**
	 * @notice Transfers previously claimed tokens to the user
	 * @dev Requires that token transfer is active and user has pending claims
	 *      User must also have sufficient balance for transfer
	 *      Emits a TokensTransfered event on successful transfer
	 */
	function transferTokens() external {
		require(claimActivationTime > 0 && block.timestamp >= claimActivationTime, "Transfer not active yet");
		require(pendingClaims[msg.sender] > 0, "No tokens to transfer");
		require(balanceOf(msg.sender) - transferAmountUsed[msg.sender] >= pendingClaims[msg.sender], "Insufficient balance to transfer, own more $WIND");

		uint256 amountToClaim = pendingClaims[msg.sender];
		pendingClaims[msg.sender] = 0;
		transferAmountUsed[msg.sender] += amountToClaim;
		_transfer(address(this), msg.sender, amountToClaim);

		emit TokensTransfered(msg.sender, amountToClaim);
	}

	/**
	 * @notice Calculates a reduction multiplier based on the number of claims
	 * @dev Uses a hyperbolic reduction function with a minimum floor
	 * @param claims Number of previous claims made by an address
	 * @return Reduction multiplier between MIN_REDUCTION and REDUCTION_BASE
	 */
	function getReductionMultiplier(uint256 claims) public pure returns (uint256) {
		if (claims == 0) return REDUCTION_BASE;
		uint256 denominator = claims + 1;
		uint256 reduction = REDUCTION_BASE / denominator;
		return reduction > MIN_REDUCTION ? reduction : MIN_REDUCTION;
	}

	/**
	 * @notice Returns the current circulating supply of tokens
	 * @dev Calculated as the difference between MAX_SUPPLY and the contract's balance
	 * @return Current circulating supply in wei
	 */
	function circulatingSupply() public view returns (uint256) {
		return MAX_SUPPLY - balanceOf(address(this));
	}

	/**
	 * @notice Activates the next phase of token minting
	 * @dev Can only be called by the contract owner
	 *      Sets the phase allocation based on the current phase number
	 */
	function activateMinting() external onlyOwner {
		require(currentPhase < 3, "All phases completed");

		salt = uint256(keccak256(abi.encodePacked(block.timestamp, block.prevrandao, msg.sender)));

		currentPhase++;
		phaseStartTime = block.timestamp;
		claimActivationTime = 0;
		activePhase = true;

		// Set the allocation for the current phase
		if (currentPhase == 1) {
			currentPhaseRemainingTokens = PHASE_1_ALLOCATION;
		} else if (currentPhase == 2) {
			currentPhaseRemainingTokens = PHASE_2_ALLOCATION;
		} else if (currentPhase == 3) {
			currentPhaseRemainingTokens = PHASE_3_ALLOCATION;
		}
	}

	/**
	 * @notice Manually ends the current phase
	 * @dev Can only be called by the contract owner
	 *      Requires an active phase to be running
	 */
	function endPhaseManually() external onlyOwner {
		require(currentPhase > 0, "No active phase");
		require(activePhase, "Phase already ended");
		_endPhase();
	}

	/**
	 * @notice Completes the claimdrop and transfers future growth allocation to owner
	 * @dev Can only be called by the contract owner and only after phase 3
	 */
	function completeClaimDrop() external onlyOwner {
		require(currentPhase == 3, "Only after 3 drops");
		require(!activePhase, "End phase first");
		claimDropCompleted = true;
		_transfer(address(this), owner(), balanceOf(address(this)));
	}

	/**
	 * @notice Updates the time thresholds used in the claimWind function
	 * @dev Can only be called by the contract owner
	 * @param _firstTimeThreshold New value for the first time threshold in seconds
	 * @param _secondTimeThreshold New value for the second time threshold in seconds
	 */
	function updateTimeThresholds(uint256 _firstTimeThreshold, uint256 _secondTimeThreshold) external onlyOwner {
		firstTimeThreshold = _firstTimeThreshold;
		secondTimeThreshold = _secondTimeThreshold;
	}

	/**
	 * @notice Adds a DEX address to the exclusion list
	 * @dev Can only be called by the contract owner
	 * @param dexAddress The address of the DEX to exclude
	 */
	function excludeDex(address dexAddress) external onlyOwner {
		require(dexAddress != address(0), "Cannot exclude zero address");
		require(!excludedDexes[dexAddress], "DEX already excluded");
		excludedDexes[dexAddress] = true;
	}

	/**
	 * @notice Removes a DEX address from the exclusion list
	 * @dev Can only be called by the contract owner
	 * @param dexAddress The address of the DEX to include
	 */
	function includeDex(address dexAddress) external onlyOwner {
		require(excludedDexes[dexAddress], "DEX not excluded");
		excludedDexes[dexAddress] = false;
	}

	/**
	 * @dev Generates a pseudo-random number between min and max (inclusive)
	 * @param min The minimum value of the random number
	 * @param max The maximum value of the random number
	 * @return A pseudo-random number between min and max
	 */
	function _randomNumber(uint256 min, uint256 max) internal returns (uint256) {
		if (min >= max) return min;

		salt = uint256(keccak256(abi.encodePacked(block.timestamp, block.prevrandao, msg.sender, salt)));

		return min + (salt % (max - min + 1));
	}

	/**
	 * @dev Ends the current phase and sets up timing for claim transfers
	 */
	function _endPhase() internal {
		activePhase = false;
		claimActivationTime = block.timestamp + 30 minutes;
	}

	/**
	 * @dev Overrides the ERC20 _update function to implement transfer restrictions
	 * @param from The address tokens are transferred from
	 * @param to The address tokens are transferred to
	 * @param amount The amount of tokens being transferred
	 */
	function _update(address from, address to, uint256 amount) internal override {
		super._update(from, to, amount);

		// Skip if minting or burning
		if (!claimDropCompleted && from != address(0) && to != address(0)) {
			if (!excludedDexes[from] && !excludedDexes[to]) {
				// can't use balance several times
				if (transferAmountUsed[from] > 0) {
					transferAmountUsed[to] += transferAmountUsed[from];
				}
			}
		}
	}
}

File 2 of 9 : Ownable.sol
// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts (last updated v5.0.0) (access/Ownable.sol)

pragma solidity ^0.8.20;

import {Context} from "../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.
 *
 * The initial owner is set to the address provided by the deployer. 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;

    /**
     * @dev The caller account is not authorized to perform an operation.
     */
    error OwnableUnauthorizedAccount(address account);

    /**
     * @dev The owner is not a valid owner account. (eg. `address(0)`)
     */
    error OwnableInvalidOwner(address owner);

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

    /**
     * @dev Initializes the contract setting the address provided by the deployer as the initial owner.
     */
    constructor(address initialOwner) {
        if (initialOwner == address(0)) {
            revert OwnableInvalidOwner(address(0));
        }
        _transferOwnership(initialOwner);
    }

    /**
     * @dev Throws if called by any account other than the owner.
     */
    modifier onlyOwner() {
        _checkOwner();
        _;
    }

    /**
     * @dev Returns the address of the current owner.
     */
    function owner() public view virtual returns (address) {
        return _owner;
    }

    /**
     * @dev Throws if the sender is not the owner.
     */
    function _checkOwner() internal view virtual {
        if (owner() != _msgSender()) {
            revert OwnableUnauthorizedAccount(_msgSender());
        }
    }

    /**
     * @dev Leaves the contract without owner. It will not be possible to call
     * `onlyOwner` functions. Can only be called by the current owner.
     *
     * NOTE: Renouncing ownership will leave the contract without an owner,
     * thereby disabling 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 {
        if (newOwner == address(0)) {
            revert OwnableInvalidOwner(address(0));
        }
        _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);
    }
}

File 3 of 9 : draft-IERC6093.sol
// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts (last updated v5.1.0) (interfaces/draft-IERC6093.sol)
pragma solidity ^0.8.20;

/**
 * @dev Standard ERC-20 Errors
 * Interface of the https://eips.ethereum.org/EIPS/eip-6093[ERC-6093] custom errors for ERC-20 tokens.
 */
interface IERC20Errors {
    /**
     * @dev Indicates an error related to the current `balance` of a `sender`. Used in transfers.
     * @param sender Address whose tokens are being transferred.
     * @param balance Current balance for the interacting account.
     * @param needed Minimum amount required to perform a transfer.
     */
    error ERC20InsufficientBalance(address sender, uint256 balance, uint256 needed);

    /**
     * @dev Indicates a failure with the token `sender`. Used in transfers.
     * @param sender Address whose tokens are being transferred.
     */
    error ERC20InvalidSender(address sender);

    /**
     * @dev Indicates a failure with the token `receiver`. Used in transfers.
     * @param receiver Address to which tokens are being transferred.
     */
    error ERC20InvalidReceiver(address receiver);

    /**
     * @dev Indicates a failure with the `spender`’s `allowance`. Used in transfers.
     * @param spender Address that may be allowed to operate on tokens without being their owner.
     * @param allowance Amount of tokens a `spender` is allowed to operate with.
     * @param needed Minimum amount required to perform a transfer.
     */
    error ERC20InsufficientAllowance(address spender, uint256 allowance, uint256 needed);

    /**
     * @dev Indicates a failure with the `approver` of a token to be approved. Used in approvals.
     * @param approver Address initiating an approval operation.
     */
    error ERC20InvalidApprover(address approver);

    /**
     * @dev Indicates a failure with the `spender` to be approved. Used in approvals.
     * @param spender Address that may be allowed to operate on tokens without being their owner.
     */
    error ERC20InvalidSpender(address spender);
}

/**
 * @dev Standard ERC-721 Errors
 * Interface of the https://eips.ethereum.org/EIPS/eip-6093[ERC-6093] custom errors for ERC-721 tokens.
 */
interface IERC721Errors {
    /**
     * @dev Indicates that an address can't be an owner. For example, `address(0)` is a forbidden owner in ERC-20.
     * Used in balance queries.
     * @param owner Address of the current owner of a token.
     */
    error ERC721InvalidOwner(address owner);

    /**
     * @dev Indicates a `tokenId` whose `owner` is the zero address.
     * @param tokenId Identifier number of a token.
     */
    error ERC721NonexistentToken(uint256 tokenId);

    /**
     * @dev Indicates an error related to the ownership over a particular token. Used in transfers.
     * @param sender Address whose tokens are being transferred.
     * @param tokenId Identifier number of a token.
     * @param owner Address of the current owner of a token.
     */
    error ERC721IncorrectOwner(address sender, uint256 tokenId, address owner);

    /**
     * @dev Indicates a failure with the token `sender`. Used in transfers.
     * @param sender Address whose tokens are being transferred.
     */
    error ERC721InvalidSender(address sender);

    /**
     * @dev Indicates a failure with the token `receiver`. Used in transfers.
     * @param receiver Address to which tokens are being transferred.
     */
    error ERC721InvalidReceiver(address receiver);

    /**
     * @dev Indicates a failure with the `operator`’s approval. Used in transfers.
     * @param operator Address that may be allowed to operate on tokens without being their owner.
     * @param tokenId Identifier number of a token.
     */
    error ERC721InsufficientApproval(address operator, uint256 tokenId);

    /**
     * @dev Indicates a failure with the `approver` of a token to be approved. Used in approvals.
     * @param approver Address initiating an approval operation.
     */
    error ERC721InvalidApprover(address approver);

    /**
     * @dev Indicates a failure with the `operator` to be approved. Used in approvals.
     * @param operator Address that may be allowed to operate on tokens without being their owner.
     */
    error ERC721InvalidOperator(address operator);
}

/**
 * @dev Standard ERC-1155 Errors
 * Interface of the https://eips.ethereum.org/EIPS/eip-6093[ERC-6093] custom errors for ERC-1155 tokens.
 */
interface IERC1155Errors {
    /**
     * @dev Indicates an error related to the current `balance` of a `sender`. Used in transfers.
     * @param sender Address whose tokens are being transferred.
     * @param balance Current balance for the interacting account.
     * @param needed Minimum amount required to perform a transfer.
     * @param tokenId Identifier number of a token.
     */
    error ERC1155InsufficientBalance(address sender, uint256 balance, uint256 needed, uint256 tokenId);

    /**
     * @dev Indicates a failure with the token `sender`. Used in transfers.
     * @param sender Address whose tokens are being transferred.
     */
    error ERC1155InvalidSender(address sender);

    /**
     * @dev Indicates a failure with the token `receiver`. Used in transfers.
     * @param receiver Address to which tokens are being transferred.
     */
    error ERC1155InvalidReceiver(address receiver);

    /**
     * @dev Indicates a failure with the `operator`’s approval. Used in transfers.
     * @param operator Address that may be allowed to operate on tokens without being their owner.
     * @param owner Address of the current owner of a token.
     */
    error ERC1155MissingApprovalForAll(address operator, address owner);

    /**
     * @dev Indicates a failure with the `approver` of a token to be approved. Used in approvals.
     * @param approver Address initiating an approval operation.
     */
    error ERC1155InvalidApprover(address approver);

    /**
     * @dev Indicates a failure with the `operator` to be approved. Used in approvals.
     * @param operator Address that may be allowed to operate on tokens without being their owner.
     */
    error ERC1155InvalidOperator(address operator);

    /**
     * @dev Indicates an array length mismatch between ids and values in a safeBatchTransferFrom operation.
     * Used in batch transfers.
     * @param idsLength Length of the array of token identifiers
     * @param valuesLength Length of the array of token amounts
     */
    error ERC1155InvalidArrayLength(uint256 idsLength, uint256 valuesLength);
}

File 4 of 9 : ERC20.sol
// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts (last updated v5.2.0) (token/ERC20/ERC20.sol)

pragma solidity ^0.8.20;

import {IERC20} from "./IERC20.sol";
import {IERC20Metadata} from "./extensions/IERC20Metadata.sol";
import {Context} from "../../utils/Context.sol";
import {IERC20Errors} from "../../interfaces/draft-IERC6093.sol";

/**
 * @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}.
 *
 * TIP: For a detailed writeup see our guide
 * https://forum.openzeppelin.com/t/how-to-implement-erc20-supply-mechanisms/226[How
 * to implement supply mechanisms].
 *
 * The default value of {decimals} is 18. To change this, you should override
 * this function so it returns a different value.
 *
 * We have followed general OpenZeppelin Contracts guidelines: functions revert
 * instead returning `false` on failure. This behavior is nonetheless
 * conventional and does not conflict with the expectations of ERC-20
 * applications.
 */
abstract contract ERC20 is Context, IERC20, IERC20Metadata, IERC20Errors {
    mapping(address account => uint256) private _balances;

    mapping(address account => mapping(address spender => uint256)) private _allowances;

    uint256 private _totalSupply;

    string private _name;
    string private _symbol;

    /**
     * @dev Sets the values for {name} and {symbol}.
     *
     * All two 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 default value returned by this function, unless
     * it's overridden.
     *
     * 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 returns (uint256) {
        return _totalSupply;
    }

    /**
     * @dev See {IERC20-balanceOf}.
     */
    function balanceOf(address account) public view virtual returns (uint256) {
        return _balances[account];
    }

    /**
     * @dev See {IERC20-transfer}.
     *
     * Requirements:
     *
     * - `to` cannot be the zero address.
     * - the caller must have a balance of at least `value`.
     */
    function transfer(address to, uint256 value) public virtual returns (bool) {
        address owner = _msgSender();
        _transfer(owner, to, value);
        return true;
    }

    /**
     * @dev See {IERC20-allowance}.
     */
    function allowance(address owner, address spender) public view virtual returns (uint256) {
        return _allowances[owner][spender];
    }

    /**
     * @dev See {IERC20-approve}.
     *
     * NOTE: If `value` is the maximum `uint256`, the allowance is not updated on
     * `transferFrom`. This is semantically equivalent to an infinite approval.
     *
     * Requirements:
     *
     * - `spender` cannot be the zero address.
     */
    function approve(address spender, uint256 value) public virtual returns (bool) {
        address owner = _msgSender();
        _approve(owner, spender, value);
        return true;
    }

    /**
     * @dev See {IERC20-transferFrom}.
     *
     * Skips emitting an {Approval} event indicating an allowance update. This is not
     * required by the ERC. See {xref-ERC20-_approve-address-address-uint256-bool-}[_approve].
     *
     * NOTE: Does not update the allowance if the current allowance
     * is the maximum `uint256`.
     *
     * Requirements:
     *
     * - `from` and `to` cannot be the zero address.
     * - `from` must have a balance of at least `value`.
     * - the caller must have allowance for ``from``'s tokens of at least
     * `value`.
     */
    function transferFrom(address from, address to, uint256 value) public virtual returns (bool) {
        address spender = _msgSender();
        _spendAllowance(from, spender, value);
        _transfer(from, to, value);
        return true;
    }

    /**
     * @dev Moves a `value` amount of tokens from `from` to `to`.
     *
     * This internal function is equivalent to {transfer}, and can be used to
     * e.g. implement automatic token fees, slashing mechanisms, etc.
     *
     * Emits a {Transfer} event.
     *
     * NOTE: This function is not virtual, {_update} should be overridden instead.
     */
    function _transfer(address from, address to, uint256 value) internal {
        if (from == address(0)) {
            revert ERC20InvalidSender(address(0));
        }
        if (to == address(0)) {
            revert ERC20InvalidReceiver(address(0));
        }
        _update(from, to, value);
    }

    /**
     * @dev Transfers a `value` amount of tokens from `from` to `to`, or alternatively mints (or burns) if `from`
     * (or `to`) is the zero address. All customizations to transfers, mints, and burns should be done by overriding
     * this function.
     *
     * Emits a {Transfer} event.
     */
    function _update(address from, address to, uint256 value) internal virtual {
        if (from == address(0)) {
            // Overflow check required: The rest of the code assumes that totalSupply never overflows
            _totalSupply += value;
        } else {
            uint256 fromBalance = _balances[from];
            if (fromBalance < value) {
                revert ERC20InsufficientBalance(from, fromBalance, value);
            }
            unchecked {
                // Overflow not possible: value <= fromBalance <= totalSupply.
                _balances[from] = fromBalance - value;
            }
        }

        if (to == address(0)) {
            unchecked {
                // Overflow not possible: value <= totalSupply or value <= fromBalance <= totalSupply.
                _totalSupply -= value;
            }
        } else {
            unchecked {
                // Overflow not possible: balance + value is at most totalSupply, which we know fits into a uint256.
                _balances[to] += value;
            }
        }

        emit Transfer(from, to, value);
    }

    /**
     * @dev Creates a `value` amount of tokens and assigns them to `account`, by transferring it from address(0).
     * Relies on the `_update` mechanism
     *
     * Emits a {Transfer} event with `from` set to the zero address.
     *
     * NOTE: This function is not virtual, {_update} should be overridden instead.
     */
    function _mint(address account, uint256 value) internal {
        if (account == address(0)) {
            revert ERC20InvalidReceiver(address(0));
        }
        _update(address(0), account, value);
    }

    /**
     * @dev Destroys a `value` amount of tokens from `account`, lowering the total supply.
     * Relies on the `_update` mechanism.
     *
     * Emits a {Transfer} event with `to` set to the zero address.
     *
     * NOTE: This function is not virtual, {_update} should be overridden instead
     */
    function _burn(address account, uint256 value) internal {
        if (account == address(0)) {
            revert ERC20InvalidSender(address(0));
        }
        _update(account, address(0), value);
    }

    /**
     * @dev Sets `value` 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.
     *
     * Overrides to this logic should be done to the variant with an additional `bool emitEvent` argument.
     */
    function _approve(address owner, address spender, uint256 value) internal {
        _approve(owner, spender, value, true);
    }

    /**
     * @dev Variant of {_approve} with an optional flag to enable or disable the {Approval} event.
     *
     * By default (when calling {_approve}) the flag is set to true. On the other hand, approval changes made by
     * `_spendAllowance` during the `transferFrom` operation set the flag to false. This saves gas by not emitting any
     * `Approval` event during `transferFrom` operations.
     *
     * Anyone who wishes to continue emitting `Approval` events on the`transferFrom` operation can force the flag to
     * true using the following override:
     *
     * ```solidity
     * function _approve(address owner, address spender, uint256 value, bool) internal virtual override {
     *     super._approve(owner, spender, value, true);
     * }
     * ```
     *
     * Requirements are the same as {_approve}.
     */
    function _approve(address owner, address spender, uint256 value, bool emitEvent) internal virtual {
        if (owner == address(0)) {
            revert ERC20InvalidApprover(address(0));
        }
        if (spender == address(0)) {
            revert ERC20InvalidSpender(address(0));
        }
        _allowances[owner][spender] = value;
        if (emitEvent) {
            emit Approval(owner, spender, value);
        }
    }

    /**
     * @dev Updates `owner` s allowance for `spender` based on spent `value`.
     *
     * Does not update the allowance value in case of infinite allowance.
     * Revert if not enough allowance is available.
     *
     * Does not emit an {Approval} event.
     */
    function _spendAllowance(address owner, address spender, uint256 value) internal virtual {
        uint256 currentAllowance = allowance(owner, spender);
        if (currentAllowance < type(uint256).max) {
            if (currentAllowance < value) {
                revert ERC20InsufficientAllowance(spender, currentAllowance, value);
            }
            unchecked {
                _approve(owner, spender, currentAllowance - value, false);
            }
        }
    }
}

File 5 of 9 : IERC20Metadata.sol
// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts (last updated v5.1.0) (token/ERC20/extensions/IERC20Metadata.sol)

pragma solidity ^0.8.20;

import {IERC20} from "../IERC20.sol";

/**
 * @dev Interface for the optional metadata functions from the ERC-20 standard.
 */
interface IERC20Metadata is IERC20 {
    /**
     * @dev Returns the name of the token.
     */
    function name() external view returns (string memory);

    /**
     * @dev Returns the symbol of the token.
     */
    function symbol() external view returns (string memory);

    /**
     * @dev Returns the decimals places of the token.
     */
    function decimals() external view returns (uint8);
}

File 6 of 9 : IERC20.sol
// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts (last updated v5.1.0) (token/ERC20/IERC20.sol)

pragma solidity ^0.8.20;

/**
 * @dev Interface of the ERC-20 standard as defined in the ERC.
 */
interface IERC20 {
    /**
     * @dev Emitted when `value` tokens are moved from one account (`from`) to
     * another (`to`).
     *
     * Note that `value` may be zero.
     */
    event Transfer(address indexed from, address indexed to, uint256 value);

    /**
     * @dev Emitted when the allowance of a `spender` for an `owner` is set by
     * a call to {approve}. `value` is the new allowance.
     */
    event Approval(address indexed owner, address indexed spender, uint256 value);

    /**
     * @dev Returns the value of tokens in existence.
     */
    function totalSupply() external view returns (uint256);

    /**
     * @dev Returns the value of tokens owned by `account`.
     */
    function balanceOf(address account) external view returns (uint256);

    /**
     * @dev Moves a `value` amount of tokens from the caller's account to `to`.
     *
     * Returns a boolean value indicating whether the operation succeeded.
     *
     * Emits a {Transfer} event.
     */
    function transfer(address to, uint256 value) 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 a `value` amount of tokens 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 value) external returns (bool);

    /**
     * @dev Moves a `value` amount of tokens from `from` to `to` using the
     * allowance mechanism. `value` is then deducted from the caller's
     * allowance.
     *
     * Returns a boolean value indicating whether the operation succeeded.
     *
     * Emits a {Transfer} event.
     */
    function transferFrom(address from, address to, uint256 value) external returns (bool);
}

File 7 of 9 : Context.sol
// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts (last updated v5.0.1) (utils/Context.sol)

pragma solidity ^0.8.20;

/**
 * @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;
    }

    function _contextSuffixLength() internal view virtual returns (uint256) {
        return 0;
    }
}

File 8 of 9 : ReentrancyGuard.sol
// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts (last updated v5.1.0) (utils/ReentrancyGuard.sol)

pragma solidity ^0.8.20;

/**
 * @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 EIP-1153 (transient storage) is available on the chain you're deploying at,
 * consider using {ReentrancyGuardTransient} instead.
 *
 * 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;

    /**
     * @dev Unauthorized reentrant call.
     */
    error ReentrancyGuardReentrantCall();

    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() {
        _nonReentrantBefore();
        _;
        _nonReentrantAfter();
    }

    function _nonReentrantBefore() private {
        // On the first call to nonReentrant, _status will be NOT_ENTERED
        if (_status == ENTERED) {
            revert ReentrancyGuardReentrantCall();
        }

        // Any calls to nonReentrant after this point will fail
        _status = ENTERED;
    }

    function _nonReentrantAfter() private {
        // By storing the original value once again, a refund is triggered (see
        // https://eips.ethereum.org/EIPS/eip-2200)
        _status = NOT_ENTERED;
    }

    /**
     * @dev Returns true if the reentrancy guard is currently set to "entered", which indicates there is a
     * `nonReentrant` function in the call stack.
     */
    function _reentrancyGuardEntered() internal view returns (bool) {
        return _status == ENTERED;
    }
}

File 9 of 9 : IWind.sol
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.20;

interface IWind{
    // Constants
    function MAX_SUPPLY() external view returns (uint256);
    function LP_ALLOCATION() external view returns (uint256);
    function FUTURE_GROWTH_ALLOCATION() external view returns (uint256);
    function PHASE_1_ALLOCATION() external view returns (uint256);
    function PHASE_2_ALLOCATION() external view returns (uint256);
    function PHASE_3_ALLOCATION() external view returns (uint256);

    // State variables
    function currentPhase() external view returns (uint8);
    function activePhase() external view returns (bool);
    function phaseStartTime() external view returns (uint256);
    function claimActivationTime() external view returns (uint256);
    function currentPhaseRemainingTokens() external view returns (uint256);
    function claimDropCompleted() external view returns (bool);
    function firstTimeThreshold() external view returns (uint256);
    function secondTimeThreshold() external view returns (uint256);

    // Mappings
    function claimCount(address user) external view returns (uint256);
    function pendingClaims(address user) external view returns (uint256);
    function transferAmountUsed(address user) external view returns (uint256);
    function excludedDexes(address dex) external view returns (bool);

    // Main functions
    function claimWind(string memory affiliate) external;
    function transferTokens() external;
    function getReductionMultiplier(uint256 claims) external pure returns (uint256);
    function circulatingSupply() external view returns (uint256);
}

Settings
{
  "evmVersion": "paris",
  "optimizer": {
    "enabled": false,
    "runs": 200
  },
  "outputSelection": {
    "*": {
      "*": [
        "evm.bytecode",
        "evm.deployedBytecode",
        "devdoc",
        "userdoc",
        "metadata",
        "abi"
      ]
    }
  },
  "metadata": {
    "useLiteralContent": true
  },
  "libraries": {}
}

Contract Security Audit

Contract ABI

API
[{"inputs":[],"stateMutability":"nonpayable","type":"constructor"},{"inputs":[{"internalType":"address","name":"spender","type":"address"},{"internalType":"uint256","name":"allowance","type":"uint256"},{"internalType":"uint256","name":"needed","type":"uint256"}],"name":"ERC20InsufficientAllowance","type":"error"},{"inputs":[{"internalType":"address","name":"sender","type":"address"},{"internalType":"uint256","name":"balance","type":"uint256"},{"internalType":"uint256","name":"needed","type":"uint256"}],"name":"ERC20InsufficientBalance","type":"error"},{"inputs":[{"internalType":"address","name":"approver","type":"address"}],"name":"ERC20InvalidApprover","type":"error"},{"inputs":[{"internalType":"address","name":"receiver","type":"address"}],"name":"ERC20InvalidReceiver","type":"error"},{"inputs":[{"internalType":"address","name":"sender","type":"address"}],"name":"ERC20InvalidSender","type":"error"},{"inputs":[{"internalType":"address","name":"spender","type":"address"}],"name":"ERC20InvalidSpender","type":"error"},{"inputs":[{"internalType":"address","name":"owner","type":"address"}],"name":"OwnableInvalidOwner","type":"error"},{"inputs":[{"internalType":"address","name":"account","type":"address"}],"name":"OwnableUnauthorizedAccount","type":"error"},{"inputs":[],"name":"ReentrancyGuardReentrantCall","type":"error"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"owner","type":"address"},{"indexed":true,"internalType":"address","name":"spender","type":"address"},{"indexed":false,"internalType":"uint256","name":"value","type":"uint256"}],"name":"Approval","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"previousOwner","type":"address"},{"indexed":true,"internalType":"address","name":"newOwner","type":"address"}],"name":"OwnershipTransferred","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"owner","type":"address"},{"indexed":false,"internalType":"uint256","name":"amount","type":"uint256"}],"name":"TokensTransfered","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"from","type":"address"},{"indexed":true,"internalType":"address","name":"to","type":"address"},{"indexed":false,"internalType":"uint256","name":"value","type":"uint256"}],"name":"Transfer","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"receiver","type":"address"},{"indexed":true,"internalType":"string","name":"affiliate","type":"string"},{"indexed":false,"internalType":"uint256","name":"amount","type":"uint256"}],"name":"WindClaimed","type":"event"},{"inputs":[],"name":"FUTURE_GROWTH_ALLOCATION","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"LP_ALLOCATION","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"MAX_SUPPLY","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"PHASE_1_ALLOCATION","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"PHASE_2_ALLOCATION","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"PHASE_3_ALLOCATION","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"activateMinting","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"activePhase","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"owner","type":"address"},{"internalType":"address","name":"spender","type":"address"}],"name":"allowance","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"spender","type":"address"},{"internalType":"uint256","name":"value","type":"uint256"}],"name":"approve","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"account","type":"address"}],"name":"balanceOf","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"circulatingSupply","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"claimActivationTime","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"claimCount","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"claimDropCompleted","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"string","name":"affiliate","type":"string"}],"name":"claimWind","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"completeClaimDrop","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"currentPhase","outputs":[{"internalType":"uint8","name":"","type":"uint8"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"currentPhaseRemainingTokens","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"decimals","outputs":[{"internalType":"uint8","name":"","type":"uint8"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"endPhaseManually","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"dexAddress","type":"address"}],"name":"excludeDex","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"excludedDexes","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"firstTimeThreshold","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"claims","type":"uint256"}],"name":"getReductionMultiplier","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"pure","type":"function"},{"inputs":[{"internalType":"address","name":"dexAddress","type":"address"}],"name":"includeDex","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"name","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"owner","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"pendingClaims","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"phaseStartTime","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"renounceOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"secondTimeThreshold","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"symbol","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"totalSupply","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"value","type":"uint256"}],"name":"transfer","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"transferAmountUsed","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"from","type":"address"},{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"value","type":"uint256"}],"name":"transferFrom","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"newOwner","type":"address"}],"name":"transferOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"transferTokens","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_firstTimeThreshold","type":"uint256"},{"internalType":"uint256","name":"_secondTimeThreshold","type":"uint256"}],"name":"updateTimeThresholds","outputs":[],"stateMutability":"nonpayable","type":"function"}]

60806040526000600760006101000a81548160ff021916908360ff1602179055506005600c55600a600d553480156200003757600080fd5b50336040518060400160405280600981526020017f57696e6420f09f92a800000000000000000000000000000000000000000000008152506040518060400160405280600481526020017f57494e44000000000000000000000000000000000000000000000000000000008152508160039081620000b6919062000aca565b508060049081620000c8919062000aca565b505050600073ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff1603620001405760006040517f1e4fbdf700000000000000000000000000000000000000000000000000000000815260040162000137919062000bf6565b60405180910390fd5b62000151816200019a60201b60201c565b50600160068190555062000177306a084595161401484a0000006200026060201b60201c565b62000194303369d3c21bcecceda1000000620002ed60201b60201c565b62000ce8565b6000600560009054906101000a900473ffffffffffffffffffffffffffffffffffffffff16905081600560006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff1602179055508173ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff167f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e060405160405180910390a35050565b600073ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff1603620002d55760006040517fec442f05000000000000000000000000000000000000000000000000000000008152600401620002cc919062000bf6565b60405180910390fd5b620002e960008383620003ef60201b60201c565b5050565b600073ffffffffffffffffffffffffffffffffffffffff168373ffffffffffffffffffffffffffffffffffffffff1603620003625760006040517f96c6fd1e00000000000000000000000000000000000000000000000000000000815260040162000359919062000bf6565b60405180910390fd5b600073ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff1603620003d75760006040517fec442f05000000000000000000000000000000000000000000000000000000008152600401620003ce919062000bf6565b60405180910390fd5b620003ea838383620003ef60201b60201c565b505050565b620004028383836200062060201b60201c565b600b60009054906101000a900460ff161580156200044d5750600073ffffffffffffffffffffffffffffffffffffffff168373ffffffffffffffffffffffffffffffffffffffff1614155b8015620004875750600073ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff1614155b156200061b57601160008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060009054906101000a900460ff16158015620005325750601160008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060009054906101000a900460ff16155b156200061a576000601060008573ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000205411156200061957601060008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002054601060008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600082825462000611919062000c42565b925050819055505b5b5b505050565b600073ffffffffffffffffffffffffffffffffffffffff168373ffffffffffffffffffffffffffffffffffffffff16036200067657806002600082825462000669919062000c42565b925050819055506200074c565b60008060008573ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000205490508181101562000705578381836040517fe450d38c000000000000000000000000000000000000000000000000000000008152600401620006fc9392919062000c8e565b60405180910390fd5b8181036000808673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002081905550505b600073ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff1603620007975780600260008282540392505081905550620007e4565b806000808473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600082825401925050819055505b8173ffffffffffffffffffffffffffffffffffffffff168373ffffffffffffffffffffffffffffffffffffffff167fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef8360405162000843919062000ccb565b60405180910390a3505050565b600081519050919050565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052604160045260246000fd5b7f4e487b7100000000000000000000000000000000000000000000000000000000600052602260045260246000fd5b60006002820490506001821680620008d257607f821691505b602082108103620008e857620008e76200088a565b5b50919050565b60008190508160005260206000209050919050565b60006020601f8301049050919050565b600082821b905092915050565b600060088302620009527fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff8262000913565b6200095e868362000913565b95508019841693508086168417925050509392505050565b6000819050919050565b6000819050919050565b6000620009ab620009a56200099f8462000976565b62000980565b62000976565b9050919050565b6000819050919050565b620009c7836200098a565b620009df620009d682620009b2565b84845462000920565b825550505050565b600090565b620009f6620009e7565b62000a03818484620009bc565b505050565b5b8181101562000a2b5762000a1f600082620009ec565b60018101905062000a09565b5050565b601f82111562000a7a5762000a4481620008ee565b62000a4f8462000903565b8101602085101562000a5f578190505b62000a7762000a6e8562000903565b83018262000a08565b50505b505050565b600082821c905092915050565b600062000a9f6000198460080262000a7f565b1980831691505092915050565b600062000aba838362000a8c565b9150826002028217905092915050565b62000ad58262000850565b67ffffffffffffffff81111562000af15762000af06200085b565b5b62000afd8254620008b9565b62000b0a82828562000a2f565b600060209050601f83116001811462000b42576000841562000b2d578287015190505b62000b39858262000aac565b86555062000ba9565b601f19841662000b5286620008ee565b60005b8281101562000b7c5784890151825560018201915060208501945060208101905062000b55565b8683101562000b9c578489015162000b98601f89168262000a8c565b8355505b6001600288020188555050505b505050505050565b600073ffffffffffffffffffffffffffffffffffffffff82169050919050565b600062000bde8262000bb1565b9050919050565b62000bf08162000bd1565b82525050565b600060208201905062000c0d600083018462000be5565b92915050565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052601160045260246000fd5b600062000c4f8262000976565b915062000c5c8362000976565b925082820190508082111562000c775762000c7662000c13565b5b92915050565b62000c888162000976565b82525050565b600060608201905062000ca5600083018662000be5565b62000cb4602083018562000c7d565b62000cc3604083018462000c7d565b949350505050565b600060208201905062000ce2600083018462000c7d565b92915050565b61311a8062000cf86000396000f3fe608060405234801561001057600080fd5b50600436106102485760003560e01c806370a082311161013b578063a9059cbb116100b8578063cd28890e1161007c578063cd28890e1461069d578063dd62ed3e146106bb578063e2e57ecc146106eb578063ebda16eb14610709578063f2fde38b1461072757610248565b8063a9059cbb146105e5578063ab7873fb14610615578063ae170fae1461061f578063c19cb26f1461063d578063ca78533d1461066d57610248565b80638da5cb5b116100ff5780638da5cb5b146105515780638e95825d1461056f5780639358928b1461058b57806395d89b41146105a95780639760b75c146105c757610248565b806370a08231146104bf578063715018a6146104ef578063747bb644146104f957806374a8ee3114610517578063796fe8e51461053557610248565b806318160ddd116101c957806332cb6b0c1161018d57806332cb6b0c146104075780633c0d0fe3146104255780635e7e68101461045557806369c12cb7146104855780636ca4fc2a146104b557610248565b806318160ddd146103735780631bc1d4e51461039157806323b872dd1461039b578063313ce567146103cb578063320e7866146103e957610248565b80630e1b26c6116102105780630e1b26c6146102f157806312f3d0c21461030f5780631402b1b41461032d578063143493731461034b5780631792b0bc1461036957610248565b8063049227801461024d578063055ad42e1461026957806306fdde0314610287578063095ea7b3146102a55780630c892edc146102d5575b600080fd5b610267600480360381019061026291906122f5565b610743565b005b6102716108a2565b60405161027e919061233e565b60405180910390f35b61028f6108b5565b60405161029c91906123e9565b60405180910390f35b6102bf60048036038101906102ba9190612441565b610947565b6040516102cc919061249c565b60405180910390f35b6102ef60048036038101906102ea91906124b7565b61096a565b005b6102f9610984565b604051610306919061249c565b60405180910390f35b610317610997565b6040516103249190612506565b60405180910390f35b61033561099d565b6040516103429190612506565b60405180910390f35b6103536109ac565b604051610360919061249c565b60405180910390f35b6103716109bf565b005b61037b610ca3565b6040516103889190612506565b60405180910390f35b610399610cad565b005b6103b560048036038101906103b09190612521565b610d63565b6040516103c2919061249c565b60405180910390f35b6103d3610d92565b6040516103e0919061233e565b60405180910390f35b6103f1610d9b565b6040516103fe9190612506565b60405180910390f35b61040f610daa565b60405161041c9190612506565b60405180910390f35b61043f600480360381019061043a9190612574565b610db9565b60405161044c9190612506565b60405180910390f35b61046f600480360381019061046a91906122f5565b610e0a565b60405161047c919061249c565b60405180910390f35b61049f600480360381019061049a91906122f5565b610e2a565b6040516104ac9190612506565b60405180910390f35b6104bd610e42565b005b6104d960048036038101906104d491906122f5565b610f26565b6040516104e69190612506565b60405180910390f35b6104f7610f6e565b005b610501610f82565b60405161050e9190612506565b60405180910390f35b61051f610f88565b60405161052c9190612506565b60405180910390f35b61054f600480360381019061054a91906122f5565b610f8e565b005b61055961107d565b60405161056691906125b0565b60405180910390f35b61058960048036038101906105849190612700565b6110a7565b005b61059361149f565b6040516105a09190612506565b60405180910390f35b6105b16114c5565b6040516105be91906123e9565b60405180910390f35b6105cf611557565b6040516105dc9190612506565b60405180910390f35b6105ff60048036038101906105fa9190612441565b61155d565b60405161060c919061249c565b60405180910390f35b61061d611580565b005b610627611705565b6040516106349190612506565b60405180910390f35b610657600480360381019061065291906122f5565b611713565b6040516106649190612506565b60405180910390f35b610687600480360381019061068291906122f5565b61172b565b6040516106949190612506565b60405180910390f35b6106a5611743565b6040516106b29190612506565b60405180910390f35b6106d560048036038101906106d09190612749565b611752565b6040516106e29190612506565b60405180910390f35b6106f36117d9565b6040516107009190612506565b60405180910390f35b6107116117df565b60405161071e9190612506565b60405180910390f35b610741600480360381019061073c91906122f5565b6117ed565b005b61074b611873565b600073ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff16036107ba576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016107b1906127d5565b60405180910390fd5b601160008273ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060009054906101000a900460ff1615610847576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161083e90612841565b60405180910390fd5b6001601160008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060006101000a81548160ff02191690831515021790555050565b600760009054906101000a900460ff1681565b6060600380546108c490612890565b80601f01602080910402602001604051908101604052809291908181526020018280546108f090612890565b801561093d5780601f106109125761010080835404028352916020019161093d565b820191906000526020600020905b81548152906001019060200180831161092057829003601f168201915b5050505050905090565b6000806109526118fa565b905061095f818585611902565b600191505092915050565b610972611873565b81600c8190555080600d819055505050565b600760019054906101000a900460ff1681565b60085481565b6a01a784379d99db4200000081565b600b60009054906101000a900460ff1681565b60006009541180156109d357506009544210155b610a12576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610a099061290d565b60405180910390fd5b6000600f60003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000205411610a94576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610a8b90612979565b60405180910390fd5b600f60003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002054601060003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002054610b1d33610f26565b610b2791906129c8565b1015610b68576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610b5f90612a6e565b60405180910390fd5b6000600f60003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000205490506000600f60003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000208190555080601060003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000206000828254610c409190612a8e565b92505081905550610c52303383611914565b3373ffffffffffffffffffffffffffffffffffffffff167fa267b5cf59cf96d97abf36d29c63cf7bdef867c208ca8575c47f2cafda39abc982604051610c989190612506565b60405180910390a250565b6000600254905090565b610cb5611873565b6000600760009054906101000a900460ff1660ff1611610d0a576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610d0190612b0e565b60405180910390fd5b600760019054906101000a900460ff16610d59576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610d5090612b7a565b60405180910390fd5b610d61611a08565b565b600080610d6e6118fa565b9050610d7b858285611a39565b610d86858585611914565b60019150509392505050565b60006012905090565b6a034f086f3b33b68400000081565b6a084595161401484a00000081565b6000808203610dcc576103e89050610e05565b6000600183610ddb9190612a8e565b90506000816103e8610ded9190612bc9565b905060648111610dfe576064610e00565b805b925050505b919050565b60116020528060005260406000206000915054906101000a900460ff1681565b600e6020528060005260406000206000915090505481565b610e4a611873565b6003600760009054906101000a900460ff1660ff1614610e9f576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610e9690612c46565b60405180910390fd5b600760019054906101000a900460ff1615610eef576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610ee690612cb2565b60405180910390fd5b6001600b60006101000a81548160ff021916908315150217905550610f2430610f1661107d565b610f1f30610f26565b611914565b565b60008060008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020549050919050565b610f76611873565b610f806000611ace565b565b600d5481565b600c5481565b610f96611873565b601160008273ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060009054906101000a900460ff16611022576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161101990612d1e565b60405180910390fd5b6000601160008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060006101000a81548160ff02191690831515021790555050565b6000600560009054906101000a900473ffffffffffffffffffffffffffffffffffffffff16905090565b6110af611b94565b3273ffffffffffffffffffffffffffffffffffffffff163373ffffffffffffffffffffffffffffffffffffffff161461111d576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161111490612d8a565b60405180910390fd5b6000600760009054906101000a900460ff1660ff1611611172576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161116990612df6565b60405180910390fd5b600760019054906101000a900460ff166111c1576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016111b890612df6565b60405180910390fd5b600854421015611206576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016111fd90612df6565b60405180910390fd5b60006008544261121691906129c8565b90506000600c54821161123357670de0b6b3a764000090506113a7565b600d54821161126257670de0b6b3a76400006112516001600a611bda565b61125b9190612e16565b90506113a6565b600061126d33610f26565b9050600080820361128157600a90506112b3565b683635c9adc5dea00000821061129b576103e890506112b2565b670de0b6b3a7640000826112af9190612bc9565b90505b5b60006112fd600e60003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002054610db9565b90506103e8818361130e9190612e16565b6113189190612bc9565b9150600a821161132957600a61132b565b815b9150670de0b6b3a7640000611341600a84611bda565b61134b9190612e16565b9350600e60003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600081548092919061139d90612e58565b91905055505050505b5b600a548111156113bf57600a5490506113be611a08565b5b80600f60003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600082825461140e9190612a8e565b9250508190555080600a600082825461142791906129c8565b925050819055508260405161143c9190612edc565b60405180910390203373ffffffffffffffffffffffffffffffffffffffff167f920a78ccdc638c996c9176021cdcc049dd8ad461c7de23754d4d5d7c232ad1518360405161148a9190612506565b60405180910390a3505061149c611c5c565b50565b60006114aa30610f26565b6a084595161401484a0000006114c091906129c8565b905090565b6060600480546114d490612890565b80601f016020809104026020016040519081016040528092919081815260200182805461150090612890565b801561154d5780601f106115225761010080835404028352916020019161154d565b820191906000526020600020905b81548152906001019060200180831161153057829003601f168201915b5050505050905090565b60095481565b6000806115686118fa565b9050611575818585611914565b600191505092915050565b611588611873565b6003600760009054906101000a900460ff1660ff16106115dd576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016115d490612f3f565b60405180910390fd5b4244336040516020016115f293929190612fc8565b6040516020818303038152906040528051906020012060001c6012819055506007600081819054906101000a900460ff168092919061163090613005565b91906101000a81548160ff021916908360ff160217905550504260088190555060006009819055506001600760016101000a81548160ff0219169083151502179055506001600760009054906101000a900460ff1660ff16036116a35769d3c21bcecceda1000000600a81905550611703565b6002600760009054906101000a900460ff1660ff16036116d4576a01a784379d99db42000000600a81905550611702565b6003600760009054906101000a900460ff1660ff1603611701576a034f086f3b33b684000000600a819055505b5b5b565b69d3c21bcecceda100000081565b60106020528060005260406000206000915090505481565b600f6020528060005260406000206000915090505481565b6a01a784379d99db4200000081565b6000600160008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002054905092915050565b600a5481565b69d3c21bcecceda100000081565b6117f5611873565b600073ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff16036118675760006040517f1e4fbdf700000000000000000000000000000000000000000000000000000000815260040161185e91906125b0565b60405180910390fd5b61187081611ace565b50565b61187b6118fa565b73ffffffffffffffffffffffffffffffffffffffff1661189961107d565b73ffffffffffffffffffffffffffffffffffffffff16146118f8576118bc6118fa565b6040517f118cdaa70000000000000000000000000000000000000000000000000000000081526004016118ef91906125b0565b60405180910390fd5b565b600033905090565b61190f8383836001611c66565b505050565b600073ffffffffffffffffffffffffffffffffffffffff168373ffffffffffffffffffffffffffffffffffffffff16036119865760006040517f96c6fd1e00000000000000000000000000000000000000000000000000000000815260040161197d91906125b0565b60405180910390fd5b600073ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff16036119f85760006040517fec442f050000000000000000000000000000000000000000000000000000000081526004016119ef91906125b0565b60405180910390fd5b611a03838383611e3d565b505050565b6000600760016101000a81548160ff02191690831515021790555061070842611a319190612a8e565b600981905550565b6000611a458484611752565b90507fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff811015611ac85781811015611ab8578281836040517ffb8f41b2000000000000000000000000000000000000000000000000000000008152600401611aaf9392919061302e565b60405180910390fd5b611ac784848484036000611c66565b5b50505050565b6000600560009054906101000a900473ffffffffffffffffffffffffffffffffffffffff16905081600560006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff1602179055508173ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff167f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e060405160405180910390a35050565b600260065403611bd0576040517f3ee5aeb500000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b6002600681905550565b6000818310611beb57829050611c56565b424433601254604051602001611c049493929190613065565b6040516020818303038152906040528051906020012060001c60128190555060018383611c3191906129c8565b611c3b9190612a8e565b601254611c4891906130b3565b83611c539190612a8e565b90505b92915050565b6001600681905550565b600073ffffffffffffffffffffffffffffffffffffffff168473ffffffffffffffffffffffffffffffffffffffff1603611cd85760006040517fe602df05000000000000000000000000000000000000000000000000000000008152600401611ccf91906125b0565b60405180910390fd5b600073ffffffffffffffffffffffffffffffffffffffff168373ffffffffffffffffffffffffffffffffffffffff1603611d4a5760006040517f94280d62000000000000000000000000000000000000000000000000000000008152600401611d4191906125b0565b60405180910390fd5b81600160008673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060008573ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020819055508015611e37578273ffffffffffffffffffffffffffffffffffffffff168473ffffffffffffffffffffffffffffffffffffffff167f8c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b92584604051611e2e9190612506565b60405180910390a35b50505050565b611e4883838361205e565b600b60009054906101000a900460ff16158015611e925750600073ffffffffffffffffffffffffffffffffffffffff168373ffffffffffffffffffffffffffffffffffffffff1614155b8015611ecb5750600073ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff1614155b1561205957601160008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060009054906101000a900460ff16158015611f745750601160008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060009054906101000a900460ff16155b15612058576000601060008573ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002054111561205757601060008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002054601060008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600082825461204f9190612a8e565b925050819055505b5b5b505050565b600073ffffffffffffffffffffffffffffffffffffffff168373ffffffffffffffffffffffffffffffffffffffff16036120b05780600260008282546120a49190612a8e565b92505081905550612183565b60008060008573ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000205490508181101561213c578381836040517fe450d38c0000000000000000000000000000000000000000000000000000000081526004016121339392919061302e565b60405180910390fd5b8181036000808673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002081905550505b600073ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff16036121cc5780600260008282540392505081905550612219565b806000808473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600082825401925050819055505b8173ffffffffffffffffffffffffffffffffffffffff168373ffffffffffffffffffffffffffffffffffffffff167fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef836040516122769190612506565b60405180910390a3505050565b6000604051905090565b600080fd5b600080fd5b600073ffffffffffffffffffffffffffffffffffffffff82169050919050565b60006122c282612297565b9050919050565b6122d2816122b7565b81146122dd57600080fd5b50565b6000813590506122ef816122c9565b92915050565b60006020828403121561230b5761230a61228d565b5b6000612319848285016122e0565b91505092915050565b600060ff82169050919050565b61233881612322565b82525050565b6000602082019050612353600083018461232f565b92915050565b600081519050919050565b600082825260208201905092915050565b60005b83811015612393578082015181840152602081019050612378565b60008484015250505050565b6000601f19601f8301169050919050565b60006123bb82612359565b6123c58185612364565b93506123d5818560208601612375565b6123de8161239f565b840191505092915050565b6000602082019050818103600083015261240381846123b0565b905092915050565b6000819050919050565b61241e8161240b565b811461242957600080fd5b50565b60008135905061243b81612415565b92915050565b600080604083850312156124585761245761228d565b5b6000612466858286016122e0565b92505060206124778582860161242c565b9150509250929050565b60008115159050919050565b61249681612481565b82525050565b60006020820190506124b1600083018461248d565b92915050565b600080604083850312156124ce576124cd61228d565b5b60006124dc8582860161242c565b92505060206124ed8582860161242c565b9150509250929050565b6125008161240b565b82525050565b600060208201905061251b60008301846124f7565b92915050565b60008060006060848603121561253a5761253961228d565b5b6000612548868287016122e0565b9350506020612559868287016122e0565b925050604061256a8682870161242c565b9150509250925092565b60006020828403121561258a5761258961228d565b5b60006125988482850161242c565b91505092915050565b6125aa816122b7565b82525050565b60006020820190506125c560008301846125a1565b92915050565b600080fd5b600080fd5b7f4e487b7100000000000000000000000000000000000000000000000000000000600052604160045260246000fd5b61260d8261239f565b810181811067ffffffffffffffff8211171561262c5761262b6125d5565b5b80604052505050565b600061263f612283565b905061264b8282612604565b919050565b600067ffffffffffffffff82111561266b5761266a6125d5565b5b6126748261239f565b9050602081019050919050565b82818337600083830152505050565b60006126a361269e84612650565b612635565b9050828152602081018484840111156126bf576126be6125d0565b5b6126ca848285612681565b509392505050565b600082601f8301126126e7576126e66125cb565b5b81356126f7848260208601612690565b91505092915050565b6000602082840312156127165761271561228d565b5b600082013567ffffffffffffffff81111561273457612733612292565b5b612740848285016126d2565b91505092915050565b600080604083850312156127605761275f61228d565b5b600061276e858286016122e0565b925050602061277f858286016122e0565b9150509250929050565b7f43616e6e6f74206578636c756465207a65726f20616464726573730000000000600082015250565b60006127bf601b83612364565b91506127ca82612789565b602082019050919050565b600060208201905081810360008301526127ee816127b2565b9050919050565b7f44455820616c7265616479206578636c75646564000000000000000000000000600082015250565b600061282b601483612364565b9150612836826127f5565b602082019050919050565b6000602082019050818103600083015261285a8161281e565b9050919050565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052602260045260246000fd5b600060028204905060018216806128a857607f821691505b6020821081036128bb576128ba612861565b5b50919050565b7f5472616e73666572206e6f742061637469766520796574000000000000000000600082015250565b60006128f7601783612364565b9150612902826128c1565b602082019050919050565b60006020820190508181036000830152612926816128ea565b9050919050565b7f4e6f20746f6b656e7320746f207472616e736665720000000000000000000000600082015250565b6000612963601583612364565b915061296e8261292d565b602082019050919050565b6000602082019050818103600083015261299281612956565b9050919050565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052601160045260246000fd5b60006129d38261240b565b91506129de8361240b565b92508282039050818111156129f6576129f5612999565b5b92915050565b7f496e73756666696369656e742062616c616e636520746f207472616e7366657260008201527f2c206f776e206d6f7265202457494e4400000000000000000000000000000000602082015250565b6000612a58603083612364565b9150612a63826129fc565b604082019050919050565b60006020820190508181036000830152612a8781612a4b565b9050919050565b6000612a998261240b565b9150612aa48361240b565b9250828201905080821115612abc57612abb612999565b5b92915050565b7f4e6f206163746976652070686173650000000000000000000000000000000000600082015250565b6000612af8600f83612364565b9150612b0382612ac2565b602082019050919050565b60006020820190508181036000830152612b2781612aeb565b9050919050565b7f506861736520616c726561647920656e64656400000000000000000000000000600082015250565b6000612b64601383612364565b9150612b6f82612b2e565b602082019050919050565b60006020820190508181036000830152612b9381612b57565b9050919050565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052601260045260246000fd5b6000612bd48261240b565b9150612bdf8361240b565b925082612bef57612bee612b9a565b5b828204905092915050565b7f4f6e6c7920616674657220332064726f70730000000000000000000000000000600082015250565b6000612c30601283612364565b9150612c3b82612bfa565b602082019050919050565b60006020820190508181036000830152612c5f81612c23565b9050919050565b7f456e642070686173652066697273740000000000000000000000000000000000600082015250565b6000612c9c600f83612364565b9150612ca782612c66565b602082019050919050565b60006020820190508181036000830152612ccb81612c8f565b9050919050565b7f444558206e6f74206578636c7564656400000000000000000000000000000000600082015250565b6000612d08601083612364565b9150612d1382612cd2565b602082019050919050565b60006020820190508181036000830152612d3781612cfb565b9050919050565b7f4e6f20636f6e7472616374730000000000000000000000000000000000000000600082015250565b6000612d74600c83612364565b9150612d7f82612d3e565b602082019050919050565b60006020820190508181036000830152612da381612d67565b9050919050565b7f436c61696d64726f70206e6f7420616374697665000000000000000000000000600082015250565b6000612de0601483612364565b9150612deb82612daa565b602082019050919050565b60006020820190508181036000830152612e0f81612dd3565b9050919050565b6000612e218261240b565b9150612e2c8361240b565b9250828202612e3a8161240b565b91508282048414831517612e5157612e50612999565b5b5092915050565b6000612e638261240b565b91507fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff8203612e9557612e94612999565b5b600182019050919050565b600081905092915050565b6000612eb682612359565b612ec08185612ea0565b9350612ed0818560208601612375565b80840191505092915050565b6000612ee88284612eab565b915081905092915050565b7f416c6c2070686173657320636f6d706c65746564000000000000000000000000600082015250565b6000612f29601483612364565b9150612f3482612ef3565b602082019050919050565b60006020820190508181036000830152612f5881612f1c565b9050919050565b6000819050919050565b612f7a612f758261240b565b612f5f565b82525050565b60008160601b9050919050565b6000612f9882612f80565b9050919050565b6000612faa82612f8d565b9050919050565b612fc2612fbd826122b7565b612f9f565b82525050565b6000612fd48286612f69565b602082019150612fe48285612f69565b602082019150612ff48284612fb1565b601482019150819050949350505050565b600061301082612322565b915060ff820361302357613022612999565b5b600182019050919050565b600060608201905061304360008301866125a1565b61305060208301856124f7565b61305d60408301846124f7565b949350505050565b60006130718287612f69565b6020820191506130818286612f69565b6020820191506130918285612fb1565b6014820191506130a18284612f69565b60208201915081905095945050505050565b60006130be8261240b565b91506130c98361240b565b9250826130d9576130d8612b9a565b5b82820690509291505056fea264697066735822122096aff99a7bbed3cbeded077cf90490c423031dc4aea6062891839e0ed6169fde64736f6c63430008140033

Deployed Bytecode

0x608060405234801561001057600080fd5b50600436106102485760003560e01c806370a082311161013b578063a9059cbb116100b8578063cd28890e1161007c578063cd28890e1461069d578063dd62ed3e146106bb578063e2e57ecc146106eb578063ebda16eb14610709578063f2fde38b1461072757610248565b8063a9059cbb146105e5578063ab7873fb14610615578063ae170fae1461061f578063c19cb26f1461063d578063ca78533d1461066d57610248565b80638da5cb5b116100ff5780638da5cb5b146105515780638e95825d1461056f5780639358928b1461058b57806395d89b41146105a95780639760b75c146105c757610248565b806370a08231146104bf578063715018a6146104ef578063747bb644146104f957806374a8ee3114610517578063796fe8e51461053557610248565b806318160ddd116101c957806332cb6b0c1161018d57806332cb6b0c146104075780633c0d0fe3146104255780635e7e68101461045557806369c12cb7146104855780636ca4fc2a146104b557610248565b806318160ddd146103735780631bc1d4e51461039157806323b872dd1461039b578063313ce567146103cb578063320e7866146103e957610248565b80630e1b26c6116102105780630e1b26c6146102f157806312f3d0c21461030f5780631402b1b41461032d578063143493731461034b5780631792b0bc1461036957610248565b8063049227801461024d578063055ad42e1461026957806306fdde0314610287578063095ea7b3146102a55780630c892edc146102d5575b600080fd5b610267600480360381019061026291906122f5565b610743565b005b6102716108a2565b60405161027e919061233e565b60405180910390f35b61028f6108b5565b60405161029c91906123e9565b60405180910390f35b6102bf60048036038101906102ba9190612441565b610947565b6040516102cc919061249c565b60405180910390f35b6102ef60048036038101906102ea91906124b7565b61096a565b005b6102f9610984565b604051610306919061249c565b60405180910390f35b610317610997565b6040516103249190612506565b60405180910390f35b61033561099d565b6040516103429190612506565b60405180910390f35b6103536109ac565b604051610360919061249c565b60405180910390f35b6103716109bf565b005b61037b610ca3565b6040516103889190612506565b60405180910390f35b610399610cad565b005b6103b560048036038101906103b09190612521565b610d63565b6040516103c2919061249c565b60405180910390f35b6103d3610d92565b6040516103e0919061233e565b60405180910390f35b6103f1610d9b565b6040516103fe9190612506565b60405180910390f35b61040f610daa565b60405161041c9190612506565b60405180910390f35b61043f600480360381019061043a9190612574565b610db9565b60405161044c9190612506565b60405180910390f35b61046f600480360381019061046a91906122f5565b610e0a565b60405161047c919061249c565b60405180910390f35b61049f600480360381019061049a91906122f5565b610e2a565b6040516104ac9190612506565b60405180910390f35b6104bd610e42565b005b6104d960048036038101906104d491906122f5565b610f26565b6040516104e69190612506565b60405180910390f35b6104f7610f6e565b005b610501610f82565b60405161050e9190612506565b60405180910390f35b61051f610f88565b60405161052c9190612506565b60405180910390f35b61054f600480360381019061054a91906122f5565b610f8e565b005b61055961107d565b60405161056691906125b0565b60405180910390f35b61058960048036038101906105849190612700565b6110a7565b005b61059361149f565b6040516105a09190612506565b60405180910390f35b6105b16114c5565b6040516105be91906123e9565b60405180910390f35b6105cf611557565b6040516105dc9190612506565b60405180910390f35b6105ff60048036038101906105fa9190612441565b61155d565b60405161060c919061249c565b60405180910390f35b61061d611580565b005b610627611705565b6040516106349190612506565b60405180910390f35b610657600480360381019061065291906122f5565b611713565b6040516106649190612506565b60405180910390f35b610687600480360381019061068291906122f5565b61172b565b6040516106949190612506565b60405180910390f35b6106a5611743565b6040516106b29190612506565b60405180910390f35b6106d560048036038101906106d09190612749565b611752565b6040516106e29190612506565b60405180910390f35b6106f36117d9565b6040516107009190612506565b60405180910390f35b6107116117df565b60405161071e9190612506565b60405180910390f35b610741600480360381019061073c91906122f5565b6117ed565b005b61074b611873565b600073ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff16036107ba576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016107b1906127d5565b60405180910390fd5b601160008273ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060009054906101000a900460ff1615610847576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161083e90612841565b60405180910390fd5b6001601160008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060006101000a81548160ff02191690831515021790555050565b600760009054906101000a900460ff1681565b6060600380546108c490612890565b80601f01602080910402602001604051908101604052809291908181526020018280546108f090612890565b801561093d5780601f106109125761010080835404028352916020019161093d565b820191906000526020600020905b81548152906001019060200180831161092057829003601f168201915b5050505050905090565b6000806109526118fa565b905061095f818585611902565b600191505092915050565b610972611873565b81600c8190555080600d819055505050565b600760019054906101000a900460ff1681565b60085481565b6a01a784379d99db4200000081565b600b60009054906101000a900460ff1681565b60006009541180156109d357506009544210155b610a12576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610a099061290d565b60405180910390fd5b6000600f60003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000205411610a94576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610a8b90612979565b60405180910390fd5b600f60003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002054601060003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002054610b1d33610f26565b610b2791906129c8565b1015610b68576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610b5f90612a6e565b60405180910390fd5b6000600f60003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000205490506000600f60003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000208190555080601060003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000206000828254610c409190612a8e565b92505081905550610c52303383611914565b3373ffffffffffffffffffffffffffffffffffffffff167fa267b5cf59cf96d97abf36d29c63cf7bdef867c208ca8575c47f2cafda39abc982604051610c989190612506565b60405180910390a250565b6000600254905090565b610cb5611873565b6000600760009054906101000a900460ff1660ff1611610d0a576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610d0190612b0e565b60405180910390fd5b600760019054906101000a900460ff16610d59576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610d5090612b7a565b60405180910390fd5b610d61611a08565b565b600080610d6e6118fa565b9050610d7b858285611a39565b610d86858585611914565b60019150509392505050565b60006012905090565b6a034f086f3b33b68400000081565b6a084595161401484a00000081565b6000808203610dcc576103e89050610e05565b6000600183610ddb9190612a8e565b90506000816103e8610ded9190612bc9565b905060648111610dfe576064610e00565b805b925050505b919050565b60116020528060005260406000206000915054906101000a900460ff1681565b600e6020528060005260406000206000915090505481565b610e4a611873565b6003600760009054906101000a900460ff1660ff1614610e9f576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610e9690612c46565b60405180910390fd5b600760019054906101000a900460ff1615610eef576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610ee690612cb2565b60405180910390fd5b6001600b60006101000a81548160ff021916908315150217905550610f2430610f1661107d565b610f1f30610f26565b611914565b565b60008060008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020549050919050565b610f76611873565b610f806000611ace565b565b600d5481565b600c5481565b610f96611873565b601160008273ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060009054906101000a900460ff16611022576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161101990612d1e565b60405180910390fd5b6000601160008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060006101000a81548160ff02191690831515021790555050565b6000600560009054906101000a900473ffffffffffffffffffffffffffffffffffffffff16905090565b6110af611b94565b3273ffffffffffffffffffffffffffffffffffffffff163373ffffffffffffffffffffffffffffffffffffffff161461111d576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161111490612d8a565b60405180910390fd5b6000600760009054906101000a900460ff1660ff1611611172576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161116990612df6565b60405180910390fd5b600760019054906101000a900460ff166111c1576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016111b890612df6565b60405180910390fd5b600854421015611206576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016111fd90612df6565b60405180910390fd5b60006008544261121691906129c8565b90506000600c54821161123357670de0b6b3a764000090506113a7565b600d54821161126257670de0b6b3a76400006112516001600a611bda565b61125b9190612e16565b90506113a6565b600061126d33610f26565b9050600080820361128157600a90506112b3565b683635c9adc5dea00000821061129b576103e890506112b2565b670de0b6b3a7640000826112af9190612bc9565b90505b5b60006112fd600e60003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002054610db9565b90506103e8818361130e9190612e16565b6113189190612bc9565b9150600a821161132957600a61132b565b815b9150670de0b6b3a7640000611341600a84611bda565b61134b9190612e16565b9350600e60003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600081548092919061139d90612e58565b91905055505050505b5b600a548111156113bf57600a5490506113be611a08565b5b80600f60003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600082825461140e9190612a8e565b9250508190555080600a600082825461142791906129c8565b925050819055508260405161143c9190612edc565b60405180910390203373ffffffffffffffffffffffffffffffffffffffff167f920a78ccdc638c996c9176021cdcc049dd8ad461c7de23754d4d5d7c232ad1518360405161148a9190612506565b60405180910390a3505061149c611c5c565b50565b60006114aa30610f26565b6a084595161401484a0000006114c091906129c8565b905090565b6060600480546114d490612890565b80601f016020809104026020016040519081016040528092919081815260200182805461150090612890565b801561154d5780601f106115225761010080835404028352916020019161154d565b820191906000526020600020905b81548152906001019060200180831161153057829003601f168201915b5050505050905090565b60095481565b6000806115686118fa565b9050611575818585611914565b600191505092915050565b611588611873565b6003600760009054906101000a900460ff1660ff16106115dd576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016115d490612f3f565b60405180910390fd5b4244336040516020016115f293929190612fc8565b6040516020818303038152906040528051906020012060001c6012819055506007600081819054906101000a900460ff168092919061163090613005565b91906101000a81548160ff021916908360ff160217905550504260088190555060006009819055506001600760016101000a81548160ff0219169083151502179055506001600760009054906101000a900460ff1660ff16036116a35769d3c21bcecceda1000000600a81905550611703565b6002600760009054906101000a900460ff1660ff16036116d4576a01a784379d99db42000000600a81905550611702565b6003600760009054906101000a900460ff1660ff1603611701576a034f086f3b33b684000000600a819055505b5b5b565b69d3c21bcecceda100000081565b60106020528060005260406000206000915090505481565b600f6020528060005260406000206000915090505481565b6a01a784379d99db4200000081565b6000600160008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002054905092915050565b600a5481565b69d3c21bcecceda100000081565b6117f5611873565b600073ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff16036118675760006040517f1e4fbdf700000000000000000000000000000000000000000000000000000000815260040161185e91906125b0565b60405180910390fd5b61187081611ace565b50565b61187b6118fa565b73ffffffffffffffffffffffffffffffffffffffff1661189961107d565b73ffffffffffffffffffffffffffffffffffffffff16146118f8576118bc6118fa565b6040517f118cdaa70000000000000000000000000000000000000000000000000000000081526004016118ef91906125b0565b60405180910390fd5b565b600033905090565b61190f8383836001611c66565b505050565b600073ffffffffffffffffffffffffffffffffffffffff168373ffffffffffffffffffffffffffffffffffffffff16036119865760006040517f96c6fd1e00000000000000000000000000000000000000000000000000000000815260040161197d91906125b0565b60405180910390fd5b600073ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff16036119f85760006040517fec442f050000000000000000000000000000000000000000000000000000000081526004016119ef91906125b0565b60405180910390fd5b611a03838383611e3d565b505050565b6000600760016101000a81548160ff02191690831515021790555061070842611a319190612a8e565b600981905550565b6000611a458484611752565b90507fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff811015611ac85781811015611ab8578281836040517ffb8f41b2000000000000000000000000000000000000000000000000000000008152600401611aaf9392919061302e565b60405180910390fd5b611ac784848484036000611c66565b5b50505050565b6000600560009054906101000a900473ffffffffffffffffffffffffffffffffffffffff16905081600560006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff1602179055508173ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff167f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e060405160405180910390a35050565b600260065403611bd0576040517f3ee5aeb500000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b6002600681905550565b6000818310611beb57829050611c56565b424433601254604051602001611c049493929190613065565b6040516020818303038152906040528051906020012060001c60128190555060018383611c3191906129c8565b611c3b9190612a8e565b601254611c4891906130b3565b83611c539190612a8e565b90505b92915050565b6001600681905550565b600073ffffffffffffffffffffffffffffffffffffffff168473ffffffffffffffffffffffffffffffffffffffff1603611cd85760006040517fe602df05000000000000000000000000000000000000000000000000000000008152600401611ccf91906125b0565b60405180910390fd5b600073ffffffffffffffffffffffffffffffffffffffff168373ffffffffffffffffffffffffffffffffffffffff1603611d4a5760006040517f94280d62000000000000000000000000000000000000000000000000000000008152600401611d4191906125b0565b60405180910390fd5b81600160008673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060008573ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020819055508015611e37578273ffffffffffffffffffffffffffffffffffffffff168473ffffffffffffffffffffffffffffffffffffffff167f8c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b92584604051611e2e9190612506565b60405180910390a35b50505050565b611e4883838361205e565b600b60009054906101000a900460ff16158015611e925750600073ffffffffffffffffffffffffffffffffffffffff168373ffffffffffffffffffffffffffffffffffffffff1614155b8015611ecb5750600073ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff1614155b1561205957601160008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060009054906101000a900460ff16158015611f745750601160008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060009054906101000a900460ff16155b15612058576000601060008573ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002054111561205757601060008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002054601060008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600082825461204f9190612a8e565b925050819055505b5b5b505050565b600073ffffffffffffffffffffffffffffffffffffffff168373ffffffffffffffffffffffffffffffffffffffff16036120b05780600260008282546120a49190612a8e565b92505081905550612183565b60008060008573ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000205490508181101561213c578381836040517fe450d38c0000000000000000000000000000000000000000000000000000000081526004016121339392919061302e565b60405180910390fd5b8181036000808673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002081905550505b600073ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff16036121cc5780600260008282540392505081905550612219565b806000808473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600082825401925050819055505b8173ffffffffffffffffffffffffffffffffffffffff168373ffffffffffffffffffffffffffffffffffffffff167fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef836040516122769190612506565b60405180910390a3505050565b6000604051905090565b600080fd5b600080fd5b600073ffffffffffffffffffffffffffffffffffffffff82169050919050565b60006122c282612297565b9050919050565b6122d2816122b7565b81146122dd57600080fd5b50565b6000813590506122ef816122c9565b92915050565b60006020828403121561230b5761230a61228d565b5b6000612319848285016122e0565b91505092915050565b600060ff82169050919050565b61233881612322565b82525050565b6000602082019050612353600083018461232f565b92915050565b600081519050919050565b600082825260208201905092915050565b60005b83811015612393578082015181840152602081019050612378565b60008484015250505050565b6000601f19601f8301169050919050565b60006123bb82612359565b6123c58185612364565b93506123d5818560208601612375565b6123de8161239f565b840191505092915050565b6000602082019050818103600083015261240381846123b0565b905092915050565b6000819050919050565b61241e8161240b565b811461242957600080fd5b50565b60008135905061243b81612415565b92915050565b600080604083850312156124585761245761228d565b5b6000612466858286016122e0565b92505060206124778582860161242c565b9150509250929050565b60008115159050919050565b61249681612481565b82525050565b60006020820190506124b1600083018461248d565b92915050565b600080604083850312156124ce576124cd61228d565b5b60006124dc8582860161242c565b92505060206124ed8582860161242c565b9150509250929050565b6125008161240b565b82525050565b600060208201905061251b60008301846124f7565b92915050565b60008060006060848603121561253a5761253961228d565b5b6000612548868287016122e0565b9350506020612559868287016122e0565b925050604061256a8682870161242c565b9150509250925092565b60006020828403121561258a5761258961228d565b5b60006125988482850161242c565b91505092915050565b6125aa816122b7565b82525050565b60006020820190506125c560008301846125a1565b92915050565b600080fd5b600080fd5b7f4e487b7100000000000000000000000000000000000000000000000000000000600052604160045260246000fd5b61260d8261239f565b810181811067ffffffffffffffff8211171561262c5761262b6125d5565b5b80604052505050565b600061263f612283565b905061264b8282612604565b919050565b600067ffffffffffffffff82111561266b5761266a6125d5565b5b6126748261239f565b9050602081019050919050565b82818337600083830152505050565b60006126a361269e84612650565b612635565b9050828152602081018484840111156126bf576126be6125d0565b5b6126ca848285612681565b509392505050565b600082601f8301126126e7576126e66125cb565b5b81356126f7848260208601612690565b91505092915050565b6000602082840312156127165761271561228d565b5b600082013567ffffffffffffffff81111561273457612733612292565b5b612740848285016126d2565b91505092915050565b600080604083850312156127605761275f61228d565b5b600061276e858286016122e0565b925050602061277f858286016122e0565b9150509250929050565b7f43616e6e6f74206578636c756465207a65726f20616464726573730000000000600082015250565b60006127bf601b83612364565b91506127ca82612789565b602082019050919050565b600060208201905081810360008301526127ee816127b2565b9050919050565b7f44455820616c7265616479206578636c75646564000000000000000000000000600082015250565b600061282b601483612364565b9150612836826127f5565b602082019050919050565b6000602082019050818103600083015261285a8161281e565b9050919050565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052602260045260246000fd5b600060028204905060018216806128a857607f821691505b6020821081036128bb576128ba612861565b5b50919050565b7f5472616e73666572206e6f742061637469766520796574000000000000000000600082015250565b60006128f7601783612364565b9150612902826128c1565b602082019050919050565b60006020820190508181036000830152612926816128ea565b9050919050565b7f4e6f20746f6b656e7320746f207472616e736665720000000000000000000000600082015250565b6000612963601583612364565b915061296e8261292d565b602082019050919050565b6000602082019050818103600083015261299281612956565b9050919050565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052601160045260246000fd5b60006129d38261240b565b91506129de8361240b565b92508282039050818111156129f6576129f5612999565b5b92915050565b7f496e73756666696369656e742062616c616e636520746f207472616e7366657260008201527f2c206f776e206d6f7265202457494e4400000000000000000000000000000000602082015250565b6000612a58603083612364565b9150612a63826129fc565b604082019050919050565b60006020820190508181036000830152612a8781612a4b565b9050919050565b6000612a998261240b565b9150612aa48361240b565b9250828201905080821115612abc57612abb612999565b5b92915050565b7f4e6f206163746976652070686173650000000000000000000000000000000000600082015250565b6000612af8600f83612364565b9150612b0382612ac2565b602082019050919050565b60006020820190508181036000830152612b2781612aeb565b9050919050565b7f506861736520616c726561647920656e64656400000000000000000000000000600082015250565b6000612b64601383612364565b9150612b6f82612b2e565b602082019050919050565b60006020820190508181036000830152612b9381612b57565b9050919050565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052601260045260246000fd5b6000612bd48261240b565b9150612bdf8361240b565b925082612bef57612bee612b9a565b5b828204905092915050565b7f4f6e6c7920616674657220332064726f70730000000000000000000000000000600082015250565b6000612c30601283612364565b9150612c3b82612bfa565b602082019050919050565b60006020820190508181036000830152612c5f81612c23565b9050919050565b7f456e642070686173652066697273740000000000000000000000000000000000600082015250565b6000612c9c600f83612364565b9150612ca782612c66565b602082019050919050565b60006020820190508181036000830152612ccb81612c8f565b9050919050565b7f444558206e6f74206578636c7564656400000000000000000000000000000000600082015250565b6000612d08601083612364565b9150612d1382612cd2565b602082019050919050565b60006020820190508181036000830152612d3781612cfb565b9050919050565b7f4e6f20636f6e7472616374730000000000000000000000000000000000000000600082015250565b6000612d74600c83612364565b9150612d7f82612d3e565b602082019050919050565b60006020820190508181036000830152612da381612d67565b9050919050565b7f436c61696d64726f70206e6f7420616374697665000000000000000000000000600082015250565b6000612de0601483612364565b9150612deb82612daa565b602082019050919050565b60006020820190508181036000830152612e0f81612dd3565b9050919050565b6000612e218261240b565b9150612e2c8361240b565b9250828202612e3a8161240b565b91508282048414831517612e5157612e50612999565b5b5092915050565b6000612e638261240b565b91507fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff8203612e9557612e94612999565b5b600182019050919050565b600081905092915050565b6000612eb682612359565b612ec08185612ea0565b9350612ed0818560208601612375565b80840191505092915050565b6000612ee88284612eab565b915081905092915050565b7f416c6c2070686173657320636f6d706c65746564000000000000000000000000600082015250565b6000612f29601483612364565b9150612f3482612ef3565b602082019050919050565b60006020820190508181036000830152612f5881612f1c565b9050919050565b6000819050919050565b612f7a612f758261240b565b612f5f565b82525050565b60008160601b9050919050565b6000612f9882612f80565b9050919050565b6000612faa82612f8d565b9050919050565b612fc2612fbd826122b7565b612f9f565b82525050565b6000612fd48286612f69565b602082019150612fe48285612f69565b602082019150612ff48284612fb1565b601482019150819050949350505050565b600061301082612322565b915060ff820361302357613022612999565b5b600182019050919050565b600060608201905061304360008301866125a1565b61305060208301856124f7565b61305d60408301846124f7565b949350505050565b60006130718287612f69565b6020820191506130818286612f69565b6020820191506130918285612fb1565b6014820191506130a18284612f69565b60208201915081905095945050505050565b60006130be8261240b565b91506130c98361240b565b9250826130d9576130d8612b9a565b5b82820690509291505056fea264697066735822122096aff99a7bbed3cbeded077cf90490c423031dc4aea6062891839e0ed6169fde64736f6c63430008140033

[ Download: CSV Export  ]
[ Download: CSV Export  ]

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