ETH Price: $3,310.07 (-0.75%)

Token

ChampagneCarbon (CCB)
 

Overview

Max Total Supply

20 CCB

Holders

16

Market

Volume (24H)

N/A

Min Price (24H)

N/A

Max Price (24H)

N/A
Balance
1 CCB
0x7C7e492f78Cbdc8755D6AFAE1f3423A84ff39cC8
Loading...
Loading
Loading...
Loading
Loading...
Loading

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

Contract Source Code Verified (Exact Match)

Contract Name:
ChampagneCarbon

Compiler Version
v0.8.18+commit.87f61d96

Optimization Enabled:
No with 200 runs

Other Settings:
default evmVersion, MIT license

Contract Source Code (Solidity)

/**
 *Submitted for verification at Etherscan.io on 2023-05-24
*/

// File: hardhat/console.sol


pragma solidity >= 0.4.22 <0.9.0;

library console {
	address constant CONSOLE_ADDRESS = address(0x000000000000000000636F6e736F6c652e6c6f67);

	function _sendLogPayload(bytes memory payload) private view {
		uint256 payloadLength = payload.length;
		address consoleAddress = CONSOLE_ADDRESS;
		assembly {
			let payloadStart := add(payload, 32)
			let r := staticcall(gas(), consoleAddress, payloadStart, payloadLength, 0, 0)
		}
	}

	function log() internal view {
		_sendLogPayload(abi.encodeWithSignature("log()"));
	}

	function logInt(int256 p0) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(int256)", p0));
	}

	function logUint(uint256 p0) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256)", p0));
	}

	function logString(string memory p0) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string)", p0));
	}

	function logBool(bool p0) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool)", p0));
	}

	function logAddress(address p0) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address)", p0));
	}

	function logBytes(bytes memory p0) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bytes)", p0));
	}

	function logBytes1(bytes1 p0) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bytes1)", p0));
	}

	function logBytes2(bytes2 p0) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bytes2)", p0));
	}

	function logBytes3(bytes3 p0) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bytes3)", p0));
	}

	function logBytes4(bytes4 p0) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bytes4)", p0));
	}

	function logBytes5(bytes5 p0) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bytes5)", p0));
	}

	function logBytes6(bytes6 p0) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bytes6)", p0));
	}

	function logBytes7(bytes7 p0) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bytes7)", p0));
	}

	function logBytes8(bytes8 p0) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bytes8)", p0));
	}

	function logBytes9(bytes9 p0) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bytes9)", p0));
	}

	function logBytes10(bytes10 p0) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bytes10)", p0));
	}

	function logBytes11(bytes11 p0) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bytes11)", p0));
	}

	function logBytes12(bytes12 p0) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bytes12)", p0));
	}

	function logBytes13(bytes13 p0) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bytes13)", p0));
	}

	function logBytes14(bytes14 p0) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bytes14)", p0));
	}

	function logBytes15(bytes15 p0) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bytes15)", p0));
	}

	function logBytes16(bytes16 p0) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bytes16)", p0));
	}

	function logBytes17(bytes17 p0) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bytes17)", p0));
	}

	function logBytes18(bytes18 p0) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bytes18)", p0));
	}

	function logBytes19(bytes19 p0) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bytes19)", p0));
	}

	function logBytes20(bytes20 p0) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bytes20)", p0));
	}

	function logBytes21(bytes21 p0) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bytes21)", p0));
	}

	function logBytes22(bytes22 p0) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bytes22)", p0));
	}

	function logBytes23(bytes23 p0) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bytes23)", p0));
	}

	function logBytes24(bytes24 p0) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bytes24)", p0));
	}

	function logBytes25(bytes25 p0) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bytes25)", p0));
	}

	function logBytes26(bytes26 p0) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bytes26)", p0));
	}

	function logBytes27(bytes27 p0) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bytes27)", p0));
	}

	function logBytes28(bytes28 p0) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bytes28)", p0));
	}

	function logBytes29(bytes29 p0) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bytes29)", p0));
	}

	function logBytes30(bytes30 p0) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bytes30)", p0));
	}

	function logBytes31(bytes31 p0) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bytes31)", p0));
	}

	function logBytes32(bytes32 p0) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bytes32)", p0));
	}

	function log(uint256 p0) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256)", p0));
	}

	function log(string memory p0) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string)", p0));
	}

	function log(bool p0) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool)", p0));
	}

	function log(address p0) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address)", p0));
	}

	function log(uint256 p0, uint256 p1) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,uint256)", p0, p1));
	}

	function log(uint256 p0, string memory p1) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,string)", p0, p1));
	}

	function log(uint256 p0, bool p1) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,bool)", p0, p1));
	}

	function log(uint256 p0, address p1) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,address)", p0, p1));
	}

	function log(string memory p0, uint256 p1) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,uint256)", p0, p1));
	}

	function log(string memory p0, string memory p1) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,string)", p0, p1));
	}

	function log(string memory p0, bool p1) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,bool)", p0, p1));
	}

	function log(string memory p0, address p1) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,address)", p0, p1));
	}

	function log(bool p0, uint256 p1) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,uint256)", p0, p1));
	}

	function log(bool p0, string memory p1) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,string)", p0, p1));
	}

	function log(bool p0, bool p1) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,bool)", p0, p1));
	}

	function log(bool p0, address p1) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,address)", p0, p1));
	}

	function log(address p0, uint256 p1) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,uint256)", p0, p1));
	}

	function log(address p0, string memory p1) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,string)", p0, p1));
	}

	function log(address p0, bool p1) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,bool)", p0, p1));
	}

	function log(address p0, address p1) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,address)", p0, p1));
	}

	function log(uint256 p0, uint256 p1, uint256 p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,uint256,uint256)", p0, p1, p2));
	}

	function log(uint256 p0, uint256 p1, string memory p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,uint256,string)", p0, p1, p2));
	}

	function log(uint256 p0, uint256 p1, bool p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,uint256,bool)", p0, p1, p2));
	}

	function log(uint256 p0, uint256 p1, address p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,uint256,address)", p0, p1, p2));
	}

	function log(uint256 p0, string memory p1, uint256 p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,string,uint256)", p0, p1, p2));
	}

	function log(uint256 p0, string memory p1, string memory p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,string,string)", p0, p1, p2));
	}

	function log(uint256 p0, string memory p1, bool p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,string,bool)", p0, p1, p2));
	}

	function log(uint256 p0, string memory p1, address p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,string,address)", p0, p1, p2));
	}

	function log(uint256 p0, bool p1, uint256 p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,bool,uint256)", p0, p1, p2));
	}

	function log(uint256 p0, bool p1, string memory p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,bool,string)", p0, p1, p2));
	}

	function log(uint256 p0, bool p1, bool p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,bool,bool)", p0, p1, p2));
	}

	function log(uint256 p0, bool p1, address p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,bool,address)", p0, p1, p2));
	}

	function log(uint256 p0, address p1, uint256 p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,address,uint256)", p0, p1, p2));
	}

	function log(uint256 p0, address p1, string memory p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,address,string)", p0, p1, p2));
	}

	function log(uint256 p0, address p1, bool p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,address,bool)", p0, p1, p2));
	}

	function log(uint256 p0, address p1, address p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,address,address)", p0, p1, p2));
	}

	function log(string memory p0, uint256 p1, uint256 p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,uint256,uint256)", p0, p1, p2));
	}

	function log(string memory p0, uint256 p1, string memory p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,uint256,string)", p0, p1, p2));
	}

	function log(string memory p0, uint256 p1, bool p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,uint256,bool)", p0, p1, p2));
	}

	function log(string memory p0, uint256 p1, address p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,uint256,address)", p0, p1, p2));
	}

	function log(string memory p0, string memory p1, uint256 p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,string,uint256)", p0, p1, p2));
	}

	function log(string memory p0, string memory p1, string memory p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,string,string)", p0, p1, p2));
	}

	function log(string memory p0, string memory p1, bool p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,string,bool)", p0, p1, p2));
	}

	function log(string memory p0, string memory p1, address p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,string,address)", p0, p1, p2));
	}

	function log(string memory p0, bool p1, uint256 p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,bool,uint256)", p0, p1, p2));
	}

	function log(string memory p0, bool p1, string memory p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,bool,string)", p0, p1, p2));
	}

	function log(string memory p0, bool p1, bool p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,bool,bool)", p0, p1, p2));
	}

	function log(string memory p0, bool p1, address p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,bool,address)", p0, p1, p2));
	}

	function log(string memory p0, address p1, uint256 p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,address,uint256)", p0, p1, p2));
	}

	function log(string memory p0, address p1, string memory p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,address,string)", p0, p1, p2));
	}

	function log(string memory p0, address p1, bool p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,address,bool)", p0, p1, p2));
	}

	function log(string memory p0, address p1, address p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,address,address)", p0, p1, p2));
	}

	function log(bool p0, uint256 p1, uint256 p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,uint256,uint256)", p0, p1, p2));
	}

	function log(bool p0, uint256 p1, string memory p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,uint256,string)", p0, p1, p2));
	}

	function log(bool p0, uint256 p1, bool p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,uint256,bool)", p0, p1, p2));
	}

	function log(bool p0, uint256 p1, address p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,uint256,address)", p0, p1, p2));
	}

	function log(bool p0, string memory p1, uint256 p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,string,uint256)", p0, p1, p2));
	}

	function log(bool p0, string memory p1, string memory p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,string,string)", p0, p1, p2));
	}

	function log(bool p0, string memory p1, bool p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,string,bool)", p0, p1, p2));
	}

	function log(bool p0, string memory p1, address p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,string,address)", p0, p1, p2));
	}

	function log(bool p0, bool p1, uint256 p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,bool,uint256)", p0, p1, p2));
	}

	function log(bool p0, bool p1, string memory p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,bool,string)", p0, p1, p2));
	}

	function log(bool p0, bool p1, bool p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,bool,bool)", p0, p1, p2));
	}

	function log(bool p0, bool p1, address p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,bool,address)", p0, p1, p2));
	}

	function log(bool p0, address p1, uint256 p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,address,uint256)", p0, p1, p2));
	}

	function log(bool p0, address p1, string memory p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,address,string)", p0, p1, p2));
	}

	function log(bool p0, address p1, bool p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,address,bool)", p0, p1, p2));
	}

	function log(bool p0, address p1, address p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,address,address)", p0, p1, p2));
	}

	function log(address p0, uint256 p1, uint256 p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,uint256,uint256)", p0, p1, p2));
	}

	function log(address p0, uint256 p1, string memory p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,uint256,string)", p0, p1, p2));
	}

	function log(address p0, uint256 p1, bool p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,uint256,bool)", p0, p1, p2));
	}

	function log(address p0, uint256 p1, address p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,uint256,address)", p0, p1, p2));
	}

	function log(address p0, string memory p1, uint256 p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,string,uint256)", p0, p1, p2));
	}

	function log(address p0, string memory p1, string memory p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,string,string)", p0, p1, p2));
	}

	function log(address p0, string memory p1, bool p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,string,bool)", p0, p1, p2));
	}

	function log(address p0, string memory p1, address p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,string,address)", p0, p1, p2));
	}

	function log(address p0, bool p1, uint256 p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,bool,uint256)", p0, p1, p2));
	}

	function log(address p0, bool p1, string memory p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,bool,string)", p0, p1, p2));
	}

	function log(address p0, bool p1, bool p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,bool,bool)", p0, p1, p2));
	}

	function log(address p0, bool p1, address p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,bool,address)", p0, p1, p2));
	}

	function log(address p0, address p1, uint256 p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,address,uint256)", p0, p1, p2));
	}

	function log(address p0, address p1, string memory p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,address,string)", p0, p1, p2));
	}

	function log(address p0, address p1, bool p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,address,bool)", p0, p1, p2));
	}

	function log(address p0, address p1, address p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,address,address)", p0, p1, p2));
	}

	function log(uint256 p0, uint256 p1, uint256 p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,uint256,uint256,uint256)", p0, p1, p2, p3));
	}

	function log(uint256 p0, uint256 p1, uint256 p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,uint256,uint256,string)", p0, p1, p2, p3));
	}

	function log(uint256 p0, uint256 p1, uint256 p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,uint256,uint256,bool)", p0, p1, p2, p3));
	}

	function log(uint256 p0, uint256 p1, uint256 p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,uint256,uint256,address)", p0, p1, p2, p3));
	}

	function log(uint256 p0, uint256 p1, string memory p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,uint256,string,uint256)", p0, p1, p2, p3));
	}

	function log(uint256 p0, uint256 p1, string memory p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,uint256,string,string)", p0, p1, p2, p3));
	}

	function log(uint256 p0, uint256 p1, string memory p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,uint256,string,bool)", p0, p1, p2, p3));
	}

	function log(uint256 p0, uint256 p1, string memory p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,uint256,string,address)", p0, p1, p2, p3));
	}

	function log(uint256 p0, uint256 p1, bool p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,uint256,bool,uint256)", p0, p1, p2, p3));
	}

	function log(uint256 p0, uint256 p1, bool p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,uint256,bool,string)", p0, p1, p2, p3));
	}

	function log(uint256 p0, uint256 p1, bool p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,uint256,bool,bool)", p0, p1, p2, p3));
	}

	function log(uint256 p0, uint256 p1, bool p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,uint256,bool,address)", p0, p1, p2, p3));
	}

	function log(uint256 p0, uint256 p1, address p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,uint256,address,uint256)", p0, p1, p2, p3));
	}

	function log(uint256 p0, uint256 p1, address p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,uint256,address,string)", p0, p1, p2, p3));
	}

	function log(uint256 p0, uint256 p1, address p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,uint256,address,bool)", p0, p1, p2, p3));
	}

	function log(uint256 p0, uint256 p1, address p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,uint256,address,address)", p0, p1, p2, p3));
	}

	function log(uint256 p0, string memory p1, uint256 p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,string,uint256,uint256)", p0, p1, p2, p3));
	}

	function log(uint256 p0, string memory p1, uint256 p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,string,uint256,string)", p0, p1, p2, p3));
	}

	function log(uint256 p0, string memory p1, uint256 p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,string,uint256,bool)", p0, p1, p2, p3));
	}

	function log(uint256 p0, string memory p1, uint256 p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,string,uint256,address)", p0, p1, p2, p3));
	}

	function log(uint256 p0, string memory p1, string memory p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,string,string,uint256)", p0, p1, p2, p3));
	}

	function log(uint256 p0, string memory p1, string memory p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,string,string,string)", p0, p1, p2, p3));
	}

	function log(uint256 p0, string memory p1, string memory p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,string,string,bool)", p0, p1, p2, p3));
	}

	function log(uint256 p0, string memory p1, string memory p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,string,string,address)", p0, p1, p2, p3));
	}

	function log(uint256 p0, string memory p1, bool p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,string,bool,uint256)", p0, p1, p2, p3));
	}

	function log(uint256 p0, string memory p1, bool p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,string,bool,string)", p0, p1, p2, p3));
	}

	function log(uint256 p0, string memory p1, bool p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,string,bool,bool)", p0, p1, p2, p3));
	}

	function log(uint256 p0, string memory p1, bool p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,string,bool,address)", p0, p1, p2, p3));
	}

	function log(uint256 p0, string memory p1, address p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,string,address,uint256)", p0, p1, p2, p3));
	}

	function log(uint256 p0, string memory p1, address p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,string,address,string)", p0, p1, p2, p3));
	}

	function log(uint256 p0, string memory p1, address p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,string,address,bool)", p0, p1, p2, p3));
	}

	function log(uint256 p0, string memory p1, address p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,string,address,address)", p0, p1, p2, p3));
	}

	function log(uint256 p0, bool p1, uint256 p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,bool,uint256,uint256)", p0, p1, p2, p3));
	}

	function log(uint256 p0, bool p1, uint256 p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,bool,uint256,string)", p0, p1, p2, p3));
	}

	function log(uint256 p0, bool p1, uint256 p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,bool,uint256,bool)", p0, p1, p2, p3));
	}

	function log(uint256 p0, bool p1, uint256 p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,bool,uint256,address)", p0, p1, p2, p3));
	}

	function log(uint256 p0, bool p1, string memory p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,bool,string,uint256)", p0, p1, p2, p3));
	}

	function log(uint256 p0, bool p1, string memory p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,bool,string,string)", p0, p1, p2, p3));
	}

	function log(uint256 p0, bool p1, string memory p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,bool,string,bool)", p0, p1, p2, p3));
	}

	function log(uint256 p0, bool p1, string memory p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,bool,string,address)", p0, p1, p2, p3));
	}

	function log(uint256 p0, bool p1, bool p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,bool,bool,uint256)", p0, p1, p2, p3));
	}

	function log(uint256 p0, bool p1, bool p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,bool,bool,string)", p0, p1, p2, p3));
	}

	function log(uint256 p0, bool p1, bool p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,bool,bool,bool)", p0, p1, p2, p3));
	}

	function log(uint256 p0, bool p1, bool p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,bool,bool,address)", p0, p1, p2, p3));
	}

	function log(uint256 p0, bool p1, address p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,bool,address,uint256)", p0, p1, p2, p3));
	}

	function log(uint256 p0, bool p1, address p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,bool,address,string)", p0, p1, p2, p3));
	}

	function log(uint256 p0, bool p1, address p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,bool,address,bool)", p0, p1, p2, p3));
	}

	function log(uint256 p0, bool p1, address p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,bool,address,address)", p0, p1, p2, p3));
	}

	function log(uint256 p0, address p1, uint256 p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,address,uint256,uint256)", p0, p1, p2, p3));
	}

	function log(uint256 p0, address p1, uint256 p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,address,uint256,string)", p0, p1, p2, p3));
	}

	function log(uint256 p0, address p1, uint256 p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,address,uint256,bool)", p0, p1, p2, p3));
	}

	function log(uint256 p0, address p1, uint256 p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,address,uint256,address)", p0, p1, p2, p3));
	}

	function log(uint256 p0, address p1, string memory p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,address,string,uint256)", p0, p1, p2, p3));
	}

	function log(uint256 p0, address p1, string memory p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,address,string,string)", p0, p1, p2, p3));
	}

	function log(uint256 p0, address p1, string memory p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,address,string,bool)", p0, p1, p2, p3));
	}

	function log(uint256 p0, address p1, string memory p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,address,string,address)", p0, p1, p2, p3));
	}

	function log(uint256 p0, address p1, bool p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,address,bool,uint256)", p0, p1, p2, p3));
	}

	function log(uint256 p0, address p1, bool p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,address,bool,string)", p0, p1, p2, p3));
	}

	function log(uint256 p0, address p1, bool p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,address,bool,bool)", p0, p1, p2, p3));
	}

	function log(uint256 p0, address p1, bool p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,address,bool,address)", p0, p1, p2, p3));
	}

	function log(uint256 p0, address p1, address p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,address,address,uint256)", p0, p1, p2, p3));
	}

	function log(uint256 p0, address p1, address p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,address,address,string)", p0, p1, p2, p3));
	}

	function log(uint256 p0, address p1, address p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,address,address,bool)", p0, p1, p2, p3));
	}

	function log(uint256 p0, address p1, address p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,address,address,address)", p0, p1, p2, p3));
	}

	function log(string memory p0, uint256 p1, uint256 p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,uint256,uint256,uint256)", p0, p1, p2, p3));
	}

	function log(string memory p0, uint256 p1, uint256 p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,uint256,uint256,string)", p0, p1, p2, p3));
	}

	function log(string memory p0, uint256 p1, uint256 p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,uint256,uint256,bool)", p0, p1, p2, p3));
	}

	function log(string memory p0, uint256 p1, uint256 p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,uint256,uint256,address)", p0, p1, p2, p3));
	}

	function log(string memory p0, uint256 p1, string memory p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,uint256,string,uint256)", p0, p1, p2, p3));
	}

	function log(string memory p0, uint256 p1, string memory p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,uint256,string,string)", p0, p1, p2, p3));
	}

	function log(string memory p0, uint256 p1, string memory p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,uint256,string,bool)", p0, p1, p2, p3));
	}

	function log(string memory p0, uint256 p1, string memory p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,uint256,string,address)", p0, p1, p2, p3));
	}

	function log(string memory p0, uint256 p1, bool p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,uint256,bool,uint256)", p0, p1, p2, p3));
	}

	function log(string memory p0, uint256 p1, bool p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,uint256,bool,string)", p0, p1, p2, p3));
	}

	function log(string memory p0, uint256 p1, bool p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,uint256,bool,bool)", p0, p1, p2, p3));
	}

	function log(string memory p0, uint256 p1, bool p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,uint256,bool,address)", p0, p1, p2, p3));
	}

	function log(string memory p0, uint256 p1, address p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,uint256,address,uint256)", p0, p1, p2, p3));
	}

	function log(string memory p0, uint256 p1, address p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,uint256,address,string)", p0, p1, p2, p3));
	}

	function log(string memory p0, uint256 p1, address p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,uint256,address,bool)", p0, p1, p2, p3));
	}

	function log(string memory p0, uint256 p1, address p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,uint256,address,address)", p0, p1, p2, p3));
	}

	function log(string memory p0, string memory p1, uint256 p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,string,uint256,uint256)", p0, p1, p2, p3));
	}

	function log(string memory p0, string memory p1, uint256 p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,string,uint256,string)", p0, p1, p2, p3));
	}

	function log(string memory p0, string memory p1, uint256 p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,string,uint256,bool)", p0, p1, p2, p3));
	}

	function log(string memory p0, string memory p1, uint256 p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,string,uint256,address)", p0, p1, p2, p3));
	}

	function log(string memory p0, string memory p1, string memory p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,string,string,uint256)", p0, p1, p2, p3));
	}

	function log(string memory p0, string memory p1, string memory p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,string,string,string)", p0, p1, p2, p3));
	}

	function log(string memory p0, string memory p1, string memory p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,string,string,bool)", p0, p1, p2, p3));
	}

	function log(string memory p0, string memory p1, string memory p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,string,string,address)", p0, p1, p2, p3));
	}

	function log(string memory p0, string memory p1, bool p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,string,bool,uint256)", p0, p1, p2, p3));
	}

	function log(string memory p0, string memory p1, bool p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,string,bool,string)", p0, p1, p2, p3));
	}

	function log(string memory p0, string memory p1, bool p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,string,bool,bool)", p0, p1, p2, p3));
	}

	function log(string memory p0, string memory p1, bool p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,string,bool,address)", p0, p1, p2, p3));
	}

	function log(string memory p0, string memory p1, address p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,string,address,uint256)", p0, p1, p2, p3));
	}

	function log(string memory p0, string memory p1, address p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,string,address,string)", p0, p1, p2, p3));
	}

	function log(string memory p0, string memory p1, address p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,string,address,bool)", p0, p1, p2, p3));
	}

	function log(string memory p0, string memory p1, address p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,string,address,address)", p0, p1, p2, p3));
	}

	function log(string memory p0, bool p1, uint256 p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,bool,uint256,uint256)", p0, p1, p2, p3));
	}

	function log(string memory p0, bool p1, uint256 p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,bool,uint256,string)", p0, p1, p2, p3));
	}

	function log(string memory p0, bool p1, uint256 p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,bool,uint256,bool)", p0, p1, p2, p3));
	}

	function log(string memory p0, bool p1, uint256 p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,bool,uint256,address)", p0, p1, p2, p3));
	}

	function log(string memory p0, bool p1, string memory p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,bool,string,uint256)", p0, p1, p2, p3));
	}

	function log(string memory p0, bool p1, string memory p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,bool,string,string)", p0, p1, p2, p3));
	}

	function log(string memory p0, bool p1, string memory p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,bool,string,bool)", p0, p1, p2, p3));
	}

	function log(string memory p0, bool p1, string memory p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,bool,string,address)", p0, p1, p2, p3));
	}

	function log(string memory p0, bool p1, bool p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,bool,bool,uint256)", p0, p1, p2, p3));
	}

	function log(string memory p0, bool p1, bool p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,bool,bool,string)", p0, p1, p2, p3));
	}

	function log(string memory p0, bool p1, bool p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,bool,bool,bool)", p0, p1, p2, p3));
	}

	function log(string memory p0, bool p1, bool p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,bool,bool,address)", p0, p1, p2, p3));
	}

	function log(string memory p0, bool p1, address p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,bool,address,uint256)", p0, p1, p2, p3));
	}

	function log(string memory p0, bool p1, address p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,bool,address,string)", p0, p1, p2, p3));
	}

	function log(string memory p0, bool p1, address p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,bool,address,bool)", p0, p1, p2, p3));
	}

	function log(string memory p0, bool p1, address p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,bool,address,address)", p0, p1, p2, p3));
	}

	function log(string memory p0, address p1, uint256 p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,address,uint256,uint256)", p0, p1, p2, p3));
	}

	function log(string memory p0, address p1, uint256 p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,address,uint256,string)", p0, p1, p2, p3));
	}

	function log(string memory p0, address p1, uint256 p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,address,uint256,bool)", p0, p1, p2, p3));
	}

	function log(string memory p0, address p1, uint256 p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,address,uint256,address)", p0, p1, p2, p3));
	}

	function log(string memory p0, address p1, string memory p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,address,string,uint256)", p0, p1, p2, p3));
	}

	function log(string memory p0, address p1, string memory p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,address,string,string)", p0, p1, p2, p3));
	}

	function log(string memory p0, address p1, string memory p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,address,string,bool)", p0, p1, p2, p3));
	}

	function log(string memory p0, address p1, string memory p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,address,string,address)", p0, p1, p2, p3));
	}

	function log(string memory p0, address p1, bool p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,address,bool,uint256)", p0, p1, p2, p3));
	}

	function log(string memory p0, address p1, bool p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,address,bool,string)", p0, p1, p2, p3));
	}

	function log(string memory p0, address p1, bool p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,address,bool,bool)", p0, p1, p2, p3));
	}

	function log(string memory p0, address p1, bool p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,address,bool,address)", p0, p1, p2, p3));
	}

	function log(string memory p0, address p1, address p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,address,address,uint256)", p0, p1, p2, p3));
	}

	function log(string memory p0, address p1, address p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,address,address,string)", p0, p1, p2, p3));
	}

	function log(string memory p0, address p1, address p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,address,address,bool)", p0, p1, p2, p3));
	}

	function log(string memory p0, address p1, address p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,address,address,address)", p0, p1, p2, p3));
	}

	function log(bool p0, uint256 p1, uint256 p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,uint256,uint256,uint256)", p0, p1, p2, p3));
	}

	function log(bool p0, uint256 p1, uint256 p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,uint256,uint256,string)", p0, p1, p2, p3));
	}

	function log(bool p0, uint256 p1, uint256 p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,uint256,uint256,bool)", p0, p1, p2, p3));
	}

	function log(bool p0, uint256 p1, uint256 p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,uint256,uint256,address)", p0, p1, p2, p3));
	}

	function log(bool p0, uint256 p1, string memory p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,uint256,string,uint256)", p0, p1, p2, p3));
	}

	function log(bool p0, uint256 p1, string memory p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,uint256,string,string)", p0, p1, p2, p3));
	}

	function log(bool p0, uint256 p1, string memory p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,uint256,string,bool)", p0, p1, p2, p3));
	}

	function log(bool p0, uint256 p1, string memory p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,uint256,string,address)", p0, p1, p2, p3));
	}

	function log(bool p0, uint256 p1, bool p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,uint256,bool,uint256)", p0, p1, p2, p3));
	}

	function log(bool p0, uint256 p1, bool p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,uint256,bool,string)", p0, p1, p2, p3));
	}

	function log(bool p0, uint256 p1, bool p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,uint256,bool,bool)", p0, p1, p2, p3));
	}

	function log(bool p0, uint256 p1, bool p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,uint256,bool,address)", p0, p1, p2, p3));
	}

	function log(bool p0, uint256 p1, address p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,uint256,address,uint256)", p0, p1, p2, p3));
	}

	function log(bool p0, uint256 p1, address p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,uint256,address,string)", p0, p1, p2, p3));
	}

	function log(bool p0, uint256 p1, address p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,uint256,address,bool)", p0, p1, p2, p3));
	}

	function log(bool p0, uint256 p1, address p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,uint256,address,address)", p0, p1, p2, p3));
	}

	function log(bool p0, string memory p1, uint256 p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,string,uint256,uint256)", p0, p1, p2, p3));
	}

	function log(bool p0, string memory p1, uint256 p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,string,uint256,string)", p0, p1, p2, p3));
	}

	function log(bool p0, string memory p1, uint256 p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,string,uint256,bool)", p0, p1, p2, p3));
	}

	function log(bool p0, string memory p1, uint256 p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,string,uint256,address)", p0, p1, p2, p3));
	}

	function log(bool p0, string memory p1, string memory p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,string,string,uint256)", p0, p1, p2, p3));
	}

	function log(bool p0, string memory p1, string memory p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,string,string,string)", p0, p1, p2, p3));
	}

	function log(bool p0, string memory p1, string memory p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,string,string,bool)", p0, p1, p2, p3));
	}

	function log(bool p0, string memory p1, string memory p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,string,string,address)", p0, p1, p2, p3));
	}

	function log(bool p0, string memory p1, bool p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,string,bool,uint256)", p0, p1, p2, p3));
	}

	function log(bool p0, string memory p1, bool p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,string,bool,string)", p0, p1, p2, p3));
	}

	function log(bool p0, string memory p1, bool p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,string,bool,bool)", p0, p1, p2, p3));
	}

	function log(bool p0, string memory p1, bool p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,string,bool,address)", p0, p1, p2, p3));
	}

	function log(bool p0, string memory p1, address p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,string,address,uint256)", p0, p1, p2, p3));
	}

	function log(bool p0, string memory p1, address p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,string,address,string)", p0, p1, p2, p3));
	}

	function log(bool p0, string memory p1, address p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,string,address,bool)", p0, p1, p2, p3));
	}

	function log(bool p0, string memory p1, address p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,string,address,address)", p0, p1, p2, p3));
	}

	function log(bool p0, bool p1, uint256 p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,bool,uint256,uint256)", p0, p1, p2, p3));
	}

	function log(bool p0, bool p1, uint256 p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,bool,uint256,string)", p0, p1, p2, p3));
	}

	function log(bool p0, bool p1, uint256 p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,bool,uint256,bool)", p0, p1, p2, p3));
	}

	function log(bool p0, bool p1, uint256 p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,bool,uint256,address)", p0, p1, p2, p3));
	}

	function log(bool p0, bool p1, string memory p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,bool,string,uint256)", p0, p1, p2, p3));
	}

	function log(bool p0, bool p1, string memory p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,bool,string,string)", p0, p1, p2, p3));
	}

	function log(bool p0, bool p1, string memory p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,bool,string,bool)", p0, p1, p2, p3));
	}

	function log(bool p0, bool p1, string memory p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,bool,string,address)", p0, p1, p2, p3));
	}

	function log(bool p0, bool p1, bool p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,bool,bool,uint256)", p0, p1, p2, p3));
	}

	function log(bool p0, bool p1, bool p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,bool,bool,string)", p0, p1, p2, p3));
	}

	function log(bool p0, bool p1, bool p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,bool,bool,bool)", p0, p1, p2, p3));
	}

	function log(bool p0, bool p1, bool p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,bool,bool,address)", p0, p1, p2, p3));
	}

	function log(bool p0, bool p1, address p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,bool,address,uint256)", p0, p1, p2, p3));
	}

	function log(bool p0, bool p1, address p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,bool,address,string)", p0, p1, p2, p3));
	}

	function log(bool p0, bool p1, address p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,bool,address,bool)", p0, p1, p2, p3));
	}

	function log(bool p0, bool p1, address p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,bool,address,address)", p0, p1, p2, p3));
	}

	function log(bool p0, address p1, uint256 p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,address,uint256,uint256)", p0, p1, p2, p3));
	}

	function log(bool p0, address p1, uint256 p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,address,uint256,string)", p0, p1, p2, p3));
	}

	function log(bool p0, address p1, uint256 p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,address,uint256,bool)", p0, p1, p2, p3));
	}

	function log(bool p0, address p1, uint256 p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,address,uint256,address)", p0, p1, p2, p3));
	}

	function log(bool p0, address p1, string memory p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,address,string,uint256)", p0, p1, p2, p3));
	}

	function log(bool p0, address p1, string memory p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,address,string,string)", p0, p1, p2, p3));
	}

	function log(bool p0, address p1, string memory p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,address,string,bool)", p0, p1, p2, p3));
	}

	function log(bool p0, address p1, string memory p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,address,string,address)", p0, p1, p2, p3));
	}

	function log(bool p0, address p1, bool p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,address,bool,uint256)", p0, p1, p2, p3));
	}

	function log(bool p0, address p1, bool p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,address,bool,string)", p0, p1, p2, p3));
	}

	function log(bool p0, address p1, bool p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,address,bool,bool)", p0, p1, p2, p3));
	}

	function log(bool p0, address p1, bool p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,address,bool,address)", p0, p1, p2, p3));
	}

	function log(bool p0, address p1, address p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,address,address,uint256)", p0, p1, p2, p3));
	}

	function log(bool p0, address p1, address p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,address,address,string)", p0, p1, p2, p3));
	}

	function log(bool p0, address p1, address p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,address,address,bool)", p0, p1, p2, p3));
	}

	function log(bool p0, address p1, address p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,address,address,address)", p0, p1, p2, p3));
	}

	function log(address p0, uint256 p1, uint256 p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,uint256,uint256,uint256)", p0, p1, p2, p3));
	}

	function log(address p0, uint256 p1, uint256 p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,uint256,uint256,string)", p0, p1, p2, p3));
	}

	function log(address p0, uint256 p1, uint256 p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,uint256,uint256,bool)", p0, p1, p2, p3));
	}

	function log(address p0, uint256 p1, uint256 p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,uint256,uint256,address)", p0, p1, p2, p3));
	}

	function log(address p0, uint256 p1, string memory p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,uint256,string,uint256)", p0, p1, p2, p3));
	}

	function log(address p0, uint256 p1, string memory p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,uint256,string,string)", p0, p1, p2, p3));
	}

	function log(address p0, uint256 p1, string memory p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,uint256,string,bool)", p0, p1, p2, p3));
	}

	function log(address p0, uint256 p1, string memory p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,uint256,string,address)", p0, p1, p2, p3));
	}

	function log(address p0, uint256 p1, bool p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,uint256,bool,uint256)", p0, p1, p2, p3));
	}

	function log(address p0, uint256 p1, bool p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,uint256,bool,string)", p0, p1, p2, p3));
	}

	function log(address p0, uint256 p1, bool p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,uint256,bool,bool)", p0, p1, p2, p3));
	}

	function log(address p0, uint256 p1, bool p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,uint256,bool,address)", p0, p1, p2, p3));
	}

	function log(address p0, uint256 p1, address p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,uint256,address,uint256)", p0, p1, p2, p3));
	}

	function log(address p0, uint256 p1, address p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,uint256,address,string)", p0, p1, p2, p3));
	}

	function log(address p0, uint256 p1, address p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,uint256,address,bool)", p0, p1, p2, p3));
	}

	function log(address p0, uint256 p1, address p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,uint256,address,address)", p0, p1, p2, p3));
	}

	function log(address p0, string memory p1, uint256 p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,string,uint256,uint256)", p0, p1, p2, p3));
	}

	function log(address p0, string memory p1, uint256 p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,string,uint256,string)", p0, p1, p2, p3));
	}

	function log(address p0, string memory p1, uint256 p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,string,uint256,bool)", p0, p1, p2, p3));
	}

	function log(address p0, string memory p1, uint256 p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,string,uint256,address)", p0, p1, p2, p3));
	}

	function log(address p0, string memory p1, string memory p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,string,string,uint256)", p0, p1, p2, p3));
	}

	function log(address p0, string memory p1, string memory p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,string,string,string)", p0, p1, p2, p3));
	}

	function log(address p0, string memory p1, string memory p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,string,string,bool)", p0, p1, p2, p3));
	}

	function log(address p0, string memory p1, string memory p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,string,string,address)", p0, p1, p2, p3));
	}

	function log(address p0, string memory p1, bool p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,string,bool,uint256)", p0, p1, p2, p3));
	}

	function log(address p0, string memory p1, bool p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,string,bool,string)", p0, p1, p2, p3));
	}

	function log(address p0, string memory p1, bool p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,string,bool,bool)", p0, p1, p2, p3));
	}

	function log(address p0, string memory p1, bool p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,string,bool,address)", p0, p1, p2, p3));
	}

	function log(address p0, string memory p1, address p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,string,address,uint256)", p0, p1, p2, p3));
	}

	function log(address p0, string memory p1, address p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,string,address,string)", p0, p1, p2, p3));
	}

	function log(address p0, string memory p1, address p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,string,address,bool)", p0, p1, p2, p3));
	}

	function log(address p0, string memory p1, address p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,string,address,address)", p0, p1, p2, p3));
	}

	function log(address p0, bool p1, uint256 p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,bool,uint256,uint256)", p0, p1, p2, p3));
	}

	function log(address p0, bool p1, uint256 p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,bool,uint256,string)", p0, p1, p2, p3));
	}

	function log(address p0, bool p1, uint256 p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,bool,uint256,bool)", p0, p1, p2, p3));
	}

	function log(address p0, bool p1, uint256 p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,bool,uint256,address)", p0, p1, p2, p3));
	}

	function log(address p0, bool p1, string memory p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,bool,string,uint256)", p0, p1, p2, p3));
	}

	function log(address p0, bool p1, string memory p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,bool,string,string)", p0, p1, p2, p3));
	}

	function log(address p0, bool p1, string memory p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,bool,string,bool)", p0, p1, p2, p3));
	}

	function log(address p0, bool p1, string memory p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,bool,string,address)", p0, p1, p2, p3));
	}

	function log(address p0, bool p1, bool p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,bool,bool,uint256)", p0, p1, p2, p3));
	}

	function log(address p0, bool p1, bool p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,bool,bool,string)", p0, p1, p2, p3));
	}

	function log(address p0, bool p1, bool p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,bool,bool,bool)", p0, p1, p2, p3));
	}

	function log(address p0, bool p1, bool p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,bool,bool,address)", p0, p1, p2, p3));
	}

	function log(address p0, bool p1, address p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,bool,address,uint256)", p0, p1, p2, p3));
	}

	function log(address p0, bool p1, address p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,bool,address,string)", p0, p1, p2, p3));
	}

	function log(address p0, bool p1, address p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,bool,address,bool)", p0, p1, p2, p3));
	}

	function log(address p0, bool p1, address p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,bool,address,address)", p0, p1, p2, p3));
	}

	function log(address p0, address p1, uint256 p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,address,uint256,uint256)", p0, p1, p2, p3));
	}

	function log(address p0, address p1, uint256 p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,address,uint256,string)", p0, p1, p2, p3));
	}

	function log(address p0, address p1, uint256 p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,address,uint256,bool)", p0, p1, p2, p3));
	}

	function log(address p0, address p1, uint256 p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,address,uint256,address)", p0, p1, p2, p3));
	}

	function log(address p0, address p1, string memory p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,address,string,uint256)", p0, p1, p2, p3));
	}

	function log(address p0, address p1, string memory p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,address,string,string)", p0, p1, p2, p3));
	}

	function log(address p0, address p1, string memory p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,address,string,bool)", p0, p1, p2, p3));
	}

	function log(address p0, address p1, string memory p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,address,string,address)", p0, p1, p2, p3));
	}

	function log(address p0, address p1, bool p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,address,bool,uint256)", p0, p1, p2, p3));
	}

	function log(address p0, address p1, bool p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,address,bool,string)", p0, p1, p2, p3));
	}

	function log(address p0, address p1, bool p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,address,bool,bool)", p0, p1, p2, p3));
	}

	function log(address p0, address p1, bool p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,address,bool,address)", p0, p1, p2, p3));
	}

	function log(address p0, address p1, address p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,address,address,uint256)", p0, p1, p2, p3));
	}

	function log(address p0, address p1, address p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,address,address,string)", p0, p1, p2, p3));
	}

	function log(address p0, address p1, address p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,address,address,bool)", p0, p1, p2, p3));
	}

	function log(address p0, address p1, address p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,address,address,address)", p0, p1, p2, p3));
	}

}

// File: @openzeppelin/contracts/utils/Context.sol


// OpenZeppelin Contracts v4.4.1 (utils/Context.sol)

pragma solidity ^0.8.0;

/**
 * @dev Provides information about the current execution context, including the
 * sender of the transaction and its data. While these are generally available
 * via msg.sender and msg.data, they should not be accessed in such a direct
 * manner, since when dealing with meta-transactions the account sending and
 * paying for execution may not be the actual sender (as far as an application
 * is concerned).
 *
 * This contract is only required for intermediate, library-like contracts.
 */
abstract contract Context {
    function _msgSender() internal view virtual returns (address) {
        return msg.sender;
    }

    function _msgData() internal view virtual returns (bytes calldata) {
        return msg.data;
    }
}

// File: @openzeppelin/contracts/security/Pausable.sol


// OpenZeppelin Contracts (last updated v4.7.0) (security/Pausable.sol)

pragma solidity ^0.8.0;


/**
 * @dev Contract module which allows children to implement an emergency stop
 * mechanism that can be triggered by an authorized account.
 *
 * This module is used through inheritance. It will make available the
 * modifiers `whenNotPaused` and `whenPaused`, which can be applied to
 * the functions of your contract. Note that they will not be pausable by
 * simply including this module, only once the modifiers are put in place.
 */
abstract contract Pausable is Context {
    /**
     * @dev Emitted when the pause is triggered by `account`.
     */
    event Paused(address account);

    /**
     * @dev Emitted when the pause is lifted by `account`.
     */
    event Unpaused(address account);

    bool private _paused;

    /**
     * @dev Initializes the contract in unpaused state.
     */
    constructor() {
        _paused = false;
    }

    /**
     * @dev Modifier to make a function callable only when the contract is not paused.
     *
     * Requirements:
     *
     * - The contract must not be paused.
     */
    modifier whenNotPaused() {
        _requireNotPaused();
        _;
    }

    /**
     * @dev Modifier to make a function callable only when the contract is paused.
     *
     * Requirements:
     *
     * - The contract must be paused.
     */
    modifier whenPaused() {
        _requirePaused();
        _;
    }

    /**
     * @dev Returns true if the contract is paused, and false otherwise.
     */
    function paused() public view virtual returns (bool) {
        return _paused;
    }

    /**
     * @dev Throws if the contract is paused.
     */
    function _requireNotPaused() internal view virtual {
        require(!paused(), "Pausable: paused");
    }

    /**
     * @dev Throws if the contract is not paused.
     */
    function _requirePaused() internal view virtual {
        require(paused(), "Pausable: not paused");
    }

    /**
     * @dev Triggers stopped state.
     *
     * Requirements:
     *
     * - The contract must not be paused.
     */
    function _pause() internal virtual whenNotPaused {
        _paused = true;
        emit Paused(_msgSender());
    }

    /**
     * @dev Returns to normal state.
     *
     * Requirements:
     *
     * - The contract must be paused.
     */
    function _unpause() internal virtual whenPaused {
        _paused = false;
        emit Unpaused(_msgSender());
    }
}

// File: @openzeppelin/contracts/access/Ownable.sol


// OpenZeppelin Contracts (last updated v4.7.0) (access/Ownable.sol)

pragma solidity ^0.8.0;


/**
 * @dev Contract module which provides a basic access control mechanism, where
 * there is an account (an owner) that can be granted exclusive access to
 * specific functions.
 *
 * By default, the owner account will be the one that deploys the contract. This
 * can later be changed with {transferOwnership}.
 *
 * This module is used through inheritance. It will make available the modifier
 * `onlyOwner`, which can be applied to your functions to restrict their use to
 * the owner.
 */
abstract contract Ownable is Context {
    address private _owner;

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

    /**
     * @dev Initializes the contract setting the deployer as the initial owner.
     */
    constructor() {
        _transferOwnership(_msgSender());
    }

    /**
     * @dev 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 {
        require(owner() == _msgSender(), "Ownable: caller is not the owner");
    }

    /**
     * @dev Leaves the contract without owner. It will not be possible to call
     * `onlyOwner` functions anymore. Can only be called by the current owner.
     *
     * NOTE: Renouncing ownership will leave the contract without an owner,
     * thereby removing any functionality that is only available to the owner.
     */
    function renounceOwnership() public virtual onlyOwner {
        _transferOwnership(address(0));
    }

    /**
     * @dev Transfers ownership of the contract to a new account (`newOwner`).
     * Can only be called by the current owner.
     */
    function transferOwnership(address newOwner) public virtual onlyOwner {
        require(newOwner != address(0), "Ownable: new owner is the zero address");
        _transferOwnership(newOwner);
    }

    /**
     * @dev Transfers ownership of the contract to a new account (`newOwner`).
     * Internal function without access restriction.
     */
    function _transferOwnership(address newOwner) internal virtual {
        address oldOwner = _owner;
        _owner = newOwner;
        emit OwnershipTransferred(oldOwner, newOwner);
    }
}

// File: erc721a/contracts/IERC721A.sol


// ERC721A Contracts v4.2.3
// Creator: Chiru Labs

pragma solidity ^0.8.4;

/**
 * @dev Interface of ERC721A.
 */
interface IERC721A {
    /**
     * The caller must own the token or be an approved operator.
     */
    error ApprovalCallerNotOwnerNorApproved();

    /**
     * The token does not exist.
     */
    error ApprovalQueryForNonexistentToken();

    /**
     * Cannot query the balance for the zero address.
     */
    error BalanceQueryForZeroAddress();

    /**
     * Cannot mint to the zero address.
     */
    error MintToZeroAddress();

    /**
     * The quantity of tokens minted must be more than zero.
     */
    error MintZeroQuantity();

    /**
     * The token does not exist.
     */
    error OwnerQueryForNonexistentToken();

    /**
     * The caller must own the token or be an approved operator.
     */
    error TransferCallerNotOwnerNorApproved();

    /**
     * The token must be owned by `from`.
     */
    error TransferFromIncorrectOwner();

    /**
     * Cannot safely transfer to a contract that does not implement the
     * ERC721Receiver interface.
     */
    error TransferToNonERC721ReceiverImplementer();

    /**
     * Cannot transfer to the zero address.
     */
    error TransferToZeroAddress();

    /**
     * The token does not exist.
     */
    error URIQueryForNonexistentToken();

    /**
     * The `quantity` minted with ERC2309 exceeds the safety limit.
     */
    error MintERC2309QuantityExceedsLimit();

    /**
     * The `extraData` cannot be set on an unintialized ownership slot.
     */
    error OwnershipNotInitializedForExtraData();

    // =============================================================
    //                            STRUCTS
    // =============================================================

    struct TokenOwnership {
        // The address of the owner.
        address addr;
        // Stores the start time of ownership with minimal overhead for tokenomics.
        uint64 startTimestamp;
        // Whether the token has been burned.
        bool burned;
        // Arbitrary data similar to `startTimestamp` that can be set via {_extraData}.
        uint24 extraData;
    }

    // =============================================================
    //                         TOKEN COUNTERS
    // =============================================================

    /**
     * @dev Returns the total number of tokens in existence.
     * Burned tokens will reduce the count.
     * To get the total number of tokens minted, please see {_totalMinted}.
     */
    function totalSupply() external view returns (uint256);

    // =============================================================
    //                            IERC165
    // =============================================================

    /**
     * @dev Returns true if this contract implements the interface defined by
     * `interfaceId`. See the corresponding
     * [EIP section](https://eips.ethereum.org/EIPS/eip-165#how-interfaces-are-identified)
     * to learn more about how these ids are created.
     *
     * This function call must use less than 30000 gas.
     */
    function supportsInterface(bytes4 interfaceId) external view returns (bool);

    // =============================================================
    //                            IERC721
    // =============================================================

    /**
     * @dev Emitted when `tokenId` token is transferred from `from` to `to`.
     */
    event Transfer(address indexed from, address indexed to, uint256 indexed tokenId);

    /**
     * @dev Emitted when `owner` enables `approved` to manage the `tokenId` token.
     */
    event Approval(address indexed owner, address indexed approved, uint256 indexed tokenId);

    /**
     * @dev Emitted when `owner` enables or disables
     * (`approved`) `operator` to manage all of its assets.
     */
    event ApprovalForAll(address indexed owner, address indexed operator, bool approved);

    /**
     * @dev Returns the number of tokens in `owner`'s account.
     */
    function balanceOf(address owner) external view returns (uint256 balance);

    /**
     * @dev Returns the owner of the `tokenId` token.
     *
     * Requirements:
     *
     * - `tokenId` must exist.
     */
    function ownerOf(uint256 tokenId) external view returns (address owner);

    /**
     * @dev Safely transfers `tokenId` token from `from` to `to`,
     * checking first that contract recipients are aware of the ERC721 protocol
     * to prevent tokens from being forever locked.
     *
     * Requirements:
     *
     * - `from` cannot be the zero address.
     * - `to` cannot be the zero address.
     * - `tokenId` token must exist and be owned by `from`.
     * - If the caller is not `from`, it must be have been allowed to move
     * this token by either {approve} or {setApprovalForAll}.
     * - If `to` refers to a smart contract, it must implement
     * {IERC721Receiver-onERC721Received}, which is called upon a safe transfer.
     *
     * Emits a {Transfer} event.
     */
    function safeTransferFrom(
        address from,
        address to,
        uint256 tokenId,
        bytes calldata data
    ) external payable;

    /**
     * @dev Equivalent to `safeTransferFrom(from, to, tokenId, '')`.
     */
    function safeTransferFrom(
        address from,
        address to,
        uint256 tokenId
    ) external payable;

    /**
     * @dev Transfers `tokenId` from `from` to `to`.
     *
     * WARNING: Usage of this method is discouraged, use {safeTransferFrom}
     * whenever possible.
     *
     * Requirements:
     *
     * - `from` cannot be the zero address.
     * - `to` cannot be the zero address.
     * - `tokenId` token must be owned by `from`.
     * - If the caller is not `from`, it must be approved to move this token
     * by either {approve} or {setApprovalForAll}.
     *
     * Emits a {Transfer} event.
     */
    function transferFrom(
        address from,
        address to,
        uint256 tokenId
    ) external payable;

    /**
     * @dev Gives permission to `to` to transfer `tokenId` token to another account.
     * The approval is cleared when the token is transferred.
     *
     * Only a single account can be approved at a time, so approving the
     * zero address clears previous approvals.
     *
     * Requirements:
     *
     * - The caller must own the token or be an approved operator.
     * - `tokenId` must exist.
     *
     * Emits an {Approval} event.
     */
    function approve(address to, uint256 tokenId) external payable;

    /**
     * @dev Approve or remove `operator` as an operator for the caller.
     * Operators can call {transferFrom} or {safeTransferFrom}
     * for any token owned by the caller.
     *
     * Requirements:
     *
     * - The `operator` cannot be the caller.
     *
     * Emits an {ApprovalForAll} event.
     */
    function setApprovalForAll(address operator, bool _approved) external;

    /**
     * @dev Returns the account approved for `tokenId` token.
     *
     * Requirements:
     *
     * - `tokenId` must exist.
     */
    function getApproved(uint256 tokenId) external view returns (address operator);

    /**
     * @dev Returns if the `operator` is allowed to manage all of the assets of `owner`.
     *
     * See {setApprovalForAll}.
     */
    function isApprovedForAll(address owner, address operator) external view returns (bool);

    // =============================================================
    //                        IERC721Metadata
    // =============================================================

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

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

    /**
     * @dev Returns the Uniform Resource Identifier (URI) for `tokenId` token.
     */
    function tokenURI(uint256 tokenId) external view returns (string memory);

    // =============================================================
    //                           IERC2309
    // =============================================================

    /**
     * @dev Emitted when tokens in `fromTokenId` to `toTokenId`
     * (inclusive) is transferred from `from` to `to`, as defined in the
     * [ERC2309](https://eips.ethereum.org/EIPS/eip-2309) standard.
     *
     * See {_mintERC2309} for more details.
     */
    event ConsecutiveTransfer(uint256 indexed fromTokenId, uint256 toTokenId, address indexed from, address indexed to);
}

// File: erc721a/contracts/ERC721A.sol


// ERC721A Contracts v4.2.3
// Creator: Chiru Labs

pragma solidity ^0.8.4;


/**
 * @dev Interface of ERC721 token receiver.
 */
interface ERC721A__IERC721Receiver {
    function onERC721Received(
        address operator,
        address from,
        uint256 tokenId,
        bytes calldata data
    ) external returns (bytes4);
}

/**
 * @title ERC721A
 *
 * @dev Implementation of the [ERC721](https://eips.ethereum.org/EIPS/eip-721)
 * Non-Fungible Token Standard, including the Metadata extension.
 * Optimized for lower gas during batch mints.
 *
 * Token IDs are minted in sequential order (e.g. 0, 1, 2, 3, ...)
 * starting from `_startTokenId()`.
 *
 * Assumptions:
 *
 * - An owner cannot have more than 2**64 - 1 (max value of uint64) of supply.
 * - The maximum token ID cannot exceed 2**256 - 1 (max value of uint256).
 */
contract ERC721A is IERC721A {
    // Bypass for a `--via-ir` bug (https://github.com/chiru-labs/ERC721A/pull/364).
    struct TokenApprovalRef {
        address value;
    }

    // =============================================================
    //                           CONSTANTS
    // =============================================================

    // Mask of an entry in packed address data.
    uint256 private constant _BITMASK_ADDRESS_DATA_ENTRY = (1 << 64) - 1;

    // The bit position of `numberMinted` in packed address data.
    uint256 private constant _BITPOS_NUMBER_MINTED = 64;

    // The bit position of `numberBurned` in packed address data.
    uint256 private constant _BITPOS_NUMBER_BURNED = 128;

    // The bit position of `aux` in packed address data.
    uint256 private constant _BITPOS_AUX = 192;

    // Mask of all 256 bits in packed address data except the 64 bits for `aux`.
    uint256 private constant _BITMASK_AUX_COMPLEMENT = (1 << 192) - 1;

    // The bit position of `startTimestamp` in packed ownership.
    uint256 private constant _BITPOS_START_TIMESTAMP = 160;

    // The bit mask of the `burned` bit in packed ownership.
    uint256 private constant _BITMASK_BURNED = 1 << 224;

    // The bit position of the `nextInitialized` bit in packed ownership.
    uint256 private constant _BITPOS_NEXT_INITIALIZED = 225;

    // The bit mask of the `nextInitialized` bit in packed ownership.
    uint256 private constant _BITMASK_NEXT_INITIALIZED = 1 << 225;

    // The bit position of `extraData` in packed ownership.
    uint256 private constant _BITPOS_EXTRA_DATA = 232;

    // Mask of all 256 bits in a packed ownership except the 24 bits for `extraData`.
    uint256 private constant _BITMASK_EXTRA_DATA_COMPLEMENT = (1 << 232) - 1;

    // The mask of the lower 160 bits for addresses.
    uint256 private constant _BITMASK_ADDRESS = (1 << 160) - 1;

    // The maximum `quantity` that can be minted with {_mintERC2309}.
    // This limit is to prevent overflows on the address data entries.
    // For a limit of 5000, a total of 3.689e15 calls to {_mintERC2309}
    // is required to cause an overflow, which is unrealistic.
    uint256 private constant _MAX_MINT_ERC2309_QUANTITY_LIMIT = 5000;

    // The `Transfer` event signature is given by:
    // `keccak256(bytes("Transfer(address,address,uint256)"))`.
    bytes32 private constant _TRANSFER_EVENT_SIGNATURE =
        0xddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef;

    // =============================================================
    //                            STORAGE
    // =============================================================

    // The next token ID to be minted.
    uint256 private _currentIndex;

    // The number of tokens burned.
    uint256 private _burnCounter;

    // Token name
    string private _name;

    // Token symbol
    string private _symbol;

    // Mapping from token ID to ownership details
    // An empty struct value does not necessarily mean the token is unowned.
    // See {_packedOwnershipOf} implementation for details.
    //
    // Bits Layout:
    // - [0..159]   `addr`
    // - [160..223] `startTimestamp`
    // - [224]      `burned`
    // - [225]      `nextInitialized`
    // - [232..255] `extraData`
    mapping(uint256 => uint256) private _packedOwnerships;

    // Mapping owner address to address data.
    //
    // Bits Layout:
    // - [0..63]    `balance`
    // - [64..127]  `numberMinted`
    // - [128..191] `numberBurned`
    // - [192..255] `aux`
    mapping(address => uint256) private _packedAddressData;

    // Mapping from token ID to approved address.
    mapping(uint256 => TokenApprovalRef) private _tokenApprovals;

    // Mapping from owner to operator approvals
    mapping(address => mapping(address => bool)) private _operatorApprovals;

    // =============================================================
    //                          CONSTRUCTOR
    // =============================================================

    constructor(string memory name_, string memory symbol_) {
        _name = name_;
        _symbol = symbol_;
        _currentIndex = _startTokenId();
    }

    // =============================================================
    //                   TOKEN COUNTING OPERATIONS
    // =============================================================

    /**
     * @dev Returns the starting token ID.
     * To change the starting token ID, please override this function.
     */
    function _startTokenId() internal view virtual returns (uint256) {
        return 0;
    }

    /**
     * @dev Returns the next token ID to be minted.
     */
    function _nextTokenId() internal view virtual returns (uint256) {
        return _currentIndex;
    }

    /**
     * @dev Returns the total number of tokens in existence.
     * Burned tokens will reduce the count.
     * To get the total number of tokens minted, please see {_totalMinted}.
     */
    function totalSupply() public view virtual override returns (uint256) {
        // Counter underflow is impossible as _burnCounter cannot be incremented
        // more than `_currentIndex - _startTokenId()` times.
        unchecked {
            return _currentIndex - _burnCounter - _startTokenId();
        }
    }

    /**
     * @dev Returns the total amount of tokens minted in the contract.
     */
    function _totalMinted() internal view virtual returns (uint256) {
        // Counter underflow is impossible as `_currentIndex` does not decrement,
        // and it is initialized to `_startTokenId()`.
        unchecked {
            return _currentIndex - _startTokenId();
        }
    }

    /**
     * @dev Returns the total number of tokens burned.
     */
    function _totalBurned() internal view virtual returns (uint256) {
        return _burnCounter;
    }

    // =============================================================
    //                    ADDRESS DATA OPERATIONS
    // =============================================================

    /**
     * @dev Returns the number of tokens in `owner`'s account.
     */
    function balanceOf(address owner) public view virtual override returns (uint256) {
        if (owner == address(0)) revert BalanceQueryForZeroAddress();
        return _packedAddressData[owner] & _BITMASK_ADDRESS_DATA_ENTRY;
    }

    /**
     * Returns the number of tokens minted by `owner`.
     */
    function _numberMinted(address owner) internal view returns (uint256) {
        return (_packedAddressData[owner] >> _BITPOS_NUMBER_MINTED) & _BITMASK_ADDRESS_DATA_ENTRY;
    }

    /**
     * Returns the number of tokens burned by or on behalf of `owner`.
     */
    function _numberBurned(address owner) internal view returns (uint256) {
        return (_packedAddressData[owner] >> _BITPOS_NUMBER_BURNED) & _BITMASK_ADDRESS_DATA_ENTRY;
    }

    /**
     * Returns the auxiliary data for `owner`. (e.g. number of whitelist mint slots used).
     */
    function _getAux(address owner) internal view returns (uint64) {
        return uint64(_packedAddressData[owner] >> _BITPOS_AUX);
    }

    /**
     * Sets the auxiliary data for `owner`. (e.g. number of whitelist mint slots used).
     * If there are multiple variables, please pack them into a uint64.
     */
    function _setAux(address owner, uint64 aux) internal virtual {
        uint256 packed = _packedAddressData[owner];
        uint256 auxCasted;
        // Cast `aux` with assembly to avoid redundant masking.
        assembly {
            auxCasted := aux
        }
        packed = (packed & _BITMASK_AUX_COMPLEMENT) | (auxCasted << _BITPOS_AUX);
        _packedAddressData[owner] = packed;
    }

    // =============================================================
    //                            IERC165
    // =============================================================

    /**
     * @dev Returns true if this contract implements the interface defined by
     * `interfaceId`. See the corresponding
     * [EIP section](https://eips.ethereum.org/EIPS/eip-165#how-interfaces-are-identified)
     * to learn more about how these ids are created.
     *
     * This function call must use less than 30000 gas.
     */
    function supportsInterface(bytes4 interfaceId) public view virtual override returns (bool) {
        // The interface IDs are constants representing the first 4 bytes
        // of the XOR of all function selectors in the interface.
        // See: [ERC165](https://eips.ethereum.org/EIPS/eip-165)
        // (e.g. `bytes4(i.functionA.selector ^ i.functionB.selector ^ ...)`)
        return
            interfaceId == 0x01ffc9a7 || // ERC165 interface ID for ERC165.
            interfaceId == 0x80ac58cd || // ERC165 interface ID for ERC721.
            interfaceId == 0x5b5e139f; // ERC165 interface ID for ERC721Metadata.
    }

    // =============================================================
    //                        IERC721Metadata
    // =============================================================

    /**
     * @dev Returns the token collection name.
     */
    function name() public view virtual override returns (string memory) {
        return _name;
    }

    /**
     * @dev Returns the token collection symbol.
     */
    function symbol() public view virtual override returns (string memory) {
        return _symbol;
    }

    /**
     * @dev Returns the Uniform Resource Identifier (URI) for `tokenId` token.
     */
    function tokenURI(uint256 tokenId) public view virtual override returns (string memory) {
        if (!_exists(tokenId)) revert URIQueryForNonexistentToken();

        string memory baseURI = _baseURI();
        return bytes(baseURI).length != 0 ? string(abi.encodePacked(baseURI, _toString(tokenId))) : '';
    }

    /**
     * @dev Base URI for computing {tokenURI}. If set, the resulting URI for each
     * token will be the concatenation of the `baseURI` and the `tokenId`. Empty
     * by default, it can be overridden in child contracts.
     */
    function _baseURI() internal view virtual returns (string memory) {
        return '';
    }

    // =============================================================
    //                     OWNERSHIPS OPERATIONS
    // =============================================================

    /**
     * @dev Returns the owner of the `tokenId` token.
     *
     * Requirements:
     *
     * - `tokenId` must exist.
     */
    function ownerOf(uint256 tokenId) public view virtual override returns (address) {
        return address(uint160(_packedOwnershipOf(tokenId)));
    }

    /**
     * @dev Gas spent here starts off proportional to the maximum mint batch size.
     * It gradually moves to O(1) as tokens get transferred around over time.
     */
    function _ownershipOf(uint256 tokenId) internal view virtual returns (TokenOwnership memory) {
        return _unpackedOwnership(_packedOwnershipOf(tokenId));
    }

    /**
     * @dev Returns the unpacked `TokenOwnership` struct at `index`.
     */
    function _ownershipAt(uint256 index) internal view virtual returns (TokenOwnership memory) {
        return _unpackedOwnership(_packedOwnerships[index]);
    }

    /**
     * @dev Initializes the ownership slot minted at `index` for efficiency purposes.
     */
    function _initializeOwnershipAt(uint256 index) internal virtual {
        if (_packedOwnerships[index] == 0) {
            _packedOwnerships[index] = _packedOwnershipOf(index);
        }
    }

    /**
     * Returns the packed ownership data of `tokenId`.
     */
    function _packedOwnershipOf(uint256 tokenId) private view returns (uint256) {
        uint256 curr = tokenId;

        unchecked {
            if (_startTokenId() <= curr)
                if (curr < _currentIndex) {
                    uint256 packed = _packedOwnerships[curr];
                    // If not burned.
                    if (packed & _BITMASK_BURNED == 0) {
                        // Invariant:
                        // There will always be an initialized ownership slot
                        // (i.e. `ownership.addr != address(0) && ownership.burned == false`)
                        // before an unintialized ownership slot
                        // (i.e. `ownership.addr == address(0) && ownership.burned == false`)
                        // Hence, `curr` will not underflow.
                        //
                        // We can directly compare the packed value.
                        // If the address is zero, packed will be zero.
                        while (packed == 0) {
                            packed = _packedOwnerships[--curr];
                        }
                        return packed;
                    }
                }
        }
        revert OwnerQueryForNonexistentToken();
    }

    /**
     * @dev Returns the unpacked `TokenOwnership` struct from `packed`.
     */
    function _unpackedOwnership(uint256 packed) private pure returns (TokenOwnership memory ownership) {
        ownership.addr = address(uint160(packed));
        ownership.startTimestamp = uint64(packed >> _BITPOS_START_TIMESTAMP);
        ownership.burned = packed & _BITMASK_BURNED != 0;
        ownership.extraData = uint24(packed >> _BITPOS_EXTRA_DATA);
    }

    /**
     * @dev Packs ownership data into a single uint256.
     */
    function _packOwnershipData(address owner, uint256 flags) private view returns (uint256 result) {
        assembly {
            // Mask `owner` to the lower 160 bits, in case the upper bits somehow aren't clean.
            owner := and(owner, _BITMASK_ADDRESS)
            // `owner | (block.timestamp << _BITPOS_START_TIMESTAMP) | flags`.
            result := or(owner, or(shl(_BITPOS_START_TIMESTAMP, timestamp()), flags))
        }
    }

    /**
     * @dev Returns the `nextInitialized` flag set if `quantity` equals 1.
     */
    function _nextInitializedFlag(uint256 quantity) private pure returns (uint256 result) {
        // For branchless setting of the `nextInitialized` flag.
        assembly {
            // `(quantity == 1) << _BITPOS_NEXT_INITIALIZED`.
            result := shl(_BITPOS_NEXT_INITIALIZED, eq(quantity, 1))
        }
    }

    // =============================================================
    //                      APPROVAL OPERATIONS
    // =============================================================

    /**
     * @dev Gives permission to `to` to transfer `tokenId` token to another account.
     * The approval is cleared when the token is transferred.
     *
     * Only a single account can be approved at a time, so approving the
     * zero address clears previous approvals.
     *
     * Requirements:
     *
     * - The caller must own the token or be an approved operator.
     * - `tokenId` must exist.
     *
     * Emits an {Approval} event.
     */
    function approve(address to, uint256 tokenId) public payable virtual override {
        address owner = ownerOf(tokenId);

        if (_msgSenderERC721A() != owner)
            if (!isApprovedForAll(owner, _msgSenderERC721A())) {
                revert ApprovalCallerNotOwnerNorApproved();
            }

        _tokenApprovals[tokenId].value = to;
        emit Approval(owner, to, tokenId);
    }

    /**
     * @dev Returns the account approved for `tokenId` token.
     *
     * Requirements:
     *
     * - `tokenId` must exist.
     */
    function getApproved(uint256 tokenId) public view virtual override returns (address) {
        if (!_exists(tokenId)) revert ApprovalQueryForNonexistentToken();

        return _tokenApprovals[tokenId].value;
    }

    /**
     * @dev Approve or remove `operator` as an operator for the caller.
     * Operators can call {transferFrom} or {safeTransferFrom}
     * for any token owned by the caller.
     *
     * Requirements:
     *
     * - The `operator` cannot be the caller.
     *
     * Emits an {ApprovalForAll} event.
     */
    function setApprovalForAll(address operator, bool approved) public virtual override {
        _operatorApprovals[_msgSenderERC721A()][operator] = approved;
        emit ApprovalForAll(_msgSenderERC721A(), operator, approved);
    }

    /**
     * @dev Returns if the `operator` is allowed to manage all of the assets of `owner`.
     *
     * See {setApprovalForAll}.
     */
    function isApprovedForAll(address owner, address operator) public view virtual override returns (bool) {
        return _operatorApprovals[owner][operator];
    }

    /**
     * @dev Returns whether `tokenId` exists.
     *
     * Tokens can be managed by their owner or approved accounts via {approve} or {setApprovalForAll}.
     *
     * Tokens start existing when they are minted. See {_mint}.
     */
    function _exists(uint256 tokenId) internal view virtual returns (bool) {
        return
            _startTokenId() <= tokenId &&
            tokenId < _currentIndex && // If within bounds,
            _packedOwnerships[tokenId] & _BITMASK_BURNED == 0; // and not burned.
    }

    /**
     * @dev Returns whether `msgSender` is equal to `approvedAddress` or `owner`.
     */
    function _isSenderApprovedOrOwner(
        address approvedAddress,
        address owner,
        address msgSender
    ) private pure returns (bool result) {
        assembly {
            // Mask `owner` to the lower 160 bits, in case the upper bits somehow aren't clean.
            owner := and(owner, _BITMASK_ADDRESS)
            // Mask `msgSender` to the lower 160 bits, in case the upper bits somehow aren't clean.
            msgSender := and(msgSender, _BITMASK_ADDRESS)
            // `msgSender == owner || msgSender == approvedAddress`.
            result := or(eq(msgSender, owner), eq(msgSender, approvedAddress))
        }
    }

    /**
     * @dev Returns the storage slot and value for the approved address of `tokenId`.
     */
    function _getApprovedSlotAndAddress(uint256 tokenId)
        private
        view
        returns (uint256 approvedAddressSlot, address approvedAddress)
    {
        TokenApprovalRef storage tokenApproval = _tokenApprovals[tokenId];
        // The following is equivalent to `approvedAddress = _tokenApprovals[tokenId].value`.
        assembly {
            approvedAddressSlot := tokenApproval.slot
            approvedAddress := sload(approvedAddressSlot)
        }
    }

    // =============================================================
    //                      TRANSFER OPERATIONS
    // =============================================================

    /**
     * @dev Transfers `tokenId` from `from` to `to`.
     *
     * Requirements:
     *
     * - `from` cannot be the zero address.
     * - `to` cannot be the zero address.
     * - `tokenId` token must be owned by `from`.
     * - If the caller is not `from`, it must be approved to move this token
     * by either {approve} or {setApprovalForAll}.
     *
     * Emits a {Transfer} event.
     */
    function transferFrom(
        address from,
        address to,
        uint256 tokenId
    ) public payable virtual override {
        uint256 prevOwnershipPacked = _packedOwnershipOf(tokenId);

        if (address(uint160(prevOwnershipPacked)) != from) revert TransferFromIncorrectOwner();

        (uint256 approvedAddressSlot, address approvedAddress) = _getApprovedSlotAndAddress(tokenId);

        // The nested ifs save around 20+ gas over a compound boolean condition.
        if (!_isSenderApprovedOrOwner(approvedAddress, from, _msgSenderERC721A()))
            if (!isApprovedForAll(from, _msgSenderERC721A())) revert TransferCallerNotOwnerNorApproved();

        if (to == address(0)) revert TransferToZeroAddress();

        _beforeTokenTransfers(from, to, tokenId, 1);

        // Clear approvals from the previous owner.
        assembly {
            if approvedAddress {
                // This is equivalent to `delete _tokenApprovals[tokenId]`.
                sstore(approvedAddressSlot, 0)
            }
        }

        // Underflow of the sender's balance is impossible because we check for
        // ownership above and the recipient's balance can't realistically overflow.
        // Counter overflow is incredibly unrealistic as `tokenId` would have to be 2**256.
        unchecked {
            // We can directly increment and decrement the balances.
            --_packedAddressData[from]; // Updates: `balance -= 1`.
            ++_packedAddressData[to]; // Updates: `balance += 1`.

            // Updates:
            // - `address` to the next owner.
            // - `startTimestamp` to the timestamp of transfering.
            // - `burned` to `false`.
            // - `nextInitialized` to `true`.
            _packedOwnerships[tokenId] = _packOwnershipData(
                to,
                _BITMASK_NEXT_INITIALIZED | _nextExtraData(from, to, prevOwnershipPacked)
            );

            // If the next slot may not have been initialized (i.e. `nextInitialized == false`) .
            if (prevOwnershipPacked & _BITMASK_NEXT_INITIALIZED == 0) {
                uint256 nextTokenId = tokenId + 1;
                // If the next slot's address is zero and not burned (i.e. packed value is zero).
                if (_packedOwnerships[nextTokenId] == 0) {
                    // If the next slot is within bounds.
                    if (nextTokenId != _currentIndex) {
                        // Initialize the next slot to maintain correctness for `ownerOf(tokenId + 1)`.
                        _packedOwnerships[nextTokenId] = prevOwnershipPacked;
                    }
                }
            }
        }

        emit Transfer(from, to, tokenId);
        _afterTokenTransfers(from, to, tokenId, 1);
    }

    /**
     * @dev Equivalent to `safeTransferFrom(from, to, tokenId, '')`.
     */
    function safeTransferFrom(
        address from,
        address to,
        uint256 tokenId
    ) public payable virtual override {
        safeTransferFrom(from, to, tokenId, '');
    }

    /**
     * @dev Safely transfers `tokenId` token from `from` to `to`.
     *
     * Requirements:
     *
     * - `from` cannot be the zero address.
     * - `to` cannot be the zero address.
     * - `tokenId` token must exist and be owned by `from`.
     * - If the caller is not `from`, it must be approved to move this token
     * by either {approve} or {setApprovalForAll}.
     * - If `to` refers to a smart contract, it must implement
     * {IERC721Receiver-onERC721Received}, which is called upon a safe transfer.
     *
     * Emits a {Transfer} event.
     */
    function safeTransferFrom(
        address from,
        address to,
        uint256 tokenId,
        bytes memory _data
    ) public payable virtual override {
        transferFrom(from, to, tokenId);
        if (to.code.length != 0)
            if (!_checkContractOnERC721Received(from, to, tokenId, _data)) {
                revert TransferToNonERC721ReceiverImplementer();
            }
    }

    /**
     * @dev Hook that is called before a set of serially-ordered token IDs
     * are about to be transferred. This includes minting.
     * And also called before burning one token.
     *
     * `startTokenId` - the first token ID to be transferred.
     * `quantity` - the amount to be transferred.
     *
     * Calling conditions:
     *
     * - When `from` and `to` are both non-zero, `from`'s `tokenId` will be
     * transferred to `to`.
     * - When `from` is zero, `tokenId` will be minted for `to`.
     * - When `to` is zero, `tokenId` will be burned by `from`.
     * - `from` and `to` are never both zero.
     */
    function _beforeTokenTransfers(
        address from,
        address to,
        uint256 startTokenId,
        uint256 quantity
    ) internal virtual {}

    /**
     * @dev Hook that is called after a set of serially-ordered token IDs
     * have been transferred. This includes minting.
     * And also called after one token has been burned.
     *
     * `startTokenId` - the first token ID to be transferred.
     * `quantity` - the amount to be transferred.
     *
     * Calling conditions:
     *
     * - When `from` and `to` are both non-zero, `from`'s `tokenId` has been
     * transferred to `to`.
     * - When `from` is zero, `tokenId` has been minted for `to`.
     * - When `to` is zero, `tokenId` has been burned by `from`.
     * - `from` and `to` are never both zero.
     */
    function _afterTokenTransfers(
        address from,
        address to,
        uint256 startTokenId,
        uint256 quantity
    ) internal virtual {}

    /**
     * @dev Private function to invoke {IERC721Receiver-onERC721Received} on a target contract.
     *
     * `from` - Previous owner of the given token ID.
     * `to` - Target address that will receive the token.
     * `tokenId` - Token ID to be transferred.
     * `_data` - Optional data to send along with the call.
     *
     * Returns whether the call correctly returned the expected magic value.
     */
    function _checkContractOnERC721Received(
        address from,
        address to,
        uint256 tokenId,
        bytes memory _data
    ) private returns (bool) {
        try ERC721A__IERC721Receiver(to).onERC721Received(_msgSenderERC721A(), from, tokenId, _data) returns (
            bytes4 retval
        ) {
            return retval == ERC721A__IERC721Receiver(to).onERC721Received.selector;
        } catch (bytes memory reason) {
            if (reason.length == 0) {
                revert TransferToNonERC721ReceiverImplementer();
            } else {
                assembly {
                    revert(add(32, reason), mload(reason))
                }
            }
        }
    }

    // =============================================================
    //                        MINT OPERATIONS
    // =============================================================

    /**
     * @dev Mints `quantity` tokens and transfers them to `to`.
     *
     * Requirements:
     *
     * - `to` cannot be the zero address.
     * - `quantity` must be greater than 0.
     *
     * Emits a {Transfer} event for each mint.
     */
    function _mint(address to, uint256 quantity) internal virtual {
        uint256 startTokenId = _currentIndex;
        if (quantity == 0) revert MintZeroQuantity();

        _beforeTokenTransfers(address(0), to, startTokenId, quantity);

        // Overflows are incredibly unrealistic.
        // `balance` and `numberMinted` have a maximum limit of 2**64.
        // `tokenId` has a maximum limit of 2**256.
        unchecked {
            // Updates:
            // - `balance += quantity`.
            // - `numberMinted += quantity`.
            //
            // We can directly add to the `balance` and `numberMinted`.
            _packedAddressData[to] += quantity * ((1 << _BITPOS_NUMBER_MINTED) | 1);

            // Updates:
            // - `address` to the owner.
            // - `startTimestamp` to the timestamp of minting.
            // - `burned` to `false`.
            // - `nextInitialized` to `quantity == 1`.
            _packedOwnerships[startTokenId] = _packOwnershipData(
                to,
                _nextInitializedFlag(quantity) | _nextExtraData(address(0), to, 0)
            );

            uint256 toMasked;
            uint256 end = startTokenId + quantity;

            // Use assembly to loop and emit the `Transfer` event for gas savings.
            // The duplicated `log4` removes an extra check and reduces stack juggling.
            // The assembly, together with the surrounding Solidity code, have been
            // delicately arranged to nudge the compiler into producing optimized opcodes.
            assembly {
                // Mask `to` to the lower 160 bits, in case the upper bits somehow aren't clean.
                toMasked := and(to, _BITMASK_ADDRESS)
                // Emit the `Transfer` event.
                log4(
                    0, // Start of data (0, since no data).
                    0, // End of data (0, since no data).
                    _TRANSFER_EVENT_SIGNATURE, // Signature.
                    0, // `address(0)`.
                    toMasked, // `to`.
                    startTokenId // `tokenId`.
                )

                // The `iszero(eq(,))` check ensures that large values of `quantity`
                // that overflows uint256 will make the loop run out of gas.
                // The compiler will optimize the `iszero` away for performance.
                for {
                    let tokenId := add(startTokenId, 1)
                } iszero(eq(tokenId, end)) {
                    tokenId := add(tokenId, 1)
                } {
                    // Emit the `Transfer` event. Similar to above.
                    log4(0, 0, _TRANSFER_EVENT_SIGNATURE, 0, toMasked, tokenId)
                }
            }
            if (toMasked == 0) revert MintToZeroAddress();

            _currentIndex = end;
        }
        _afterTokenTransfers(address(0), to, startTokenId, quantity);
    }

    /**
     * @dev Mints `quantity` tokens and transfers them to `to`.
     *
     * This function is intended for efficient minting only during contract creation.
     *
     * It emits only one {ConsecutiveTransfer} as defined in
     * [ERC2309](https://eips.ethereum.org/EIPS/eip-2309),
     * instead of a sequence of {Transfer} event(s).
     *
     * Calling this function outside of contract creation WILL make your contract
     * non-compliant with the ERC721 standard.
     * For full ERC721 compliance, substituting ERC721 {Transfer} event(s) with the ERC2309
     * {ConsecutiveTransfer} event is only permissible during contract creation.
     *
     * Requirements:
     *
     * - `to` cannot be the zero address.
     * - `quantity` must be greater than 0.
     *
     * Emits a {ConsecutiveTransfer} event.
     */
    function _mintERC2309(address to, uint256 quantity) internal virtual {
        uint256 startTokenId = _currentIndex;
        if (to == address(0)) revert MintToZeroAddress();
        if (quantity == 0) revert MintZeroQuantity();
        if (quantity > _MAX_MINT_ERC2309_QUANTITY_LIMIT) revert MintERC2309QuantityExceedsLimit();

        _beforeTokenTransfers(address(0), to, startTokenId, quantity);

        // Overflows are unrealistic due to the above check for `quantity` to be below the limit.
        unchecked {
            // Updates:
            // - `balance += quantity`.
            // - `numberMinted += quantity`.
            //
            // We can directly add to the `balance` and `numberMinted`.
            _packedAddressData[to] += quantity * ((1 << _BITPOS_NUMBER_MINTED) | 1);

            // Updates:
            // - `address` to the owner.
            // - `startTimestamp` to the timestamp of minting.
            // - `burned` to `false`.
            // - `nextInitialized` to `quantity == 1`.
            _packedOwnerships[startTokenId] = _packOwnershipData(
                to,
                _nextInitializedFlag(quantity) | _nextExtraData(address(0), to, 0)
            );

            emit ConsecutiveTransfer(startTokenId, startTokenId + quantity - 1, address(0), to);

            _currentIndex = startTokenId + quantity;
        }
        _afterTokenTransfers(address(0), to, startTokenId, quantity);
    }

    /**
     * @dev Safely mints `quantity` tokens and transfers them to `to`.
     *
     * Requirements:
     *
     * - If `to` refers to a smart contract, it must implement
     * {IERC721Receiver-onERC721Received}, which is called for each safe transfer.
     * - `quantity` must be greater than 0.
     *
     * See {_mint}.
     *
     * Emits a {Transfer} event for each mint.
     */
    function _safeMint(
        address to,
        uint256 quantity,
        bytes memory _data
    ) internal virtual {
        _mint(to, quantity);

        unchecked {
            if (to.code.length != 0) {
                uint256 end = _currentIndex;
                uint256 index = end - quantity;
                do {
                    if (!_checkContractOnERC721Received(address(0), to, index++, _data)) {
                        revert TransferToNonERC721ReceiverImplementer();
                    }
                } while (index < end);
                // Reentrancy protection.
                if (_currentIndex != end) revert();
            }
        }
    }

    /**
     * @dev Equivalent to `_safeMint(to, quantity, '')`.
     */
    function _safeMint(address to, uint256 quantity) internal virtual {
        _safeMint(to, quantity, '');
    }

    // =============================================================
    //                        BURN OPERATIONS
    // =============================================================

    /**
     * @dev Equivalent to `_burn(tokenId, false)`.
     */
    function _burn(uint256 tokenId) internal virtual {
        _burn(tokenId, false);
    }

    /**
     * @dev Destroys `tokenId`.
     * The approval is cleared when the token is burned.
     *
     * Requirements:
     *
     * - `tokenId` must exist.
     *
     * Emits a {Transfer} event.
     */
    function _burn(uint256 tokenId, bool approvalCheck) internal virtual {
        uint256 prevOwnershipPacked = _packedOwnershipOf(tokenId);

        address from = address(uint160(prevOwnershipPacked));

        (uint256 approvedAddressSlot, address approvedAddress) = _getApprovedSlotAndAddress(tokenId);

        if (approvalCheck) {
            // The nested ifs save around 20+ gas over a compound boolean condition.
            if (!_isSenderApprovedOrOwner(approvedAddress, from, _msgSenderERC721A()))
                if (!isApprovedForAll(from, _msgSenderERC721A())) revert TransferCallerNotOwnerNorApproved();
        }

        _beforeTokenTransfers(from, address(0), tokenId, 1);

        // Clear approvals from the previous owner.
        assembly {
            if approvedAddress {
                // This is equivalent to `delete _tokenApprovals[tokenId]`.
                sstore(approvedAddressSlot, 0)
            }
        }

        // Underflow of the sender's balance is impossible because we check for
        // ownership above and the recipient's balance can't realistically overflow.
        // Counter overflow is incredibly unrealistic as `tokenId` would have to be 2**256.
        unchecked {
            // Updates:
            // - `balance -= 1`.
            // - `numberBurned += 1`.
            //
            // We can directly decrement the balance, and increment the number burned.
            // This is equivalent to `packed -= 1; packed += 1 << _BITPOS_NUMBER_BURNED;`.
            _packedAddressData[from] += (1 << _BITPOS_NUMBER_BURNED) - 1;

            // Updates:
            // - `address` to the last owner.
            // - `startTimestamp` to the timestamp of burning.
            // - `burned` to `true`.
            // - `nextInitialized` to `true`.
            _packedOwnerships[tokenId] = _packOwnershipData(
                from,
                (_BITMASK_BURNED | _BITMASK_NEXT_INITIALIZED) | _nextExtraData(from, address(0), prevOwnershipPacked)
            );

            // If the next slot may not have been initialized (i.e. `nextInitialized == false`) .
            if (prevOwnershipPacked & _BITMASK_NEXT_INITIALIZED == 0) {
                uint256 nextTokenId = tokenId + 1;
                // If the next slot's address is zero and not burned (i.e. packed value is zero).
                if (_packedOwnerships[nextTokenId] == 0) {
                    // If the next slot is within bounds.
                    if (nextTokenId != _currentIndex) {
                        // Initialize the next slot to maintain correctness for `ownerOf(tokenId + 1)`.
                        _packedOwnerships[nextTokenId] = prevOwnershipPacked;
                    }
                }
            }
        }

        emit Transfer(from, address(0), tokenId);
        _afterTokenTransfers(from, address(0), tokenId, 1);

        // Overflow not possible, as _burnCounter cannot be exceed _currentIndex times.
        unchecked {
            _burnCounter++;
        }
    }

    // =============================================================
    //                     EXTRA DATA OPERATIONS
    // =============================================================

    /**
     * @dev Directly sets the extra data for the ownership data `index`.
     */
    function _setExtraDataAt(uint256 index, uint24 extraData) internal virtual {
        uint256 packed = _packedOwnerships[index];
        if (packed == 0) revert OwnershipNotInitializedForExtraData();
        uint256 extraDataCasted;
        // Cast `extraData` with assembly to avoid redundant masking.
        assembly {
            extraDataCasted := extraData
        }
        packed = (packed & _BITMASK_EXTRA_DATA_COMPLEMENT) | (extraDataCasted << _BITPOS_EXTRA_DATA);
        _packedOwnerships[index] = packed;
    }

    /**
     * @dev Called during each token transfer to set the 24bit `extraData` field.
     * Intended to be overridden by the cosumer contract.
     *
     * `previousExtraData` - the value of `extraData` before transfer.
     *
     * Calling conditions:
     *
     * - When `from` and `to` are both non-zero, `from`'s `tokenId` will be
     * transferred to `to`.
     * - When `from` is zero, `tokenId` will be minted for `to`.
     * - When `to` is zero, `tokenId` will be burned by `from`.
     * - `from` and `to` are never both zero.
     */
    function _extraData(
        address from,
        address to,
        uint24 previousExtraData
    ) internal view virtual returns (uint24) {}

    /**
     * @dev Returns the next extra data for the packed ownership data.
     * The returned result is shifted into position.
     */
    function _nextExtraData(
        address from,
        address to,
        uint256 prevOwnershipPacked
    ) private view returns (uint256) {
        uint24 extraData = uint24(prevOwnershipPacked >> _BITPOS_EXTRA_DATA);
        return uint256(_extraData(from, to, extraData)) << _BITPOS_EXTRA_DATA;
    }

    // =============================================================
    //                       OTHER OPERATIONS
    // =============================================================

    /**
     * @dev Returns the message sender (defaults to `msg.sender`).
     *
     * If you are writing GSN compatible contracts, you need to override this function.
     */
    function _msgSenderERC721A() internal view virtual returns (address) {
        return msg.sender;
    }

    /**
     * @dev Converts a uint256 to its ASCII string decimal representation.
     */
    function _toString(uint256 value) internal pure virtual returns (string memory str) {
        assembly {
            // The maximum value of a uint256 contains 78 digits (1 byte per digit), but
            // we allocate 0xa0 bytes to keep the free memory pointer 32-byte word aligned.
            // We will need 1 word for the trailing zeros padding, 1 word for the length,
            // and 3 words for a maximum of 78 digits. Total: 5 * 0x20 = 0xa0.
            let m := add(mload(0x40), 0xa0)
            // Update the free memory pointer to allocate.
            mstore(0x40, m)
            // Assign the `str` to the end.
            str := sub(m, 0x20)
            // Zeroize the slot after the string.
            mstore(str, 0)

            // Cache the end of the memory to calculate the length later.
            let end := str

            // We write the string from rightmost digit to leftmost digit.
            // The following is essentially a do-while loop that also handles the zero case.
            // prettier-ignore
            for { let temp := value } 1 {} {
                str := sub(str, 1)
                // Write the character to the pointer.
                // The ASCII index of the '0' character is 48.
                mstore8(str, add(48, mod(temp, 10)))
                // Keep dividing `temp` until zero.
                temp := div(temp, 10)
                // prettier-ignore
                if iszero(temp) { break }
            }

            let length := sub(end, str)
            // Move the pointer 32 bytes leftwards to make room for the length.
            str := sub(str, 0x20)
            // Store the length.
            mstore(str, length)
        }
    }
}

// File: @chainlink/contracts/src/v0.8/interfaces/AggregatorV3Interface.sol


pragma solidity ^0.8.0;

interface AggregatorV3Interface {
  function decimals() external view returns (uint8);

  function description() external view returns (string memory);

  function version() external view returns (uint256);

  function getRoundData(uint80 _roundId)
    external
    view
    returns (
      uint80 roundId,
      int256 answer,
      uint256 startedAt,
      uint256 updatedAt,
      uint80 answeredInRound
    );

  function latestRoundData()
    external
    view
    returns (
      uint80 roundId,
      int256 answer,
      uint256 startedAt,
      uint256 updatedAt,
      uint80 answeredInRound
    );
}

// File: @openzeppelin/contracts/utils/math/Math.sol


// OpenZeppelin Contracts (last updated v4.8.0) (utils/math/Math.sol)

pragma solidity ^0.8.0;

/**
 * @dev Standard math utilities missing in the Solidity language.
 */
library Math {
    enum Rounding {
        Down, // Toward negative infinity
        Up, // Toward infinity
        Zero // Toward zero
    }

    /**
     * @dev Returns the largest of two numbers.
     */
    function max(uint256 a, uint256 b) internal pure returns (uint256) {
        return a > b ? a : b;
    }

    /**
     * @dev Returns the smallest of two numbers.
     */
    function min(uint256 a, uint256 b) internal pure returns (uint256) {
        return a < b ? a : b;
    }

    /**
     * @dev Returns the average of two numbers. The result is rounded towards
     * zero.
     */
    function average(uint256 a, uint256 b) internal pure returns (uint256) {
        // (a + b) / 2 can overflow.
        return (a & b) + (a ^ b) / 2;
    }

    /**
     * @dev Returns the ceiling of the division of two numbers.
     *
     * This differs from standard division with `/` in that it rounds up instead
     * of rounding down.
     */
    function ceilDiv(uint256 a, uint256 b) internal pure returns (uint256) {
        // (a + b - 1) / b can overflow on addition, so we distribute.
        return a == 0 ? 0 : (a - 1) / b + 1;
    }

    /**
     * @notice Calculates floor(x * y / denominator) with full precision. Throws if result overflows a uint256 or denominator == 0
     * @dev Original credit to Remco Bloemen under MIT license (https://xn--2-umb.com/21/muldiv)
     * with further edits by Uniswap Labs also under MIT license.
     */
    function mulDiv(
        uint256 x,
        uint256 y,
        uint256 denominator
    ) internal pure returns (uint256 result) {
        unchecked {
            // 512-bit multiply [prod1 prod0] = x * y. Compute the product mod 2^256 and mod 2^256 - 1, then use
            // use the Chinese Remainder Theorem to reconstruct the 512 bit result. The result is stored in two 256
            // variables such that product = prod1 * 2^256 + prod0.
            uint256 prod0; // Least significant 256 bits of the product
            uint256 prod1; // Most significant 256 bits of the product
            assembly {
                let mm := mulmod(x, y, not(0))
                prod0 := mul(x, y)
                prod1 := sub(sub(mm, prod0), lt(mm, prod0))
            }

            // Handle non-overflow cases, 256 by 256 division.
            if (prod1 == 0) {
                return prod0 / denominator;
            }

            // Make sure the result is less than 2^256. Also prevents denominator == 0.
            require(denominator > prod1);

            ///////////////////////////////////////////////
            // 512 by 256 division.
            ///////////////////////////////////////////////

            // Make division exact by subtracting the remainder from [prod1 prod0].
            uint256 remainder;
            assembly {
                // Compute remainder using mulmod.
                remainder := mulmod(x, y, denominator)

                // Subtract 256 bit number from 512 bit number.
                prod1 := sub(prod1, gt(remainder, prod0))
                prod0 := sub(prod0, remainder)
            }

            // Factor powers of two out of denominator and compute largest power of two divisor of denominator. Always >= 1.
            // See https://cs.stackexchange.com/q/138556/92363.

            // Does not overflow because the denominator cannot be zero at this stage in the function.
            uint256 twos = denominator & (~denominator + 1);
            assembly {
                // Divide denominator by twos.
                denominator := div(denominator, twos)

                // Divide [prod1 prod0] by twos.
                prod0 := div(prod0, twos)

                // Flip twos such that it is 2^256 / twos. If twos is zero, then it becomes one.
                twos := add(div(sub(0, twos), twos), 1)
            }

            // Shift in bits from prod1 into prod0.
            prod0 |= prod1 * twos;

            // Invert denominator mod 2^256. Now that denominator is an odd number, it has an inverse modulo 2^256 such
            // that denominator * inv = 1 mod 2^256. Compute the inverse by starting with a seed that is correct for
            // four bits. That is, denominator * inv = 1 mod 2^4.
            uint256 inverse = (3 * denominator) ^ 2;

            // Use the Newton-Raphson iteration to improve the precision. Thanks to Hensel's lifting lemma, this also works
            // in modular arithmetic, doubling the correct bits in each step.
            inverse *= 2 - denominator * inverse; // inverse mod 2^8
            inverse *= 2 - denominator * inverse; // inverse mod 2^16
            inverse *= 2 - denominator * inverse; // inverse mod 2^32
            inverse *= 2 - denominator * inverse; // inverse mod 2^64
            inverse *= 2 - denominator * inverse; // inverse mod 2^128
            inverse *= 2 - denominator * inverse; // inverse mod 2^256

            // Because the division is now exact we can divide by multiplying with the modular inverse of denominator.
            // This will give us the correct result modulo 2^256. Since the preconditions guarantee that the outcome is
            // less than 2^256, this is the final result. We don't need to compute the high bits of the result and prod1
            // is no longer required.
            result = prod0 * inverse;
            return result;
        }
    }

    /**
     * @notice Calculates x * y / denominator with full precision, following the selected rounding direction.
     */
    function mulDiv(
        uint256 x,
        uint256 y,
        uint256 denominator,
        Rounding rounding
    ) internal pure returns (uint256) {
        uint256 result = mulDiv(x, y, denominator);
        if (rounding == Rounding.Up && mulmod(x, y, denominator) > 0) {
            result += 1;
        }
        return result;
    }

    /**
     * @dev Returns the square root of a number. If the number is not a perfect square, the value is rounded down.
     *
     * Inspired by Henry S. Warren, Jr.'s "Hacker's Delight" (Chapter 11).
     */
    function sqrt(uint256 a) internal pure returns (uint256) {
        if (a == 0) {
            return 0;
        }

        // For our first guess, we get the biggest power of 2 which is smaller than the square root of the target.
        //
        // We know that the "msb" (most significant bit) of our target number `a` is a power of 2 such that we have
        // `msb(a) <= a < 2*msb(a)`. This value can be written `msb(a)=2**k` with `k=log2(a)`.
        //
        // This can be rewritten `2**log2(a) <= a < 2**(log2(a) + 1)`
        // → `sqrt(2**k) <= sqrt(a) < sqrt(2**(k+1))`
        // → `2**(k/2) <= sqrt(a) < 2**((k+1)/2) <= 2**(k/2 + 1)`
        //
        // Consequently, `2**(log2(a) / 2)` is a good first approximation of `sqrt(a)` with at least 1 correct bit.
        uint256 result = 1 << (log2(a) >> 1);

        // At this point `result` is an estimation with one bit of precision. We know the true value is a uint128,
        // since it is the square root of a uint256. Newton's method converges quadratically (precision doubles at
        // every iteration). We thus need at most 7 iteration to turn our partial result with one bit of precision
        // into the expected uint128 result.
        unchecked {
            result = (result + a / result) >> 1;
            result = (result + a / result) >> 1;
            result = (result + a / result) >> 1;
            result = (result + a / result) >> 1;
            result = (result + a / result) >> 1;
            result = (result + a / result) >> 1;
            result = (result + a / result) >> 1;
            return min(result, a / result);
        }
    }

    /**
     * @notice Calculates sqrt(a), following the selected rounding direction.
     */
    function sqrt(uint256 a, Rounding rounding) internal pure returns (uint256) {
        unchecked {
            uint256 result = sqrt(a);
            return result + (rounding == Rounding.Up && result * result < a ? 1 : 0);
        }
    }

    /**
     * @dev Return the log in base 2, rounded down, of a positive value.
     * Returns 0 if given 0.
     */
    function log2(uint256 value) internal pure returns (uint256) {
        uint256 result = 0;
        unchecked {
            if (value >> 128 > 0) {
                value >>= 128;
                result += 128;
            }
            if (value >> 64 > 0) {
                value >>= 64;
                result += 64;
            }
            if (value >> 32 > 0) {
                value >>= 32;
                result += 32;
            }
            if (value >> 16 > 0) {
                value >>= 16;
                result += 16;
            }
            if (value >> 8 > 0) {
                value >>= 8;
                result += 8;
            }
            if (value >> 4 > 0) {
                value >>= 4;
                result += 4;
            }
            if (value >> 2 > 0) {
                value >>= 2;
                result += 2;
            }
            if (value >> 1 > 0) {
                result += 1;
            }
        }
        return result;
    }

    /**
     * @dev Return the log in base 2, following the selected rounding direction, of a positive value.
     * Returns 0 if given 0.
     */
    function log2(uint256 value, Rounding rounding) internal pure returns (uint256) {
        unchecked {
            uint256 result = log2(value);
            return result + (rounding == Rounding.Up && 1 << result < value ? 1 : 0);
        }
    }

    /**
     * @dev Return the log in base 10, rounded down, of a positive value.
     * Returns 0 if given 0.
     */
    function log10(uint256 value) internal pure returns (uint256) {
        uint256 result = 0;
        unchecked {
            if (value >= 10**64) {
                value /= 10**64;
                result += 64;
            }
            if (value >= 10**32) {
                value /= 10**32;
                result += 32;
            }
            if (value >= 10**16) {
                value /= 10**16;
                result += 16;
            }
            if (value >= 10**8) {
                value /= 10**8;
                result += 8;
            }
            if (value >= 10**4) {
                value /= 10**4;
                result += 4;
            }
            if (value >= 10**2) {
                value /= 10**2;
                result += 2;
            }
            if (value >= 10**1) {
                result += 1;
            }
        }
        return result;
    }

    /**
     * @dev Return the log in base 10, following the selected rounding direction, of a positive value.
     * Returns 0 if given 0.
     */
    function log10(uint256 value, Rounding rounding) internal pure returns (uint256) {
        unchecked {
            uint256 result = log10(value);
            return result + (rounding == Rounding.Up && 10**result < value ? 1 : 0);
        }
    }

    /**
     * @dev Return the log in base 256, rounded down, of a positive value.
     * Returns 0 if given 0.
     *
     * Adding one to the result gives the number of pairs of hex symbols needed to represent `value` as a hex string.
     */
    function log256(uint256 value) internal pure returns (uint256) {
        uint256 result = 0;
        unchecked {
            if (value >> 128 > 0) {
                value >>= 128;
                result += 16;
            }
            if (value >> 64 > 0) {
                value >>= 64;
                result += 8;
            }
            if (value >> 32 > 0) {
                value >>= 32;
                result += 4;
            }
            if (value >> 16 > 0) {
                value >>= 16;
                result += 2;
            }
            if (value >> 8 > 0) {
                result += 1;
            }
        }
        return result;
    }

    /**
     * @dev Return the log in base 10, following the selected rounding direction, of a positive value.
     * Returns 0 if given 0.
     */
    function log256(uint256 value, Rounding rounding) internal pure returns (uint256) {
        unchecked {
            uint256 result = log256(value);
            return result + (rounding == Rounding.Up && 1 << (result * 8) < value ? 1 : 0);
        }
    }
}

// File: @openzeppelin/contracts/utils/Strings.sol


// OpenZeppelin Contracts (last updated v4.8.0) (utils/Strings.sol)

pragma solidity ^0.8.0;


/**
 * @dev String operations.
 */
library Strings {
    bytes16 private constant _SYMBOLS = "0123456789abcdef";
    uint8 private constant _ADDRESS_LENGTH = 20;

    /**
     * @dev Converts a `uint256` to its ASCII `string` decimal representation.
     */
    function toString(uint256 value) internal pure returns (string memory) {
        unchecked {
            uint256 length = Math.log10(value) + 1;
            string memory buffer = new string(length);
            uint256 ptr;
            /// @solidity memory-safe-assembly
            assembly {
                ptr := add(buffer, add(32, length))
            }
            while (true) {
                ptr--;
                /// @solidity memory-safe-assembly
                assembly {
                    mstore8(ptr, byte(mod(value, 10), _SYMBOLS))
                }
                value /= 10;
                if (value == 0) break;
            }
            return buffer;
        }
    }

    /**
     * @dev Converts a `uint256` to its ASCII `string` hexadecimal representation.
     */
    function toHexString(uint256 value) internal pure returns (string memory) {
        unchecked {
            return toHexString(value, Math.log256(value) + 1);
        }
    }

    /**
     * @dev Converts a `uint256` to its ASCII `string` hexadecimal representation with fixed length.
     */
    function toHexString(uint256 value, uint256 length) internal pure returns (string memory) {
        bytes memory buffer = new bytes(2 * length + 2);
        buffer[0] = "0";
        buffer[1] = "x";
        for (uint256 i = 2 * length + 1; i > 1; --i) {
            buffer[i] = _SYMBOLS[value & 0xf];
            value >>= 4;
        }
        require(value == 0, "Strings: hex length insufficient");
        return string(buffer);
    }

    /**
     * @dev Converts an `address` with fixed length of 20 bytes to its not checksummed ASCII `string` hexadecimal representation.
     */
    function toHexString(address addr) internal pure returns (string memory) {
        return toHexString(uint256(uint160(addr)), _ADDRESS_LENGTH);
    }
}

// File: @openzeppelin/contracts/utils/cryptography/ECDSA.sol


// OpenZeppelin Contracts (last updated v4.8.0) (utils/cryptography/ECDSA.sol)

pragma solidity ^0.8.0;


/**
 * @dev Elliptic Curve Digital Signature Algorithm (ECDSA) operations.
 *
 * These functions can be used to verify that a message was signed by the holder
 * of the private keys of a given address.
 */
library ECDSA {
    enum RecoverError {
        NoError,
        InvalidSignature,
        InvalidSignatureLength,
        InvalidSignatureS,
        InvalidSignatureV // Deprecated in v4.8
    }

    function _throwError(RecoverError error) private pure {
        if (error == RecoverError.NoError) {
            return; // no error: do nothing
        } else if (error == RecoverError.InvalidSignature) {
            revert("ECDSA: invalid signature");
        } else if (error == RecoverError.InvalidSignatureLength) {
            revert("ECDSA: invalid signature length");
        } else if (error == RecoverError.InvalidSignatureS) {
            revert("ECDSA: invalid signature 's' value");
        }
    }

    /**
     * @dev Returns the address that signed a hashed message (`hash`) with
     * `signature` or error string. This address can then be used for verification purposes.
     *
     * The `ecrecover` EVM opcode allows for malleable (non-unique) signatures:
     * this function rejects them by requiring the `s` value to be in the lower
     * half order, and the `v` value to be either 27 or 28.
     *
     * IMPORTANT: `hash` _must_ be the result of a hash operation for the
     * verification to be secure: it is possible to craft signatures that
     * recover to arbitrary addresses for non-hashed data. A safe way to ensure
     * this is by receiving a hash of the original message (which may otherwise
     * be too long), and then calling {toEthSignedMessageHash} on it.
     *
     * Documentation for signature generation:
     * - with https://web3js.readthedocs.io/en/v1.3.4/web3-eth-accounts.html#sign[Web3.js]
     * - with https://docs.ethers.io/v5/api/signer/#Signer-signMessage[ethers]
     *
     * _Available since v4.3._
     */
    function tryRecover(bytes32 hash, bytes memory signature) internal pure returns (address, RecoverError) {
        if (signature.length == 65) {
            bytes32 r;
            bytes32 s;
            uint8 v;
            // ecrecover takes the signature parameters, and the only way to get them
            // currently is to use assembly.
            /// @solidity memory-safe-assembly
            assembly {
                r := mload(add(signature, 0x20))
                s := mload(add(signature, 0x40))
                v := byte(0, mload(add(signature, 0x60)))
            }
            return tryRecover(hash, v, r, s);
        } else {
            return (address(0), RecoverError.InvalidSignatureLength);
        }
    }

    /**
     * @dev Returns the address that signed a hashed message (`hash`) with
     * `signature`. This address can then be used for verification purposes.
     *
     * The `ecrecover` EVM opcode allows for malleable (non-unique) signatures:
     * this function rejects them by requiring the `s` value to be in the lower
     * half order, and the `v` value to be either 27 or 28.
     *
     * IMPORTANT: `hash` _must_ be the result of a hash operation for the
     * verification to be secure: it is possible to craft signatures that
     * recover to arbitrary addresses for non-hashed data. A safe way to ensure
     * this is by receiving a hash of the original message (which may otherwise
     * be too long), and then calling {toEthSignedMessageHash} on it.
     */
    function recover(bytes32 hash, bytes memory signature) internal pure returns (address) {
        (address recovered, RecoverError error) = tryRecover(hash, signature);
        _throwError(error);
        return recovered;
    }

    /**
     * @dev Overload of {ECDSA-tryRecover} that receives the `r` and `vs` short-signature fields separately.
     *
     * See https://eips.ethereum.org/EIPS/eip-2098[EIP-2098 short signatures]
     *
     * _Available since v4.3._
     */
    function tryRecover(
        bytes32 hash,
        bytes32 r,
        bytes32 vs
    ) internal pure returns (address, RecoverError) {
        bytes32 s = vs & bytes32(0x7fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff);
        uint8 v = uint8((uint256(vs) >> 255) + 27);
        return tryRecover(hash, v, r, s);
    }

    /**
     * @dev Overload of {ECDSA-recover} that receives the `r and `vs` short-signature fields separately.
     *
     * _Available since v4.2._
     */
    function recover(
        bytes32 hash,
        bytes32 r,
        bytes32 vs
    ) internal pure returns (address) {
        (address recovered, RecoverError error) = tryRecover(hash, r, vs);
        _throwError(error);
        return recovered;
    }

    /**
     * @dev Overload of {ECDSA-tryRecover} that receives the `v`,
     * `r` and `s` signature fields separately.
     *
     * _Available since v4.3._
     */
    function tryRecover(
        bytes32 hash,
        uint8 v,
        bytes32 r,
        bytes32 s
    ) internal pure returns (address, RecoverError) {
        // EIP-2 still allows signature malleability for ecrecover(). Remove this possibility and make the signature
        // unique. Appendix F in the Ethereum Yellow paper (https://ethereum.github.io/yellowpaper/paper.pdf), defines
        // the valid range for s in (301): 0 < s < secp256k1n ÷ 2 + 1, and for v in (302): v ∈ {27, 28}. Most
        // signatures from current libraries generate a unique signature with an s-value in the lower half order.
        //
        // If your library generates malleable signatures, such as s-values in the upper range, calculate a new s-value
        // with 0xFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFEBAAEDCE6AF48A03BBFD25E8CD0364141 - s1 and flip v from 27 to 28 or
        // vice versa. If your library also generates signatures with 0/1 for v instead 27/28, add 27 to v to accept
        // these malleable signatures as well.
        if (uint256(s) > 0x7FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF5D576E7357A4501DDFE92F46681B20A0) {
            return (address(0), RecoverError.InvalidSignatureS);
        }

        // If the signature is valid (and not malleable), return the signer address
        address signer = ecrecover(hash, v, r, s);
        if (signer == address(0)) {
            return (address(0), RecoverError.InvalidSignature);
        }

        return (signer, RecoverError.NoError);
    }

    /**
     * @dev Overload of {ECDSA-recover} that receives the `v`,
     * `r` and `s` signature fields separately.
     */
    function recover(
        bytes32 hash,
        uint8 v,
        bytes32 r,
        bytes32 s
    ) internal pure returns (address) {
        (address recovered, RecoverError error) = tryRecover(hash, v, r, s);
        _throwError(error);
        return recovered;
    }

    /**
     * @dev Returns an Ethereum Signed Message, created from a `hash`. This
     * produces hash corresponding to the one signed with the
     * https://eth.wiki/json-rpc/API#eth_sign[`eth_sign`]
     * JSON-RPC method as part of EIP-191.
     *
     * See {recover}.
     */
    function toEthSignedMessageHash(bytes32 hash) internal pure returns (bytes32) {
        // 32 is the length in bytes of hash,
        // enforced by the type signature above
        return keccak256(abi.encodePacked("\x19Ethereum Signed Message:\n32", hash));
    }

    /**
     * @dev Returns an Ethereum Signed Message, created from `s`. This
     * produces hash corresponding to the one signed with the
     * https://eth.wiki/json-rpc/API#eth_sign[`eth_sign`]
     * JSON-RPC method as part of EIP-191.
     *
     * See {recover}.
     */
    function toEthSignedMessageHash(bytes memory s) internal pure returns (bytes32) {
        return keccak256(abi.encodePacked("\x19Ethereum Signed Message:\n", Strings.toString(s.length), s));
    }

    /**
     * @dev Returns an Ethereum Signed Typed Data, created from a
     * `domainSeparator` and a `structHash`. This produces hash corresponding
     * to the one signed with the
     * https://eips.ethereum.org/EIPS/eip-712[`eth_signTypedData`]
     * JSON-RPC method as part of EIP-712.
     *
     * See {recover}.
     */
    function toTypedDataHash(bytes32 domainSeparator, bytes32 structHash) internal pure returns (bytes32) {
        return keccak256(abi.encodePacked("\x19\x01", domainSeparator, structHash));
    }
}

// File: @openzeppelin/contracts/utils/cryptography/EIP712.sol


// OpenZeppelin Contracts (last updated v4.8.0) (utils/cryptography/EIP712.sol)

pragma solidity ^0.8.0;


/**
 * @dev https://eips.ethereum.org/EIPS/eip-712[EIP 712] is a standard for hashing and signing of typed structured data.
 *
 * The encoding specified in the EIP is very generic, and such a generic implementation in Solidity is not feasible,
 * thus this contract does not implement the encoding itself. Protocols need to implement the type-specific encoding
 * they need in their contracts using a combination of `abi.encode` and `keccak256`.
 *
 * This contract implements the EIP 712 domain separator ({_domainSeparatorV4}) that is used as part of the encoding
 * scheme, and the final step of the encoding to obtain the message digest that is then signed via ECDSA
 * ({_hashTypedDataV4}).
 *
 * The implementation of the domain separator was designed to be as efficient as possible while still properly updating
 * the chain id to protect against replay attacks on an eventual fork of the chain.
 *
 * NOTE: This contract implements the version of the encoding known as "v4", as implemented by the JSON RPC method
 * https://docs.metamask.io/guide/signing-data.html[`eth_signTypedDataV4` in MetaMask].
 *
 * _Available since v3.4._
 */
abstract contract EIP712 {
    /* solhint-disable var-name-mixedcase */
    // Cache the domain separator as an immutable value, but also store the chain id that it corresponds to, in order to
    // invalidate the cached domain separator if the chain id changes.
    bytes32 private immutable _CACHED_DOMAIN_SEPARATOR;
    uint256 private immutable _CACHED_CHAIN_ID;
    address private immutable _CACHED_THIS;

    bytes32 private immutable _HASHED_NAME;
    bytes32 private immutable _HASHED_VERSION;
    bytes32 private immutable _TYPE_HASH;

    /* solhint-enable var-name-mixedcase */

    /**
     * @dev Initializes the domain separator and parameter caches.
     *
     * The meaning of `name` and `version` is specified in
     * https://eips.ethereum.org/EIPS/eip-712#definition-of-domainseparator[EIP 712]:
     *
     * - `name`: the user readable name of the signing domain, i.e. the name of the DApp or the protocol.
     * - `version`: the current major version of the signing domain.
     *
     * NOTE: These parameters cannot be changed except through a xref:learn::upgrading-smart-contracts.adoc[smart
     * contract upgrade].
     */
    constructor(string memory name, string memory version) {
        bytes32 hashedName = keccak256(bytes(name));
        bytes32 hashedVersion = keccak256(bytes(version));
        bytes32 typeHash = keccak256(
            "EIP712Domain(string name,string version,uint256 chainId,address verifyingContract)"
        );
        _HASHED_NAME = hashedName;
        _HASHED_VERSION = hashedVersion;
        _CACHED_CHAIN_ID = block.chainid;
        _CACHED_DOMAIN_SEPARATOR = _buildDomainSeparator(typeHash, hashedName, hashedVersion);
        _CACHED_THIS = address(this);
        _TYPE_HASH = typeHash;
    }

    /**
     * @dev Returns the domain separator for the current chain.
     */
    function _domainSeparatorV4() internal view returns (bytes32) {
        if (address(this) == _CACHED_THIS && block.chainid == _CACHED_CHAIN_ID) {
            return _CACHED_DOMAIN_SEPARATOR;
        } else {
            return _buildDomainSeparator(_TYPE_HASH, _HASHED_NAME, _HASHED_VERSION);
        }
    }

    function _buildDomainSeparator(
        bytes32 typeHash,
        bytes32 nameHash,
        bytes32 versionHash
    ) private view returns (bytes32) {
        return keccak256(abi.encode(typeHash, nameHash, versionHash, block.chainid, address(this)));
    }

    /**
     * @dev Given an already https://eips.ethereum.org/EIPS/eip-712#definition-of-hashstruct[hashed struct], this
     * function returns the hash of the fully encoded EIP712 message for this domain.
     *
     * This hash can be used together with {ECDSA-recover} to obtain the signer of a message. For example:
     *
     * ```solidity
     * bytes32 digest = _hashTypedDataV4(keccak256(abi.encode(
     *     keccak256("Mail(address to,string contents)"),
     *     mailTo,
     *     keccak256(bytes(mailContents))
     * )));
     * address signer = ECDSA.recover(digest, signature);
     * ```
     */
    function _hashTypedDataV4(bytes32 structHash) internal view virtual returns (bytes32) {
        return ECDSA.toTypedDataHash(_domainSeparatorV4(), structHash);
    }
}

// File: @openzeppelin/contracts/utils/cryptography/draft-EIP712.sol


// OpenZeppelin Contracts (last updated v4.8.0) (utils/cryptography/draft-EIP712.sol)

pragma solidity ^0.8.0;

// EIP-712 is Final as of 2022-08-11. This file is deprecated.


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


// OpenZeppelin Contracts (last updated v4.8.0) (utils/cryptography/MerkleProof.sol)

pragma solidity ^0.8.0;

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

// File: contracts/ChampagneCarbon.sol


pragma solidity ^0.8.4;








contract ChampagneCarbon is ERC721A, Ownable, EIP712, Pausable {
    using Strings for uint256;

    //SUPPLY
    uint256 public constant MAX_SUPPLY = 20;
    uint256 public constant MAX_MINT_PER_WALLET = 2;

    //METADATA
    string public baseURI;
    string private collectionURI;

    //MALARTIC SALE CONTRACT ADDRESS
    address private saleAddress;

    //REDEEM MAPPING
    mapping(uint8 => uint256) public redeemedTimestamp;

    //MINT PER ADDRESS COUNTER MAPPING
    mapping(address => uint256) public mintedPerWallet;

    //MINT EVENT
    event Mint(address to, uint256 amount);

    //METADATA UPDATE EVENT
    event MetadataUpdate(uint256 _tokenId);

    //MODIFIERS
    modifier checkMintPerAddress(address account_, uint256 amount_) {
        require(
            mintedPerWallet[account_] + amount_ <= MAX_MINT_PER_WALLET,
            "ChampagneCarbon: mint limit per wallet was exceeded"
        );
        _;
    }

    modifier checkSupply(uint256 amount_) {
        require(
            totalSupply() + amount_ <= MAX_SUPPLY,
            "ChampagneCarbon: Mint supply limit was exceeded"
        );
        _;
    }

    constructor()
        ERC721A("ChampagneCarbon", "CCB")
        EIP712("ChampagneCarbon", "1")
    {}

    function batchMint(
        address to_,
        uint256 amount_
    ) external checkMintPerAddress(to_, amount_) checkSupply(amount_) {
        require(msg.sender == saleAddress, "Not allowed");
        _safeMint(to_, amount_);
        mintedPerWallet[to_] = mintedPerWallet[to_] + amount_;
        emit Mint(to_, amount_);
    }

    function drop(
        address to_,
        uint256 amount_
    ) public onlyOwner checkSupply(amount_) {
        _safeMint(to_, amount_);
    }

    //REDEEM FUNCTIONS

    // EIP-712 typed structured data signature
    bytes32 private constant CHAMPAGNE_CARBON_SHIPPING_DATA_TYPEHASH =
        keccak256(
            "ChampagneCarbonShippingData(bytes32 hashdata,uint64 timestamp)"
        );

    /**
     * @dev Utility function for EIP712
     * @param hashdata The hash of the redeem data
     * @param timestamp The redeem timestamp
     * @param signer The wallet who signed the hashdata
     * @param signature The result of when the "signer" signs the "hashdata"
     */
    function verifyShippingData(
        bytes32 hashdata,
        uint64 timestamp,
        address signer,
        bytes calldata signature
    ) public view returns (bool) {
        bytes32 digest = _hashTypedDataV4(
            keccak256(
                abi.encode(
                    CHAMPAGNE_CARBON_SHIPPING_DATA_TYPEHASH,
                    hashdata,
                    timestamp
                )
            )
        );
        (address a, ECDSA.RecoverError e) = ECDSA.tryRecover(digest, signature);
        return (a == signer) && (e == ECDSA.RecoverError.NoError);
    }

    /**
     * @dev Utility function for EIP712
     * @param tokenId The token Id to be redeemed
     * @param hashdata The hash of the redeem data
     * @param timestamp The redeem timestamp
     * @param signature The result of when the "signer" signs the "hashdata"
     */
    function redeem(
        uint8 tokenId,
        bytes32 hashdata,
        uint64 timestamp,
        bytes calldata signature
    ) public {
        require(ownerOf(tokenId) == msg.sender, "ChampagneCarbon: Not allowed");
        require(
            redeemedTimestamp[tokenId] == 0,
            "ChampagneCarbon: NFT already redeemed"
        );
        require(
            verifyShippingData(hashdata, timestamp, msg.sender, signature),
            "Invalid signature"
        );
        redeemedTimestamp[tokenId] = block.timestamp;
        emit MetadataUpdate(tokenId);
    }

    //PAUSE/UNPAUSE FUNCTIONS
    function pause() public onlyOwner {
        _pause();
    }

    function unpause() public onlyOwner {
        _unpause();
    }

    function contractURI() public view returns (string memory) {
        return collectionURI;
    }

    //SETTERS
    function setBaseURI(string calldata uri) external onlyOwner {
        baseURI = uri;
    }

    function setContractURI(string calldata uri) external onlyOwner {
        collectionURI = uri;
    }

    function setSaleAddress(address _saleAddress) public onlyOwner {
        saleAddress = _saleAddress;
    }

    function _baseURI() internal view override returns (string memory) {
        return baseURI;
    }

    //OVERRIDES
    function _startTokenId()
        internal
        view
        virtual
        override(ERC721A)
        returns (uint256)
    {
        return 1;
    }

    function _beforeTokenTransfers(
        address from,
        address to,
        uint256 startTokenId,
        uint256 quantity
    ) internal virtual override(ERC721A) {
        super._beforeTokenTransfers(from, to, startTokenId, quantity);
        require(!paused(), "Pausable: token transfer while paused");
    }

    //supports ERC4906
    function supportsInterface(
        bytes4 interfaceId
    ) public view virtual override(ERC721A) returns (bool) {
        return
            interfaceId == bytes4(0x49064906) ||
            super.supportsInterface(interfaceId);
    }
}

Contract Security Audit

Contract ABI

[{"inputs":[],"stateMutability":"nonpayable","type":"constructor"},{"inputs":[],"name":"ApprovalCallerNotOwnerNorApproved","type":"error"},{"inputs":[],"name":"ApprovalQueryForNonexistentToken","type":"error"},{"inputs":[],"name":"BalanceQueryForZeroAddress","type":"error"},{"inputs":[],"name":"MintERC2309QuantityExceedsLimit","type":"error"},{"inputs":[],"name":"MintToZeroAddress","type":"error"},{"inputs":[],"name":"MintZeroQuantity","type":"error"},{"inputs":[],"name":"OwnerQueryForNonexistentToken","type":"error"},{"inputs":[],"name":"OwnershipNotInitializedForExtraData","type":"error"},{"inputs":[],"name":"TransferCallerNotOwnerNorApproved","type":"error"},{"inputs":[],"name":"TransferFromIncorrectOwner","type":"error"},{"inputs":[],"name":"TransferToNonERC721ReceiverImplementer","type":"error"},{"inputs":[],"name":"TransferToZeroAddress","type":"error"},{"inputs":[],"name":"URIQueryForNonexistentToken","type":"error"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"owner","type":"address"},{"indexed":true,"internalType":"address","name":"approved","type":"address"},{"indexed":true,"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"Approval","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"owner","type":"address"},{"indexed":true,"internalType":"address","name":"operator","type":"address"},{"indexed":false,"internalType":"bool","name":"approved","type":"bool"}],"name":"ApprovalForAll","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"uint256","name":"fromTokenId","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"toTokenId","type":"uint256"},{"indexed":true,"internalType":"address","name":"from","type":"address"},{"indexed":true,"internalType":"address","name":"to","type":"address"}],"name":"ConsecutiveTransfer","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"_tokenId","type":"uint256"}],"name":"MetadataUpdate","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"to","type":"address"},{"indexed":false,"internalType":"uint256","name":"amount","type":"uint256"}],"name":"Mint","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"previousOwner","type":"address"},{"indexed":true,"internalType":"address","name":"newOwner","type":"address"}],"name":"OwnershipTransferred","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"account","type":"address"}],"name":"Paused","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"from","type":"address"},{"indexed":true,"internalType":"address","name":"to","type":"address"},{"indexed":true,"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"Transfer","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"account","type":"address"}],"name":"Unpaused","type":"event"},{"inputs":[],"name":"MAX_MINT_PER_WALLET","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":[{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"approve","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[{"internalType":"address","name":"owner","type":"address"}],"name":"balanceOf","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"baseURI","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"to_","type":"address"},{"internalType":"uint256","name":"amount_","type":"uint256"}],"name":"batchMint","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"contractURI","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"to_","type":"address"},{"internalType":"uint256","name":"amount_","type":"uint256"}],"name":"drop","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"getApproved","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"owner","type":"address"},{"internalType":"address","name":"operator","type":"address"}],"name":"isApprovedForAll","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"mintedPerWallet","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","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":"uint256","name":"tokenId","type":"uint256"}],"name":"ownerOf","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"pause","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"paused","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint8","name":"tokenId","type":"uint8"},{"internalType":"bytes32","name":"hashdata","type":"bytes32"},{"internalType":"uint64","name":"timestamp","type":"uint64"},{"internalType":"bytes","name":"signature","type":"bytes"}],"name":"redeem","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint8","name":"","type":"uint8"}],"name":"redeemedTimestamp","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"renounceOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"from","type":"address"},{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"safeTransferFrom","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[{"internalType":"address","name":"from","type":"address"},{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"tokenId","type":"uint256"},{"internalType":"bytes","name":"_data","type":"bytes"}],"name":"safeTransferFrom","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[{"internalType":"address","name":"operator","type":"address"},{"internalType":"bool","name":"approved","type":"bool"}],"name":"setApprovalForAll","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"string","name":"uri","type":"string"}],"name":"setBaseURI","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"string","name":"uri","type":"string"}],"name":"setContractURI","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_saleAddress","type":"address"}],"name":"setSaleAddress","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"bytes4","name":"interfaceId","type":"bytes4"}],"name":"supportsInterface","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"symbol","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"tokenURI","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"totalSupply","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":"tokenId","type":"uint256"}],"name":"transferFrom","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[{"internalType":"address","name":"newOwner","type":"address"}],"name":"transferOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"unpause","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"bytes32","name":"hashdata","type":"bytes32"},{"internalType":"uint64","name":"timestamp","type":"uint64"},{"internalType":"address","name":"signer","type":"address"},{"internalType":"bytes","name":"signature","type":"bytes"}],"name":"verifyShippingData","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"}]

6101406040523480156200001257600080fd5b506040518060400160405280600f81526020017f4368616d7061676e65436172626f6e00000000000000000000000000000000008152506040518060400160405280600181526020017f31000000000000000000000000000000000000000000000000000000000000008152506040518060400160405280600f81526020017f4368616d7061676e65436172626f6e00000000000000000000000000000000008152506040518060400160405280600381526020017f43434200000000000000000000000000000000000000000000000000000000008152508160029081620000fc9190620005a7565b5080600390816200010e9190620005a7565b506200011f6200021a60201b60201c565b6000819055505050620001476200013b6200022360201b60201c565b6200022b60201b60201c565b60008280519060200120905060008280519060200120905060007f8b73c3c69bb8fe3d512ecc4cf759cc79239f7b179b0ffacaa9a75d522b39400f90508260e081815250508161010081815250504660a08181525050620001b0818484620002f160201b60201c565b608081815250503073ffffffffffffffffffffffffffffffffffffffff1660c08173ffffffffffffffffffffffffffffffffffffffff168152505080610120818152505050505050506000600860146101000a81548160ff0219169083151502179055506200075c565b60006001905090565b600033905090565b6000600860009054906101000a900473ffffffffffffffffffffffffffffffffffffffff16905081600860006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff1602179055508173ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff167f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e060405160405180910390a35050565b600083838346306040516020016200030e959493929190620006ff565b6040516020818303038152906040528051906020012090509392505050565b600081519050919050565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052604160045260246000fd5b7f4e487b7100000000000000000000000000000000000000000000000000000000600052602260045260246000fd5b60006002820490506001821680620003af57607f821691505b602082108103620003c557620003c462000367565b5b50919050565b60008190508160005260206000209050919050565b60006020601f8301049050919050565b600082821b905092915050565b6000600883026200042f7fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff82620003f0565b6200043b8683620003f0565b95508019841693508086168417925050509392505050565b6000819050919050565b6000819050919050565b600062000488620004826200047c8462000453565b6200045d565b62000453565b9050919050565b6000819050919050565b620004a48362000467565b620004bc620004b3826200048f565b848454620003fd565b825550505050565b600090565b620004d3620004c4565b620004e081848462000499565b505050565b5b818110156200050857620004fc600082620004c9565b600181019050620004e6565b5050565b601f82111562000557576200052181620003cb565b6200052c84620003e0565b810160208510156200053c578190505b620005546200054b85620003e0565b830182620004e5565b50505b505050565b600082821c905092915050565b60006200057c600019846008026200055c565b1980831691505092915050565b600062000597838362000569565b9150826002028217905092915050565b620005b2826200032d565b67ffffffffffffffff811115620005ce57620005cd62000338565b5b620005da825462000396565b620005e78282856200050c565b600060209050601f8311600181146200061f57600084156200060a578287015190505b62000616858262000589565b86555062000686565b601f1984166200062f86620003cb565b60005b82811015620006595784890151825560018201915060208501945060208101905062000632565b8683101562000679578489015162000675601f89168262000569565b8355505b6001600288020188555050505b505050505050565b6000819050919050565b620006a3816200068e565b82525050565b620006b48162000453565b82525050565b600073ffffffffffffffffffffffffffffffffffffffff82169050919050565b6000620006e782620006ba565b9050919050565b620006f981620006da565b82525050565b600060a08201905062000716600083018862000698565b62000725602083018762000698565b62000734604083018662000698565b620007436060830185620006a9565b620007526080830184620006ee565b9695505050505050565b60805160a05160c05160e0516101005161012051613c23620007ac600039600061238b015260006123cd015260006123ac015260006122e101526000612337015260006123600152613c236000f3fe6080604052600436106101ee5760003560e01c806370a082311161010d578063b19960e6116100a0578063c87b56dd1161006f578063c87b56dd146106a1578063e8a3d485146106de578063e985e9c514610709578063f2fde38b14610746578063f8fb491f1461076f576101ee565b8063b19960e6146105f4578063b277caaa1461061f578063b88d4fde1461065c578063be9c6ee114610678576101ee565b80638da5cb5b116100dc5780638da5cb5b1461054c578063938e3d7b1461057757806395d89b41146105a0578063a22cb465146105cb576101ee565b806370a08231146104b8578063715018a6146104f55780638456cb591461050c5780638d0f8cef14610523576101ee565b80633dfcc40a1161018557806355f804b31161015457806355f804b3146103fc5780635c975abb146104255780636352211e146104505780636c0360eb1461048d576101ee565b80633dfcc40a146103635780633f4ba83a146103a057806342842e0e146103b757806343508b05146103d3576101ee565b806318160ddd116101c157806318160ddd146102b457806323b872dd146102df57806332cb6b0c146102fb5780633a602b4d14610326576101ee565b806301ffc9a7146101f357806306fdde0314610230578063081812fc1461025b578063095ea7b314610298575b600080fd5b3480156101ff57600080fd5b5061021a6004803603810190610215919061277d565b610798565b60405161022791906127c5565b60405180910390f35b34801561023c57600080fd5b506102456107f9565b6040516102529190612870565b60405180910390f35b34801561026757600080fd5b50610282600480360381019061027d91906128c8565b61088b565b60405161028f9190612936565b60405180910390f35b6102b260048036038101906102ad919061297d565b61090a565b005b3480156102c057600080fd5b506102c9610a4e565b6040516102d691906129cc565b60405180910390f35b6102f960048036038101906102f491906129e7565b610a65565b005b34801561030757600080fd5b50610310610d87565b60405161031d91906129cc565b60405180910390f35b34801561033257600080fd5b5061034d60048036038101906103489190612a3a565b610d8c565b60405161035a91906129cc565b60405180910390f35b34801561036f57600080fd5b5061038a60048036038101906103859190612aa0565b610da4565b60405161039791906129cc565b60405180910390f35b3480156103ac57600080fd5b506103b5610dbc565b005b6103d160048036038101906103cc91906129e7565b610dce565b005b3480156103df57600080fd5b506103fa60048036038101906103f5919061297d565b610dee565b005b34801561040857600080fd5b50610423600480360381019061041e9190612b32565b61103d565b005b34801561043157600080fd5b5061043a61105b565b60405161044791906127c5565b60405180910390f35b34801561045c57600080fd5b50610477600480360381019061047291906128c8565b611072565b6040516104849190612936565b60405180910390f35b34801561049957600080fd5b506104a2611084565b6040516104af9190612870565b60405180910390f35b3480156104c457600080fd5b506104df60048036038101906104da9190612a3a565b611112565b6040516104ec91906129cc565b60405180910390f35b34801561050157600080fd5b5061050a6111ca565b005b34801561051857600080fd5b506105216111de565b005b34801561052f57600080fd5b5061054a6004803603810190610545919061297d565b6111f0565b005b34801561055857600080fd5b5061056161125e565b60405161056e9190612936565b60405180910390f35b34801561058357600080fd5b5061059e60048036038101906105999190612b32565b611288565b005b3480156105ac57600080fd5b506105b56112a6565b6040516105c29190612870565b60405180910390f35b3480156105d757600080fd5b506105f260048036038101906105ed9190612bab565b611338565b005b34801561060057600080fd5b50610609611443565b60405161061691906129cc565b60405180910390f35b34801561062b57600080fd5b5061064660048036038101906106419190612cb7565b611448565b60405161065391906127c5565b60405180910390f35b61067660048036038101906106719190612e6f565b611562565b005b34801561068457600080fd5b5061069f600480360381019061069a9190612ef2565b6115d5565b005b3480156106ad57600080fd5b506106c860048036038101906106c391906128c8565b611752565b6040516106d59190612870565b60405180910390f35b3480156106ea57600080fd5b506106f36117f0565b6040516107009190612870565b60405180910390f35b34801561071557600080fd5b50610730600480360381019061072b9190612f7a565b611882565b60405161073d91906127c5565b60405180910390f35b34801561075257600080fd5b5061076d60048036038101906107689190612a3a565b611916565b005b34801561077b57600080fd5b5061079660048036038101906107919190612a3a565b611999565b005b6000634906490660e01b7bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916827bffffffffffffffffffffffffffffffffffffffffffffffffffffffff191614806107f257506107f1826119e5565b5b9050919050565b60606002805461080890612fe9565b80601f016020809104026020016040519081016040528092919081815260200182805461083490612fe9565b80156108815780601f1061085657610100808354040283529160200191610881565b820191906000526020600020905b81548152906001019060200180831161086457829003601f168201915b5050505050905090565b600061089682611a77565b6108cc576040517fcf4700e400000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b6006600083815260200190815260200160002060000160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff169050919050565b600061091582611072565b90508073ffffffffffffffffffffffffffffffffffffffff16610936611ad6565b73ffffffffffffffffffffffffffffffffffffffff1614610999576109628161095d611ad6565b611882565b610998576040517fcfb3b94200000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b5b826006600084815260200190815260200160002060000160006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff160217905550818373ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff167f8c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b92560405160405180910390a4505050565b6000610a58611ade565b6001546000540303905090565b6000610a7082611ae7565b90508373ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff1614610ad7576040517fa114810000000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b600080610ae384611bb3565b91509150610af98187610af4611ad6565b611bda565b610b4557610b0e86610b09611ad6565b611882565b610b44576040517f59c896be00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b5b600073ffffffffffffffffffffffffffffffffffffffff168573ffffffffffffffffffffffffffffffffffffffff1603610bab576040517fea553b3400000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b610bb88686866001611c1e565b8015610bc357600082555b600560008773ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600081546001900391905081905550600560008673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000206000815460010191905081905550610c9185610c6d888887611c78565b7c020000000000000000000000000000000000000000000000000000000017611ca0565b600460008681526020019081526020016000208190555060007c0200000000000000000000000000000000000000000000000000000000841603610d175760006001850190506000600460008381526020019081526020016000205403610d15576000548114610d14578360046000838152602001908152602001600020819055505b5b505b838573ffffffffffffffffffffffffffffffffffffffff168773ffffffffffffffffffffffffffffffffffffffff167fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef60405160405180910390a4610d7f8686866001611ccb565b505050505050565b601481565b600d6020528060005260406000206000915090505481565b600c6020528060005260406000206000915090505481565b610dc4611cd1565b610dcc611d4f565b565b610de983838360405180602001604052806000815250611562565b505050565b8181600281600d60008573ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002054610e3d9190613049565b1115610e7e576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610e75906130ef565b60405180910390fd5b82601481610e8a610a4e565b610e949190613049565b1115610ed5576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610ecc90613181565b60405180910390fd5b600b60009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff163373ffffffffffffffffffffffffffffffffffffffff1614610f65576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610f5c906131ed565b60405180910390fd5b610f6f8585611db2565b83600d60008773ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002054610fba9190613049565b600d60008773ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020819055507f0f6798a560793a54c3bcfe86a93cde1e73087d944c0ea20544137d4121396885858560405161102e92919061320d565b60405180910390a15050505050565b611045611cd1565b8181600991826110569291906133ed565b505050565b6000600860149054906101000a900460ff16905090565b600061107d82611ae7565b9050919050565b6009805461109190612fe9565b80601f01602080910402602001604051908101604052809291908181526020018280546110bd90612fe9565b801561110a5780601f106110df5761010080835404028352916020019161110a565b820191906000526020600020905b8154815290600101906020018083116110ed57829003601f168201915b505050505081565b60008073ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff1603611179576040517f8f4eb60400000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b67ffffffffffffffff600560008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002054169050919050565b6111d2611cd1565b6111dc6000611dd0565b565b6111e6611cd1565b6111ee611e96565b565b6111f8611cd1565b80601481611204610a4e565b61120e9190613049565b111561124f576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161124690613181565b60405180910390fd5b6112598383611db2565b505050565b6000600860009054906101000a900473ffffffffffffffffffffffffffffffffffffffff16905090565b611290611cd1565b8181600a91826112a19291906133ed565b505050565b6060600380546112b590612fe9565b80601f01602080910402602001604051908101604052809291908181526020018280546112e190612fe9565b801561132e5780601f106113035761010080835404028352916020019161132e565b820191906000526020600020905b81548152906001019060200180831161131157829003601f168201915b5050505050905090565b8060076000611345611ad6565b73ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060006101000a81548160ff0219169083151502179055508173ffffffffffffffffffffffffffffffffffffffff166113f2611ad6565b73ffffffffffffffffffffffffffffffffffffffff167f17307eab39ab6107e8899845ad3d59bd9653f200f220920489ca2b5937696c318360405161143791906127c5565b60405180910390a35050565b600281565b60008061149e7f9067465cd589b9fb125b4fb7b6fbf2ed93fd28c348210854bbba666b918b612f8888604051602001611483939291906134db565b60405160208183030381529060405280519060200120611ef9565b90506000806114f18387878080601f016020809104026020016040519081016040528093929190818152602001838380828437600081840152601f19601f82011690508083019250505050505050611f13565b915091508673ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff1614801561155457506000600481111561153f5761153e613512565b5b81600481111561155257611551613512565b5b145b935050505095945050505050565b61156d848484610a65565b60008373ffffffffffffffffffffffffffffffffffffffff163b146115cf5761159884848484611f64565b6115ce576040517fd1a57ed600000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b5b50505050565b3373ffffffffffffffffffffffffffffffffffffffff166115f88660ff16611072565b73ffffffffffffffffffffffffffffffffffffffff161461164e576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016116459061358d565b60405180910390fd5b6000600c60008760ff1660ff16815260200190815260200160002054146116aa576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016116a19061361f565b60405180910390fd5b6116b78484338585611448565b6116f6576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016116ed9061368b565b60405180910390fd5b42600c60008760ff1660ff168152602001908152602001600020819055507ff8e1a15aba9398e019f0b49df1a4fde98ee17ae345cb5f6b5e2c27f5033e8ce78560405161174391906136dc565b60405180910390a15050505050565b606061175d82611a77565b611793576040517fa14c4b5000000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b600061179d6120b4565b905060008151036117bd57604051806020016040528060008152506117e8565b806117c784612146565b6040516020016117d8929190613733565b6040516020818303038152906040525b915050919050565b6060600a80546117ff90612fe9565b80601f016020809104026020016040519081016040528092919081815260200182805461182b90612fe9565b80156118785780601f1061184d57610100808354040283529160200191611878565b820191906000526020600020905b81548152906001019060200180831161185b57829003601f168201915b5050505050905090565b6000600760008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060009054906101000a900460ff16905092915050565b61191e611cd1565b600073ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff160361198d576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401611984906137c9565b60405180910390fd5b61199681611dd0565b50565b6119a1611cd1565b80600b60006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff16021790555050565b60006301ffc9a760e01b827bffffffffffffffffffffffffffffffffffffffffffffffffffffffff19161480611a4057506380ac58cd60e01b827bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916145b80611a705750635b5e139f60e01b827bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916145b9050919050565b600081611a82611ade565b11158015611a91575060005482105b8015611acf575060007c0100000000000000000000000000000000000000000000000000000000600460008581526020019081526020016000205416145b9050919050565b600033905090565b60006001905090565b60008082905080611af6611ade565b11611b7c57600054811015611b7b5760006004600083815260200190815260200160002054905060007c0100000000000000000000000000000000000000000000000000000000821603611b79575b60008103611b6f576004600083600190039350838152602001908152602001600020549050611b45565b8092505050611bae565b505b5b6040517fdf2d9b4200000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b919050565b60008060006006600085815260200190815260200160002090508092508254915050915091565b600073ffffffffffffffffffffffffffffffffffffffff8316925073ffffffffffffffffffffffffffffffffffffffff821691508382148383141790509392505050565b611c2a84848484612196565b611c3261105b565b15611c72576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401611c699061385b565b60405180910390fd5b50505050565b60008060e883901c905060e8611c8f86868461219c565b62ffffff16901b9150509392505050565b600073ffffffffffffffffffffffffffffffffffffffff83169250814260a01b178317905092915050565b50505050565b611cd96121a5565b73ffffffffffffffffffffffffffffffffffffffff16611cf761125e565b73ffffffffffffffffffffffffffffffffffffffff1614611d4d576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401611d44906138c7565b60405180910390fd5b565b611d576121ad565b6000600860146101000a81548160ff0219169083151502179055507f5db9ee0a495bf2e6ff9c91a7834c1ba4fdd244a5e8aa4e537bd38aeae4b073aa611d9b6121a5565b604051611da89190612936565b60405180910390a1565b611dcc8282604051806020016040528060008152506121f6565b5050565b6000600860009054906101000a900473ffffffffffffffffffffffffffffffffffffffff16905081600860006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff1602179055508173ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff167f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e060405160405180910390a35050565b611e9e612293565b6001600860146101000a81548160ff0219169083151502179055507f62e78cea01bee320cd4e420270b5ea74000d11b0c9f74754ebdbfc544b05a258611ee26121a5565b604051611eef9190612936565b60405180910390a1565b6000611f0c611f066122dd565b836123f7565b9050919050565b6000806041835103611f545760008060006020860151925060408601519150606086015160001a9050611f488782858561242a565b94509450505050611f5d565b60006002915091505b9250929050565b60008373ffffffffffffffffffffffffffffffffffffffff1663150b7a02611f8a611ad6565b8786866040518563ffffffff1660e01b8152600401611fac949392919061393c565b6020604051808303816000875af1925050508015611fe857506040513d601f19601f82011682018060405250810190611fe5919061399d565b60015b612061573d8060008114612018576040519150601f19603f3d011682016040523d82523d6000602084013e61201d565b606091505b506000815103612059576040517fd1a57ed600000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b805181602001fd5b63150b7a0260e01b7bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916817bffffffffffffffffffffffffffffffffffffffffffffffffffffffff191614915050949350505050565b6060600980546120c390612fe9565b80601f01602080910402602001604051908101604052809291908181526020018280546120ef90612fe9565b801561213c5780601f106121115761010080835404028352916020019161213c565b820191906000526020600020905b81548152906001019060200180831161211f57829003601f168201915b5050505050905090565b606060a060405101806040526020810391506000825281835b60011561218157600184039350600a81066030018453600a810490508061215f575b50828103602084039350808452505050919050565b50505050565b60009392505050565b600033905090565b6121b561105b565b6121f4576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016121eb90613a16565b60405180910390fd5b565b612200838361250c565b60008373ffffffffffffffffffffffffffffffffffffffff163b1461228e57600080549050600083820390505b6122406000868380600101945086611f64565b612276576040517fd1a57ed600000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b81811061222d57816000541461228b57600080fd5b50505b505050565b61229b61105b565b156122db576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016122d290613a82565b60405180910390fd5b565b60007f000000000000000000000000000000000000000000000000000000000000000073ffffffffffffffffffffffffffffffffffffffff163073ffffffffffffffffffffffffffffffffffffffff1614801561235957507f000000000000000000000000000000000000000000000000000000000000000046145b15612386577f000000000000000000000000000000000000000000000000000000000000000090506123f4565b6123f17f00000000000000000000000000000000000000000000000000000000000000007f00000000000000000000000000000000000000000000000000000000000000007f00000000000000000000000000000000000000000000000000000000000000006126c7565b90505b90565b6000828260405160200161240c929190613b0f565b60405160208183030381529060405280519060200120905092915050565b6000807f7fffffffffffffffffffffffffffffff5d576e7357a4501ddfe92f46681b20a08360001c1115612465576000600391509150612503565b60006001878787876040516000815260200160405260405161248a9493929190613b55565b6020604051602081039080840390855afa1580156124ac573d6000803e3d6000fd5b505050602060405103519050600073ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff16036124fa57600060019250925050612503565b80600092509250505b94509492505050565b6000805490506000820361254c576040517fb562e8dd00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b6125596000848385611c1e565b600160406001901b178202600560008573ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600082825401925050819055506125d0836125c16000866000611c78565b6125ca85612701565b17611ca0565b6004600083815260200190815260200160002081905550600080838301905073ffffffffffffffffffffffffffffffffffffffff85169150828260007fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef600080a4600183015b81811461267157808360007fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef600080a4600181019050612636565b50600082036126ac576040517f2e07630000000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b8060008190555050506126c26000848385611ccb565b505050565b600083838346306040516020016126e2959493929190613b9a565b6040516020818303038152906040528051906020012090509392505050565b60006001821460e11b9050919050565b6000604051905090565b600080fd5b600080fd5b60007fffffffff0000000000000000000000000000000000000000000000000000000082169050919050565b61275a81612725565b811461276557600080fd5b50565b60008135905061277781612751565b92915050565b6000602082840312156127935761279261271b565b5b60006127a184828501612768565b91505092915050565b60008115159050919050565b6127bf816127aa565b82525050565b60006020820190506127da60008301846127b6565b92915050565b600081519050919050565b600082825260208201905092915050565b60005b8381101561281a5780820151818401526020810190506127ff565b60008484015250505050565b6000601f19601f8301169050919050565b6000612842826127e0565b61284c81856127eb565b935061285c8185602086016127fc565b61286581612826565b840191505092915050565b6000602082019050818103600083015261288a8184612837565b905092915050565b6000819050919050565b6128a581612892565b81146128b057600080fd5b50565b6000813590506128c28161289c565b92915050565b6000602082840312156128de576128dd61271b565b5b60006128ec848285016128b3565b91505092915050565b600073ffffffffffffffffffffffffffffffffffffffff82169050919050565b6000612920826128f5565b9050919050565b61293081612915565b82525050565b600060208201905061294b6000830184612927565b92915050565b61295a81612915565b811461296557600080fd5b50565b60008135905061297781612951565b92915050565b600080604083850312156129945761299361271b565b5b60006129a285828601612968565b92505060206129b3858286016128b3565b9150509250929050565b6129c681612892565b82525050565b60006020820190506129e160008301846129bd565b92915050565b600080600060608486031215612a00576129ff61271b565b5b6000612a0e86828701612968565b9350506020612a1f86828701612968565b9250506040612a30868287016128b3565b9150509250925092565b600060208284031215612a5057612a4f61271b565b5b6000612a5e84828501612968565b91505092915050565b600060ff82169050919050565b612a7d81612a67565b8114612a8857600080fd5b50565b600081359050612a9a81612a74565b92915050565b600060208284031215612ab657612ab561271b565b5b6000612ac484828501612a8b565b91505092915050565b600080fd5b600080fd5b600080fd5b60008083601f840112612af257612af1612acd565b5b8235905067ffffffffffffffff811115612b0f57612b0e612ad2565b5b602083019150836001820283011115612b2b57612b2a612ad7565b5b9250929050565b60008060208385031215612b4957612b4861271b565b5b600083013567ffffffffffffffff811115612b6757612b66612720565b5b612b7385828601612adc565b92509250509250929050565b612b88816127aa565b8114612b9357600080fd5b50565b600081359050612ba581612b7f565b92915050565b60008060408385031215612bc257612bc161271b565b5b6000612bd085828601612968565b9250506020612be185828601612b96565b9150509250929050565b6000819050919050565b612bfe81612beb565b8114612c0957600080fd5b50565b600081359050612c1b81612bf5565b92915050565b600067ffffffffffffffff82169050919050565b612c3e81612c21565b8114612c4957600080fd5b50565b600081359050612c5b81612c35565b92915050565b60008083601f840112612c7757612c76612acd565b5b8235905067ffffffffffffffff811115612c9457612c93612ad2565b5b602083019150836001820283011115612cb057612caf612ad7565b5b9250929050565b600080600080600060808688031215612cd357612cd261271b565b5b6000612ce188828901612c0c565b9550506020612cf288828901612c4c565b9450506040612d0388828901612968565b935050606086013567ffffffffffffffff811115612d2457612d23612720565b5b612d3088828901612c61565b92509250509295509295909350565b600080fd5b7f4e487b7100000000000000000000000000000000000000000000000000000000600052604160045260246000fd5b612d7c82612826565b810181811067ffffffffffffffff82111715612d9b57612d9a612d44565b5b80604052505050565b6000612dae612711565b9050612dba8282612d73565b919050565b600067ffffffffffffffff821115612dda57612dd9612d44565b5b612de382612826565b9050602081019050919050565b82818337600083830152505050565b6000612e12612e0d84612dbf565b612da4565b905082815260208101848484011115612e2e57612e2d612d3f565b5b612e39848285612df0565b509392505050565b600082601f830112612e5657612e55612acd565b5b8135612e66848260208601612dff565b91505092915050565b60008060008060808587031215612e8957612e8861271b565b5b6000612e9787828801612968565b9450506020612ea887828801612968565b9350506040612eb9878288016128b3565b925050606085013567ffffffffffffffff811115612eda57612ed9612720565b5b612ee687828801612e41565b91505092959194509250565b600080600080600060808688031215612f0e57612f0d61271b565b5b6000612f1c88828901612a8b565b9550506020612f2d88828901612c0c565b9450506040612f3e88828901612c4c565b935050606086013567ffffffffffffffff811115612f5f57612f5e612720565b5b612f6b88828901612c61565b92509250509295509295909350565b60008060408385031215612f9157612f9061271b565b5b6000612f9f85828601612968565b9250506020612fb085828601612968565b9150509250929050565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052602260045260246000fd5b6000600282049050600182168061300157607f821691505b60208210810361301457613013612fba565b5b50919050565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052601160045260246000fd5b600061305482612892565b915061305f83612892565b92508282019050808211156130775761307661301a565b5b92915050565b7f4368616d7061676e65436172626f6e3a206d696e74206c696d6974207065722060008201527f77616c6c65742077617320657863656564656400000000000000000000000000602082015250565b60006130d96033836127eb565b91506130e48261307d565b604082019050919050565b60006020820190508181036000830152613108816130cc565b9050919050565b7f4368616d7061676e65436172626f6e3a204d696e7420737570706c79206c696d60008201527f6974207761732065786365656465640000000000000000000000000000000000602082015250565b600061316b602f836127eb565b91506131768261310f565b604082019050919050565b6000602082019050818103600083015261319a8161315e565b9050919050565b7f4e6f7420616c6c6f776564000000000000000000000000000000000000000000600082015250565b60006131d7600b836127eb565b91506131e2826131a1565b602082019050919050565b60006020820190508181036000830152613206816131ca565b9050919050565b60006040820190506132226000830185612927565b61322f60208301846129bd565b9392505050565b600082905092915050565b60008190508160005260206000209050919050565b60006020601f8301049050919050565b600082821b905092915050565b6000600883026132a37fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff82613266565b6132ad8683613266565b95508019841693508086168417925050509392505050565b6000819050919050565b60006132ea6132e56132e084612892565b6132c5565b612892565b9050919050565b6000819050919050565b613304836132cf565b613318613310826132f1565b848454613273565b825550505050565b600090565b61332d613320565b6133388184846132fb565b505050565b5b8181101561335c57613351600082613325565b60018101905061333e565b5050565b601f8211156133a15761337281613241565b61337b84613256565b8101602085101561338a578190505b61339e61339685613256565b83018261333d565b50505b505050565b600082821c905092915050565b60006133c4600019846008026133a6565b1980831691505092915050565b60006133dd83836133b3565b9150826002028217905092915050565b6133f78383613236565b67ffffffffffffffff8111156134105761340f612d44565b5b61341a8254612fe9565b613425828285613360565b6000601f8311600181146134545760008415613442578287013590505b61344c85826133d1565b8655506134b4565b601f19841661346286613241565b60005b8281101561348a57848901358255600182019150602085019450602081019050613465565b868310156134a757848901356134a3601f8916826133b3565b8355505b6001600288020188555050505b50505050505050565b6134c681612beb565b82525050565b6134d581612c21565b82525050565b60006060820190506134f060008301866134bd565b6134fd60208301856134bd565b61350a60408301846134cc565b949350505050565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052602160045260246000fd5b7f4368616d7061676e65436172626f6e3a204e6f7420616c6c6f77656400000000600082015250565b6000613577601c836127eb565b915061358282613541565b602082019050919050565b600060208201905081810360008301526135a68161356a565b9050919050565b7f4368616d7061676e65436172626f6e3a204e465420616c72656164792072656460008201527f65656d6564000000000000000000000000000000000000000000000000000000602082015250565b60006136096025836127eb565b9150613614826135ad565b604082019050919050565b60006020820190508181036000830152613638816135fc565b9050919050565b7f496e76616c6964207369676e6174757265000000000000000000000000000000600082015250565b60006136756011836127eb565b91506136808261363f565b602082019050919050565b600060208201905081810360008301526136a481613668565b9050919050565b60006136c66136c16136bc84612a67565b6132c5565b612892565b9050919050565b6136d6816136ab565b82525050565b60006020820190506136f160008301846136cd565b92915050565b600081905092915050565b600061370d826127e0565b61371781856136f7565b93506137278185602086016127fc565b80840191505092915050565b600061373f8285613702565b915061374b8284613702565b91508190509392505050565b7f4f776e61626c653a206e6577206f776e657220697320746865207a65726f206160008201527f6464726573730000000000000000000000000000000000000000000000000000602082015250565b60006137b36026836127eb565b91506137be82613757565b604082019050919050565b600060208201905081810360008301526137e2816137a6565b9050919050565b7f5061757361626c653a20746f6b656e207472616e73666572207768696c65207060008201527f6175736564000000000000000000000000000000000000000000000000000000602082015250565b60006138456025836127eb565b9150613850826137e9565b604082019050919050565b6000602082019050818103600083015261387481613838565b9050919050565b7f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572600082015250565b60006138b16020836127eb565b91506138bc8261387b565b602082019050919050565b600060208201905081810360008301526138e0816138a4565b9050919050565b600081519050919050565b600082825260208201905092915050565b600061390e826138e7565b61391881856138f2565b93506139288185602086016127fc565b61393181612826565b840191505092915050565b60006080820190506139516000830187612927565b61395e6020830186612927565b61396b60408301856129bd565b818103606083015261397d8184613903565b905095945050505050565b60008151905061399781612751565b92915050565b6000602082840312156139b3576139b261271b565b5b60006139c184828501613988565b91505092915050565b7f5061757361626c653a206e6f7420706175736564000000000000000000000000600082015250565b6000613a006014836127eb565b9150613a0b826139ca565b602082019050919050565b60006020820190508181036000830152613a2f816139f3565b9050919050565b7f5061757361626c653a2070617573656400000000000000000000000000000000600082015250565b6000613a6c6010836127eb565b9150613a7782613a36565b602082019050919050565b60006020820190508181036000830152613a9b81613a5f565b9050919050565b7f1901000000000000000000000000000000000000000000000000000000000000600082015250565b6000613ad86002836136f7565b9150613ae382613aa2565b600282019050919050565b6000819050919050565b613b09613b0482612beb565b613aee565b82525050565b6000613b1a82613acb565b9150613b268285613af8565b602082019150613b368284613af8565b6020820191508190509392505050565b613b4f81612a67565b82525050565b6000608082019050613b6a60008301876134bd565b613b776020830186613b46565b613b8460408301856134bd565b613b9160608301846134bd565b95945050505050565b600060a082019050613baf60008301886134bd565b613bbc60208301876134bd565b613bc960408301866134bd565b613bd660608301856129bd565b613be36080830184612927565b969550505050505056fea2646970667358221220b534d7e3cbd1c027442c64cf214fc46967ca6e6656ba5be217be4e862a9fab6a64736f6c63430008120033

Deployed Bytecode

0x6080604052600436106101ee5760003560e01c806370a082311161010d578063b19960e6116100a0578063c87b56dd1161006f578063c87b56dd146106a1578063e8a3d485146106de578063e985e9c514610709578063f2fde38b14610746578063f8fb491f1461076f576101ee565b8063b19960e6146105f4578063b277caaa1461061f578063b88d4fde1461065c578063be9c6ee114610678576101ee565b80638da5cb5b116100dc5780638da5cb5b1461054c578063938e3d7b1461057757806395d89b41146105a0578063a22cb465146105cb576101ee565b806370a08231146104b8578063715018a6146104f55780638456cb591461050c5780638d0f8cef14610523576101ee565b80633dfcc40a1161018557806355f804b31161015457806355f804b3146103fc5780635c975abb146104255780636352211e146104505780636c0360eb1461048d576101ee565b80633dfcc40a146103635780633f4ba83a146103a057806342842e0e146103b757806343508b05146103d3576101ee565b806318160ddd116101c157806318160ddd146102b457806323b872dd146102df57806332cb6b0c146102fb5780633a602b4d14610326576101ee565b806301ffc9a7146101f357806306fdde0314610230578063081812fc1461025b578063095ea7b314610298575b600080fd5b3480156101ff57600080fd5b5061021a6004803603810190610215919061277d565b610798565b60405161022791906127c5565b60405180910390f35b34801561023c57600080fd5b506102456107f9565b6040516102529190612870565b60405180910390f35b34801561026757600080fd5b50610282600480360381019061027d91906128c8565b61088b565b60405161028f9190612936565b60405180910390f35b6102b260048036038101906102ad919061297d565b61090a565b005b3480156102c057600080fd5b506102c9610a4e565b6040516102d691906129cc565b60405180910390f35b6102f960048036038101906102f491906129e7565b610a65565b005b34801561030757600080fd5b50610310610d87565b60405161031d91906129cc565b60405180910390f35b34801561033257600080fd5b5061034d60048036038101906103489190612a3a565b610d8c565b60405161035a91906129cc565b60405180910390f35b34801561036f57600080fd5b5061038a60048036038101906103859190612aa0565b610da4565b60405161039791906129cc565b60405180910390f35b3480156103ac57600080fd5b506103b5610dbc565b005b6103d160048036038101906103cc91906129e7565b610dce565b005b3480156103df57600080fd5b506103fa60048036038101906103f5919061297d565b610dee565b005b34801561040857600080fd5b50610423600480360381019061041e9190612b32565b61103d565b005b34801561043157600080fd5b5061043a61105b565b60405161044791906127c5565b60405180910390f35b34801561045c57600080fd5b50610477600480360381019061047291906128c8565b611072565b6040516104849190612936565b60405180910390f35b34801561049957600080fd5b506104a2611084565b6040516104af9190612870565b60405180910390f35b3480156104c457600080fd5b506104df60048036038101906104da9190612a3a565b611112565b6040516104ec91906129cc565b60405180910390f35b34801561050157600080fd5b5061050a6111ca565b005b34801561051857600080fd5b506105216111de565b005b34801561052f57600080fd5b5061054a6004803603810190610545919061297d565b6111f0565b005b34801561055857600080fd5b5061056161125e565b60405161056e9190612936565b60405180910390f35b34801561058357600080fd5b5061059e60048036038101906105999190612b32565b611288565b005b3480156105ac57600080fd5b506105b56112a6565b6040516105c29190612870565b60405180910390f35b3480156105d757600080fd5b506105f260048036038101906105ed9190612bab565b611338565b005b34801561060057600080fd5b50610609611443565b60405161061691906129cc565b60405180910390f35b34801561062b57600080fd5b5061064660048036038101906106419190612cb7565b611448565b60405161065391906127c5565b60405180910390f35b61067660048036038101906106719190612e6f565b611562565b005b34801561068457600080fd5b5061069f600480360381019061069a9190612ef2565b6115d5565b005b3480156106ad57600080fd5b506106c860048036038101906106c391906128c8565b611752565b6040516106d59190612870565b60405180910390f35b3480156106ea57600080fd5b506106f36117f0565b6040516107009190612870565b60405180910390f35b34801561071557600080fd5b50610730600480360381019061072b9190612f7a565b611882565b60405161073d91906127c5565b60405180910390f35b34801561075257600080fd5b5061076d60048036038101906107689190612a3a565b611916565b005b34801561077b57600080fd5b5061079660048036038101906107919190612a3a565b611999565b005b6000634906490660e01b7bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916827bffffffffffffffffffffffffffffffffffffffffffffffffffffffff191614806107f257506107f1826119e5565b5b9050919050565b60606002805461080890612fe9565b80601f016020809104026020016040519081016040528092919081815260200182805461083490612fe9565b80156108815780601f1061085657610100808354040283529160200191610881565b820191906000526020600020905b81548152906001019060200180831161086457829003601f168201915b5050505050905090565b600061089682611a77565b6108cc576040517fcf4700e400000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b6006600083815260200190815260200160002060000160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff169050919050565b600061091582611072565b90508073ffffffffffffffffffffffffffffffffffffffff16610936611ad6565b73ffffffffffffffffffffffffffffffffffffffff1614610999576109628161095d611ad6565b611882565b610998576040517fcfb3b94200000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b5b826006600084815260200190815260200160002060000160006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff160217905550818373ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff167f8c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b92560405160405180910390a4505050565b6000610a58611ade565b6001546000540303905090565b6000610a7082611ae7565b90508373ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff1614610ad7576040517fa114810000000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b600080610ae384611bb3565b91509150610af98187610af4611ad6565b611bda565b610b4557610b0e86610b09611ad6565b611882565b610b44576040517f59c896be00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b5b600073ffffffffffffffffffffffffffffffffffffffff168573ffffffffffffffffffffffffffffffffffffffff1603610bab576040517fea553b3400000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b610bb88686866001611c1e565b8015610bc357600082555b600560008773ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600081546001900391905081905550600560008673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000206000815460010191905081905550610c9185610c6d888887611c78565b7c020000000000000000000000000000000000000000000000000000000017611ca0565b600460008681526020019081526020016000208190555060007c0200000000000000000000000000000000000000000000000000000000841603610d175760006001850190506000600460008381526020019081526020016000205403610d15576000548114610d14578360046000838152602001908152602001600020819055505b5b505b838573ffffffffffffffffffffffffffffffffffffffff168773ffffffffffffffffffffffffffffffffffffffff167fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef60405160405180910390a4610d7f8686866001611ccb565b505050505050565b601481565b600d6020528060005260406000206000915090505481565b600c6020528060005260406000206000915090505481565b610dc4611cd1565b610dcc611d4f565b565b610de983838360405180602001604052806000815250611562565b505050565b8181600281600d60008573ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002054610e3d9190613049565b1115610e7e576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610e75906130ef565b60405180910390fd5b82601481610e8a610a4e565b610e949190613049565b1115610ed5576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610ecc90613181565b60405180910390fd5b600b60009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff163373ffffffffffffffffffffffffffffffffffffffff1614610f65576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610f5c906131ed565b60405180910390fd5b610f6f8585611db2565b83600d60008773ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002054610fba9190613049565b600d60008773ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020819055507f0f6798a560793a54c3bcfe86a93cde1e73087d944c0ea20544137d4121396885858560405161102e92919061320d565b60405180910390a15050505050565b611045611cd1565b8181600991826110569291906133ed565b505050565b6000600860149054906101000a900460ff16905090565b600061107d82611ae7565b9050919050565b6009805461109190612fe9565b80601f01602080910402602001604051908101604052809291908181526020018280546110bd90612fe9565b801561110a5780601f106110df5761010080835404028352916020019161110a565b820191906000526020600020905b8154815290600101906020018083116110ed57829003601f168201915b505050505081565b60008073ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff1603611179576040517f8f4eb60400000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b67ffffffffffffffff600560008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002054169050919050565b6111d2611cd1565b6111dc6000611dd0565b565b6111e6611cd1565b6111ee611e96565b565b6111f8611cd1565b80601481611204610a4e565b61120e9190613049565b111561124f576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161124690613181565b60405180910390fd5b6112598383611db2565b505050565b6000600860009054906101000a900473ffffffffffffffffffffffffffffffffffffffff16905090565b611290611cd1565b8181600a91826112a19291906133ed565b505050565b6060600380546112b590612fe9565b80601f01602080910402602001604051908101604052809291908181526020018280546112e190612fe9565b801561132e5780601f106113035761010080835404028352916020019161132e565b820191906000526020600020905b81548152906001019060200180831161131157829003601f168201915b5050505050905090565b8060076000611345611ad6565b73ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060006101000a81548160ff0219169083151502179055508173ffffffffffffffffffffffffffffffffffffffff166113f2611ad6565b73ffffffffffffffffffffffffffffffffffffffff167f17307eab39ab6107e8899845ad3d59bd9653f200f220920489ca2b5937696c318360405161143791906127c5565b60405180910390a35050565b600281565b60008061149e7f9067465cd589b9fb125b4fb7b6fbf2ed93fd28c348210854bbba666b918b612f8888604051602001611483939291906134db565b60405160208183030381529060405280519060200120611ef9565b90506000806114f18387878080601f016020809104026020016040519081016040528093929190818152602001838380828437600081840152601f19601f82011690508083019250505050505050611f13565b915091508673ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff1614801561155457506000600481111561153f5761153e613512565b5b81600481111561155257611551613512565b5b145b935050505095945050505050565b61156d848484610a65565b60008373ffffffffffffffffffffffffffffffffffffffff163b146115cf5761159884848484611f64565b6115ce576040517fd1a57ed600000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b5b50505050565b3373ffffffffffffffffffffffffffffffffffffffff166115f88660ff16611072565b73ffffffffffffffffffffffffffffffffffffffff161461164e576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016116459061358d565b60405180910390fd5b6000600c60008760ff1660ff16815260200190815260200160002054146116aa576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016116a19061361f565b60405180910390fd5b6116b78484338585611448565b6116f6576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016116ed9061368b565b60405180910390fd5b42600c60008760ff1660ff168152602001908152602001600020819055507ff8e1a15aba9398e019f0b49df1a4fde98ee17ae345cb5f6b5e2c27f5033e8ce78560405161174391906136dc565b60405180910390a15050505050565b606061175d82611a77565b611793576040517fa14c4b5000000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b600061179d6120b4565b905060008151036117bd57604051806020016040528060008152506117e8565b806117c784612146565b6040516020016117d8929190613733565b6040516020818303038152906040525b915050919050565b6060600a80546117ff90612fe9565b80601f016020809104026020016040519081016040528092919081815260200182805461182b90612fe9565b80156118785780601f1061184d57610100808354040283529160200191611878565b820191906000526020600020905b81548152906001019060200180831161185b57829003601f168201915b5050505050905090565b6000600760008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060009054906101000a900460ff16905092915050565b61191e611cd1565b600073ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff160361198d576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401611984906137c9565b60405180910390fd5b61199681611dd0565b50565b6119a1611cd1565b80600b60006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff16021790555050565b60006301ffc9a760e01b827bffffffffffffffffffffffffffffffffffffffffffffffffffffffff19161480611a4057506380ac58cd60e01b827bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916145b80611a705750635b5e139f60e01b827bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916145b9050919050565b600081611a82611ade565b11158015611a91575060005482105b8015611acf575060007c0100000000000000000000000000000000000000000000000000000000600460008581526020019081526020016000205416145b9050919050565b600033905090565b60006001905090565b60008082905080611af6611ade565b11611b7c57600054811015611b7b5760006004600083815260200190815260200160002054905060007c0100000000000000000000000000000000000000000000000000000000821603611b79575b60008103611b6f576004600083600190039350838152602001908152602001600020549050611b45565b8092505050611bae565b505b5b6040517fdf2d9b4200000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b919050565b60008060006006600085815260200190815260200160002090508092508254915050915091565b600073ffffffffffffffffffffffffffffffffffffffff8316925073ffffffffffffffffffffffffffffffffffffffff821691508382148383141790509392505050565b611c2a84848484612196565b611c3261105b565b15611c72576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401611c699061385b565b60405180910390fd5b50505050565b60008060e883901c905060e8611c8f86868461219c565b62ffffff16901b9150509392505050565b600073ffffffffffffffffffffffffffffffffffffffff83169250814260a01b178317905092915050565b50505050565b611cd96121a5565b73ffffffffffffffffffffffffffffffffffffffff16611cf761125e565b73ffffffffffffffffffffffffffffffffffffffff1614611d4d576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401611d44906138c7565b60405180910390fd5b565b611d576121ad565b6000600860146101000a81548160ff0219169083151502179055507f5db9ee0a495bf2e6ff9c91a7834c1ba4fdd244a5e8aa4e537bd38aeae4b073aa611d9b6121a5565b604051611da89190612936565b60405180910390a1565b611dcc8282604051806020016040528060008152506121f6565b5050565b6000600860009054906101000a900473ffffffffffffffffffffffffffffffffffffffff16905081600860006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff1602179055508173ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff167f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e060405160405180910390a35050565b611e9e612293565b6001600860146101000a81548160ff0219169083151502179055507f62e78cea01bee320cd4e420270b5ea74000d11b0c9f74754ebdbfc544b05a258611ee26121a5565b604051611eef9190612936565b60405180910390a1565b6000611f0c611f066122dd565b836123f7565b9050919050565b6000806041835103611f545760008060006020860151925060408601519150606086015160001a9050611f488782858561242a565b94509450505050611f5d565b60006002915091505b9250929050565b60008373ffffffffffffffffffffffffffffffffffffffff1663150b7a02611f8a611ad6565b8786866040518563ffffffff1660e01b8152600401611fac949392919061393c565b6020604051808303816000875af1925050508015611fe857506040513d601f19601f82011682018060405250810190611fe5919061399d565b60015b612061573d8060008114612018576040519150601f19603f3d011682016040523d82523d6000602084013e61201d565b606091505b506000815103612059576040517fd1a57ed600000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b805181602001fd5b63150b7a0260e01b7bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916817bffffffffffffffffffffffffffffffffffffffffffffffffffffffff191614915050949350505050565b6060600980546120c390612fe9565b80601f01602080910402602001604051908101604052809291908181526020018280546120ef90612fe9565b801561213c5780601f106121115761010080835404028352916020019161213c565b820191906000526020600020905b81548152906001019060200180831161211f57829003601f168201915b5050505050905090565b606060a060405101806040526020810391506000825281835b60011561218157600184039350600a81066030018453600a810490508061215f575b50828103602084039350808452505050919050565b50505050565b60009392505050565b600033905090565b6121b561105b565b6121f4576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016121eb90613a16565b60405180910390fd5b565b612200838361250c565b60008373ffffffffffffffffffffffffffffffffffffffff163b1461228e57600080549050600083820390505b6122406000868380600101945086611f64565b612276576040517fd1a57ed600000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b81811061222d57816000541461228b57600080fd5b50505b505050565b61229b61105b565b156122db576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016122d290613a82565b60405180910390fd5b565b60007f000000000000000000000000204cdf46ea5b0c27d1ac8fc9e56b44e14359943273ffffffffffffffffffffffffffffffffffffffff163073ffffffffffffffffffffffffffffffffffffffff1614801561235957507f000000000000000000000000000000000000000000000000000000000000000146145b15612386577fadea77d3f7fd33c14a2f46e3e404e2dc61a48cedba3a1877922d726bb09776d290506123f4565b6123f17f8b73c3c69bb8fe3d512ecc4cf759cc79239f7b179b0ffacaa9a75d522b39400f7fd9e4274902883a0f166dfe3cca0afed353f6be6ccc511beab631082d6333663b7fc89efdaa54c0f20c7adf612882df0950f5a951637e0307cdcb4c672f298b8bc66126c7565b90505b90565b6000828260405160200161240c929190613b0f565b60405160208183030381529060405280519060200120905092915050565b6000807f7fffffffffffffffffffffffffffffff5d576e7357a4501ddfe92f46681b20a08360001c1115612465576000600391509150612503565b60006001878787876040516000815260200160405260405161248a9493929190613b55565b6020604051602081039080840390855afa1580156124ac573d6000803e3d6000fd5b505050602060405103519050600073ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff16036124fa57600060019250925050612503565b80600092509250505b94509492505050565b6000805490506000820361254c576040517fb562e8dd00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b6125596000848385611c1e565b600160406001901b178202600560008573ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600082825401925050819055506125d0836125c16000866000611c78565b6125ca85612701565b17611ca0565b6004600083815260200190815260200160002081905550600080838301905073ffffffffffffffffffffffffffffffffffffffff85169150828260007fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef600080a4600183015b81811461267157808360007fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef600080a4600181019050612636565b50600082036126ac576040517f2e07630000000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b8060008190555050506126c26000848385611ccb565b505050565b600083838346306040516020016126e2959493929190613b9a565b6040516020818303038152906040528051906020012090509392505050565b60006001821460e11b9050919050565b6000604051905090565b600080fd5b600080fd5b60007fffffffff0000000000000000000000000000000000000000000000000000000082169050919050565b61275a81612725565b811461276557600080fd5b50565b60008135905061277781612751565b92915050565b6000602082840312156127935761279261271b565b5b60006127a184828501612768565b91505092915050565b60008115159050919050565b6127bf816127aa565b82525050565b60006020820190506127da60008301846127b6565b92915050565b600081519050919050565b600082825260208201905092915050565b60005b8381101561281a5780820151818401526020810190506127ff565b60008484015250505050565b6000601f19601f8301169050919050565b6000612842826127e0565b61284c81856127eb565b935061285c8185602086016127fc565b61286581612826565b840191505092915050565b6000602082019050818103600083015261288a8184612837565b905092915050565b6000819050919050565b6128a581612892565b81146128b057600080fd5b50565b6000813590506128c28161289c565b92915050565b6000602082840312156128de576128dd61271b565b5b60006128ec848285016128b3565b91505092915050565b600073ffffffffffffffffffffffffffffffffffffffff82169050919050565b6000612920826128f5565b9050919050565b61293081612915565b82525050565b600060208201905061294b6000830184612927565b92915050565b61295a81612915565b811461296557600080fd5b50565b60008135905061297781612951565b92915050565b600080604083850312156129945761299361271b565b5b60006129a285828601612968565b92505060206129b3858286016128b3565b9150509250929050565b6129c681612892565b82525050565b60006020820190506129e160008301846129bd565b92915050565b600080600060608486031215612a00576129ff61271b565b5b6000612a0e86828701612968565b9350506020612a1f86828701612968565b9250506040612a30868287016128b3565b9150509250925092565b600060208284031215612a5057612a4f61271b565b5b6000612a5e84828501612968565b91505092915050565b600060ff82169050919050565b612a7d81612a67565b8114612a8857600080fd5b50565b600081359050612a9a81612a74565b92915050565b600060208284031215612ab657612ab561271b565b5b6000612ac484828501612a8b565b91505092915050565b600080fd5b600080fd5b600080fd5b60008083601f840112612af257612af1612acd565b5b8235905067ffffffffffffffff811115612b0f57612b0e612ad2565b5b602083019150836001820283011115612b2b57612b2a612ad7565b5b9250929050565b60008060208385031215612b4957612b4861271b565b5b600083013567ffffffffffffffff811115612b6757612b66612720565b5b612b7385828601612adc565b92509250509250929050565b612b88816127aa565b8114612b9357600080fd5b50565b600081359050612ba581612b7f565b92915050565b60008060408385031215612bc257612bc161271b565b5b6000612bd085828601612968565b9250506020612be185828601612b96565b9150509250929050565b6000819050919050565b612bfe81612beb565b8114612c0957600080fd5b50565b600081359050612c1b81612bf5565b92915050565b600067ffffffffffffffff82169050919050565b612c3e81612c21565b8114612c4957600080fd5b50565b600081359050612c5b81612c35565b92915050565b60008083601f840112612c7757612c76612acd565b5b8235905067ffffffffffffffff811115612c9457612c93612ad2565b5b602083019150836001820283011115612cb057612caf612ad7565b5b9250929050565b600080600080600060808688031215612cd357612cd261271b565b5b6000612ce188828901612c0c565b9550506020612cf288828901612c4c565b9450506040612d0388828901612968565b935050606086013567ffffffffffffffff811115612d2457612d23612720565b5b612d3088828901612c61565b92509250509295509295909350565b600080fd5b7f4e487b7100000000000000000000000000000000000000000000000000000000600052604160045260246000fd5b612d7c82612826565b810181811067ffffffffffffffff82111715612d9b57612d9a612d44565b5b80604052505050565b6000612dae612711565b9050612dba8282612d73565b919050565b600067ffffffffffffffff821115612dda57612dd9612d44565b5b612de382612826565b9050602081019050919050565b82818337600083830152505050565b6000612e12612e0d84612dbf565b612da4565b905082815260208101848484011115612e2e57612e2d612d3f565b5b612e39848285612df0565b509392505050565b600082601f830112612e5657612e55612acd565b5b8135612e66848260208601612dff565b91505092915050565b60008060008060808587031215612e8957612e8861271b565b5b6000612e9787828801612968565b9450506020612ea887828801612968565b9350506040612eb9878288016128b3565b925050606085013567ffffffffffffffff811115612eda57612ed9612720565b5b612ee687828801612e41565b91505092959194509250565b600080600080600060808688031215612f0e57612f0d61271b565b5b6000612f1c88828901612a8b565b9550506020612f2d88828901612c0c565b9450506040612f3e88828901612c4c565b935050606086013567ffffffffffffffff811115612f5f57612f5e612720565b5b612f6b88828901612c61565b92509250509295509295909350565b60008060408385031215612f9157612f9061271b565b5b6000612f9f85828601612968565b9250506020612fb085828601612968565b9150509250929050565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052602260045260246000fd5b6000600282049050600182168061300157607f821691505b60208210810361301457613013612fba565b5b50919050565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052601160045260246000fd5b600061305482612892565b915061305f83612892565b92508282019050808211156130775761307661301a565b5b92915050565b7f4368616d7061676e65436172626f6e3a206d696e74206c696d6974207065722060008201527f77616c6c65742077617320657863656564656400000000000000000000000000602082015250565b60006130d96033836127eb565b91506130e48261307d565b604082019050919050565b60006020820190508181036000830152613108816130cc565b9050919050565b7f4368616d7061676e65436172626f6e3a204d696e7420737570706c79206c696d60008201527f6974207761732065786365656465640000000000000000000000000000000000602082015250565b600061316b602f836127eb565b91506131768261310f565b604082019050919050565b6000602082019050818103600083015261319a8161315e565b9050919050565b7f4e6f7420616c6c6f776564000000000000000000000000000000000000000000600082015250565b60006131d7600b836127eb565b91506131e2826131a1565b602082019050919050565b60006020820190508181036000830152613206816131ca565b9050919050565b60006040820190506132226000830185612927565b61322f60208301846129bd565b9392505050565b600082905092915050565b60008190508160005260206000209050919050565b60006020601f8301049050919050565b600082821b905092915050565b6000600883026132a37fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff82613266565b6132ad8683613266565b95508019841693508086168417925050509392505050565b6000819050919050565b60006132ea6132e56132e084612892565b6132c5565b612892565b9050919050565b6000819050919050565b613304836132cf565b613318613310826132f1565b848454613273565b825550505050565b600090565b61332d613320565b6133388184846132fb565b505050565b5b8181101561335c57613351600082613325565b60018101905061333e565b5050565b601f8211156133a15761337281613241565b61337b84613256565b8101602085101561338a578190505b61339e61339685613256565b83018261333d565b50505b505050565b600082821c905092915050565b60006133c4600019846008026133a6565b1980831691505092915050565b60006133dd83836133b3565b9150826002028217905092915050565b6133f78383613236565b67ffffffffffffffff8111156134105761340f612d44565b5b61341a8254612fe9565b613425828285613360565b6000601f8311600181146134545760008415613442578287013590505b61344c85826133d1565b8655506134b4565b601f19841661346286613241565b60005b8281101561348a57848901358255600182019150602085019450602081019050613465565b868310156134a757848901356134a3601f8916826133b3565b8355505b6001600288020188555050505b50505050505050565b6134c681612beb565b82525050565b6134d581612c21565b82525050565b60006060820190506134f060008301866134bd565b6134fd60208301856134bd565b61350a60408301846134cc565b949350505050565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052602160045260246000fd5b7f4368616d7061676e65436172626f6e3a204e6f7420616c6c6f77656400000000600082015250565b6000613577601c836127eb565b915061358282613541565b602082019050919050565b600060208201905081810360008301526135a68161356a565b9050919050565b7f4368616d7061676e65436172626f6e3a204e465420616c72656164792072656460008201527f65656d6564000000000000000000000000000000000000000000000000000000602082015250565b60006136096025836127eb565b9150613614826135ad565b604082019050919050565b60006020820190508181036000830152613638816135fc565b9050919050565b7f496e76616c6964207369676e6174757265000000000000000000000000000000600082015250565b60006136756011836127eb565b91506136808261363f565b602082019050919050565b600060208201905081810360008301526136a481613668565b9050919050565b60006136c66136c16136bc84612a67565b6132c5565b612892565b9050919050565b6136d6816136ab565b82525050565b60006020820190506136f160008301846136cd565b92915050565b600081905092915050565b600061370d826127e0565b61371781856136f7565b93506137278185602086016127fc565b80840191505092915050565b600061373f8285613702565b915061374b8284613702565b91508190509392505050565b7f4f776e61626c653a206e6577206f776e657220697320746865207a65726f206160008201527f6464726573730000000000000000000000000000000000000000000000000000602082015250565b60006137b36026836127eb565b91506137be82613757565b604082019050919050565b600060208201905081810360008301526137e2816137a6565b9050919050565b7f5061757361626c653a20746f6b656e207472616e73666572207768696c65207060008201527f6175736564000000000000000000000000000000000000000000000000000000602082015250565b60006138456025836127eb565b9150613850826137e9565b604082019050919050565b6000602082019050818103600083015261387481613838565b9050919050565b7f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572600082015250565b60006138b16020836127eb565b91506138bc8261387b565b602082019050919050565b600060208201905081810360008301526138e0816138a4565b9050919050565b600081519050919050565b600082825260208201905092915050565b600061390e826138e7565b61391881856138f2565b93506139288185602086016127fc565b61393181612826565b840191505092915050565b60006080820190506139516000830187612927565b61395e6020830186612927565b61396b60408301856129bd565b818103606083015261397d8184613903565b905095945050505050565b60008151905061399781612751565b92915050565b6000602082840312156139b3576139b261271b565b5b60006139c184828501613988565b91505092915050565b7f5061757361626c653a206e6f7420706175736564000000000000000000000000600082015250565b6000613a006014836127eb565b9150613a0b826139ca565b602082019050919050565b60006020820190508181036000830152613a2f816139f3565b9050919050565b7f5061757361626c653a2070617573656400000000000000000000000000000000600082015250565b6000613a6c6010836127eb565b9150613a7782613a36565b602082019050919050565b60006020820190508181036000830152613a9b81613a5f565b9050919050565b7f1901000000000000000000000000000000000000000000000000000000000000600082015250565b6000613ad86002836136f7565b9150613ae382613aa2565b600282019050919050565b6000819050919050565b613b09613b0482612beb565b613aee565b82525050565b6000613b1a82613acb565b9150613b268285613af8565b602082019150613b368284613af8565b6020820191508190509392505050565b613b4f81612a67565b82525050565b6000608082019050613b6a60008301876134bd565b613b776020830186613b46565b613b8460408301856134bd565b613b9160608301846134bd565b95945050505050565b600060a082019050613baf60008301886134bd565b613bbc60208301876134bd565b613bc960408301866134bd565b613bd660608301856129bd565b613be36080830184612927565b969550505050505056fea2646970667358221220b534d7e3cbd1c027442c64cf214fc46967ca6e6656ba5be217be4e862a9fab6a64736f6c63430008120033

Deployed Bytecode Sourcemap

162404:5364:0:-:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;167524:241;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;91027:100;;;;;;;;;;;;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;97518:218;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;96951:408;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;86778:323;;;;;;;;;;;;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;101157:2825;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;162522:39;;;;;;;;;;;;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;162900:50;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;162801;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;166354:65;;;;;;;;;;;;;:::i;:::-;;104078:193;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;163702:338;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;166548:92;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;68039:86;;;;;;;;;;;;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;92420:152;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;162640:21;;;;;;;;;;;;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;87962:233;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;70904:103;;;;;;;;;;;;;:::i;:::-;;166285:61;;;;;;;;;;;;;:::i;:::-;;164048:149;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;70256:87;;;;;;;;;;;;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;166648:102;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;91203:104;;;;;;;;;;;;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;98076:234;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;162568:47;;;;;;;;;;;;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;164755:601;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;104869:407;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;165650:596;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;91413:318;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;166427:98;;;;;;;;;;;;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;98467:164;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;71162:201;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;166758:108;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;167524:241;167634:4;167693:10;167686:18;;167671:33;;;:11;:33;;;;:86;;;;167721:36;167745:11;167721:23;:36::i;:::-;167671:86;167651:106;;167524:241;;;:::o;91027:100::-;91081:13;91114:5;91107:12;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;91027:100;:::o;97518:218::-;97594:7;97619:16;97627:7;97619;:16::i;:::-;97614:64;;97644:34;;;;;;;;;;;;;;97614:64;97698:15;:24;97714:7;97698:24;;;;;;;;;;;:30;;;;;;;;;;;;97691:37;;97518:218;;;:::o;96951:408::-;97040:13;97056:16;97064:7;97056;:16::i;:::-;97040:32;;97112:5;97089:28;;:19;:17;:19::i;:::-;:28;;;97085:175;;97137:44;97154:5;97161:19;:17;:19::i;:::-;97137:16;:44::i;:::-;97132:128;;97209:35;;;;;;;;;;;;;;97132:128;97085:175;97305:2;97272:15;:24;97288:7;97272:24;;;;;;;;;;;:30;;;:35;;;;;;;;;;;;;;;;;;97343:7;97339:2;97323:28;;97332:5;97323:28;;;;;;;;;;;;97029:330;96951:408;;:::o;86778:323::-;86839:7;87067:15;:13;:15::i;:::-;87052:12;;87036:13;;:28;:46;87029:53;;86778:323;:::o;101157:2825::-;101299:27;101329;101348:7;101329:18;:27::i;:::-;101299:57;;101414:4;101373:45;;101389:19;101373:45;;;101369:86;;101427:28;;;;;;;;;;;;;;101369:86;101469:27;101498:23;101525:35;101552:7;101525:26;:35::i;:::-;101468:92;;;;101660:68;101685:15;101702:4;101708:19;:17;:19::i;:::-;101660:24;:68::i;:::-;101655:180;;101748:43;101765:4;101771:19;:17;:19::i;:::-;101748:16;:43::i;:::-;101743:92;;101800:35;;;;;;;;;;;;;;101743:92;101655:180;101866:1;101852:16;;:2;:16;;;101848:52;;101877:23;;;;;;;;;;;;;;101848:52;101913:43;101935:4;101941:2;101945:7;101954:1;101913:21;:43::i;:::-;102049:15;102046:160;;;102189:1;102168:19;102161:30;102046:160;102586:18;:24;102605:4;102586:24;;;;;;;;;;;;;;;;102584:26;;;;;;;;;;;;102655:18;:22;102674:2;102655:22;;;;;;;;;;;;;;;;102653:24;;;;;;;;;;;102977:146;103014:2;103063:45;103078:4;103084:2;103088:19;103063:14;:45::i;:::-;83177:8;103035:73;102977:18;:146::i;:::-;102948:17;:26;102966:7;102948:26;;;;;;;;;;;:175;;;;103294:1;83177:8;103243:19;:47;:52;103239:627;;103316:19;103348:1;103338:7;:11;103316:33;;103505:1;103471:17;:30;103489:11;103471:30;;;;;;;;;;;;:35;103467:384;;103609:13;;103594:11;:28;103590:242;;103789:19;103756:17;:30;103774:11;103756:30;;;;;;;;;;;:52;;;;103590:242;103467:384;103297:569;103239:627;103913:7;103909:2;103894:27;;103903:4;103894:27;;;;;;;;;;;;103932:42;103953:4;103959:2;103963:7;103972:1;103932:20;:42::i;:::-;101288:2694;;;101157:2825;;;:::o;162522:39::-;162559:2;162522:39;:::o;162900:50::-;;;;;;;;;;;;;;;;;:::o;162801:::-;;;;;;;;;;;;;;;;;:::o;166354:65::-;70142:13;:11;:13::i;:::-;166401:10:::1;:8;:10::i;:::-;166354:65::o:0;104078:193::-;104224:39;104241:4;104247:2;104251:7;104224:39;;;;;;;;;;;;:16;:39::i;:::-;104078:193;;;:::o;163702:338::-;163805:3;163810:7;162614:1;163242:7;163214:15;:25;163230:8;163214:25;;;;;;;;;;;;;;;;:35;;;;:::i;:::-;:58;;163192:159;;;;;;;;;;;;:::i;:::-;;;;;;;;;163831:7:::1;162559:2;163466:7;163450:13;:11;:13::i;:::-;:23;;;;:::i;:::-;:37;;163428:134;;;;;;;;;;;;:::i;:::-;;;;;;;;;163873:11:::2;;;;;;;;;;;163859:25;;:10;:25;;;163851:49;;;;;;;;;;;;:::i;:::-;;;;;;;;;163911:23;163921:3;163926:7;163911:9;:23::i;:::-;163991:7;163968:15;:20;163984:3;163968:20;;;;;;;;;;;;;;;;:30;;;;:::i;:::-;163945:15;:20;163961:3;163945:20;;;;;;;;;;;;;;;:53;;;;164014:18;164019:3;164024:7;164014:18;;;;;;;:::i;:::-;;;;;;;;163362:1:::1;163702:338:::0;;;;:::o;166548:92::-;70142:13;:11;:13::i;:::-;166629:3:::1;;166619:7;:13;;;;;;;:::i;:::-;;166548:92:::0;;:::o;68039:86::-;68086:4;68110:7;;;;;;;;;;;68103:14;;68039:86;:::o;92420:152::-;92492:7;92535:27;92554:7;92535:18;:27::i;:::-;92512:52;;92420:152;;;:::o;162640:21::-;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::o;87962:233::-;88034:7;88075:1;88058:19;;:5;:19;;;88054:60;;88086:28;;;;;;;;;;;;;;88054:60;82121:13;88132:18;:25;88151:5;88132:25;;;;;;;;;;;;;;;;:55;88125:62;;87962:233;;;:::o;70904:103::-;70142:13;:11;:13::i;:::-;70969:30:::1;70996:1;70969:18;:30::i;:::-;70904:103::o:0;166285:61::-;70142:13;:11;:13::i;:::-;166330:8:::1;:6;:8::i;:::-;166285:61::o:0;164048:149::-;70142:13;:11;:13::i;:::-;164146:7:::1;162559:2;163466:7;163450:13;:11;:13::i;:::-;:23;;;;:::i;:::-;:37;;163428:134;;;;;;;;;;;;:::i;:::-;;;;;;;;;164166:23:::2;164176:3;164181:7;164166:9;:23::i;:::-;70166:1:::1;164048:149:::0;;:::o;70256:87::-;70302:7;70329:6;;;;;;;;;;;70322:13;;70256:87;:::o;166648:102::-;70142:13;:11;:13::i;:::-;166739:3:::1;;166723:13;:19;;;;;;;:::i;:::-;;166648:102:::0;;:::o;91203:104::-;91259:13;91292:7;91285:14;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;91203:104;:::o;98076:234::-;98223:8;98171:18;:39;98190:19;:17;:19::i;:::-;98171:39;;;;;;;;;;;;;;;:49;98211:8;98171:49;;;;;;;;;;;;;;;;:60;;;;;;;;;;;;;;;;;;98283:8;98247:55;;98262:19;:17;:19::i;:::-;98247:55;;;98293:8;98247:55;;;;;;:::i;:::-;;;;;;;;98076:234;;:::o;162568:47::-;162614:1;162568:47;:::o;164755:601::-;164925:4;164942:14;164959:239;164355:99;165113:8;165144:9;165018:154;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;164990:197;;;;;;164959:16;:239::i;:::-;164942:256;;165210:9;165221:20;165245:35;165262:6;165270:9;;165245:35;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:16;:35::i;:::-;165209:71;;;;165304:6;165299:11;;:1;:11;;;165298:50;;;;;165321:26;165316:31;;;;;;;;:::i;:::-;;:1;:31;;;;;;;;:::i;:::-;;;165298:50;165291:57;;;;;164755:601;;;;;;;:::o;104869:407::-;105044:31;105057:4;105063:2;105067:7;105044:12;:31::i;:::-;105108:1;105090:2;:14;;;:19;105086:183;;105129:56;105160:4;105166:2;105170:7;105179:5;105129:30;:56::i;:::-;105124:145;;105213:40;;;;;;;;;;;;;;105124:145;105086:183;104869:407;;;;:::o;165650:596::-;165832:10;165812:30;;:16;165820:7;165812:16;;:7;:16::i;:::-;:30;;;165804:71;;;;;;;;;;;;:::i;:::-;;;;;;;;;165938:1;165908:17;:26;165926:7;165908:26;;;;;;;;;;;;;;;;:31;165886:118;;;;;;;;;;;;:::i;:::-;;;;;;;;;166037:62;166056:8;166066:9;166077:10;166089:9;;166037:18;:62::i;:::-;166015:129;;;;;;;;;;;;:::i;:::-;;;;;;;;;166184:15;166155:17;:26;166173:7;166155:26;;;;;;;;;;;;;;;:44;;;;166215:23;166230:7;166215:23;;;;;;:::i;:::-;;;;;;;;165650:596;;;;;:::o;91413:318::-;91486:13;91517:16;91525:7;91517;:16::i;:::-;91512:59;;91542:29;;;;;;;;;;;;;;91512:59;91584:21;91608:10;:8;:10::i;:::-;91584:34;;91661:1;91642:7;91636:21;:26;:87;;;;;;;;;;;;;;;;;91689:7;91698:18;91708:7;91698:9;:18::i;:::-;91672:45;;;;;;;;;:::i;:::-;;;;;;;;;;;;;91636:87;91629:94;;;91413:318;;;:::o;166427:98::-;166471:13;166504;166497:20;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;166427:98;:::o;98467:164::-;98564:4;98588:18;:25;98607:5;98588:25;;;;;;;;;;;;;;;:35;98614:8;98588:35;;;;;;;;;;;;;;;;;;;;;;;;;98581:42;;98467:164;;;;:::o;71162:201::-;70142:13;:11;:13::i;:::-;71271:1:::1;71251:22;;:8;:22;;::::0;71243:73:::1;;;;;;;;;;;;:::i;:::-;;;;;;;;;71327:28;71346:8;71327:18;:28::i;:::-;71162:201:::0;:::o;166758:108::-;70142:13;:11;:13::i;:::-;166846:12:::1;166832:11;;:26;;;;;;;;;;;;;;;;;;166758:108:::0;:::o;90125:639::-;90210:4;90549:10;90534:25;;:11;:25;;;;:102;;;;90626:10;90611:25;;:11;:25;;;;90534:102;:179;;;;90703:10;90688:25;;:11;:25;;;;90534:179;90514:199;;90125:639;;;:::o;98889:282::-;98954:4;99010:7;98991:15;:13;:15::i;:::-;:26;;:66;;;;;99044:13;;99034:7;:23;98991:66;:153;;;;;99143:1;82897:8;99095:17;:26;99113:7;99095:26;;;;;;;;;;;;:44;:49;98991:153;98971:173;;98889:282;;;:::o;121197:105::-;121257:7;121284:10;121277:17;;121197:105;:::o;166999:160::-;167118:7;167150:1;167143:8;;166999:160;:::o;93575:1275::-;93642:7;93662:12;93677:7;93662:22;;93745:4;93726:15;:13;:15::i;:::-;:23;93722:1061;;93779:13;;93772:4;:20;93768:1015;;;93817:14;93834:17;:23;93852:4;93834:23;;;;;;;;;;;;93817:40;;93951:1;82897:8;93923:6;:24;:29;93919:845;;94588:113;94605:1;94595:6;:11;94588:113;;94648:17;:25;94666:6;;;;;;;94648:25;;;;;;;;;;;;94639:34;;94588:113;;;94734:6;94727:13;;;;;;93919:845;93794:989;93768:1015;93722:1061;94811:31;;;;;;;;;;;;;;93575:1275;;;;:::o;100052:485::-;100154:27;100183:23;100224:38;100265:15;:24;100281:7;100265:24;;;;;;;;;;;100224:65;;100442:18;100419:41;;100499:19;100493:26;100474:45;;100404:126;100052:485;;;:::o;99280:659::-;99429:11;99594:16;99587:5;99583:28;99574:37;;99754:16;99743:9;99739:32;99726:45;;99904:15;99893:9;99890:30;99882:5;99871:9;99868:20;99865:56;99855:66;;99280:659;;;;;:::o;167167:325::-;167353:61;167381:4;167387:2;167391:12;167405:8;167353:27;:61::i;:::-;167434:8;:6;:8::i;:::-;167433:9;167425:59;;;;;;;;;;;;:::i;:::-;;;;;;;;;167167:325;;;;:::o;120506:311::-;120641:7;120661:16;83301:3;120687:19;:41;;120661:68;;83301:3;120755:31;120766:4;120772:2;120776:9;120755:10;:31::i;:::-;120747:40;;:62;;120740:69;;;120506:311;;;;;:::o;95398:450::-;95478:14;95646:16;95639:5;95635:28;95626:37;;95823:5;95809:11;95784:23;95780:41;95777:52;95770:5;95767:63;95757:73;;95398:450;;;;:::o;106762:158::-;;;;;:::o;70421:132::-;70496:12;:10;:12::i;:::-;70485:23;;:7;:5;:7::i;:::-;:23;;;70477:68;;;;;;;;;;;;:::i;:::-;;;;;;;;;70421:132::o;68894:120::-;67903:16;:14;:16::i;:::-;68963:5:::1;68953:7;;:15;;;;;;;;;;;;;;;;;;68984:22;68993:12;:10;:12::i;:::-;68984:22;;;;;;:::i;:::-;;;;;;;;68894:120::o:0;115029:112::-;115106:27;115116:2;115120:8;115106:27;;;;;;;;;;;;:9;:27::i;:::-;115029:112;;:::o;71523:191::-;71597:16;71616:6;;;;;;;;;;;71597:25;;71642:8;71633:6;;:17;;;;;;;;;;;;;;;;;;71697:8;71666:40;;71687:8;71666:40;;;;;;;;;;;;71586:128;71523:191;:::o;68635:118::-;67644:19;:17;:19::i;:::-;68705:4:::1;68695:7;;:14;;;;;;;;;;;;;;;;;;68725:20;68732:12;:10;:12::i;:::-;68725:20;;;;;;:::i;:::-;;;;;;;;68635:118::o:0;152326:167::-;152403:7;152430:55;152452:20;:18;:20::i;:::-;152474:10;152430:21;:55::i;:::-;152423:62;;152326:167;;;:::o;141312:747::-;141393:7;141402:12;141451:2;141431:9;:16;:22;141427:625;;141470:9;141494;141518:7;141775:4;141764:9;141760:20;141754:27;141749:32;;141825:4;141814:9;141810:20;141804:27;141799:32;;141883:4;141872:9;141868:20;141862:27;141859:1;141854:36;141849:41;;141926:25;141937:4;141943:1;141946;141949;141926:10;:25::i;:::-;141919:32;;;;;;;;;141427:625;142000:1;142004:35;141984:56;;;;141312:747;;;;;;:::o;107360:716::-;107523:4;107569:2;107544:45;;;107590:19;:17;:19::i;:::-;107611:4;107617:7;107626:5;107544:88;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;107540:529;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;107844:1;107827:6;:13;:18;107823:235;;107873:40;;;;;;;;;;;;;;107823:235;108016:6;108010:13;108001:6;107997:2;107993:15;107986:38;107540:529;107713:54;;;107703:64;;;:6;:64;;;;107696:71;;;107360:716;;;;;;:::o;166874:100::-;166926:13;166959:7;166952:14;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;166874:100;:::o;121404:1745::-;121469:17;121903:4;121896;121890:11;121886:22;121995:1;121989:4;121982:15;122070:4;122067:1;122063:12;122056:19;;122152:1;122147:3;122140:14;122256:3;122495:5;122477:428;122503:1;122477:428;;;122543:1;122538:3;122534:11;122527:18;;122714:2;122708:4;122704:13;122700:2;122696:22;122691:3;122683:36;122808:2;122802:4;122798:13;122790:21;;122875:4;122477:428;122865:25;122477:428;122481:21;122944:3;122939;122935:13;123059:4;123054:3;123050:14;123043:21;;123124:6;123119:3;123112:19;121508:1634;;;121404:1745;;;:::o;105938:159::-;;;;;:::o;120207:147::-;120344:6;120207:147;;;;;:::o;66152:98::-;66205:7;66232:10;66225:17;;66152:98;:::o;68383:108::-;68450:8;:6;:8::i;:::-;68442:41;;;;;;;;;;;;:::i;:::-;;;;;;;;;68383:108::o;114256:689::-;114387:19;114393:2;114397:8;114387:5;:19::i;:::-;114466:1;114448:2;:14;;;:19;114444:483;;114488:11;114502:13;;114488:27;;114534:13;114556:8;114550:3;:14;114534:30;;114583:233;114614:62;114653:1;114657:2;114661:7;;;;;;114670:5;114614:30;:62::i;:::-;114609:167;;114712:40;;;;;;;;;;;;;;114609:167;114811:3;114803:5;:11;114583:233;;114898:3;114881:13;;:20;114877:34;;114903:8;;;114877:34;114469:458;;114444:483;114256:689;;;:::o;68198:108::-;68269:8;:6;:8::i;:::-;68268:9;68260:38;;;;;;;;;;;;:::i;:::-;;;;;;;;;68198:108::o;151099:314::-;151152:7;151193:12;151176:29;;151184:4;151176:29;;;:66;;;;;151226:16;151209:13;:33;151176:66;151172:234;;;151266:24;151259:31;;;;151172:234;151330:64;151352:10;151364:12;151378:15;151330:21;:64::i;:::-;151323:71;;151099:314;;:::o;147663:196::-;147756:7;147822:15;147839:10;147793:57;;;;;;;;;:::i;:::-;;;;;;;;;;;;;147783:68;;;;;;147776:75;;147663:196;;;;:::o;144313:1520::-;144444:7;144453:12;145378:66;145373:1;145365:10;;:79;145361:163;;;145477:1;145481:30;145461:51;;;;;;145361:163;145621:14;145638:24;145648:4;145654:1;145657;145660;145638:24;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;145621:41;;145695:1;145677:20;;:6;:20;;;145673:103;;145730:1;145734:29;145714:50;;;;;;;145673:103;145796:6;145804:20;145788:37;;;;;144313:1520;;;;;;;;:::o;108538:2966::-;108611:20;108634:13;;108611:36;;108674:1;108662:8;:13;108658:44;;108684:18;;;;;;;;;;;;;;108658:44;108715:61;108745:1;108749:2;108753:12;108767:8;108715:21;:61::i;:::-;109259:1;82259:2;109229:1;:26;;109228:32;109216:8;:45;109190:18;:22;109209:2;109190:22;;;;;;;;;;;;;;;;:71;;;;;;;;;;;109538:139;109575:2;109629:33;109652:1;109656:2;109660:1;109629:14;:33::i;:::-;109596:30;109617:8;109596:20;:30::i;:::-;:66;109538:18;:139::i;:::-;109504:17;:31;109522:12;109504:31;;;;;;;;;;;:173;;;;109694:16;109725:11;109754:8;109739:12;:23;109725:37;;110275:16;110271:2;110267:25;110255:37;;110647:12;110607:8;110566:1;110504:25;110445:1;110384;110357:335;111018:1;111004:12;111000:20;110958:346;111059:3;111050:7;111047:16;110958:346;;111277:7;111267:8;111264:1;111237:25;111234:1;111231;111226:59;111112:1;111103:7;111099:15;111088:26;;110958:346;;;110962:77;111349:1;111337:8;:13;111333:45;;111359:19;;;;;;;;;;;;;;111333:45;111411:3;111395:13;:19;;;;108964:2462;;111436:60;111465:1;111469:2;111473:12;111487:8;111436:20;:60::i;:::-;108600:2904;108538:2966;;:::o;151421:263::-;151565:7;151613:8;151623;151633:11;151646:13;151669:4;151602:73;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;151592:84;;;;;;151585:91;;151421:263;;;;;:::o;95950:324::-;96020:14;96253:1;96243:8;96240:15;96214:24;96210:46;96200:56;;95950:324;;;:::o;7:75:1:-;40:6;73:2;67:9;57:19;;7:75;:::o;88:117::-;197:1;194;187:12;211:117;320:1;317;310:12;334:149;370:7;410:66;403:5;399:78;388:89;;334:149;;;:::o;489:120::-;561:23;578:5;561:23;:::i;:::-;554:5;551:34;541:62;;599:1;596;589:12;541:62;489:120;:::o;615:137::-;660:5;698:6;685:20;676:29;;714:32;740:5;714:32;:::i;:::-;615:137;;;;:::o;758:327::-;816:6;865:2;853:9;844:7;840:23;836:32;833:119;;;871:79;;:::i;:::-;833:119;991:1;1016:52;1060:7;1051:6;1040:9;1036:22;1016:52;:::i;:::-;1006:62;;962:116;758:327;;;;:::o;1091:90::-;1125:7;1168:5;1161:13;1154:21;1143:32;;1091:90;;;:::o;1187:109::-;1268:21;1283:5;1268:21;:::i;:::-;1263:3;1256:34;1187:109;;:::o;1302:210::-;1389:4;1427:2;1416:9;1412:18;1404:26;;1440:65;1502:1;1491:9;1487:17;1478:6;1440:65;:::i;:::-;1302:210;;;;:::o;1518:99::-;1570:6;1604:5;1598:12;1588:22;;1518:99;;;:::o;1623:169::-;1707:11;1741:6;1736:3;1729:19;1781:4;1776:3;1772:14;1757:29;;1623:169;;;;:::o;1798:246::-;1879:1;1889:113;1903:6;1900:1;1897:13;1889:113;;;1988:1;1983:3;1979:11;1973:18;1969:1;1964:3;1960:11;1953:39;1925:2;1922:1;1918:10;1913:15;;1889:113;;;2036:1;2027:6;2022:3;2018:16;2011:27;1860:184;1798:246;;;:::o;2050:102::-;2091:6;2142:2;2138:7;2133:2;2126:5;2122:14;2118:28;2108:38;;2050:102;;;:::o;2158:377::-;2246:3;2274:39;2307:5;2274:39;:::i;:::-;2329:71;2393:6;2388:3;2329:71;:::i;:::-;2322:78;;2409:65;2467:6;2462:3;2455:4;2448:5;2444:16;2409:65;:::i;:::-;2499:29;2521:6;2499:29;:::i;:::-;2494:3;2490:39;2483:46;;2250:285;2158:377;;;;:::o;2541:313::-;2654:4;2692:2;2681:9;2677:18;2669:26;;2741:9;2735:4;2731:20;2727:1;2716:9;2712:17;2705:47;2769:78;2842:4;2833:6;2769:78;:::i;:::-;2761:86;;2541:313;;;;:::o;2860:77::-;2897:7;2926:5;2915:16;;2860:77;;;:::o;2943:122::-;3016:24;3034:5;3016:24;:::i;:::-;3009:5;3006:35;2996:63;;3055:1;3052;3045:12;2996:63;2943:122;:::o;3071:139::-;3117:5;3155:6;3142:20;3133:29;;3171:33;3198:5;3171:33;:::i;:::-;3071:139;;;;:::o;3216:329::-;3275:6;3324:2;3312:9;3303:7;3299:23;3295:32;3292:119;;;3330:79;;:::i;:::-;3292:119;3450:1;3475:53;3520:7;3511:6;3500:9;3496:22;3475:53;:::i;:::-;3465:63;;3421:117;3216:329;;;;:::o;3551:126::-;3588:7;3628:42;3621:5;3617:54;3606:65;;3551:126;;;:::o;3683:96::-;3720:7;3749:24;3767:5;3749:24;:::i;:::-;3738:35;;3683:96;;;:::o;3785:118::-;3872:24;3890:5;3872:24;:::i;:::-;3867:3;3860:37;3785:118;;:::o;3909:222::-;4002:4;4040:2;4029:9;4025:18;4017:26;;4053:71;4121:1;4110:9;4106:17;4097:6;4053:71;:::i;:::-;3909:222;;;;:::o;4137:122::-;4210:24;4228:5;4210:24;:::i;:::-;4203:5;4200:35;4190:63;;4249:1;4246;4239:12;4190:63;4137:122;:::o;4265:139::-;4311:5;4349:6;4336:20;4327:29;;4365:33;4392:5;4365:33;:::i;:::-;4265:139;;;;:::o;4410:474::-;4478:6;4486;4535:2;4523:9;4514:7;4510:23;4506:32;4503:119;;;4541:79;;:::i;:::-;4503:119;4661:1;4686:53;4731:7;4722:6;4711:9;4707:22;4686:53;:::i;:::-;4676:63;;4632:117;4788:2;4814:53;4859:7;4850:6;4839:9;4835:22;4814:53;:::i;:::-;4804:63;;4759:118;4410:474;;;;;:::o;4890:118::-;4977:24;4995:5;4977:24;:::i;:::-;4972:3;4965:37;4890:118;;:::o;5014:222::-;5107:4;5145:2;5134:9;5130:18;5122:26;;5158:71;5226:1;5215:9;5211:17;5202:6;5158:71;:::i;:::-;5014:222;;;;:::o;5242:619::-;5319:6;5327;5335;5384:2;5372:9;5363:7;5359:23;5355:32;5352:119;;;5390:79;;:::i;:::-;5352:119;5510:1;5535:53;5580:7;5571:6;5560:9;5556:22;5535:53;:::i;:::-;5525:63;;5481:117;5637:2;5663:53;5708:7;5699:6;5688:9;5684:22;5663:53;:::i;:::-;5653:63;;5608:118;5765:2;5791:53;5836:7;5827:6;5816:9;5812:22;5791:53;:::i;:::-;5781:63;;5736:118;5242:619;;;;;:::o;5867:329::-;5926:6;5975:2;5963:9;5954:7;5950:23;5946:32;5943:119;;;5981:79;;:::i;:::-;5943:119;6101:1;6126:53;6171:7;6162:6;6151:9;6147:22;6126:53;:::i;:::-;6116:63;;6072:117;5867:329;;;;:::o;6202:86::-;6237:7;6277:4;6270:5;6266:16;6255:27;;6202:86;;;:::o;6294:118::-;6365:22;6381:5;6365:22;:::i;:::-;6358:5;6355:33;6345:61;;6402:1;6399;6392:12;6345:61;6294:118;:::o;6418:135::-;6462:5;6500:6;6487:20;6478:29;;6516:31;6541:5;6516:31;:::i;:::-;6418:135;;;;:::o;6559:325::-;6616:6;6665:2;6653:9;6644:7;6640:23;6636:32;6633:119;;;6671:79;;:::i;:::-;6633:119;6791:1;6816:51;6859:7;6850:6;6839:9;6835:22;6816:51;:::i;:::-;6806:61;;6762:115;6559:325;;;;:::o;6890:117::-;6999:1;6996;6989:12;7013:117;7122:1;7119;7112:12;7136:117;7245:1;7242;7235:12;7273:553;7331:8;7341:6;7391:3;7384:4;7376:6;7372:17;7368:27;7358:122;;7399:79;;:::i;:::-;7358:122;7512:6;7499:20;7489:30;;7542:18;7534:6;7531:30;7528:117;;;7564:79;;:::i;:::-;7528:117;7678:4;7670:6;7666:17;7654:29;;7732:3;7724:4;7716:6;7712:17;7702:8;7698:32;7695:41;7692:128;;;7739:79;;:::i;:::-;7692:128;7273:553;;;;;:::o;7832:529::-;7903:6;7911;7960:2;7948:9;7939:7;7935:23;7931:32;7928:119;;;7966:79;;:::i;:::-;7928:119;8114:1;8103:9;8099:17;8086:31;8144:18;8136:6;8133:30;8130:117;;;8166:79;;:::i;:::-;8130:117;8279:65;8336:7;8327:6;8316:9;8312:22;8279:65;:::i;:::-;8261:83;;;;8057:297;7832:529;;;;;:::o;8367:116::-;8437:21;8452:5;8437:21;:::i;:::-;8430:5;8427:32;8417:60;;8473:1;8470;8463:12;8417:60;8367:116;:::o;8489:133::-;8532:5;8570:6;8557:20;8548:29;;8586:30;8610:5;8586:30;:::i;:::-;8489:133;;;;:::o;8628:468::-;8693:6;8701;8750:2;8738:9;8729:7;8725:23;8721:32;8718:119;;;8756:79;;:::i;:::-;8718:119;8876:1;8901:53;8946:7;8937:6;8926:9;8922:22;8901:53;:::i;:::-;8891:63;;8847:117;9003:2;9029:50;9071:7;9062:6;9051:9;9047:22;9029:50;:::i;:::-;9019:60;;8974:115;8628:468;;;;;:::o;9102:77::-;9139:7;9168:5;9157:16;;9102:77;;;:::o;9185:122::-;9258:24;9276:5;9258:24;:::i;:::-;9251:5;9248:35;9238:63;;9297:1;9294;9287:12;9238:63;9185:122;:::o;9313:139::-;9359:5;9397:6;9384:20;9375:29;;9413:33;9440:5;9413:33;:::i;:::-;9313:139;;;;:::o;9458:101::-;9494:7;9534:18;9527:5;9523:30;9512:41;;9458:101;;;:::o;9565:120::-;9637:23;9654:5;9637:23;:::i;:::-;9630:5;9627:34;9617:62;;9675:1;9672;9665:12;9617:62;9565:120;:::o;9691:137::-;9736:5;9774:6;9761:20;9752:29;;9790:32;9816:5;9790:32;:::i;:::-;9691:137;;;;:::o;9847:552::-;9904:8;9914:6;9964:3;9957:4;9949:6;9945:17;9941:27;9931:122;;9972:79;;:::i;:::-;9931:122;10085:6;10072:20;10062:30;;10115:18;10107:6;10104:30;10101:117;;;10137:79;;:::i;:::-;10101:117;10251:4;10243:6;10239:17;10227:29;;10305:3;10297:4;10289:6;10285:17;10275:8;10271:32;10268:41;10265:128;;;10312:79;;:::i;:::-;10265:128;9847:552;;;;;:::o;10405:961::-;10501:6;10509;10517;10525;10533;10582:3;10570:9;10561:7;10557:23;10553:33;10550:120;;;10589:79;;:::i;:::-;10550:120;10709:1;10734:53;10779:7;10770:6;10759:9;10755:22;10734:53;:::i;:::-;10724:63;;10680:117;10836:2;10862:52;10906:7;10897:6;10886:9;10882:22;10862:52;:::i;:::-;10852:62;;10807:117;10963:2;10989:53;11034:7;11025:6;11014:9;11010:22;10989:53;:::i;:::-;10979:63;;10934:118;11119:2;11108:9;11104:18;11091:32;11150:18;11142:6;11139:30;11136:117;;;11172:79;;:::i;:::-;11136:117;11285:64;11341:7;11332:6;11321:9;11317:22;11285:64;:::i;:::-;11267:82;;;;11062:297;10405:961;;;;;;;;:::o;11372:117::-;11481:1;11478;11471:12;11495:180;11543:77;11540:1;11533:88;11640:4;11637:1;11630:15;11664:4;11661:1;11654:15;11681:281;11764:27;11786:4;11764:27;:::i;:::-;11756:6;11752:40;11894:6;11882:10;11879:22;11858:18;11846:10;11843:34;11840:62;11837:88;;;11905:18;;:::i;:::-;11837:88;11945:10;11941:2;11934:22;11724:238;11681:281;;:::o;11968:129::-;12002:6;12029:20;;:::i;:::-;12019:30;;12058:33;12086:4;12078:6;12058:33;:::i;:::-;11968:129;;;:::o;12103:307::-;12164:4;12254:18;12246:6;12243:30;12240:56;;;12276:18;;:::i;:::-;12240:56;12314:29;12336:6;12314:29;:::i;:::-;12306:37;;12398:4;12392;12388:15;12380:23;;12103:307;;;:::o;12416:146::-;12513:6;12508:3;12503;12490:30;12554:1;12545:6;12540:3;12536:16;12529:27;12416:146;;;:::o;12568:423::-;12645:5;12670:65;12686:48;12727:6;12686:48;:::i;:::-;12670:65;:::i;:::-;12661:74;;12758:6;12751:5;12744:21;12796:4;12789:5;12785:16;12834:3;12825:6;12820:3;12816:16;12813:25;12810:112;;;12841:79;;:::i;:::-;12810:112;12931:54;12978:6;12973:3;12968;12931:54;:::i;:::-;12651:340;12568:423;;;;;:::o;13010:338::-;13065:5;13114:3;13107:4;13099:6;13095:17;13091:27;13081:122;;13122:79;;:::i;:::-;13081:122;13239:6;13226:20;13264:78;13338:3;13330:6;13323:4;13315:6;13311:17;13264:78;:::i;:::-;13255:87;;13071:277;13010:338;;;;:::o;13354:943::-;13449:6;13457;13465;13473;13522:3;13510:9;13501:7;13497:23;13493:33;13490:120;;;13529:79;;:::i;:::-;13490:120;13649:1;13674:53;13719:7;13710:6;13699:9;13695:22;13674:53;:::i;:::-;13664:63;;13620:117;13776:2;13802:53;13847:7;13838:6;13827:9;13823:22;13802:53;:::i;:::-;13792:63;;13747:118;13904:2;13930:53;13975:7;13966:6;13955:9;13951:22;13930:53;:::i;:::-;13920:63;;13875:118;14060:2;14049:9;14045:18;14032:32;14091:18;14083:6;14080:30;14077:117;;;14113:79;;:::i;:::-;14077:117;14218:62;14272:7;14263:6;14252:9;14248:22;14218:62;:::i;:::-;14208:72;;14003:287;13354:943;;;;;;;:::o;14303:957::-;14397:6;14405;14413;14421;14429;14478:3;14466:9;14457:7;14453:23;14449:33;14446:120;;;14485:79;;:::i;:::-;14446:120;14605:1;14630:51;14673:7;14664:6;14653:9;14649:22;14630:51;:::i;:::-;14620:61;;14576:115;14730:2;14756:53;14801:7;14792:6;14781:9;14777:22;14756:53;:::i;:::-;14746:63;;14701:118;14858:2;14884:52;14928:7;14919:6;14908:9;14904:22;14884:52;:::i;:::-;14874:62;;14829:117;15013:2;15002:9;14998:18;14985:32;15044:18;15036:6;15033:30;15030:117;;;15066:79;;:::i;:::-;15030:117;15179:64;15235:7;15226:6;15215:9;15211:22;15179:64;:::i;:::-;15161:82;;;;14956:297;14303:957;;;;;;;;:::o;15266:474::-;15334:6;15342;15391:2;15379:9;15370:7;15366:23;15362:32;15359:119;;;15397:79;;:::i;:::-;15359:119;15517:1;15542:53;15587:7;15578:6;15567:9;15563:22;15542:53;:::i;:::-;15532:63;;15488:117;15644:2;15670:53;15715:7;15706:6;15695:9;15691:22;15670:53;:::i;:::-;15660:63;;15615:118;15266:474;;;;;:::o;15746:180::-;15794:77;15791:1;15784:88;15891:4;15888:1;15881:15;15915:4;15912:1;15905:15;15932:320;15976:6;16013:1;16007:4;16003:12;15993:22;;16060:1;16054:4;16050:12;16081:18;16071:81;;16137:4;16129:6;16125:17;16115:27;;16071:81;16199:2;16191:6;16188:14;16168:18;16165:38;16162:84;;16218:18;;:::i;:::-;16162:84;15983:269;15932:320;;;:::o;16258:180::-;16306:77;16303:1;16296:88;16403:4;16400:1;16393:15;16427:4;16424:1;16417:15;16444:191;16484:3;16503:20;16521:1;16503:20;:::i;:::-;16498:25;;16537:20;16555:1;16537:20;:::i;:::-;16532:25;;16580:1;16577;16573:9;16566:16;;16601:3;16598:1;16595:10;16592:36;;;16608:18;;:::i;:::-;16592:36;16444:191;;;;:::o;16641:238::-;16781:34;16777:1;16769:6;16765:14;16758:58;16850:21;16845:2;16837:6;16833:15;16826:46;16641:238;:::o;16885:366::-;17027:3;17048:67;17112:2;17107:3;17048:67;:::i;:::-;17041:74;;17124:93;17213:3;17124:93;:::i;:::-;17242:2;17237:3;17233:12;17226:19;;16885:366;;;:::o;17257:419::-;17423:4;17461:2;17450:9;17446:18;17438:26;;17510:9;17504:4;17500:20;17496:1;17485:9;17481:17;17474:47;17538:131;17664:4;17538:131;:::i;:::-;17530:139;;17257:419;;;:::o;17682:234::-;17822:34;17818:1;17810:6;17806:14;17799:58;17891:17;17886:2;17878:6;17874:15;17867:42;17682:234;:::o;17922:366::-;18064:3;18085:67;18149:2;18144:3;18085:67;:::i;:::-;18078:74;;18161:93;18250:3;18161:93;:::i;:::-;18279:2;18274:3;18270:12;18263:19;;17922:366;;;:::o;18294:419::-;18460:4;18498:2;18487:9;18483:18;18475:26;;18547:9;18541:4;18537:20;18533:1;18522:9;18518:17;18511:47;18575:131;18701:4;18575:131;:::i;:::-;18567:139;;18294:419;;;:::o;18719:161::-;18859:13;18855:1;18847:6;18843:14;18836:37;18719:161;:::o;18886:366::-;19028:3;19049:67;19113:2;19108:3;19049:67;:::i;:::-;19042:74;;19125:93;19214:3;19125:93;:::i;:::-;19243:2;19238:3;19234:12;19227:19;;18886:366;;;:::o;19258:419::-;19424:4;19462:2;19451:9;19447:18;19439:26;;19511:9;19505:4;19501:20;19497:1;19486:9;19482:17;19475:47;19539:131;19665:4;19539:131;:::i;:::-;19531:139;;19258:419;;;:::o;19683:332::-;19804:4;19842:2;19831:9;19827:18;19819:26;;19855:71;19923:1;19912:9;19908:17;19899:6;19855:71;:::i;:::-;19936:72;20004:2;19993:9;19989:18;19980:6;19936:72;:::i;:::-;19683:332;;;;;:::o;20021:97::-;20080:6;20108:3;20098:13;;20021:97;;;;:::o;20124:141::-;20173:4;20196:3;20188:11;;20219:3;20216:1;20209:14;20253:4;20250:1;20240:18;20232:26;;20124:141;;;:::o;20271:93::-;20308:6;20355:2;20350;20343:5;20339:14;20335:23;20325:33;;20271:93;;;:::o;20370:107::-;20414:8;20464:5;20458:4;20454:16;20433:37;;20370:107;;;;:::o;20483:393::-;20552:6;20602:1;20590:10;20586:18;20625:97;20655:66;20644:9;20625:97;:::i;:::-;20743:39;20773:8;20762:9;20743:39;:::i;:::-;20731:51;;20815:4;20811:9;20804:5;20800:21;20791:30;;20864:4;20854:8;20850:19;20843:5;20840:30;20830:40;;20559:317;;20483:393;;;;;:::o;20882:60::-;20910:3;20931:5;20924:12;;20882:60;;;:::o;20948:142::-;20998:9;21031:53;21049:34;21058:24;21076:5;21058:24;:::i;:::-;21049:34;:::i;:::-;21031:53;:::i;:::-;21018:66;;20948:142;;;:::o;21096:75::-;21139:3;21160:5;21153:12;;21096:75;;;:::o;21177:269::-;21287:39;21318:7;21287:39;:::i;:::-;21348:91;21397:41;21421:16;21397:41;:::i;:::-;21389:6;21382:4;21376:11;21348:91;:::i;:::-;21342:4;21335:105;21253:193;21177:269;;;:::o;21452:73::-;21497:3;21452:73;:::o;21531:189::-;21608:32;;:::i;:::-;21649:65;21707:6;21699;21693:4;21649:65;:::i;:::-;21584:136;21531:189;;:::o;21726:186::-;21786:120;21803:3;21796:5;21793:14;21786:120;;;21857:39;21894:1;21887:5;21857:39;:::i;:::-;21830:1;21823:5;21819:13;21810:22;;21786:120;;;21726:186;;:::o;21918:543::-;22019:2;22014:3;22011:11;22008:446;;;22053:38;22085:5;22053:38;:::i;:::-;22137:29;22155:10;22137:29;:::i;:::-;22127:8;22123:44;22320:2;22308:10;22305:18;22302:49;;;22341:8;22326:23;;22302:49;22364:80;22420:22;22438:3;22420:22;:::i;:::-;22410:8;22406:37;22393:11;22364:80;:::i;:::-;22023:431;;22008:446;21918:543;;;:::o;22467:117::-;22521:8;22571:5;22565:4;22561:16;22540:37;;22467:117;;;;:::o;22590:169::-;22634:6;22667:51;22715:1;22711:6;22703:5;22700:1;22696:13;22667:51;:::i;:::-;22663:56;22748:4;22742;22738:15;22728:25;;22641:118;22590:169;;;;:::o;22764:295::-;22840:4;22986:29;23011:3;23005:4;22986:29;:::i;:::-;22978:37;;23048:3;23045:1;23041:11;23035:4;23032:21;23024:29;;22764:295;;;;:::o;23064:1403::-;23188:44;23228:3;23223;23188:44;:::i;:::-;23297:18;23289:6;23286:30;23283:56;;;23319:18;;:::i;:::-;23283:56;23363:38;23395:4;23389:11;23363:38;:::i;:::-;23448:67;23508:6;23500;23494:4;23448:67;:::i;:::-;23542:1;23571:2;23563:6;23560:14;23588:1;23583:632;;;;24259:1;24276:6;24273:84;;;24332:9;24327:3;24323:19;24310:33;24301:42;;24273:84;24383:67;24443:6;24436:5;24383:67;:::i;:::-;24377:4;24370:81;24232:229;23553:908;;23583:632;23635:4;23631:9;23623:6;23619:22;23669:37;23701:4;23669:37;:::i;:::-;23728:1;23742:215;23756:7;23753:1;23750:14;23742:215;;;23842:9;23837:3;23833:19;23820:33;23812:6;23805:49;23893:1;23885:6;23881:14;23871:24;;23940:2;23929:9;23925:18;23912:31;;23779:4;23776:1;23772:12;23767:17;;23742:215;;;23985:6;23976:7;23973:19;23970:186;;;24050:9;24045:3;24041:19;24028:33;24093:48;24135:4;24127:6;24123:17;24112:9;24093:48;:::i;:::-;24085:6;24078:64;23993:163;23970:186;24202:1;24198;24190:6;24186:14;24182:22;24176:4;24169:36;23590:625;;;23553:908;;23163:1304;;;23064:1403;;;:::o;24473:118::-;24560:24;24578:5;24560:24;:::i;:::-;24555:3;24548:37;24473:118;;:::o;24597:115::-;24682:23;24699:5;24682:23;:::i;:::-;24677:3;24670:36;24597:115;;:::o;24718:438::-;24865:4;24903:2;24892:9;24888:18;24880:26;;24916:71;24984:1;24973:9;24969:17;24960:6;24916:71;:::i;:::-;24997:72;25065:2;25054:9;25050:18;25041:6;24997:72;:::i;:::-;25079:70;25145:2;25134:9;25130:18;25121:6;25079:70;:::i;:::-;24718:438;;;;;;:::o;25162:180::-;25210:77;25207:1;25200:88;25307:4;25304:1;25297:15;25331:4;25328:1;25321:15;25348:178;25488:30;25484:1;25476:6;25472:14;25465:54;25348:178;:::o;25532:366::-;25674:3;25695:67;25759:2;25754:3;25695:67;:::i;:::-;25688:74;;25771:93;25860:3;25771:93;:::i;:::-;25889:2;25884:3;25880:12;25873:19;;25532:366;;;:::o;25904:419::-;26070:4;26108:2;26097:9;26093:18;26085:26;;26157:9;26151:4;26147:20;26143:1;26132:9;26128:17;26121:47;26185:131;26311:4;26185:131;:::i;:::-;26177:139;;25904:419;;;:::o;26329:224::-;26469:34;26465:1;26457:6;26453:14;26446:58;26538:7;26533:2;26525:6;26521:15;26514:32;26329:224;:::o;26559:366::-;26701:3;26722:67;26786:2;26781:3;26722:67;:::i;:::-;26715:74;;26798:93;26887:3;26798:93;:::i;:::-;26916:2;26911:3;26907:12;26900:19;;26559:366;;;:::o;26931:419::-;27097:4;27135:2;27124:9;27120:18;27112:26;;27184:9;27178:4;27174:20;27170:1;27159:9;27155:17;27148:47;27212:131;27338:4;27212:131;:::i;:::-;27204:139;;26931:419;;;:::o;27356:167::-;27496:19;27492:1;27484:6;27480:14;27473:43;27356:167;:::o;27529:366::-;27671:3;27692:67;27756:2;27751:3;27692:67;:::i;:::-;27685:74;;27768:93;27857:3;27768:93;:::i;:::-;27886:2;27881:3;27877:12;27870:19;;27529:366;;;:::o;27901:419::-;28067:4;28105:2;28094:9;28090:18;28082:26;;28154:9;28148:4;28144:20;28140:1;28129:9;28125:17;28118:47;28182:131;28308:4;28182:131;:::i;:::-;28174:139;;27901:419;;;:::o;28326:138::-;28374:9;28407:51;28425:32;28434:22;28450:5;28434:22;:::i;:::-;28425:32;:::i;:::-;28407:51;:::i;:::-;28394:64;;28326:138;;;:::o;28470:127::-;28555:35;28584:5;28555:35;:::i;:::-;28550:3;28543:48;28470:127;;:::o;28603:218::-;28694:4;28732:2;28721:9;28717:18;28709:26;;28745:69;28811:1;28800:9;28796:17;28787:6;28745:69;:::i;:::-;28603:218;;;;:::o;28827:148::-;28929:11;28966:3;28951:18;;28827:148;;;;:::o;28981:390::-;29087:3;29115:39;29148:5;29115:39;:::i;:::-;29170:89;29252:6;29247:3;29170:89;:::i;:::-;29163:96;;29268:65;29326:6;29321:3;29314:4;29307:5;29303:16;29268:65;:::i;:::-;29358:6;29353:3;29349:16;29342:23;;29091:280;28981:390;;;;:::o;29377:435::-;29557:3;29579:95;29670:3;29661:6;29579:95;:::i;:::-;29572:102;;29691:95;29782:3;29773:6;29691:95;:::i;:::-;29684:102;;29803:3;29796:10;;29377:435;;;;;:::o;29818:225::-;29958:34;29954:1;29946:6;29942:14;29935:58;30027:8;30022:2;30014:6;30010:15;30003:33;29818:225;:::o;30049:366::-;30191:3;30212:67;30276:2;30271:3;30212:67;:::i;:::-;30205:74;;30288:93;30377:3;30288:93;:::i;:::-;30406:2;30401:3;30397:12;30390:19;;30049:366;;;:::o;30421:419::-;30587:4;30625:2;30614:9;30610:18;30602:26;;30674:9;30668:4;30664:20;30660:1;30649:9;30645:17;30638:47;30702:131;30828:4;30702:131;:::i;:::-;30694:139;;30421:419;;;:::o;30846:224::-;30986:34;30982:1;30974:6;30970:14;30963:58;31055:7;31050:2;31042:6;31038:15;31031:32;30846:224;:::o;31076:366::-;31218:3;31239:67;31303:2;31298:3;31239:67;:::i;:::-;31232:74;;31315:93;31404:3;31315:93;:::i;:::-;31433:2;31428:3;31424:12;31417:19;;31076:366;;;:::o;31448:419::-;31614:4;31652:2;31641:9;31637:18;31629:26;;31701:9;31695:4;31691:20;31687:1;31676:9;31672:17;31665:47;31729:131;31855:4;31729:131;:::i;:::-;31721:139;;31448:419;;;:::o;31873:182::-;32013:34;32009:1;32001:6;31997:14;31990:58;31873:182;:::o;32061:366::-;32203:3;32224:67;32288:2;32283:3;32224:67;:::i;:::-;32217:74;;32300:93;32389:3;32300:93;:::i;:::-;32418:2;32413:3;32409:12;32402:19;;32061:366;;;:::o;32433:419::-;32599:4;32637:2;32626:9;32622:18;32614:26;;32686:9;32680:4;32676:20;32672:1;32661:9;32657:17;32650:47;32714:131;32840:4;32714:131;:::i;:::-;32706:139;;32433:419;;;:::o;32858:98::-;32909:6;32943:5;32937:12;32927:22;;32858:98;;;:::o;32962:168::-;33045:11;33079:6;33074:3;33067:19;33119:4;33114:3;33110:14;33095:29;;32962:168;;;;:::o;33136:373::-;33222:3;33250:38;33282:5;33250:38;:::i;:::-;33304:70;33367:6;33362:3;33304:70;:::i;:::-;33297:77;;33383:65;33441:6;33436:3;33429:4;33422:5;33418:16;33383:65;:::i;:::-;33473:29;33495:6;33473:29;:::i;:::-;33468:3;33464:39;33457:46;;33226:283;33136:373;;;;:::o;33515:640::-;33710:4;33748:3;33737:9;33733:19;33725:27;;33762:71;33830:1;33819:9;33815:17;33806:6;33762:71;:::i;:::-;33843:72;33911:2;33900:9;33896:18;33887:6;33843:72;:::i;:::-;33925;33993:2;33982:9;33978:18;33969:6;33925:72;:::i;:::-;34044:9;34038:4;34034:20;34029:2;34018:9;34014:18;34007:48;34072:76;34143:4;34134:6;34072:76;:::i;:::-;34064:84;;33515:640;;;;;;;:::o;34161:141::-;34217:5;34248:6;34242:13;34233:22;;34264:32;34290:5;34264:32;:::i;:::-;34161:141;;;;:::o;34308:349::-;34377:6;34426:2;34414:9;34405:7;34401:23;34397:32;34394:119;;;34432:79;;:::i;:::-;34394:119;34552:1;34577:63;34632:7;34623:6;34612:9;34608:22;34577:63;:::i;:::-;34567:73;;34523:127;34308:349;;;;:::o;34663:170::-;34803:22;34799:1;34791:6;34787:14;34780:46;34663:170;:::o;34839:366::-;34981:3;35002:67;35066:2;35061:3;35002:67;:::i;:::-;34995:74;;35078:93;35167:3;35078:93;:::i;:::-;35196:2;35191:3;35187:12;35180:19;;34839:366;;;:::o;35211:419::-;35377:4;35415:2;35404:9;35400:18;35392:26;;35464:9;35458:4;35454:20;35450:1;35439:9;35435:17;35428:47;35492:131;35618:4;35492:131;:::i;:::-;35484:139;;35211:419;;;:::o;35636:166::-;35776:18;35772:1;35764:6;35760:14;35753:42;35636:166;:::o;35808:366::-;35950:3;35971:67;36035:2;36030:3;35971:67;:::i;:::-;35964:74;;36047:93;36136:3;36047:93;:::i;:::-;36165:2;36160:3;36156:12;36149:19;;35808:366;;;:::o;36180:419::-;36346:4;36384:2;36373:9;36369:18;36361:26;;36433:9;36427:4;36423:20;36419:1;36408:9;36404:17;36397:47;36461:131;36587:4;36461:131;:::i;:::-;36453:139;;36180:419;;;:::o;36605:214::-;36745:66;36741:1;36733:6;36729:14;36722:90;36605:214;:::o;36825:400::-;36985:3;37006:84;37088:1;37083:3;37006:84;:::i;:::-;36999:91;;37099:93;37188:3;37099:93;:::i;:::-;37217:1;37212:3;37208:11;37201:18;;36825:400;;;:::o;37231:79::-;37270:7;37299:5;37288:16;;37231:79;;;:::o;37316:157::-;37421:45;37441:24;37459:5;37441:24;:::i;:::-;37421:45;:::i;:::-;37416:3;37409:58;37316:157;;:::o;37479:663::-;37720:3;37742:148;37886:3;37742:148;:::i;:::-;37735:155;;37900:75;37971:3;37962:6;37900:75;:::i;:::-;38000:2;37995:3;37991:12;37984:19;;38013:75;38084:3;38075:6;38013:75;:::i;:::-;38113:2;38108:3;38104:12;38097:19;;38133:3;38126:10;;37479:663;;;;;:::o;38148:112::-;38231:22;38247:5;38231:22;:::i;:::-;38226:3;38219:35;38148:112;;:::o;38266:545::-;38439:4;38477:3;38466:9;38462:19;38454:27;;38491:71;38559:1;38548:9;38544:17;38535:6;38491:71;:::i;:::-;38572:68;38636:2;38625:9;38621:18;38612:6;38572:68;:::i;:::-;38650:72;38718:2;38707:9;38703:18;38694:6;38650:72;:::i;:::-;38732;38800:2;38789:9;38785:18;38776:6;38732:72;:::i;:::-;38266:545;;;;;;;:::o;38817:664::-;39022:4;39060:3;39049:9;39045:19;39037:27;;39074:71;39142:1;39131:9;39127:17;39118:6;39074:71;:::i;:::-;39155:72;39223:2;39212:9;39208:18;39199:6;39155:72;:::i;:::-;39237;39305:2;39294:9;39290:18;39281:6;39237:72;:::i;:::-;39319;39387:2;39376:9;39372:18;39363:6;39319:72;:::i;:::-;39401:73;39469:3;39458:9;39454:19;39445:6;39401:73;:::i;:::-;38817:664;;;;;;;;:::o

Swarm Source

ipfs://b534d7e3cbd1c027442c64cf214fc46967ca6e6656ba5be217be4e862a9fab6a
Loading...
Loading
Loading...
Loading
[ Download: CSV Export  ]
[ Download: CSV Export  ]

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