Feature Tip: Add private address tag to any address under My Name Tag !
ERC-721
Overview
Max Total Supply
3,309 SEA
Holders
1,444
Total Transfers
-
Market
Volume (24H)
N/A
Min Price (24H)
N/A
Max Price (24H)
N/A
Other Info
Token Contract
Loading...
Loading
Loading...
Loading
Loading...
Loading
# | Exchange | Pair | Price | 24H Volume | % Volume |
---|
Contract Name:
OnChainSeasides
Compiler Version
v0.8.7+commit.e28d00a7
Optimization Enabled:
Yes with 200 runs
Other Settings:
default evmVersion
Contract Source Code (Solidity Standard Json-Input format)
// SPDX-License-Identifier: MIT pragma solidity ^0.8.2; import "@openzeppelin/contracts/token/ERC721/ERC721.sol"; import "@openzeppelin/contracts/token/ERC721/extensions/ERC721Enumerable.sol"; import "@openzeppelin/contracts/access/Ownable.sol"; import "@openzeppelin/contracts/utils/Counters.sol"; import "@openzeppelin/contracts/utils/math/SafeMath.sol"; contract OnChainSeasides is ERC721, ERC721Enumerable, Ownable { using Counters for Counters.Counter; using SafeMath for uint256; Counters.Counter private _tokenIdCounter; uint private constant maxTokensPerTransaction = 30; uint256 private tokenPrice = 30000000000000000; //0.03 ETH uint256 private constant nftsNumber = 3333; uint256 private constant nftsPublicNumber = 3300; constructor() ERC721("OnChain Seasides", "SEA") { _tokenIdCounter.increment(); } function toString(uint256 value) internal pure returns (string memory) { // Inspired by OraclizeAPI's implementation - MIT license // https://github.com/oraclize/ethereum-api/blob/b42146b063c7d6ee1358846c198246239e9360e8/oraclizeAPI_0.4.25.sol if (value == 0) { return "0"; } uint256 temp = value; uint256 digits; while (temp != 0) { digits++; temp /= 10; } bytes memory buffer = new bytes(digits); while (value != 0) { digits -= 1; buffer[digits] = bytes1(uint8(48 + uint256(value % 10))); value /= 10; } return string(buffer); } function safeMint(address to) public onlyOwner { _safeMint(to, _tokenIdCounter.current()); _tokenIdCounter.increment(); } // The following functions are overrides required by Solidity. function _beforeTokenTransfer(address from, address to, uint256 tokenId) internal override(ERC721, ERC721Enumerable) { super._beforeTokenTransfer(from, to, tokenId); } function supportsInterface(bytes4 interfaceId) public view override(ERC721, ERC721Enumerable) returns (bool) { return super.supportsInterface(interfaceId); } function random(string memory input) internal pure returns (uint256) { return uint256(keccak256(abi.encodePacked(input))); } function toHashCode(uint256 value) internal pure returns (string memory) { if (value == 0) { return "000000"; } uint256 i; bytes memory buffer = "000000"; for(i=6;i>0;i--) { if (value % 16 < 10) buffer[i-1] = bytes1(uint8(48 + uint256(value % 16))); else buffer[i-1] = bytes1(uint8(55 + uint256(value % 16))); value /= 16; } return string(buffer); } function getGrass(uint256 value, uint256 is_mirror) public pure returns (string memory) { uint256 i; uint256 prt; int256 sym; uint256 pos; uint256 seed; uint256 ps; uint256 psx; uint256 finale=1; bytes memory buffer = new bytes(250); uint256[6] memory gtypes; gtypes[0] = 563891157643448204000202434637890680663763947839870365052682593692495482209; gtypes[1] = 1002862760941180669446230775124823108274825368678245159754096610961814; gtypes[2] = 563871000271014004664722370698789260236686074535032282236861359775136052288; gtypes[3] = 1003329826285104973259814183739832352502149867278345049274004408492605; gtypes[4] = 564017709498396033660054424834310877655732749413952956046917965314005819035; gtypes[5] = 1002137238139483205643623885946750710585681910777426838629649942020236; finale = 0; seed = gtypes[value*2-2]; for(psx=ps=pos=i=0;i<70;i++) { prt = seed / (750 ** ps++); sym = int256(prt % 750); if (sym == 749) { if (finale == 1) break; finale = 1; seed = gtypes[value*2-1]; ps=0; continue; } if (psx%2==0) { if (is_mirror==1) sym = 400 - sym; else sym += 550; if (i>0) buffer[pos++] = ' '; } else { if (i>0) buffer[pos++] = ','; sym += 500; } if (sym<0) { sym=-sym; buffer[pos++] = bytes1(uint8(45)); } if (sym<10) buffer[pos++] = bytes1(uint8(48 + uint256(sym % 10))); else if (sym<100) { buffer[pos++] = bytes1(uint8(48 + uint256(sym / 10))); buffer[pos++] = bytes1(uint8(48 + uint256(sym % 10))); } else if (sym<1000) { buffer[pos++] = bytes1(uint8(48 + uint256(sym / 100))); buffer[pos++] = bytes1(uint8(48 + uint256((sym / 10)%10))); buffer[pos++] = bytes1(uint8(48 + uint256(sym % 10))); } else { buffer[pos++] = bytes1(uint8(48 + uint256(sym / 1000))); buffer[pos++] = bytes1(uint8(48 + uint256((sym / 100)%10))); buffer[pos++] = bytes1(uint8(48 + uint256((sym / 10)%10))); buffer[pos++] = bytes1(uint8(48 + uint256(sym % 10))); } psx++; } bytes memory buffer2 = new bytes(pos); for(i=0;i<pos;i++) buffer2[i] = buffer[i]; return string(buffer2); } function getShip(uint256 value, uint dir, uint is_mirror) internal pure returns (string memory) { uint256 i; uint256 prt; uint256 sym; uint256 pos; uint256 seed; uint256 ps; uint256 finale=1; bytes memory buffer = new bytes(300); uint256[10] memory stypes; stypes[0] = 4039953958419306012627391607069051694356360220713754655043968660016; stypes[1] = 63106266081136485859406661386793985037851232673778661; stypes[2] = 161568093605294050650559196328221936718722000674656809180508552713619040; stypes[3] = 63096482884095845761084638724950672236119832596577537; stypes[4] = 6461049781969783938359086369026346968766711308334899858451023002616666897676; stypes[5] = 63120086296825990194702960480301658791387552688699008; stypes[6] = 161512793322334258241850556133413063336821761930288295517787616106338882; stypes[7] = 986070141659197590531764122903072004020; stypes[8] = 161390119281273703232398765386984498139365471992159755174679367096177400; stypes[9] = 100972004999646208506838990436000093602246360157725215130052251; if (value < 7) { seed = stypes[value-1]; } else { finale = 0; seed = stypes[value==7?6:8]; } uint256 shift_y = is_mirror==1 ? 560 : 460; for(ps=pos=i=0;i<70;i++) { prt = seed / (200 ** ps++); sym = prt % 200; if (sym == 150) { if (finale == 1) break; finale = 1; seed = stypes[value==7?7:9]; ps=0; continue; } if (ps%2==0) { if (is_mirror==1) sym = 1000 - (sym + 320); else sym = sym + shift_y; if (i>0) buffer[pos++] = ','; } else { if (dir == 1) sym += 100; else sym = 1000 - (sym + 100); if (i>0) buffer[pos++] = ' '; } if (sym<10) buffer[pos++] = bytes1(uint8(48 + uint256(sym % 10))); else if (sym<100) { buffer[pos++] = bytes1(uint8(48 + uint256(sym / 10))); buffer[pos++] = bytes1(uint8(48 + uint256(sym % 10))); } else { buffer[pos++] = bytes1(uint8(48 + uint256(sym / 100))); buffer[pos++] = bytes1(uint8(48 + uint256((sym / 10)%10))); buffer[pos++] = bytes1(uint8(48 + uint256(sym % 10))); } } bytes memory buffer2 = new bytes(pos); for(i=0;i<pos;i++) buffer2[i] = buffer[i]; return string(buffer2); } function getPalm(uint256 num) internal pure returns (string memory) { string[4] memory palm; palm[0]='M-137,104C-32,27,573,28,762,263 C582,95,180,57,23,87c105-7,598,17,757,244 c-60-74-241-182-524-212c241,55,435,181,475,265 C587,207,260,141,220,133c18,4,375,100,461,262 C504,164,19,125-4,123c141,15,524,86,641,281 C493,229,162,170,122,163c141,51,245,127,273,197 C340,270,68,139-39,146c139,51,242,127,269,196 C117,195-84,150-84,150L-758,97L-137,104z'; palm[1]='M-144,304c9-106,334-319,580-175 C236,41-2,150-67,232c52-44,331-195,555-47 c-77-45-241-78-410-5c162-35,344,8,417,70 C307,140,92,197,65,204c12-2,263-42,407,72 c-236-144-520-8-534-1C24,239,273,167,455,301 C270,193,57,257,31,265c107-3,209,27,267,80 C214,284-11,263-65,308c106-3,207,28,264,80 C48,296-87,326-87,326L-533,433L-144,304z'; palm[2]='M173,299c80-78,288-143,525-156c123-6,235,2,323,22 c11,1,21,2,32,4c-87-30-205-48-339-49c-210-0-400,43-495,107 c84-74,293-128,528-127c217,0,395,47,466,114l435,66 c0,0-500-18-509-26c-175,21-328,81-402,151c56-69,193-132,358-163 c-15-2-30-4-47-5c-22,1-46,2-70,5C796,262,636,324,560,396 c56-70,195-133,362-164c-44,0-91,3-139,8 c-213,23-400,87-490,161c76-82,282-158,521-184 c92-10,179-11,255-5c-0-0-0-0-0-0c-89-14-204-14-330,1 C524,241,338,309,250,384c75-83,280-163,519-193 c21-2,42-4,63-6c-45,1-92,5-141,11c-213,27-400,95-488,170 c75-83,280-163,519-193c26-3,53-6,78-8c-42-1-87-1-133,0 C454,175,265,230,173,299z'; palm[3]='M1086,14C982-63,415-11,256,269C412,64,785-11,934,9 c-99,0-559,70-690,339c50-88,212-225,474-283 c-221,82-393,240-424,339C415,191,716,89,753,77 c-16,6-344,144-412,334C490,136,941,50,963,47 C832,76,478,189,384,418c120-210,426-305,463-316 c-128,69-220,164-241,245c44-106,289-276,391-277 c-126,69-217,163-237,243c94-175,279-243,279-243l438-99L1086,14z'; return palm[num-1]; } function getGround(uint256 num) internal pure returns (string memory) { string[7] memory ground; ground[0] = '-30,898,-1357,1016,1298,1016'; ground[1] = '-11,927,431,952,663,921,845,945,1043,934,1043,1011,-8,1011'; ground[2] = '-11,927,156,912,222,941,284,931,1043,934,1043,1011,-8,1011'; ground[3] = '-11,927,425,969,875,952,1046,880,1043,934,1043,1011,-8,1011'; ground[4] = '-12,962,136,940,239,960,648,966,951,931,1041,938,1043,1011,-8,1011'; ground[5] = '894,933,722,948,666,900,666,900,664,890,646,890,645,900,482,900,481,890,462,890,461,900,461,900,364,967,247,961,102,946,-9,912,-12,1007,1039,1007,1043,930'; ground[6] = '800,951,1014,927,1039,1007,-12,1007,-10,940,81,941,175,962,647,927'; return ground[num-1]; } function getSun(uint256 num) internal pure returns (string memory) { uint256 suns; suns = 712316151692331099684323100692331075717224135713226117576244139582262105; if (num > 0) suns = suns / (1000 ** (num*3)); string memory output = string(abi.encodePacked('cx="',toString((suns/1000000)%1000),'" cy="',toString((suns/1000)%1000), '" r="',toString(suns%1000),'"')); return output; } function tokenURI(uint256 tokenId) pure public override(ERC721) returns (string memory) { uint256[16] memory xtypes; string[4] memory colors; string[17] memory parts; string[8] memory mount1; string[8] memory mount2; uint256[8] memory params; uint256 pos; uint256 rand = random(string(abi.encodePacked('Seasides',toString(tokenId)))); params[0] = 1 + ((rand/10) % 8);// ship params[1] = 1 + (rand/100) % 2; // dir params[2] = 1 + ((rand/10000) % 37); // pallette params[3] = 1 + ((rand/1000000) % 8); // mounts params[4] = 1 + ((rand/100000000) % 6); // grass params[5] = 1 + ((rand/1000000000) % 4); // palm params[6] = 1 + ((rand/10000000000) % 7); // ground params[7] = 1 + ((rand/100000000000) % 4); // sun mount1[0] = '78,444 -494,478 651,478'; mount1[1] = '999,392 865,417 806,420 743,451 529,478 1468,478'; mount1[2] = '463,449 403,457 351,431 177,478 681,478'; mount1[3] = '1004,457 848,464 760,450 608,455 419,433 320,454 135,409 -8,424 -8,478 1004,478'; mount1[4] = '226,422 177,414 83,344 -8,367 -12,478 328,478'; mount1[5] = '999,392 865,417 806,420 743,451 529,478 1468,478'; mount1[6] = '564,443 463,457 375,478 793,478 721,467 612,414'; mount1[7] = '1013,420 853,410 732,441 637,431 608,446 390,466 240,457 186,430 94,447 -16,420 -87,478 1016,478'; mount2[0] = '162,392 -307,478 632,478'; mount2[1] = ''; mount2[2] = ''; mount2[3] = ''; mount2[4] = '991,410 826,454 611,478 1262,478'; mount2[5] = ''; mount2[6] = '156,438 47,478 321,478 214,461'; mount2[7] = ''; xtypes[0] = 5165462586977505248984271025794477445148782908573069521325498340212639; xtypes[1] = 490024044101034400102396419179934085738779419751960710510484619019681904; xtypes[2] = 4043991994607814950473362577238312297018937036519365143342896853810329; xtypes[3] = 1379064599573736476104814799994272434465744258265921437097553789059202; xtypes[4] = 6056088629583070600596400423476580059718415009582353316298341503465113; xtypes[5] = 138040297937156288773826099078203347749133755730926697092887565253488215; xtypes[6] = 1763486549546207954426324916291393168705773800679768336312337964145309337; xtypes[7] = 948653233183009513098268805292360185252612190882203913941189109236825; xtypes[8] = 1765596160030049122294337924707755907300568572202546387173451426705702110; xtypes[9] = 571213962168160818623884462797953331024439701527741670201332697661529; xtypes[10] = 1759945423641250114310949500884067660757318022344968985317724270290468863; xtypes[11] = 634962523324887909409742165431514640856016078396772822011217464449368064; xtypes[12] = 1318233657466206738337996551415773989873200879281323137549919882303230302; xtypes[13] = 20786449684869734852663949139680728584860474635680334261919687729872949; xtypes[14] = 321076936879265745525548114627410433723373509773187045336; pos = (params[2]-1) * 4; colors[0] = toHashCode(xtypes[pos/10] / (16777216 ** (pos%10)) % 16777216); pos = (params[2]-1) * 4 + 1; colors[1] = toHashCode(xtypes[pos/10] / (16777216 ** (pos%10)) % 16777216); pos = (params[2]-1) * 4 + 2; colors[2] = toHashCode(xtypes[pos/10] / (16777216 ** (pos%10)) % 16777216); pos = (params[2]-1) * 4 + 3; colors[3] = toHashCode(xtypes[pos/10] / (16777216 ** (pos%10)) % 16777216); parts[0] = '<svg width="1000px" height="1000px" viewBox="0 0 1000 1000" version="1.1" xmlns="http://www.w3.org/2000/svg"> <linearGradient id="SkyGradient" gradientUnits="userSpaceOnUse" x1="500.001" y1="999.8105" x2="500.0009" y2="4.882813e-004"> <stop offset="0.5604" style="stop-color:#'; // 2 parts[1] = '"/> <stop offset="1" style="stop-color:#'; // 3 parts[2] = '"/> </linearGradient> <rect x="0.001" fill="url(#SkyGradient)" width="1000" height="999.811"/> <polygon opacity="0.15" fill="#'; // 3 parts[3] = string(abi.encodePacked('" points="',mount2[params[3]-1],'"/> <polygon opacity="0.1" fill="#')); // 3 parts[4] = string(abi.encodePacked('" points="',mount1[params[3]-1],'"/> <rect x="0" y="478" opacity="0.2" fill="#')); // 3 parts[5] = '" width="1000" height="734.531"/> <rect x="0" y="563.156" opacity="0.3" fill="#'; // 3 parts[6] = '" width="1000" height="649.315"/> <g> <path xmlns="http://www.w3.org/2000/svg" opacity="0.55" fill="#'; // 3 parts[7] = '" d="M8087,687c-158,0-320-3.15-469-3 c-293,0-616,10-701,10c-261,0-600-17-809-17 c-118,0-246,11-376,11c-158,0-320-10-469-10 c-293,0-379,10-574,10c-195,0-331-11-540-11 c-118,0-246,11-376,11c-158,0-320-10-469-10 c-293,0-616,17-701,17c-261,0-600-12-809-12 c-118,0-246,12-376,12c-103,0-263-9-469-9 c-92,0-181,2-260,2c-171,0-304,0-362,0c-261,0-330-0-330-0 v525l9053-6V688C9039,688,8217,687,8087,687z"/> <animateMotion path="M 0 0 L -8050 20 Z" dur="70s" repeatCount="indefinite" /> </g> <g> <path xmlns="http://www.w3.org/2000/svg" fill="#'; // 3 parts[8] = '" d="M8097,846c-158,0-319-7-470-7c-285,0-443,20-651,20 c-172,0-353-5-449-9c-101-4-247-20-413-20c-116,0-243,26-373,26 c-158,0-320-31-471-31c-285,0-352,36-560,36c-172,0-390-31-556-31 c-116,0-243,26-373,26c-158,0-320-31-471-31c-285,0-442,35-650,35 c-172,0-353-5-449-9c-101-4-247-20-413-20c-116,0-245,25-375,25 c-158,0-322-13-474-13c-107,0-197,2-277,3c-133,1-243,0-372,0 c-172,0-308-0-308-0v364h9053V846C9038,846,8227,846,8097,846z"/> <animateMotion path="M 0 0 L -8050 40 Z" dur="70s" repeatCount="indefinite" /> </g> <g> <polygon fill="#'; // 3 parts[9] = string(abi.encodePacked('" points="',getShip(params[0],params[1],0), '"/> <polygon opacity="0.2" fill="#')); // 3 parts[10] = string(abi.encodePacked('" points="',getShip(params[0],params[1],1),'"/> <animateMotion path="m 0 0 h ',(params[1]==1 ? '':'-'),'5000" dur="1500s" repeatCount="indefinite" /> </g> <radialGradient id="SunGradient" ',getSun(params[7]*2-2),' gradientUnits="userSpaceOnUse"> <stop offset="0.7604" style="stop-color:#')); // 1 parts[11] = '"/> <stop offset="0.9812" style="stop-color:#'; // 2 parts[12] = '"/> </radialGradient> <circle opacity="0.1" fill="#'; // 2 parts[13] = string(abi.encodePacked('" ',getSun(params[7]*2-1),'/> <circle fill="url(#SunGradient)" ',getSun(params[7]*2-2),'/> <g> <polygon fill="#')); // 4 parts[14] = string(abi.encodePacked('" points="',getGrass(params[4]/3+1, params[4]%2),'"/> <animateMotion path="M 0 0 H 10 Z" dur="4s" repeatCount="indefinite" /> </g> <g> <path fill="#',colors[3],'" d="',getPalm(params[5]),'"/> <animateMotion path="M 0 0 H 15 Z" dur="5s" repeatCount="indefinite"/> </g> <polygon fill="#')); // 4 parts[15] = '" points="'; parts[16] = '"/></svg> '; string memory output = string(abi.encodePacked(parts[0],colors[1],parts[1],colors[2])); output = string(abi.encodePacked(output,parts[2],colors[2],parts[3] )); output = string(abi.encodePacked(output,colors[2],parts[4],colors[2] )); output = string(abi.encodePacked(output,parts[5],colors[2],parts[6] )); output = string(abi.encodePacked(output,colors[2],parts[7],colors[2] )); output = string(abi.encodePacked(output,parts[8],colors[2],parts[9] )); output = string(abi.encodePacked(output,colors[2],parts[10],colors[0] )); output = string(abi.encodePacked(output,parts[11],colors[1],parts[12] )); output = string(abi.encodePacked(output,colors[1],parts[13],colors[3] )); output = string(abi.encodePacked(output,parts[14],colors[3],parts[15])); output = string(abi.encodePacked(output,getGround(params[6]), parts[16])); string[11] memory aparts; aparts[0] = '[{ "trait_type": "Ship", "value": "'; aparts[1] = toString(params[0]); aparts[2] = '" }, { "trait_type": "Palette", "value": "'; aparts[3] = toString(params[2]); aparts[4] = '" }, { "trait_type": "Hills", "value": "'; aparts[5] = toString(params[3]); aparts[6] = '" }, { "trait_type": "Sun", "value": "'; aparts[7] = toString(params[7]); aparts[8] = '" }, { "trait_type": "Coast", "value": "'; aparts[9] = toString(params[6]); aparts[10] = '" }]'; string memory strparams = string(abi.encodePacked(aparts[0], aparts[1], aparts[2], aparts[3], aparts[4], aparts[5])); strparams = string(abi.encodePacked(strparams, aparts[6], aparts[7], aparts[8], aparts[9], aparts[10])); string memory json = Base64.encode(bytes(string(abi.encodePacked('{"name": "OnChain Seaside", "description": "Beautiful views, completely generated OnChain","attributes":', strparams, ', "image": "data:image/svg+xml;base64,', Base64.encode(bytes(output)), '"}')))); output = string(abi.encodePacked('data:application/json;base64,', json)); return output; } function claim() public { require(_tokenIdCounter.current() <= 333, "Tokens number to mint exceeds number of public tokens"); _safeMint(msg.sender, _tokenIdCounter.current()); _tokenIdCounter.increment(); } function withdraw() public onlyOwner { uint balance = address(this).balance; payable(msg.sender).transfer(balance); } function buySunsets(uint tokensNumber) public payable { require(tokensNumber > 0, "Wrong amount"); require(tokensNumber <= maxTokensPerTransaction, "Max tokens per transaction number exceeded"); require(_tokenIdCounter.current().add(tokensNumber) <= nftsPublicNumber, "Tokens number to mint exceeds number of public tokens"); require(tokenPrice.mul(tokensNumber) <= msg.value, "Ether value sent is too low"); for(uint i = 0; i < tokensNumber; i++) { _safeMint(msg.sender, _tokenIdCounter.current()); _tokenIdCounter.increment(); } } } /// [MIT License] /// @title Base64 /// @notice Provides a function for encoding some bytes in base64 /// @author Brecht Devos <[email protected]> library Base64 { bytes internal constant TABLE = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/"; /// @notice Encodes some bytes to the base64 representation function encode(bytes memory data) internal pure returns (string memory) { uint256 len = data.length; if (len == 0) return ""; // multiply by 4/3 rounded up uint256 encodedLen = 4 * ((len + 2) / 3); // Add some extra buffer at the end bytes memory result = new bytes(encodedLen + 32); bytes memory table = TABLE; assembly { let tablePtr := add(table, 1) let resultPtr := add(result, 32) for { let i := 0 } lt(i, len) { } { i := add(i, 3) let input := and(mload(add(data, i)), 0xffffff) let out := mload(add(tablePtr, and(shr(18, input), 0x3F))) out := shl(8, out) out := add(out, and(mload(add(tablePtr, and(shr(12, input), 0x3F))), 0xFF)) out := shl(8, out) out := add(out, and(mload(add(tablePtr, and(shr(6, input), 0x3F))), 0xFF)) out := shl(8, out) out := add(out, and(mload(add(tablePtr, and(input, 0x3F))), 0xFF)) out := shl(224, out) mstore(resultPtr, out) resultPtr := add(resultPtr, 4) } switch mod(len, 3) case 1 { mstore(sub(resultPtr, 2), shl(240, 0x3d3d)) } case 2 { mstore(sub(resultPtr, 1), shl(248, 0x3d)) } mstore(result, encodedLen) } return string(result); } }
// SPDX-License-Identifier: MIT pragma solidity ^0.8.0; // CAUTION // This version of SafeMath should only be used with Solidity 0.8 or later, // because it relies on the compiler's built in overflow checks. /** * @dev Wrappers over Solidity's arithmetic operations. * * NOTE: `SafeMath` is no longer needed starting with Solidity 0.8. The compiler * now has built in overflow checking. */ library SafeMath { /** * @dev Returns the addition of two unsigned integers, with an overflow flag. * * _Available since v3.4._ */ function tryAdd(uint256 a, uint256 b) internal pure returns (bool, uint256) { unchecked { uint256 c = a + b; if (c < a) return (false, 0); return (true, c); } } /** * @dev Returns the substraction of two unsigned integers, with an overflow flag. * * _Available since v3.4._ */ function trySub(uint256 a, uint256 b) internal pure returns (bool, uint256) { unchecked { if (b > a) return (false, 0); return (true, a - b); } } /** * @dev Returns the multiplication of two unsigned integers, with an overflow flag. * * _Available since v3.4._ */ function tryMul(uint256 a, uint256 b) internal pure returns (bool, uint256) { unchecked { // Gas optimization: this is cheaper than requiring 'a' not being zero, but the // benefit is lost if 'b' is also tested. // See: https://github.com/OpenZeppelin/openzeppelin-contracts/pull/522 if (a == 0) return (true, 0); uint256 c = a * b; if (c / a != b) return (false, 0); return (true, c); } } /** * @dev Returns the division of two unsigned integers, with a division by zero flag. * * _Available since v3.4._ */ function tryDiv(uint256 a, uint256 b) internal pure returns (bool, uint256) { unchecked { if (b == 0) return (false, 0); return (true, a / b); } } /** * @dev Returns the remainder of dividing two unsigned integers, with a division by zero flag. * * _Available since v3.4._ */ function tryMod(uint256 a, uint256 b) internal pure returns (bool, uint256) { unchecked { if (b == 0) return (false, 0); return (true, a % b); } } /** * @dev Returns the addition of two unsigned integers, reverting on * overflow. * * Counterpart to Solidity's `+` operator. * * Requirements: * * - Addition cannot overflow. */ function add(uint256 a, uint256 b) internal pure returns (uint256) { return a + b; } /** * @dev Returns the subtraction of two unsigned integers, reverting on * overflow (when the result is negative). * * Counterpart to Solidity's `-` operator. * * Requirements: * * - Subtraction cannot overflow. */ function sub(uint256 a, uint256 b) internal pure returns (uint256) { return a - b; } /** * @dev Returns the multiplication of two unsigned integers, reverting on * overflow. * * Counterpart to Solidity's `*` operator. * * Requirements: * * - Multiplication cannot overflow. */ function mul(uint256 a, uint256 b) internal pure returns (uint256) { return a * b; } /** * @dev Returns the integer division of two unsigned integers, reverting on * division by zero. The result is rounded towards zero. * * Counterpart to Solidity's `/` operator. * * Requirements: * * - The divisor cannot be zero. */ function div(uint256 a, uint256 b) internal pure returns (uint256) { return a / b; } /** * @dev Returns the remainder of dividing two unsigned integers. (unsigned integer modulo), * reverting when dividing by zero. * * Counterpart to Solidity's `%` operator. This function uses a `revert` * opcode (which leaves remaining gas untouched) while Solidity uses an * invalid opcode to revert (consuming all remaining gas). * * Requirements: * * - The divisor cannot be zero. */ function mod(uint256 a, uint256 b) internal pure returns (uint256) { return a % b; } /** * @dev Returns the subtraction of two unsigned integers, reverting with custom message on * overflow (when the result is negative). * * CAUTION: This function is deprecated because it requires allocating memory for the error * message unnecessarily. For custom revert reasons use {trySub}. * * Counterpart to Solidity's `-` operator. * * Requirements: * * - Subtraction cannot overflow. */ function sub( uint256 a, uint256 b, string memory errorMessage ) internal pure returns (uint256) { unchecked { require(b <= a, errorMessage); return a - b; } } /** * @dev Returns the integer division of two unsigned integers, reverting with custom message on * division by zero. The result is rounded towards zero. * * Counterpart to Solidity's `/` operator. Note: this function uses a * `revert` opcode (which leaves remaining gas untouched) while Solidity * uses an invalid opcode to revert (consuming all remaining gas). * * Requirements: * * - The divisor cannot be zero. */ function div( uint256 a, uint256 b, string memory errorMessage ) internal pure returns (uint256) { unchecked { require(b > 0, errorMessage); return a / b; } } /** * @dev Returns the remainder of dividing two unsigned integers. (unsigned integer modulo), * reverting with custom message when dividing by zero. * * CAUTION: This function is deprecated because it requires allocating memory for the error * message unnecessarily. For custom revert reasons use {tryMod}. * * Counterpart to Solidity's `%` operator. This function uses a `revert` * opcode (which leaves remaining gas untouched) while Solidity uses an * invalid opcode to revert (consuming all remaining gas). * * Requirements: * * - The divisor cannot be zero. */ function mod( uint256 a, uint256 b, string memory errorMessage ) internal pure returns (uint256) { unchecked { require(b > 0, errorMessage); return a % b; } } }
// SPDX-License-Identifier: MIT pragma solidity ^0.8.0; /** * @title Counters * @author Matt Condon (@shrugs) * @dev Provides counters that can only be incremented, decremented or reset. This can be used e.g. to track the number * of elements in a mapping, issuing ERC721 ids, or counting request ids. * * Include with `using Counters for Counters.Counter;` */ library Counters { struct Counter { // This variable should never be directly accessed by users of the library: interactions must be restricted to // the library's function. As of Solidity v0.5.2, this cannot be enforced, though there is a proposal to add // this feature: see https://github.com/ethereum/solidity/issues/4637 uint256 _value; // default: 0 } function current(Counter storage counter) internal view returns (uint256) { return counter._value; } function increment(Counter storage counter) internal { unchecked { counter._value += 1; } } function decrement(Counter storage counter) internal { uint256 value = counter._value; require(value > 0, "Counter: decrement overflow"); unchecked { counter._value = value - 1; } } function reset(Counter storage counter) internal { counter._value = 0; } }
// SPDX-License-Identifier: MIT pragma solidity ^0.8.0; import "../utils/Context.sol"; /** * @dev Contract module which provides a basic access control mechanism, where * there is an account (an owner) that can be granted exclusive access to * specific functions. * * By default, the owner account will be the one that deploys the contract. This * can later be changed with {transferOwnership}. * * This module is used through inheritance. It will make available the modifier * `onlyOwner`, which can be applied to your functions to restrict their use to * the owner. */ abstract contract Ownable is Context { address private _owner; event OwnershipTransferred(address indexed previousOwner, address indexed newOwner); /** * @dev Initializes the contract setting the deployer as the initial owner. */ constructor() { _setOwner(_msgSender()); } /** * @dev Returns the address of the current owner. */ function owner() public view virtual returns (address) { return _owner; } /** * @dev Throws if called by any account other than the owner. */ modifier onlyOwner() { require(owner() == _msgSender(), "Ownable: caller is not the owner"); _; } /** * @dev Leaves the contract without owner. It will not be possible to call * `onlyOwner` functions anymore. Can only be called by the current owner. * * NOTE: Renouncing ownership will leave the contract without an owner, * thereby removing any functionality that is only available to the owner. */ function renounceOwnership() public virtual onlyOwner { _setOwner(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"); _setOwner(newOwner); } function _setOwner(address newOwner) private { address oldOwner = _owner; _owner = newOwner; emit OwnershipTransferred(oldOwner, newOwner); } }
// SPDX-License-Identifier: MIT pragma solidity ^0.8.0; import "../ERC721.sol"; import "./IERC721Enumerable.sol"; /** * @dev This implements an optional extension of {ERC721} defined in the EIP that adds * enumerability of all the token ids in the contract as well as all token ids owned by each * account. */ abstract contract ERC721Enumerable is ERC721, IERC721Enumerable { // Mapping from owner to list of owned token IDs mapping(address => mapping(uint256 => uint256)) private _ownedTokens; // Mapping from token ID to index of the owner tokens list mapping(uint256 => uint256) private _ownedTokensIndex; // Array with all token ids, used for enumeration uint256[] private _allTokens; // Mapping from token id to position in the allTokens array mapping(uint256 => uint256) private _allTokensIndex; /** * @dev See {IERC165-supportsInterface}. */ function supportsInterface(bytes4 interfaceId) public view virtual override(IERC165, ERC721) returns (bool) { return interfaceId == type(IERC721Enumerable).interfaceId || super.supportsInterface(interfaceId); } /** * @dev See {IERC721Enumerable-tokenOfOwnerByIndex}. */ function tokenOfOwnerByIndex(address owner, uint256 index) public view virtual override returns (uint256) { require(index < ERC721.balanceOf(owner), "ERC721Enumerable: owner index out of bounds"); return _ownedTokens[owner][index]; } /** * @dev See {IERC721Enumerable-totalSupply}. */ function totalSupply() public view virtual override returns (uint256) { return _allTokens.length; } /** * @dev See {IERC721Enumerable-tokenByIndex}. */ function tokenByIndex(uint256 index) public view virtual override returns (uint256) { require(index < ERC721Enumerable.totalSupply(), "ERC721Enumerable: global index out of bounds"); return _allTokens[index]; } /** * @dev Hook that is called before any token transfer. This includes minting * and burning. * * 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, ``from``'s `tokenId` will be burned. * - `from` cannot be the zero address. * - `to` cannot be the zero address. * * To learn more about hooks, head to xref:ROOT:extending-contracts.adoc#using-hooks[Using Hooks]. */ function _beforeTokenTransfer( address from, address to, uint256 tokenId ) internal virtual override { super._beforeTokenTransfer(from, to, tokenId); if (from == address(0)) { _addTokenToAllTokensEnumeration(tokenId); } else if (from != to) { _removeTokenFromOwnerEnumeration(from, tokenId); } if (to == address(0)) { _removeTokenFromAllTokensEnumeration(tokenId); } else if (to != from) { _addTokenToOwnerEnumeration(to, tokenId); } } /** * @dev Private function to add a token to this extension's ownership-tracking data structures. * @param to address representing the new owner of the given token ID * @param tokenId uint256 ID of the token to be added to the tokens list of the given address */ function _addTokenToOwnerEnumeration(address to, uint256 tokenId) private { uint256 length = ERC721.balanceOf(to); _ownedTokens[to][length] = tokenId; _ownedTokensIndex[tokenId] = length; } /** * @dev Private function to add a token to this extension's token tracking data structures. * @param tokenId uint256 ID of the token to be added to the tokens list */ function _addTokenToAllTokensEnumeration(uint256 tokenId) private { _allTokensIndex[tokenId] = _allTokens.length; _allTokens.push(tokenId); } /** * @dev Private function to remove a token from this extension's ownership-tracking data structures. Note that * while the token is not assigned a new owner, the `_ownedTokensIndex` mapping is _not_ updated: this allows for * gas optimizations e.g. when performing a transfer operation (avoiding double writes). * This has O(1) time complexity, but alters the order of the _ownedTokens array. * @param from address representing the previous owner of the given token ID * @param tokenId uint256 ID of the token to be removed from the tokens list of the given address */ function _removeTokenFromOwnerEnumeration(address from, uint256 tokenId) private { // To prevent a gap in from's tokens array, we store the last token in the index of the token to delete, and // then delete the last slot (swap and pop). uint256 lastTokenIndex = ERC721.balanceOf(from) - 1; uint256 tokenIndex = _ownedTokensIndex[tokenId]; // When the token to delete is the last token, the swap operation is unnecessary if (tokenIndex != lastTokenIndex) { uint256 lastTokenId = _ownedTokens[from][lastTokenIndex]; _ownedTokens[from][tokenIndex] = lastTokenId; // Move the last token to the slot of the to-delete token _ownedTokensIndex[lastTokenId] = tokenIndex; // Update the moved token's index } // This also deletes the contents at the last position of the array delete _ownedTokensIndex[tokenId]; delete _ownedTokens[from][lastTokenIndex]; } /** * @dev Private function to remove a token from this extension's token tracking data structures. * This has O(1) time complexity, but alters the order of the _allTokens array. * @param tokenId uint256 ID of the token to be removed from the tokens list */ function _removeTokenFromAllTokensEnumeration(uint256 tokenId) private { // To prevent a gap in the tokens array, we store the last token in the index of the token to delete, and // then delete the last slot (swap and pop). uint256 lastTokenIndex = _allTokens.length - 1; uint256 tokenIndex = _allTokensIndex[tokenId]; // When the token to delete is the last token, the swap operation is unnecessary. However, since this occurs so // rarely (when the last minted token is burnt) that we still do the swap here to avoid the gas cost of adding // an 'if' statement (like in _removeTokenFromOwnerEnumeration) uint256 lastTokenId = _allTokens[lastTokenIndex]; _allTokens[tokenIndex] = lastTokenId; // Move the last token to the slot of the to-delete token _allTokensIndex[lastTokenId] = tokenIndex; // Update the moved token's index // This also deletes the contents at the last position of the array delete _allTokensIndex[tokenId]; _allTokens.pop(); } }
// SPDX-License-Identifier: MIT pragma solidity ^0.8.0; import "./IERC721.sol"; import "./IERC721Receiver.sol"; import "./extensions/IERC721Metadata.sol"; import "../../utils/Address.sol"; import "../../utils/Context.sol"; import "../../utils/Strings.sol"; import "../../utils/introspection/ERC165.sol"; /** * @dev Implementation of https://eips.ethereum.org/EIPS/eip-721[ERC721] Non-Fungible Token Standard, including * the Metadata extension, but not including the Enumerable extension, which is available separately as * {ERC721Enumerable}. */ contract ERC721 is Context, ERC165, IERC721, IERC721Metadata { using Address for address; using Strings for uint256; // Token name string private _name; // Token symbol string private _symbol; // Mapping from token ID to owner address mapping(uint256 => address) private _owners; // Mapping owner address to token count mapping(address => uint256) private _balances; // Mapping from token ID to approved address mapping(uint256 => address) private _tokenApprovals; // Mapping from owner to operator approvals mapping(address => mapping(address => bool)) private _operatorApprovals; /** * @dev Initializes the contract by setting a `name` and a `symbol` to the token collection. */ constructor(string memory name_, string memory symbol_) { _name = name_; _symbol = symbol_; } /** * @dev See {IERC165-supportsInterface}. */ function supportsInterface(bytes4 interfaceId) public view virtual override(ERC165, IERC165) returns (bool) { return interfaceId == type(IERC721).interfaceId || interfaceId == type(IERC721Metadata).interfaceId || super.supportsInterface(interfaceId); } /** * @dev See {IERC721-balanceOf}. */ function balanceOf(address owner) public view virtual override returns (uint256) { require(owner != address(0), "ERC721: balance query for the zero address"); return _balances[owner]; } /** * @dev See {IERC721-ownerOf}. */ function ownerOf(uint256 tokenId) public view virtual override returns (address) { address owner = _owners[tokenId]; require(owner != address(0), "ERC721: owner query for nonexistent token"); return owner; } /** * @dev See {IERC721Metadata-name}. */ function name() public view virtual override returns (string memory) { return _name; } /** * @dev See {IERC721Metadata-symbol}. */ function symbol() public view virtual override returns (string memory) { return _symbol; } /** * @dev See {IERC721Metadata-tokenURI}. */ function tokenURI(uint256 tokenId) public view virtual override returns (string memory) { require(_exists(tokenId), "ERC721Metadata: URI query for nonexistent token"); string memory baseURI = _baseURI(); return bytes(baseURI).length > 0 ? string(abi.encodePacked(baseURI, tokenId.toString())) : ""; } /** * @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, can be overriden in child contracts. */ function _baseURI() internal view virtual returns (string memory) { return ""; } /** * @dev See {IERC721-approve}. */ function approve(address to, uint256 tokenId) public virtual override { address owner = ERC721.ownerOf(tokenId); require(to != owner, "ERC721: approval to current owner"); require( _msgSender() == owner || isApprovedForAll(owner, _msgSender()), "ERC721: approve caller is not owner nor approved for all" ); _approve(to, tokenId); } /** * @dev See {IERC721-getApproved}. */ function getApproved(uint256 tokenId) public view virtual override returns (address) { require(_exists(tokenId), "ERC721: approved query for nonexistent token"); return _tokenApprovals[tokenId]; } /** * @dev See {IERC721-setApprovalForAll}. */ function setApprovalForAll(address operator, bool approved) public virtual override { require(operator != _msgSender(), "ERC721: approve to caller"); _operatorApprovals[_msgSender()][operator] = approved; emit ApprovalForAll(_msgSender(), operator, approved); } /** * @dev See {IERC721-isApprovedForAll}. */ function isApprovedForAll(address owner, address operator) public view virtual override returns (bool) { return _operatorApprovals[owner][operator]; } /** * @dev See {IERC721-transferFrom}. */ function transferFrom( address from, address to, uint256 tokenId ) public virtual override { //solhint-disable-next-line max-line-length require(_isApprovedOrOwner(_msgSender(), tokenId), "ERC721: transfer caller is not owner nor approved"); _transfer(from, to, tokenId); } /** * @dev See {IERC721-safeTransferFrom}. */ function safeTransferFrom( address from, address to, uint256 tokenId ) public virtual override { safeTransferFrom(from, to, tokenId, ""); } /** * @dev See {IERC721-safeTransferFrom}. */ function safeTransferFrom( address from, address to, uint256 tokenId, bytes memory _data ) public virtual override { require(_isApprovedOrOwner(_msgSender(), tokenId), "ERC721: transfer caller is not owner nor approved"); _safeTransfer(from, to, tokenId, _data); } /** * @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. * * `_data` is additional data, it has no specified format and it is sent in call to `to`. * * This internal function is equivalent to {safeTransferFrom}, and can be used to e.g. * implement alternative mechanisms to perform token transfer, such as signature-based. * * Requirements: * * - `from` cannot be the zero address. * - `to` cannot be the zero address. * - `tokenId` token must exist and be owned by `from`. * - If `to` refers to a smart contract, it must implement {IERC721Receiver-onERC721Received}, which is called upon a safe transfer. * * Emits a {Transfer} event. */ function _safeTransfer( address from, address to, uint256 tokenId, bytes memory _data ) internal virtual { _transfer(from, to, tokenId); require(_checkOnERC721Received(from, to, tokenId, _data), "ERC721: transfer to non ERC721Receiver implementer"); } /** * @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 (`_mint`), * and stop existing when they are burned (`_burn`). */ function _exists(uint256 tokenId) internal view virtual returns (bool) { return _owners[tokenId] != address(0); } /** * @dev Returns whether `spender` is allowed to manage `tokenId`. * * Requirements: * * - `tokenId` must exist. */ function _isApprovedOrOwner(address spender, uint256 tokenId) internal view virtual returns (bool) { require(_exists(tokenId), "ERC721: operator query for nonexistent token"); address owner = ERC721.ownerOf(tokenId); return (spender == owner || getApproved(tokenId) == spender || isApprovedForAll(owner, spender)); } /** * @dev Safely mints `tokenId` and transfers it to `to`. * * Requirements: * * - `tokenId` must not exist. * - If `to` refers to a smart contract, it must implement {IERC721Receiver-onERC721Received}, which is called upon a safe transfer. * * Emits a {Transfer} event. */ function _safeMint(address to, uint256 tokenId) internal virtual { _safeMint(to, tokenId, ""); } /** * @dev Same as {xref-ERC721-_safeMint-address-uint256-}[`_safeMint`], with an additional `data` parameter which is * forwarded in {IERC721Receiver-onERC721Received} to contract recipients. */ function _safeMint( address to, uint256 tokenId, bytes memory _data ) internal virtual { _mint(to, tokenId); require( _checkOnERC721Received(address(0), to, tokenId, _data), "ERC721: transfer to non ERC721Receiver implementer" ); } /** * @dev Mints `tokenId` and transfers it to `to`. * * WARNING: Usage of this method is discouraged, use {_safeMint} whenever possible * * Requirements: * * - `tokenId` must not exist. * - `to` cannot be the zero address. * * Emits a {Transfer} event. */ function _mint(address to, uint256 tokenId) internal virtual { require(to != address(0), "ERC721: mint to the zero address"); require(!_exists(tokenId), "ERC721: token already minted"); _beforeTokenTransfer(address(0), to, tokenId); _balances[to] += 1; _owners[tokenId] = to; emit Transfer(address(0), to, tokenId); } /** * @dev Destroys `tokenId`. * The approval is cleared when the token is burned. * * Requirements: * * - `tokenId` must exist. * * Emits a {Transfer} event. */ function _burn(uint256 tokenId) internal virtual { address owner = ERC721.ownerOf(tokenId); _beforeTokenTransfer(owner, address(0), tokenId); // Clear approvals _approve(address(0), tokenId); _balances[owner] -= 1; delete _owners[tokenId]; emit Transfer(owner, address(0), tokenId); } /** * @dev Transfers `tokenId` from `from` to `to`. * As opposed to {transferFrom}, this imposes no restrictions on msg.sender. * * Requirements: * * - `to` cannot be the zero address. * - `tokenId` token must be owned by `from`. * * Emits a {Transfer} event. */ function _transfer( address from, address to, uint256 tokenId ) internal virtual { require(ERC721.ownerOf(tokenId) == from, "ERC721: transfer of token that is not own"); require(to != address(0), "ERC721: transfer to the zero address"); _beforeTokenTransfer(from, to, tokenId); // Clear approvals from the previous owner _approve(address(0), tokenId); _balances[from] -= 1; _balances[to] += 1; _owners[tokenId] = to; emit Transfer(from, to, tokenId); } /** * @dev Approve `to` to operate on `tokenId` * * Emits a {Approval} event. */ function _approve(address to, uint256 tokenId) internal virtual { _tokenApprovals[tokenId] = to; emit Approval(ERC721.ownerOf(tokenId), to, tokenId); } /** * @dev Internal function to invoke {IERC721Receiver-onERC721Received} on a target address. * The call is not executed if the target address is not a contract. * * @param from address representing the previous owner of the given token ID * @param to target address that will receive the tokens * @param tokenId uint256 ID of the token to be transferred * @param _data bytes optional data to send along with the call * @return bool whether the call correctly returned the expected magic value */ function _checkOnERC721Received( address from, address to, uint256 tokenId, bytes memory _data ) private returns (bool) { if (to.isContract()) { try IERC721Receiver(to).onERC721Received(_msgSender(), from, tokenId, _data) returns (bytes4 retval) { return retval == IERC721Receiver.onERC721Received.selector; } catch (bytes memory reason) { if (reason.length == 0) { revert("ERC721: transfer to non ERC721Receiver implementer"); } else { assembly { revert(add(32, reason), mload(reason)) } } } } else { return true; } } /** * @dev Hook that is called before any token transfer. This includes minting * and burning. * * 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, ``from``'s `tokenId` will be burned. * - `from` and `to` are never both zero. * * To learn more about hooks, head to xref:ROOT:extending-contracts.adoc#using-hooks[Using Hooks]. */ function _beforeTokenTransfer( address from, address to, uint256 tokenId ) internal virtual {} }
// SPDX-License-Identifier: MIT pragma solidity ^0.8.0; import "../IERC721.sol"; /** * @title ERC-721 Non-Fungible Token Standard, optional enumeration extension * @dev See https://eips.ethereum.org/EIPS/eip-721 */ interface IERC721Enumerable is IERC721 { /** * @dev Returns the total amount of tokens stored by the contract. */ function totalSupply() external view returns (uint256); /** * @dev Returns a token ID owned by `owner` at a given `index` of its token list. * Use along with {balanceOf} to enumerate all of ``owner``'s tokens. */ function tokenOfOwnerByIndex(address owner, uint256 index) external view returns (uint256 tokenId); /** * @dev Returns a token ID at a given `index` of all the tokens stored by the contract. * Use along with {totalSupply} to enumerate all tokens. */ function tokenByIndex(uint256 index) external view returns (uint256); }
// SPDX-License-Identifier: MIT pragma solidity ^0.8.0; import "./IERC165.sol"; /** * @dev Implementation of the {IERC165} interface. * * Contracts that want to implement ERC165 should inherit from this contract and override {supportsInterface} to check * for the additional interface id that will be supported. For example: * * ```solidity * function supportsInterface(bytes4 interfaceId) public view virtual override returns (bool) { * return interfaceId == type(MyInterface).interfaceId || super.supportsInterface(interfaceId); * } * ``` * * Alternatively, {ERC165Storage} provides an easier to use but more expensive implementation. */ abstract contract ERC165 is IERC165 { /** * @dev See {IERC165-supportsInterface}. */ function supportsInterface(bytes4 interfaceId) public view virtual override returns (bool) { return interfaceId == type(IERC165).interfaceId; } }
// SPDX-License-Identifier: MIT pragma solidity ^0.8.0; /** * @dev String operations. */ library Strings { bytes16 private constant _HEX_SYMBOLS = "0123456789abcdef"; /** * @dev Converts a `uint256` to its ASCII `string` decimal representation. */ function toString(uint256 value) internal pure returns (string memory) { // Inspired by OraclizeAPI's implementation - MIT licence // https://github.com/oraclize/ethereum-api/blob/b42146b063c7d6ee1358846c198246239e9360e8/oraclizeAPI_0.4.25.sol if (value == 0) { return "0"; } uint256 temp = value; uint256 digits; while (temp != 0) { digits++; temp /= 10; } bytes memory buffer = new bytes(digits); while (value != 0) { digits -= 1; buffer[digits] = bytes1(uint8(48 + uint256(value % 10))); value /= 10; } return string(buffer); } /** * @dev Converts a `uint256` to its ASCII `string` hexadecimal representation. */ function toHexString(uint256 value) internal pure returns (string memory) { if (value == 0) { return "0x00"; } uint256 temp = value; uint256 length = 0; while (temp != 0) { length++; temp >>= 8; } return toHexString(value, length); } /** * @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] = _HEX_SYMBOLS[value & 0xf]; value >>= 4; } require(value == 0, "Strings: hex length insufficient"); return string(buffer); } }
// SPDX-License-Identifier: MIT pragma solidity ^0.8.0; /** * @dev Provides information about the current execution context, including the * sender of the transaction and its data. While these are generally available * via msg.sender and msg.data, they should not be accessed in such a direct * manner, since when dealing with meta-transactions the account sending and * paying for execution may not be the actual sender (as far as an application * is concerned). * * This contract is only required for intermediate, library-like contracts. */ abstract contract Context { function _msgSender() internal view virtual returns (address) { return msg.sender; } function _msgData() internal view virtual returns (bytes calldata) { return msg.data; } }
// SPDX-License-Identifier: MIT pragma solidity ^0.8.0; /** * @dev Collection of functions related to the address type */ library Address { /** * @dev Returns true if `account` is a contract. * * [IMPORTANT] * ==== * It is unsafe to assume that an address for which this function returns * false is an externally-owned account (EOA) and not a contract. * * Among others, `isContract` will return false for the following * types of addresses: * * - an externally-owned account * - a contract in construction * - an address where a contract will be created * - an address where a contract lived, but was destroyed * ==== */ function isContract(address account) internal view returns (bool) { // This method relies on extcodesize, which returns 0 for contracts in // construction, since the code is only stored at the end of the // constructor execution. uint256 size; assembly { size := extcodesize(account) } return size > 0; } /** * @dev Replacement for Solidity's `transfer`: sends `amount` wei to * `recipient`, forwarding all available gas and reverting on errors. * * https://eips.ethereum.org/EIPS/eip-1884[EIP1884] increases the gas cost * of certain opcodes, possibly making contracts go over the 2300 gas limit * imposed by `transfer`, making them unable to receive funds via * `transfer`. {sendValue} removes this limitation. * * https://diligence.consensys.net/posts/2019/09/stop-using-soliditys-transfer-now/[Learn more]. * * IMPORTANT: because control is transferred to `recipient`, care must be * taken to not create reentrancy vulnerabilities. Consider using * {ReentrancyGuard} or the * https://solidity.readthedocs.io/en/v0.5.11/security-considerations.html#use-the-checks-effects-interactions-pattern[checks-effects-interactions pattern]. */ function sendValue(address payable recipient, uint256 amount) internal { require(address(this).balance >= amount, "Address: insufficient balance"); (bool success, ) = recipient.call{value: amount}(""); require(success, "Address: unable to send value, recipient may have reverted"); } /** * @dev Performs a Solidity function call using a low level `call`. A * plain `call` is an unsafe replacement for a function call: use this * function instead. * * If `target` reverts with a revert reason, it is bubbled up by this * function (like regular Solidity function calls). * * Returns the raw returned data. To convert to the expected return value, * use https://solidity.readthedocs.io/en/latest/units-and-global-variables.html?highlight=abi.decode#abi-encoding-and-decoding-functions[`abi.decode`]. * * Requirements: * * - `target` must be a contract. * - calling `target` with `data` must not revert. * * _Available since v3.1._ */ function functionCall(address target, bytes memory data) internal returns (bytes memory) { return functionCall(target, data, "Address: low-level call failed"); } /** * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], but with * `errorMessage` as a fallback revert reason when `target` reverts. * * _Available since v3.1._ */ function functionCall( address target, bytes memory data, string memory errorMessage ) internal returns (bytes memory) { return functionCallWithValue(target, data, 0, errorMessage); } /** * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], * but also transferring `value` wei to `target`. * * Requirements: * * - the calling contract must have an ETH balance of at least `value`. * - the called Solidity function must be `payable`. * * _Available since v3.1._ */ function functionCallWithValue( address target, bytes memory data, uint256 value ) internal returns (bytes memory) { return functionCallWithValue(target, data, value, "Address: low-level call with value failed"); } /** * @dev Same as {xref-Address-functionCallWithValue-address-bytes-uint256-}[`functionCallWithValue`], but * with `errorMessage` as a fallback revert reason when `target` reverts. * * _Available since v3.1._ */ function functionCallWithValue( address target, bytes memory data, uint256 value, string memory errorMessage ) internal returns (bytes memory) { require(address(this).balance >= value, "Address: insufficient balance for call"); require(isContract(target), "Address: call to non-contract"); (bool success, bytes memory returndata) = target.call{value: value}(data); return verifyCallResult(success, returndata, errorMessage); } /** * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], * but performing a static call. * * _Available since v3.3._ */ function functionStaticCall(address target, bytes memory data) internal view returns (bytes memory) { return functionStaticCall(target, data, "Address: low-level static call failed"); } /** * @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`], * but performing a static call. * * _Available since v3.3._ */ function functionStaticCall( address target, bytes memory data, string memory errorMessage ) internal view returns (bytes memory) { require(isContract(target), "Address: static call to non-contract"); (bool success, bytes memory returndata) = target.staticcall(data); return verifyCallResult(success, returndata, errorMessage); } /** * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], * but performing a delegate call. * * _Available since v3.4._ */ function functionDelegateCall(address target, bytes memory data) internal returns (bytes memory) { return functionDelegateCall(target, data, "Address: low-level delegate call failed"); } /** * @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`], * but performing a delegate call. * * _Available since v3.4._ */ function functionDelegateCall( address target, bytes memory data, string memory errorMessage ) internal returns (bytes memory) { require(isContract(target), "Address: delegate call to non-contract"); (bool success, bytes memory returndata) = target.delegatecall(data); return verifyCallResult(success, returndata, errorMessage); } /** * @dev Tool to verifies that a low level call was successful, and revert if it wasn't, either by bubbling the * revert reason using the provided one. * * _Available since v4.3._ */ function verifyCallResult( bool success, bytes memory returndata, string memory errorMessage ) internal pure returns (bytes memory) { if (success) { return returndata; } else { // Look for revert reason and bubble it up if present if (returndata.length > 0) { // The easiest way to bubble the revert reason is using memory via assembly assembly { let returndata_size := mload(returndata) revert(add(32, returndata), returndata_size) } } else { revert(errorMessage); } } } }
// SPDX-License-Identifier: MIT pragma solidity ^0.8.0; import "../IERC721.sol"; /** * @title ERC-721 Non-Fungible Token Standard, optional metadata extension * @dev See https://eips.ethereum.org/EIPS/eip-721 */ interface IERC721Metadata is IERC721 { /** * @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); }
// SPDX-License-Identifier: MIT pragma solidity ^0.8.0; /** * @title ERC721 token receiver interface * @dev Interface for any contract that wants to support safeTransfers * from ERC721 asset contracts. */ interface IERC721Receiver { /** * @dev Whenever an {IERC721} `tokenId` token is transferred to this contract via {IERC721-safeTransferFrom} * by `operator` from `from`, this function is called. * * It must return its Solidity selector to confirm the token transfer. * If any other value is returned or the interface is not implemented by the recipient, the transfer will be reverted. * * The selector can be obtained in Solidity with `IERC721.onERC721Received.selector`. */ function onERC721Received( address operator, address from, uint256 tokenId, bytes calldata data ) external returns (bytes4); }
// SPDX-License-Identifier: MIT pragma solidity ^0.8.0; import "../../utils/introspection/IERC165.sol"; /** * @dev Required interface of an ERC721 compliant contract. */ interface IERC721 is IERC165 { /** * @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 ) external; /** * @dev Transfers `tokenId` token 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; /** * @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; /** * @dev Returns the account approved for `tokenId` token. * * Requirements: * * - `tokenId` must exist. */ function getApproved(uint256 tokenId) external view returns (address operator); /** * @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 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); /** * @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 calldata data ) external; }
// SPDX-License-Identifier: MIT pragma solidity ^0.8.0; /** * @dev Interface of the ERC165 standard, as defined in the * https://eips.ethereum.org/EIPS/eip-165[EIP]. * * Implementers can declare support of contract interfaces, which can then be * queried by others ({ERC165Checker}). * * For an implementation, see {ERC165}. */ interface IERC165 { /** * @dev Returns true if this contract implements the interface defined by * `interfaceId`. See the corresponding * https://eips.ethereum.org/EIPS/eip-165#how-interfaces-are-identified[EIP section] * to learn more about how these ids are created. * * This function call must use less than 30 000 gas. */ function supportsInterface(bytes4 interfaceId) external view returns (bool); }
{ "optimizer": { "enabled": true, "runs": 200 }, "outputSelection": { "*": { "*": [ "evm.bytecode", "evm.deployedBytecode", "abi" ] } } }
Contract Security Audit
- No Contract Security Audit Submitted- Submit Audit Here
[{"inputs":[],"stateMutability":"nonpayable","type":"constructor"},{"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":"address","name":"previousOwner","type":"address"},{"indexed":true,"internalType":"address","name":"newOwner","type":"address"}],"name":"OwnershipTransferred","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"from","type":"address"},{"indexed":true,"internalType":"address","name":"to","type":"address"},{"indexed":true,"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"Transfer","type":"event"},{"inputs":[{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"approve","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"owner","type":"address"}],"name":"balanceOf","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"tokensNumber","type":"uint256"}],"name":"buySunsets","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[],"name":"claim","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":"uint256","name":"value","type":"uint256"},{"internalType":"uint256","name":"is_mirror","type":"uint256"}],"name":"getGrass","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"pure","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":[],"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":"renounceOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"to","type":"address"}],"name":"safeMint","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":"nonpayable","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":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"operator","type":"address"},{"internalType":"bool","name":"approved","type":"bool"}],"name":"setApprovalForAll","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"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":"index","type":"uint256"}],"name":"tokenByIndex","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"owner","type":"address"},{"internalType":"uint256","name":"index","type":"uint256"}],"name":"tokenOfOwnerByIndex","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"tokenURI","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"pure","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":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"newOwner","type":"address"}],"name":"transferOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"withdraw","outputs":[],"stateMutability":"nonpayable","type":"function"}]
Contract Creation Code
6080604052666a94d74f430000600c553480156200001c57600080fd5b50604080518082018252601081526f4f6e436861696e20536561736964657360801b60208083019182528351808501909452600384526253454160e81b90840152815191929162000070916000916200011f565b508051620000869060019060208401906200011f565b505050620000a36200009d620000c060201b60201c565b620000c4565b620000ba600b6200011660201b620022811760201c565b62000202565b3390565b600a80546001600160a01b038381166001600160a01b0319831681179093556040519116919082907f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e090600090a35050565b80546001019055565b8280546200012d90620001c5565b90600052602060002090601f0160209004810192826200015157600085556200019c565b82601f106200016c57805160ff19168380011785556200019c565b828001600101855582156200019c579182015b828111156200019c5782518255916020019190600101906200017f565b50620001aa929150620001ae565b5090565b5b80821115620001aa5760008155600101620001af565b600181811c90821680620001da57607f821691505b60208210811415620001fc57634e487b7160e01b600052602260045260246000fd5b50919050565b6159c480620002126000396000f3fe60806040526004361061014b5760003560e01c80634f6ccce7116100b657806398bd87631161006f57806398bd876314610385578063a22cb465146103a5578063b88d4fde146103c5578063c87b56dd146103e5578063e985e9c514610405578063f2fde38b1461044e57600080fd5b80634f6ccce7146102dd5780636352211e146102fd57806370a082311461031d578063715018a61461033d5780638da5cb5b1461035257806395d89b411461037057600080fd5b806323b872dd1161010857806323b872dd146102335780632f745c59146102535780633ccfd60b1461027357806340d097c31461028857806342842e0e146102a85780634e71d92d146102c857600080fd5b806301ffc9a71461015057806306fdde0314610185578063081812fc146101a7578063095ea7b3146101df5780631172219e1461020157806318160ddd14610214575b600080fd5b34801561015c57600080fd5b5061017061016b366004613ae9565b61046e565b60405190151581526020015b60405180910390f35b34801561019157600080fd5b5061019a61047f565b60405161017c919061436c565b3480156101b357600080fd5b506101c76101c2366004613b23565b610511565b6040516001600160a01b03909116815260200161017c565b3480156101eb57600080fd5b506101ff6101fa366004613abf565b6105ab565b005b6101ff61020f366004613b23565b6106c1565b34801561022057600080fd5b506008545b60405190815260200161017c565b34801561023f57600080fd5b506101ff61024e36600461396b565b610836565b34801561025f57600080fd5b5061022561026e366004613abf565b610867565b34801561027f57600080fd5b506101ff6108fd565b34801561029457600080fd5b506101ff6102a336600461391d565b610956565b3480156102b457600080fd5b506101ff6102c336600461396b565b61099e565b3480156102d457600080fd5b506101ff6109b9565b3480156102e957600080fd5b506102256102f8366004613b23565b610a00565b34801561030957600080fd5b506101c7610318366004613b23565b610a93565b34801561032957600080fd5b5061022561033836600461391d565b610b0a565b34801561034957600080fd5b506101ff610b91565b34801561035e57600080fd5b50600a546001600160a01b03166101c7565b34801561037c57600080fd5b5061019a610bc5565b34801561039157600080fd5b5061019a6103a0366004613b3c565b610bd4565b3480156103b157600080fd5b506101ff6103c0366004613a83565b61122e565b3480156103d157600080fd5b506101ff6103e03660046139a7565b6112f3565b3480156103f157600080fd5b5061019a610400366004613b23565b61132b565b34801561041157600080fd5b50610170610420366004613938565b6001600160a01b03918216600090815260056020908152604080832093909416825291909152205460ff1690565b34801561045a57600080fd5b506101ff61046936600461391d565b6121e9565b60006104798261228a565b92915050565b60606000805461048e906146ec565b80601f01602080910402602001604051908101604052809291908181526020018280546104ba906146ec565b80156105075780601f106104dc57610100808354040283529160200191610507565b820191906000526020600020905b8154815290600101906020018083116104ea57829003601f168201915b5050505050905090565b6000818152600260205260408120546001600160a01b031661058f5760405162461bcd60e51b815260206004820152602c60248201527f4552433732313a20617070726f76656420717565727920666f72206e6f6e657860448201526b34b9ba32b73a103a37b5b2b760a11b60648201526084015b60405180910390fd5b506000908152600460205260409020546001600160a01b031690565b60006105b682610a93565b9050806001600160a01b0316836001600160a01b031614156106245760405162461bcd60e51b815260206004820152602160248201527f4552433732313a20617070726f76616c20746f2063757272656e74206f776e656044820152603960f91b6064820152608401610586565b336001600160a01b038216148061064057506106408133610420565b6106b25760405162461bcd60e51b815260206004820152603860248201527f4552433732313a20617070726f76652063616c6c6572206973206e6f74206f7760448201527f6e6572206e6f7220617070726f76656420666f7220616c6c00000000000000006064820152608401610586565b6106bc83836122af565b505050565b600081116107005760405162461bcd60e51b815260206004820152600c60248201526b15dc9bdb99c8185b5bdd5b9d60a21b6044820152606401610586565b601e8111156107645760405162461bcd60e51b815260206004820152602a60248201527f4d617820746f6b656e7320706572207472616e73616374696f6e206e756d62656044820152691c88195e18d95959195960b21b6064820152608401610586565b610ce461077a82610774600b5490565b9061231d565b11156107985760405162461bcd60e51b8152600401610586906143d1565b600c5434906107a79083612330565b11156107f55760405162461bcd60e51b815260206004820152601b60248201527f45746865722076616c75652073656e7420697320746f6f206c6f7700000000006044820152606401610586565b60005b81811015610832576108123361080d600b5490565b61233c565b610820600b80546001019055565b8061082a81614727565b9150506107f8565b5050565b6108403382612356565b61085c5760405162461bcd60e51b81526004016105869061445b565b6106bc83838361244d565b600061087283610b0a565b82106108d45760405162461bcd60e51b815260206004820152602b60248201527f455243373231456e756d657261626c653a206f776e657220696e646578206f7560448201526a74206f6620626f756e647360a81b6064820152608401610586565b506001600160a01b03919091166000908152600660209081526040808320938352929052205490565b600a546001600160a01b031633146109275760405162461bcd60e51b815260040161058690614426565b6040514790339082156108fc029083906000818181858888f19350505050158015610832573d6000803e3d6000fd5b600a546001600160a01b031633146109805760405162461bcd60e51b815260040161058690614426565b61098d8161080d600b5490565b61099b600b80546001019055565b50565b6106bc838383604051806020016040528060008152506112f3565b61014d6109c5600b5490565b11156109e35760405162461bcd60e51b8152600401610586906143d1565b6109f03361080d600b5490565b6109fe600b80546001019055565b565b6000610a0b60085490565b8210610a6e5760405162461bcd60e51b815260206004820152602c60248201527f455243373231456e756d657261626c653a20676c6f62616c20696e646578206f60448201526b7574206f6620626f756e647360a01b6064820152608401610586565b60088281548110610a8157610a816147c9565b90600052602060002001549050919050565b6000818152600260205260408120546001600160a01b0316806104795760405162461bcd60e51b815260206004820152602960248201527f4552433732313a206f776e657220717565727920666f72206e6f6e657869737460448201526832b73a103a37b5b2b760b91b6064820152608401610586565b60006001600160a01b038216610b755760405162461bcd60e51b815260206004820152602a60248201527f4552433732313a2062616c616e636520717565727920666f7220746865207a65604482015269726f206164647265737360b01b6064820152608401610586565b506001600160a01b031660009081526003602052604090205490565b600a546001600160a01b03163314610bbb5760405162461bcd60e51b815260040161058690614426565b6109fe60006125f8565b60606001805461048e906146ec565b6060600080808080808060018160fa6040519080825280601f01601f191660200182016040528015610c0d576020820181803683370190505b509050610c186137f4565b7f013f26ad80a3665e606c4eb755430dfa59a6131a4231e4002f9594eb6f8d8d6181527c2532c0544b22a3db43ebb53ac374bfa1a8856080f009cac8bb8a5ce99660208201527f013f23c1d2f15869a74f46157a0d9ce5483c1b6d416baa1c1ba2cbfbe13b744060408201527c25372fb3c545de4cccfe1d373df28795888191dffc101dc2434bf1e23d60608201527f013f39039259ac91dc503b9e331c79e866e2a21c236775ad088d0b209aea5e9b60808201527c252bdcae587671d3dfaba908d50dd7e1a0daf274e0c640d85e24ef808c60a082015260009250806002610d018f82614634565b610d0b9190614692565b60068110610d1b57610d1b6147c9565b60200201519550600099508996508694508493505b60468a10156111705784610d4381614727565b9550610d51906102ee61458c565b610d5b9087614533565b9850610d696102ee8a614756565b9750876102ed1415610dbe578260011415610d8357611170565b600192508083610d948f6002614634565b610d9e9190614692565b60068110610dae57610dae6147c9565b602002015195506000945061115e565b610dc9600285614756565b610e3d578b60011415610de957610de288610190614653565b9750610df8565b610df5610226896144ac565b97505b8915610e3857600160fd1b8288610e0e81614727565b995081518110610e2057610e206147c9565b60200101906001600160f81b031916908160001a9053505b610e8c565b8915610e7d57600b60fa1b8288610e5381614727565b995081518110610e6557610e656147c9565b60200101906001600160f81b031916908160001a9053505b610e896101f4896144ac565b97505b6000881215610eda57610e9e8861476a565b9750602d60f81b8288610eb081614727565b995081518110610ec257610ec26147c9565b60200101906001600160f81b031916908160001a9053505b600a881215610f3557610eee600a89614742565b610ef99060306144ed565b60f81b8288610f0781614727565b995081518110610f1957610f196147c9565b60200101906001600160f81b031916908160001a905350611150565b6064881215610f9657610f49600a89614505565b610f549060306144ed565b60f81b8288610f6281614727565b995081518110610f7457610f746147c9565b60200101906001600160f81b031916908160001a905350610eee600a89614742565b6103e888121561100357610fab606489614505565b610fb69060306144ed565b60f81b8288610fc481614727565b995081518110610fd657610fd66147c9565b60200101906001600160f81b031916908160001a905350600a610ff9818a614505565b610f499190614742565b61100f6103e889614505565b61101a9060306144ed565b60f81b828861102881614727565b99508151811061103a5761103a6147c9565b60200101906001600160f81b031916908160001a905350600a61105e60648a614505565b6110689190614742565b6110739060306144ed565b60f81b828861108181614727565b995081518110611093576110936147c9565b60200101906001600160f81b031916908160001a905350600a6110b6818a614505565b6110c09190614742565b6110cb9060306144ed565b60f81b82886110d981614727565b9950815181106110eb576110eb6147c9565b60200101906001600160f81b031916908160001a90535061110d600a89614742565b6111189060306144ed565b60f81b828861112681614727565b995081518110611138576111386147c9565b60200101906001600160f81b031916908160001a9053505b8361115a81614727565b9450505b8961116881614727565b9a5050610d30565b60008767ffffffffffffffff81111561118b5761118b6147df565b6040519080825280601f01601f1916602001820160405280156111b5576020820181803683370190505b50905060009a505b878b101561121d57828b815181106111d7576111d76147c9565b602001015160f81c60f81b818c815181106111f4576111f46147c9565b60200101906001600160f81b031916908160001a9053508a61121581614727565b9b50506111bd565b9d9c50505050505050505050505050565b6001600160a01b0382163314156112875760405162461bcd60e51b815260206004820152601960248201527f4552433732313a20617070726f766520746f2063616c6c6572000000000000006044820152606401610586565b3360008181526005602090815260408083206001600160a01b03871680855290835292819020805460ff191686151590811790915590519081529192917f17307eab39ab6107e8899845ad3d59bd9653f200f220920489ca2b5937696c31910160405180910390a35050565b6112fd3383612356565b6113195760405162461bcd60e51b81526004016105869061445b565b6113258484848461264a565b50505050565b6060611335613812565b61133d613831565b611345613858565b61134d613873565b611355613873565b61135d61388e565b60008061139061136c8b61267d565b60405160200161137c9190613f25565b60405160208183030381529060405261277b565b9050600861139f600a83614533565b6113a99190614756565b6113b49060016144ed565b835260026113c3606483614533565b6113cd9190614756565b6113d89060016144ed565b602084015260256113eb61271083614533565b6113f59190614756565b6114009060016144ed565b60408401526008611414620f424083614533565b61141e9190614756565b6114299060016144ed565b6060840152600661143e6305f5e10083614533565b6114489190614756565b6114539060016144ed565b60808401526004611468633b9aca0083614533565b6114729190614756565b61147d9060016144ed565b60a084015260076114936402540be40083614533565b61149d9190614756565b6114a89060016144ed565b60c084015260046114be64174876e80083614533565b6114c89190614756565b6114d39060016144ed565b60e0840152604080518082018252601781527f37382c343434202d3439342c343738203635312c343738000000000000000000602080830191909152908752815160608101909252603080835290615413908301398560016020020181905250604051806060016040528060278152602001614b4c60279139604080870191909152805160808101909152604f808252614f246020830139606080870191909152604080519182019052602d808252614f736020830139608086015260408051606081019091526030808252615413602083013960a08601526040805160608101909152602f8082526148ab602083013960c086015260408051608081019091526060808252614e5c602083013960e0860152604080518082018252601881527f3136322c333932202d3330372c343738203633322c3437380000000000000000602080830191909152908652815180820183526000808252878301919091528251808301845281815287840152825180830184528181526060880152825180840184528281527f3939312c343130203832362c343534203631312c34373820313236322c3437388184015260808801528251808301845281815260a088015282518084018452601e81527f3135362c3433382034372c343738203332312c343738203231342c34363100008184015260c08801528251918201909252908152846007602090810291909101919091527cbf98fffb991e1362256591ffa922dbfbfd55003975004eff997affff9f89527d470000820000ff3b3bffc79926134a500070ff3b3bfffb9926134a500070908901527c95fffffc99690000c4000affc300fffc998a0000c44800ffd13bfffc996040808a01919091527c332700573b00ddfffffc9900023647006b0095fffffc990002362e008260608a01527ce0a1fff69900332700573e00e0a1fff6990033270a570000e0a1fff69960808a01527d140033002e57bdff52fff699140033002e57a473fffff699140033002e5760a08a01527dff8359fffcde3e0047870068ff9914fff699140033290087ff9914fff69960c08a01527c2330004459599cfffffcde002330004459ee59fffffcde00233000445960e08a01527dffd199f2ffff002330004459ffd199f2ffff002330004459aaff99fffcde6101008a01527c153000395999ffe9f2ffff13003044005999ffe9f2ffff1300304400596101208a01527dff0000ffc021400c16750000ff7824fff58200153000395980d4fff2ffff6101408a01527d5c0011a80000ffa100ffd15e330009610000ff0077ffc0213300096100006101608a01527dbf0000ffcc4c2d10384d0085ff66f0ffe0914a1b5c4931a8ff66f0ffd15e6101808a01527d030303242424c4c4c4ebebeb3000234a0035ff5900ffcc4c3000234a00356101a08a0152770d18331b2a610083c429ffa62d0047a754ff5200c4ff57d86101c08a015283015161190990600190614692565b611914906004614634565b91506119716301000000611929600a85614756565b61193790630100000061458c565b8a611943600a87614533565b60108110611953576119536147c9565b60200201516119629190614533565b61196c9190614756565b6127ac565b8752604083015161198490600190614692565b61198f906004614634565b61199a9060016144ed565b91506119af6301000000611929600a85614756565b602088015260408301516119c590600190614692565b6119d0906004614634565b6119db9060026144ed565b91506119f06301000000611929600a85614756565b604080890191909152830151611a0890600190614692565b611a13906004614634565b611a1e9060036144ed565b9150611a336301000000611929600a85614756565b606088015260408051610140810190915261011480825261547e6020830139865260408051606081019091526028808252614883602083013986600160200201819052506040518060a00160405280607e81526020016150fa607e9139604087015260608301518490611aa890600190614692565b60088110611ab857611ab86147c9565b6020020151604051602001611acd9190613f55565b60408051601f1981840301815291905286600360200201528460018460036020020151611afa9190614692565b60088110611b0a57611b0a6147c9565b6020020151604051602001611b1f9190614161565b60408051808303601f1901815291815260808089019290925280519182019052604f808252614834602083013960a0808801919091526040805191820190526065808252614abd602083013960c0870152604080516102408101909152610215808252614c47602083013960e08701526040805161024081019091526102178082526151fc602083013961010087015282516020840151611bc2919060006128cb565b604051602001611bd291906140fc565b60408051808303601f1901815291905261012087015282516020840151611bfb919060016128cb565b6020840151600114611c2657604051806040016040528060018152602001602d60f81b815250611c37565b604051806020016040528060008152505b611c5f60028660075b6020020151611c50906002614634565b611c5a9190614692565b612ed9565b604051602001611c71939291906141d1565b60408051808303601f19018152918152610140880191909152805160608101909152602d808252614a6d6020830139610160870152604080516060810190915260338082526155926020830139610180870152611cd16001846007611c40565b611cde6002856007611c40565b604051602001611cef929190613d46565b60408051808303601f190181529190526101a08701526080830151611d3690611d1a90600390614533565b611d259060016144ed565b60808501516103a090600290614756565b606088015160a0850151611d4990612f91565b604051602001611d5b93929190613fba565b60408051808303601f190181529181526101c088019190915280518082018252600a8082526911103837b4b73a399e9160b11b6020808401919091526101e08a01929092528251808401845290815269011179f1e17b9bb339f160b51b81830152610200890152875189820151828a01518b8501519451600095611de195909101613be9565b60408051808303601f1901815282825289820151918b015160608b0151919450611e119385939290602001613be9565b60408051808303601f19018152828252908a015160808a0151919350611e3e928492908290602001613be9565b60408051808303601f1901815282825260a08a0151918b015160c08b0151919450611e6f9385939290602001613be9565b60408051808303601f19018152828252908a015160e08a0151919350611e9c928492908290602001613be9565b60408051808303601f190181528282526101008a0151918b01516101208b0151919450611ecf9385939290602001613be9565b60408051808303601f19018152828252908a01516101408a01518b51929450611efd93859390602001613be9565b60408051808303601f190181529082905261016089015160208b8101516101808c0151939550611f31948694919201613be9565b60408051808303601f19018152908290526020808b01516101a08b015160608d0151939550611f67948694929391929101613be9565b60408051808303601f19018152908290526101c089015160608b01516101e08b0151929450611f9b93859390602001613be9565b60408051601f19818403018152919052905080611fbe8560066020020151613048565b610200890151604051611fd693929190602001613ba6565b6040516020818303038152906040529050611fef6138ad565b604051806060016040528060238152602001614a9a60239139815261201b8560005b602002015161267d565b81600160200201819052506040518060600160405280602a8152602001614b22602a9139604082015261204f856002612011565b606080830191909152604080519182019052602880825261480c6020830139608082015261207e856003612011565b60a082015260408051606081019091526026808252615969602083013960c08201526120ab856007612011565b60e082015260408051606081019091526028808252614efc60208301396101008201526120d9856006612011565b610120820152604080518082018252600481526322207d5d60e01b602080830191909152610140840191909152825181840151838501516060860151608087015160a0880151965160009761213097909101613c40565b60408051808303601f190181529082905260c084015160e085015161010086015161012087015161014088015194965061216f95879590602001613c40565b604051602081830303815290604052905060006121b48261218f8661315d565b6040516020016121a0929190613de4565b60405160208183030381529060405261315d565b9050806040516020016121c79190613ee0565b60408051601f198184030181529190529e9d5050505050505050505050505050565b600a546001600160a01b031633146122135760405162461bcd60e51b815260040161058690614426565b6001600160a01b0381166122785760405162461bcd60e51b815260206004820152602660248201527f4f776e61626c653a206e6577206f776e657220697320746865207a65726f206160448201526564647265737360d01b6064820152608401610586565b61099b816125f8565b80546001019055565b60006001600160e01b0319821663780e9d6360e01b14806104795750610479826132c3565b600081815260046020526040902080546001600160a01b0319166001600160a01b03841690811790915581906122e482610a93565b6001600160a01b03167f8c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b92560405160405180910390a45050565b600061232982846144ed565b9392505050565b60006123298284614634565b610832828260405180602001604052806000815250613313565b6000818152600260205260408120546001600160a01b03166123cf5760405162461bcd60e51b815260206004820152602c60248201527f4552433732313a206f70657261746f7220717565727920666f72206e6f6e657860448201526b34b9ba32b73a103a37b5b2b760a11b6064820152608401610586565b60006123da83610a93565b9050806001600160a01b0316846001600160a01b031614806124155750836001600160a01b031661240a84610511565b6001600160a01b0316145b8061244557506001600160a01b0380821660009081526005602090815260408083209388168352929052205460ff165b949350505050565b826001600160a01b031661246082610a93565b6001600160a01b0316146124c85760405162461bcd60e51b815260206004820152602960248201527f4552433732313a207472616e73666572206f6620746f6b656e2074686174206960448201526839903737ba1037bbb760b91b6064820152608401610586565b6001600160a01b03821661252a5760405162461bcd60e51b8152602060048201526024808201527f4552433732313a207472616e7366657220746f20746865207a65726f206164646044820152637265737360e01b6064820152608401610586565b612535838383613346565b6125406000826122af565b6001600160a01b0383166000908152600360205260408120805460019290612569908490614692565b90915550506001600160a01b03821660009081526003602052604081208054600192906125979084906144ed565b909155505060008181526002602052604080822080546001600160a01b0319166001600160a01b0386811691821790925591518493918716917fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef91a4505050565b600a80546001600160a01b038381166001600160a01b0319831681179093556040519116919082907f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e090600090a35050565b61265584848461244d565b61266184848484613351565b6113255760405162461bcd60e51b81526004016105869061437f565b6060816126a15750506040805180820190915260018152600360fc1b602082015290565b8160005b81156126cb57806126b581614727565b91506126c49050600a83614533565b91506126a5565b60008167ffffffffffffffff8111156126e6576126e66147df565b6040519080825280601f01601f191660200182016040528015612710576020820181803683370190505b5090505b841561244557612725600183614692565b9150612732600a86614756565b61273d9060306144ed565b60f81b818381518110612752576127526147c9565b60200101906001600160f81b031916908160001a905350612774600a86614533565b9450612714565b60008160405160200161278e9190613b8a565b60408051601f19818403018152919052805160209091012092915050565b6060816127d557505060408051808201909152600681526503030303030360d41b602082015290565b6040805180820190915260068082526503030303030360d41b6020830152905b811561232957600a612808601086614756565b101561285f57612819601085614756565b6128249060306144ed565b60f81b81612833600185614692565b81518110612843576128436147c9565b60200101906001600160f81b031916908160001a9053506128ac565b61286a601085614756565b6128759060376144ed565b60f81b81612884600185614692565b81518110612894576128946147c9565b60200101906001600160f81b031916908160001a9053505b6128b7601085614533565b9350816128c3816146d5565b9250506127f5565b6040805161012c80825261016082019092526060916000918291829182918291829160019183916020820181803683370190505090506129096138c8565b7b265c95854022eaa8a826db38647b201739ce4f24204af6e0aeb97230815275a8ab0f37f5ec2ffcaac252f15949a052559bd538cbe560208201527d1768e4c744107bec6966e48b6070bd558da53030af9c8422b1af5a6e4660604082015275a8a45d923ba48d952c5e519add1ee679dc58aa97fd0160608201527f0e48d317e96bb0676c73e1d91d25620a84c8154c68ff2ed985ef7adc27c1bd0c608082015275a8b483ffdd8795098e01fc17b102a5a54877a6d4f68060a08201527d1766d7abb97dd372b1290c2d7e4fd5586924479bed6a1e1763d811595a4260c08201527002e5d63388f891d6087aea8c6e0084d7b460e08201527d17624acfa651e03c326754582994d729a8f7b92398e67cc3d71b62a792f8610100820152793ed5c4b8964e9feddde8a45588adebd10888fab0bbdb9052e69b61012082015260078d1015612a785780612a5c60018f614692565b600a8110612a6c57612a6c6147c9565b60200201519450612aaa565b60009250808d600714612a8c576008612a8f565b60065b60ff16600a8110612aa257612aa26147c9565b602002015194505b60008b600114612abc576101cc612ac0565b6102305b61ffff169050600099508996508694505b60468a1015612e1a5784612ae481614727565b9550612af19060c861458c565b612afb9087614533565b9850612b0860c88a614756565b97508760961415612b5b578360011415612b2157612e1a565b60019350818e600714612b35576009612b38565b60075b60ff16600a8110612b4b57612b4b6147c9565b6020020151955060009450612e08565b612b66600286614756565b612be4578b60011415612b9257612b7f886101406144ed565b612b8b906103e8614692565b9750612b9f565b612b9c81896144ed565b97505b8915612bdf57600b60fa1b8388612bb581614727565b995081518110612bc757612bc76147c9565b60200101906001600160f81b031916908160001a9053505b612c59565b8c60011415612bff57612bf86064896144ed565b9750612c19565b612c0a8860646144ed565b612c16906103e8614692565b97505b8915612c5957600160fd1b8388612c2f81614727565b995081518110612c4157612c416147c9565b60200101906001600160f81b031916908160001a9053505b600a881015612cb457612c6d600a89614756565b612c789060306144ed565b60f81b8388612c8681614727565b995081518110612c9857612c986147c9565b60200101906001600160f81b031916908160001a905350612e08565b6064881015612d1557612cc8600a89614533565b612cd39060306144ed565b60f81b8388612ce181614727565b995081518110612cf357612cf36147c9565b60200101906001600160f81b031916908160001a905350612c6d600a89614756565b612d20606489614533565b612d2b9060306144ed565b60f81b8388612d3981614727565b995081518110612d4b57612d4b6147c9565b60200101906001600160f81b031916908160001a905350600a612d6e818a614533565b612d789190614756565b612d839060306144ed565b60f81b8388612d9181614727565b995081518110612da357612da36147c9565b60200101906001600160f81b031916908160001a905350612dc5600a89614756565b612dd09060306144ed565b60f81b8388612dde81614727565b995081518110612df057612df06147c9565b60200101906001600160f81b031916908160001a9053505b89612e1281614727565b9a5050612ad1565b60008767ffffffffffffffff811115612e3557612e356147df565b6040519080825280601f01601f191660200182016040528015612e5f576020820181803683370190505b50905060009a505b878b1015612ec757838b81518110612e8157612e816147c9565b602001015160f81c60f81b818c81518110612e9e57612e9e6147c9565b60200101906001600160f81b031916908160001a9053508a612ebf81614727565b9b5050612e67565b9e9d5050505050505050505050505050565b60607d6735464658dcf4043d5da3a3ecc8c16957e326064fbc66daeee5b6a78b598215612f2457612f0b836003614634565b612f17906103e861458c565b612f219082614533565b90505b6000612f486103e8612f39620f424085614533565b612f439190614756565b61267d565b612f586103e8612f398186614533565b612f67612f436103e886614756565b604051602001612f7993929190613cbf565b60408051601f19818403018152919052949350505050565b6060612f9b613831565b60405180610180016040528061015981526020016148da610159913981526040805161018081019091526101488082526158216020830139816001602002018190525060405180610280016040528061025c81526020016155c561025c91396040808301919091528051610180810190915261015a808252614fa0602083013960608201528061302c600185614692565b6004811061303c5761303c6147c9565b60200201519392505050565b60606130526138e7565b604080518082018252601c81527f2d33302c3839382c2d313335372c313031362c313239382c3130313600000000602080830191909152908352815160608101909252603a80835290614c0d9083013981600160200201819052506040518060600160405280603a8152602001614a33603a9139604080830191909152805160608101909152603b80825261544360208301396060820152604080516080810190915260428082526151ba602083013960808201526040805160c08101909152609a808252614b73602083013960a082015260408051608081019091526042808252615178602083013960c08201528061314d600185614692565b6007811061303c5761303c6147c9565b80516060908061317d575050604080516020810190915260008152919050565b6000600361318c8360026144ed565b6131969190614533565b6131a1906004614634565b905060006131b08260206144ed565b67ffffffffffffffff8111156131c8576131c86147df565b6040519080825280601f01601f1916602001820160405280156131f2576020820181803683370190505b5090506000604051806060016040528060408152602001614ebc604091399050600181016020830160005b8681101561327e576003818a01810151603f601282901c8116860151600c83901c8216870151600684901c831688015192909316870151600891821b60ff94851601821b92841692909201901b91160160e01b83526004909201910161321d565b50600386066001811461329857600281146132a9576132b5565b613d3d60f01b6001198301526132b5565b603d60f81b6000198301525b505050918152949350505050565b60006001600160e01b031982166380ac58cd60e01b14806132f457506001600160e01b03198216635b5e139f60e01b145b8061047957506301ffc9a760e01b6001600160e01b0319831614610479565b61331d838361345e565b61332a6000848484613351565b6106bc5760405162461bcd60e51b81526004016105869061437f565b6106bc8383836135ac565b60006001600160a01b0384163b1561345357604051630a85bd0160e11b81526001600160a01b0385169063150b7a029061339590339089908890889060040161432f565b602060405180830381600087803b1580156133af57600080fd5b505af19250505080156133df575060408051601f3d908101601f191682019092526133dc91810190613b06565b60015b613439573d80801561340d576040519150601f19603f3d011682016040523d82523d6000602084013e613412565b606091505b5080516134315760405162461bcd60e51b81526004016105869061437f565b805181602001fd5b6001600160e01b031916630a85bd0160e11b149050612445565b506001949350505050565b6001600160a01b0382166134b45760405162461bcd60e51b815260206004820181905260248201527f4552433732313a206d696e7420746f20746865207a65726f20616464726573736044820152606401610586565b6000818152600260205260409020546001600160a01b0316156135195760405162461bcd60e51b815260206004820152601c60248201527f4552433732313a20746f6b656e20616c7265616479206d696e746564000000006044820152606401610586565b61352560008383613346565b6001600160a01b038216600090815260036020526040812080546001929061354e9084906144ed565b909155505060008181526002602052604080822080546001600160a01b0319166001600160a01b03861690811790915590518392907fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef908290a45050565b6001600160a01b0383166136075761360281600880546000838152600960205260408120829055600182018355919091527ff3f7a9fe364faab93b216da50a3214154f22a0a2b415b23a84c8169e8b636ee30155565b61362a565b816001600160a01b0316836001600160a01b03161461362a5761362a8382613664565b6001600160a01b038216613641576106bc81613701565b826001600160a01b0316826001600160a01b0316146106bc576106bc82826137b0565b6000600161367184610b0a565b61367b9190614692565b6000838152600760205260409020549091508082146136ce576001600160a01b03841660009081526006602090815260408083208584528252808320548484528184208190558352600790915290208190555b5060009182526007602090815260408084208490556001600160a01b039094168352600681528383209183525290812055565b60085460009061371390600190614692565b6000838152600960205260408120546008805493945090928490811061373b5761373b6147c9565b90600052602060002001549050806008838154811061375c5761375c6147c9565b6000918252602080832090910192909255828152600990915260408082208490558582528120556008805480613794576137946147b3565b6001900381819060005260206000200160009055905550505050565b60006137bb83610b0a565b6001600160a01b039093166000908152600660209081526040808320868452825280832085905593825260079052919091209190915550565b6040518060c001604052806006906020820280368337509192915050565b6040518061020001604052806010906020820280368337509192915050565b60405180608001604052806004905b60608152602001906001900390816138405790505090565b60408051610220810190915260608152601060208201613840565b60408051610100810190915260608152600760208201613840565b6040518061010001604052806008906020820280368337509192915050565b60408051610160810190915260608152600a60208201613840565b604051806101400160405280600a906020820280368337509192915050565b6040805160e0810190915260608152600660208201613840565b80356001600160a01b038116811461391857600080fd5b919050565b60006020828403121561392f57600080fd5b61232982613901565b6000806040838503121561394b57600080fd5b61395483613901565b915061396260208401613901565b90509250929050565b60008060006060848603121561398057600080fd5b61398984613901565b925061399760208501613901565b9150604084013590509250925092565b600080600080608085870312156139bd57600080fd5b6139c685613901565b93506139d460208601613901565b925060408501359150606085013567ffffffffffffffff808211156139f857600080fd5b818701915087601f830112613a0c57600080fd5b813581811115613a1e57613a1e6147df565b604051601f8201601f19908116603f01168101908382118183101715613a4657613a466147df565b816040528281528a6020848701011115613a5f57600080fd5b82602086016020830137600060208483010152809550505050505092959194509250565b60008060408385031215613a9657600080fd5b613a9f83613901565b915060208301358015158114613ab457600080fd5b809150509250929050565b60008060408385031215613ad257600080fd5b613adb83613901565b946020939093013593505050565b600060208284031215613afb57600080fd5b8135612329816147f5565b600060208284031215613b1857600080fd5b8151612329816147f5565b600060208284031215613b3557600080fd5b5035919050565b60008060408385031215613b4f57600080fd5b50508035926020909101359150565b60008151808452613b768160208601602086016146a9565b601f01601f19169290920160200192915050565b60008251613b9c8184602087016146a9565b9190910192915050565b60008451613bb88184602089016146a9565b845190830190613bcc8183602089016146a9565b8451910190613bdf8183602088016146a9565b0195945050505050565b60008551613bfb818460208a016146a9565b855190830190613c0f818360208a016146a9565b8551910190613c228183602089016146a9565b8451910190613c358183602088016146a9565b019695505050505050565b600087516020613c538285838d016146a9565b885191840191613c668184848d016146a9565b8851920191613c788184848c016146a9565b8751920191613c8a8184848b016146a9565b8651920191613c9c8184848a016146a9565b8551920191613cae81848489016146a9565b919091019998505050505050505050565b6331bc1e9160e11b81528351600090613cdf8160048501602089016146a9565b65111031bc9e9160d11b6004918401918201528451613d0581600a8401602089016146a9565b641110391e9160d91b600a92909101918201528351613d2b81600f8401602088016146a9565b601160f91b600f929091019182015260100195945050505050565b61011160f51b81528251600090613d648160028501602088016146a9565b7f2f3e203c636972636c652066696c6c3d2275726c282353756e4772616469656e6002918401918201526303a1491160e51b60228201528351613dae8160268401602088016146a9565b7f2f3e203c673e203c706f6c79676f6e2066696c6c3d222300000000000000000060269290910191820152603d01949350505050565b7f7b226e616d65223a20224f6e436861696e2053656173696465222c202264657381527f6372697074696f6e223a202242656175746966756c2076696577732c20636f6d60208201527f706c6574656c792067656e657261746564204f6e436861696e222c226174747260408201526734b13aba32b9911d60c11b606082015260008351613e798160688501602088016146a9565b7f2c2022696d616765223a2022646174613a696d6167652f7376672b786d6c3b6260689184019182015265185cd94d8d0b60d21b60888201528351613ec581608e8401602088016146a9565b61227d60f01b608e9290910191820152609001949350505050565b7f646174613a6170706c69636174696f6e2f6a736f6e3b6261736536342c000000815260008251613f1881601d8501602087016146a9565b91909101601d0192915050565b67536561736964657360c01b815260008251613f488160088501602087016146a9565b9190910160080192915050565b6911103837b4b73a399e9160b11b81528151600090613f7b81600a8501602087016146a9565b7f222f3e203c706f6c79676f6e206f7061636974793d22302e31222066696c6c3d600a93909101928301525061222360f01b602a820152602c01919050565b6911103837b4b73a399e9160b11b81528351600090613fe081600a8501602089016146a9565b7f222f3e203c616e696d6174654d6f74696f6e20706174683d224d203020302048600a9184019182018190527f203130205a22206475723d2234732220726570656174436f756e743d22696e64602a8301527f6566696e69746522202f3e203c2f673e203c673e203c706174682066696c6c3d604a83015261222360f01b606a830152855161407681606c850160208a016146a9565b641110321e9160d91b606c9390910192830152845161409c8160718501602089016146a9565b60719201918201527f203135205a22206475723d2235732220726570656174436f756e743d22696e6460918201527f6566696e697465222f3e203c2f673e203c706f6c79676f6e2066696c6c3d222360b182015260d10195945050505050565b6911103837b4b73a399e9160b11b8152815160009061412281600a8501602087016146a9565b7f222f3e203c706f6c79676f6e206f7061636974793d22302e32222066696c6c3d600a93909101928301525061222360f01b602a820152602c01919050565b6911103837b4b73a399e9160b11b8152815160009061418781600a8501602087016146a9565b7f222f3e203c7265637420783d22302220793d2234373822206f7061636974793d600a9390910192830152506c22302e32222066696c6c3d222360981b602a820152603701919050565b6911103837b4b73a399e9160b11b815283516000906141f781600a8501602089016146a9565b7f222f3e203c616e696d6174654d6f74696f6e20706174683d226d203020302068600a91840191820152600160fd1b602a820152845161423e81602b8401602089016146a9565b7f3530303022206475723d2231353030732220726570656174436f756e743d2269602b92909101918201527f6e646566696e69746522202f3e203c2f673e203c72616469616c477261646965604b820152730373a1034b21e9129bab723b930b234b2b73a11160651b606b82015283516142bf81607f8401602088016146a9565b7f206772616469656e74556e6974733d227573657253706163654f6e557365223e607f92909101918201527f203c73746f70206f66667365743d22302e3736303422207374796c653d227374609f820152696f702d636f6c6f723a2360b01b60bf82015260c90195945050505050565b6001600160a01b038581168252841660208201526040810183905260806060820181905260009061436290830184613b5e565b9695505050505050565b6020815260006123296020830184613b5e565b60208082526032908201527f4552433732313a207472616e7366657220746f206e6f6e20455243373231526560408201527131b2b4bb32b91034b6b83632b6b2b73a32b960711b606082015260800190565b60208082526035908201527f546f6b656e73206e756d62657220746f206d696e742065786365656473206e756040820152746d626572206f66207075626c696320746f6b656e7360581b606082015260800190565b6020808252818101527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572604082015260600190565b60208082526031908201527f4552433732313a207472616e736665722063616c6c6572206973206e6f74206f6040820152701ddb995c881b9bdc88185c1c1c9bdd9959607a1b606082015260800190565b600080821280156001600160ff1b03849003851316156144ce576144ce614787565b600160ff1b83900384128116156144e7576144e7614787565b50500190565b6000821982111561450057614500614787565b500190565b6000826145145761451461479d565b600160ff1b82146000198414161561452e5761452e614787565b500590565b6000826145425761454261479d565b500490565b600181600019825b808611156145835782820483111561456957614569614787565b8086161561457657928202925b94851c949180029161454f565b50509250929050565b600061232983836000826145a257506001610479565b816145af57506000610479565b81600181146145c557600281146145cf576145eb565b6001915050610479565b60ff8411156145e0576145e0614787565b50506001821b610479565b5060208310610133831016604e8410600b841016171561460e575081810a610479565b6146188383614547565b806000190482111561462c5761462c614787565b029392505050565b600081600019048311821515161561464e5761464e614787565b500290565b60008083128015600160ff1b85018412161561467157614671614787565b6001600160ff1b038401831381161561468c5761468c614787565b50500390565b6000828210156146a4576146a4614787565b500390565b60005b838110156146c45781810151838201526020016146ac565b838111156113255750506000910152565b6000816146e4576146e4614787565b506000190190565b600181811c9082168061470057607f821691505b6020821081141561472157634e487b7160e01b600052602260045260246000fd5b50919050565b600060001982141561473b5761473b614787565b5060010190565b6000826147515761475161479d565b500790565b6000826147655761476561479d565b500690565b6000600160ff1b82141561478057614780614787565b5060000390565b634e487b7160e01b600052601160045260246000fd5b634e487b7160e01b600052601260045260246000fd5b634e487b7160e01b600052603160045260246000fd5b634e487b7160e01b600052603260045260246000fd5b634e487b7160e01b600052604160045260246000fd5b6001600160e01b03198116811461099b57600080fdfe22207d2c207b202274726169745f74797065223a202248696c6c73222c202276616c7565223a2022222077696474683d223130303022206865696768743d223733342e353331222f3e203c7265637420783d22302220793d223536332e31353622206f7061636974793d22302e33222066696c6c3d2223222f3e203c73746f70206f66667365743d223122207374796c653d2273746f702d636f6c6f723a233536342c343433203436332c343537203337352c343738203739332c343738203732312c343637203631322c3431344d2d3133372c313034432d33322c32372c3537332c32382c3736322c32363320433538322c39352c3138302c35372c32332c3837633130352d372c3539382c31372c3735372c32343420632d36302d37342d3234312d3138322d3532342d323132633234312c35352c3433352c3138312c3437352c32363520433538372c3230372c3236302c3134312c3232302c3133336331382c342c3337352c3130302c3436312c32363220433530342c3136342c31392c3132352d342c313233633134312c31352c3532342c38362c3634312c32383120433439332c3232392c3136322c3137302c3132322c313633633134312c35312c3234352c3132372c3237332c31393720433334302c3237302c36382c3133392d33392c313436633133392c35312c3234322c3132372c3236392c31393620433131372c3139352d38342c3135302d38342c3135304c2d3735382c39374c2d3133372c3130347a2d31312c3932372c3135362c3931322c3232322c3934312c3238342c3933312c313034332c3933342c313034332c313031312c2d382c31303131222f3e203c73746f70206f66667365743d22302e3938313222207374796c653d2273746f702d636f6c6f723a235b7b202274726169745f74797065223a202253686970222c202276616c7565223a2022222077696474683d223130303022206865696768743d223634392e333135222f3e203c673e203c7061746820786d6c6e733d22687474703a2f2f7777772e77332e6f72672f323030302f73766722206f7061636974793d22302e3535222066696c6c3d222322207d2c207b202274726169745f74797065223a202250616c65747465222c202276616c7565223a20223436332c343439203430332c343537203335312c343331203137372c343738203638312c3437383839342c3933332c3732322c3934382c3636362c3930302c3636362c3930302c3636342c3839302c3634362c3839302c3634352c3930302c3438322c3930302c3438312c3839302c3436322c3839302c3436312c3930302c3436312c3930302c3336342c3936372c3234372c3936312c3130322c3934362c2d392c3931322c2d31322c313030372c313033392c313030372c313034332c3933302d31312c3932372c3433312c3935322c3636332c3932312c3834352c3934352c313034332c3933342c313034332c313031312c2d382c313031312220643d224d383038372c363837632d3135382c302d3332302d332e31352d3436392d3320632d3239332c302d3631362c31302d3730312c3130632d3236312c302d3630302d31372d3830392d313720632d3131382c302d3234362c31312d3337362c3131632d3135382c302d3332302d31302d3436392d313020632d3239332c302d3337392c31302d3537342c3130632d3139352c302d3333312d31312d3534302d313120632d3131382c302d3234362c31312d3337362c3131632d3135382c302d3332302d31302d3436392d313020632d3239332c302d3631362c31372d3730312c3137632d3236312c302d3630302d31322d3830392d313220632d3131382c302d3234362c31322d3337362c3132632d3130332c302d3236332d392d3436392d3920632d39322c302d3138312c322d3236302c32632d3137312c302d3330342c302d3336322c30632d3236312c302d3333302d302d3333302d3020763532356c393035332d365636383843393033392c3638382c383231372c3638372c383038372c3638377a222f3e203c616e696d6174654d6f74696f6e20706174683d224d20302030204c202d38303530203230205a22206475723d223730732220726570656174436f756e743d22696e646566696e69746522202f3e203c2f673e203c673e203c7061746820786d6c6e733d22687474703a2f2f7777772e77332e6f72672f323030302f737667222066696c6c3d2223313031332c343230203835332c343130203733322c343431203633372c343331203630382c343436203339302c343636203234302c343537203138362c3433302039342c343437202d31362c343230202d38372c34373820313031362c3437384142434445464748494a4b4c4d4e4f505152535455565758595a6162636465666768696a6b6c6d6e6f707172737475767778797a303132333435363738392b2f22207d2c207b202274726169745f74797065223a2022436f617374222c202276616c7565223a2022313030342c343537203834382c343634203736302c343530203630382c343535203431392c343333203332302c343534203133352c343039202d382c343234202d382c34373820313030342c3437383232362c343232203137372c3431342038332c333434202d382c333637202d31322c343738203332382c3437384d313038362c3134433938322d36332c3431352d31312c3235362c323639433431322c36342c3738352d31312c3933342c3920632d39392c302d3535392c37302d3639302c3333396335302d38382c3231322d3232352c3437342d32383320632d3232312c38322d3339332c3234302d3432342c333339433431352c3139312c3731362c38392c3735332c373720632d31362c362d3334342c3134342d3431322c333334433439302c3133362c3934312c35302c3936332c343720433833322c37362c3437382c3138392c3338342c343138633132302d3231302c3432362d3330352c3436332d33313620632d3132382c36392d3232302c3136342d3234312c3234356334342d3130362c3238392d3237362c3339312d32373720632d3132362c36392d3231372c3136332d3233372c3234336339342d3137352c3237392d3234332c3237392d3234336c3433382d39394c313038362c31347a222f3e203c2f6c696e6561724772616469656e743e203c7265637420783d22302e303031222066696c6c3d2275726c2823536b794772616469656e7429222077696474683d223130303022206865696768743d223939392e383131222f3e203c706f6c79676f6e206f7061636974793d22302e3135222066696c6c3d22233830302c3935312c313031342c3932372c313033392c313030372c2d31322c313030372c2d31302c3934302c38312c3934312c3137352c3936322c3634372c3932372d31322c3936322c3133362c3934302c3233392c3936302c3634382c3936362c3935312c3933312c313034312c3933382c313034332c313031312c2d382c313031312220643d224d383039372c383436632d3135382c302d3331392d372d3437302d37632d3238352c302d3434332c32302d3635312c323020632d3137322c302d3335332d352d3434392d39632d3130312d342d3234372d32302d3431332d3230632d3131362c302d3234332c32362d3337332c323620632d3135382c302d3332302d33312d3437312d3331632d3238352c302d3335322c33362d3536302c3336632d3137322c302d3339302d33312d3535362d333120632d3131362c302d3234332c32362d3337332c3236632d3135382c302d3332302d33312d3437312d3331632d3238352c302d3434322c33352d3635302c333520632d3137322c302d3335332d352d3434392d39632d3130312d342d3234372d32302d3431332d3230632d3131362c302d3234352c32352d3337352c323520632d3135382c302d3332322d31332d3437342d3133632d3130372c302d3139372c322d3237372c33632d3133332c312d3234332c302d3337322c3020632d3137322c302d3330382d302d3330382d307633363468393035335638343643393033382c3834362c383232372c3834362c383039372c3834367a222f3e203c616e696d6174654d6f74696f6e20706174683d224d20302030204c202d38303530203430205a22206475723d223730732220726570656174436f756e743d22696e646566696e69746522202f3e203c2f673e203c673e203c706f6c79676f6e2066696c6c3d22233939392c333932203836352c343137203830362c343230203734332c343531203532392c34373820313436382c3437382d31312c3932372c3432352c3936392c3837352c3935322c313034362c3838302c313034332c3933342c313034332c313031312c2d382c313031313c7376672077696474683d2231303030707822206865696768743d22313030307078222076696577426f783d2230203020313030302031303030222076657273696f6e3d22312e312220786d6c6e733d22687474703a2f2f7777772e77332e6f72672f323030302f737667223e203c6c696e6561724772616469656e742069643d22536b794772616469656e7422206772616469656e74556e6974733d227573657253706163654f6e557365222078313d223530302e303031222079313d223939392e38313035222078323d223530302e30303039222079323d22342e383832383133652d303034223e203c73746f70206f66667365743d22302e3536303422207374796c653d2273746f702d636f6c6f723a23222f3e203c2f72616469616c4772616469656e743e203c636972636c65206f7061636974793d22302e31222066696c6c3d22234d3137332c3239396338302d37382c3238382d3134332c3532352d313536633132332d362c3233352c322c3332332c3232206331312c312c32312c322c33322c34632d38372d33302d3230352d34382d3333392d3439632d3231302d302d3430302c34332d3439352c313037206338342d37342c3239332d3132382c3532382d313237633231372c302c3339352c34372c3436362c3131346c3433352c36362063302c302d3530302d31382d3530392d3236632d3137352c32312d3332382c38312d3430322c3135316335362d36392c3139332d3133322c3335382d31363320632d31352d322d33302d342d34372d35632d32322c312d34362c322d37302c35433739362c3236322c3633362c3332342c3536302c333936206335362d37302c3139352d3133332c3336322d313634632d34342c302d39312c332d3133392c3820632d3231332c32332d3430302c38372d3439302c3136316337362d38322c3238322d3135382c3532312d313834206339322d31302c3137392d31312c3235352d35632d302d302d302d302d302d30632d38392d31342d3230342d31342d3333302c3120433532342c3234312c3333382c3330392c3235302c3338346337352d38332c3238302d3136332c3531392d313933206332312d322c34322d342c36332d36632d34352c312d39322c352d3134312c3131632d3231332c32372d3430302c39352d3438382c313730206337352d38332c3238302d3136332c3531392d3139336332362d332c35332d362c37382d38632d34322d312d38372d312d3133332c3020433435342c3137352c3236352c3233302c3137332c3239397a4d2d3134342c33303463392d3130362c3333342d3331392c3538302d31373520433233362c34312d322c3135302d36372c3233326335322d34342c3333312d3139352c3535352d343720632d37372d34352d3234312d37382d3431302d35633136322d33352c3334342c382c3431372c373020433330372c3134302c39322c3139372c36352c3230346331322d322c3236332d34322c3430372c373220632d3233362d3134342d3532302d382d3533342d314332342c3233392c3237332c3136372c3435352c33303120433237302c3139332c35372c3235372c33312c323635633130372d332c3230392c32372c3236372c383020433231342c3238342d31312c3236332d36352c333038633130362d332c3230372c32382c3236342c3830204334382c3239362d38372c3332362d38372c3332364c2d3533332c3433334c2d3134342c3330347a22207d2c207b202274726169745f74797065223a202253756e222c202276616c7565223a2022a2646970667358221220320ccb54eac3242f1a83645c7f4cf208147f0f6967d12376960be692ba6301ea64736f6c63430008070033
Deployed Bytecode
0x60806040526004361061014b5760003560e01c80634f6ccce7116100b657806398bd87631161006f57806398bd876314610385578063a22cb465146103a5578063b88d4fde146103c5578063c87b56dd146103e5578063e985e9c514610405578063f2fde38b1461044e57600080fd5b80634f6ccce7146102dd5780636352211e146102fd57806370a082311461031d578063715018a61461033d5780638da5cb5b1461035257806395d89b411461037057600080fd5b806323b872dd1161010857806323b872dd146102335780632f745c59146102535780633ccfd60b1461027357806340d097c31461028857806342842e0e146102a85780634e71d92d146102c857600080fd5b806301ffc9a71461015057806306fdde0314610185578063081812fc146101a7578063095ea7b3146101df5780631172219e1461020157806318160ddd14610214575b600080fd5b34801561015c57600080fd5b5061017061016b366004613ae9565b61046e565b60405190151581526020015b60405180910390f35b34801561019157600080fd5b5061019a61047f565b60405161017c919061436c565b3480156101b357600080fd5b506101c76101c2366004613b23565b610511565b6040516001600160a01b03909116815260200161017c565b3480156101eb57600080fd5b506101ff6101fa366004613abf565b6105ab565b005b6101ff61020f366004613b23565b6106c1565b34801561022057600080fd5b506008545b60405190815260200161017c565b34801561023f57600080fd5b506101ff61024e36600461396b565b610836565b34801561025f57600080fd5b5061022561026e366004613abf565b610867565b34801561027f57600080fd5b506101ff6108fd565b34801561029457600080fd5b506101ff6102a336600461391d565b610956565b3480156102b457600080fd5b506101ff6102c336600461396b565b61099e565b3480156102d457600080fd5b506101ff6109b9565b3480156102e957600080fd5b506102256102f8366004613b23565b610a00565b34801561030957600080fd5b506101c7610318366004613b23565b610a93565b34801561032957600080fd5b5061022561033836600461391d565b610b0a565b34801561034957600080fd5b506101ff610b91565b34801561035e57600080fd5b50600a546001600160a01b03166101c7565b34801561037c57600080fd5b5061019a610bc5565b34801561039157600080fd5b5061019a6103a0366004613b3c565b610bd4565b3480156103b157600080fd5b506101ff6103c0366004613a83565b61122e565b3480156103d157600080fd5b506101ff6103e03660046139a7565b6112f3565b3480156103f157600080fd5b5061019a610400366004613b23565b61132b565b34801561041157600080fd5b50610170610420366004613938565b6001600160a01b03918216600090815260056020908152604080832093909416825291909152205460ff1690565b34801561045a57600080fd5b506101ff61046936600461391d565b6121e9565b60006104798261228a565b92915050565b60606000805461048e906146ec565b80601f01602080910402602001604051908101604052809291908181526020018280546104ba906146ec565b80156105075780601f106104dc57610100808354040283529160200191610507565b820191906000526020600020905b8154815290600101906020018083116104ea57829003601f168201915b5050505050905090565b6000818152600260205260408120546001600160a01b031661058f5760405162461bcd60e51b815260206004820152602c60248201527f4552433732313a20617070726f76656420717565727920666f72206e6f6e657860448201526b34b9ba32b73a103a37b5b2b760a11b60648201526084015b60405180910390fd5b506000908152600460205260409020546001600160a01b031690565b60006105b682610a93565b9050806001600160a01b0316836001600160a01b031614156106245760405162461bcd60e51b815260206004820152602160248201527f4552433732313a20617070726f76616c20746f2063757272656e74206f776e656044820152603960f91b6064820152608401610586565b336001600160a01b038216148061064057506106408133610420565b6106b25760405162461bcd60e51b815260206004820152603860248201527f4552433732313a20617070726f76652063616c6c6572206973206e6f74206f7760448201527f6e6572206e6f7220617070726f76656420666f7220616c6c00000000000000006064820152608401610586565b6106bc83836122af565b505050565b600081116107005760405162461bcd60e51b815260206004820152600c60248201526b15dc9bdb99c8185b5bdd5b9d60a21b6044820152606401610586565b601e8111156107645760405162461bcd60e51b815260206004820152602a60248201527f4d617820746f6b656e7320706572207472616e73616374696f6e206e756d62656044820152691c88195e18d95959195960b21b6064820152608401610586565b610ce461077a82610774600b5490565b9061231d565b11156107985760405162461bcd60e51b8152600401610586906143d1565b600c5434906107a79083612330565b11156107f55760405162461bcd60e51b815260206004820152601b60248201527f45746865722076616c75652073656e7420697320746f6f206c6f7700000000006044820152606401610586565b60005b81811015610832576108123361080d600b5490565b61233c565b610820600b80546001019055565b8061082a81614727565b9150506107f8565b5050565b6108403382612356565b61085c5760405162461bcd60e51b81526004016105869061445b565b6106bc83838361244d565b600061087283610b0a565b82106108d45760405162461bcd60e51b815260206004820152602b60248201527f455243373231456e756d657261626c653a206f776e657220696e646578206f7560448201526a74206f6620626f756e647360a81b6064820152608401610586565b506001600160a01b03919091166000908152600660209081526040808320938352929052205490565b600a546001600160a01b031633146109275760405162461bcd60e51b815260040161058690614426565b6040514790339082156108fc029083906000818181858888f19350505050158015610832573d6000803e3d6000fd5b600a546001600160a01b031633146109805760405162461bcd60e51b815260040161058690614426565b61098d8161080d600b5490565b61099b600b80546001019055565b50565b6106bc838383604051806020016040528060008152506112f3565b61014d6109c5600b5490565b11156109e35760405162461bcd60e51b8152600401610586906143d1565b6109f03361080d600b5490565b6109fe600b80546001019055565b565b6000610a0b60085490565b8210610a6e5760405162461bcd60e51b815260206004820152602c60248201527f455243373231456e756d657261626c653a20676c6f62616c20696e646578206f60448201526b7574206f6620626f756e647360a01b6064820152608401610586565b60088281548110610a8157610a816147c9565b90600052602060002001549050919050565b6000818152600260205260408120546001600160a01b0316806104795760405162461bcd60e51b815260206004820152602960248201527f4552433732313a206f776e657220717565727920666f72206e6f6e657869737460448201526832b73a103a37b5b2b760b91b6064820152608401610586565b60006001600160a01b038216610b755760405162461bcd60e51b815260206004820152602a60248201527f4552433732313a2062616c616e636520717565727920666f7220746865207a65604482015269726f206164647265737360b01b6064820152608401610586565b506001600160a01b031660009081526003602052604090205490565b600a546001600160a01b03163314610bbb5760405162461bcd60e51b815260040161058690614426565b6109fe60006125f8565b60606001805461048e906146ec565b6060600080808080808060018160fa6040519080825280601f01601f191660200182016040528015610c0d576020820181803683370190505b509050610c186137f4565b7f013f26ad80a3665e606c4eb755430dfa59a6131a4231e4002f9594eb6f8d8d6181527c2532c0544b22a3db43ebb53ac374bfa1a8856080f009cac8bb8a5ce99660208201527f013f23c1d2f15869a74f46157a0d9ce5483c1b6d416baa1c1ba2cbfbe13b744060408201527c25372fb3c545de4cccfe1d373df28795888191dffc101dc2434bf1e23d60608201527f013f39039259ac91dc503b9e331c79e866e2a21c236775ad088d0b209aea5e9b60808201527c252bdcae587671d3dfaba908d50dd7e1a0daf274e0c640d85e24ef808c60a082015260009250806002610d018f82614634565b610d0b9190614692565b60068110610d1b57610d1b6147c9565b60200201519550600099508996508694508493505b60468a10156111705784610d4381614727565b9550610d51906102ee61458c565b610d5b9087614533565b9850610d696102ee8a614756565b9750876102ed1415610dbe578260011415610d8357611170565b600192508083610d948f6002614634565b610d9e9190614692565b60068110610dae57610dae6147c9565b602002015195506000945061115e565b610dc9600285614756565b610e3d578b60011415610de957610de288610190614653565b9750610df8565b610df5610226896144ac565b97505b8915610e3857600160fd1b8288610e0e81614727565b995081518110610e2057610e206147c9565b60200101906001600160f81b031916908160001a9053505b610e8c565b8915610e7d57600b60fa1b8288610e5381614727565b995081518110610e6557610e656147c9565b60200101906001600160f81b031916908160001a9053505b610e896101f4896144ac565b97505b6000881215610eda57610e9e8861476a565b9750602d60f81b8288610eb081614727565b995081518110610ec257610ec26147c9565b60200101906001600160f81b031916908160001a9053505b600a881215610f3557610eee600a89614742565b610ef99060306144ed565b60f81b8288610f0781614727565b995081518110610f1957610f196147c9565b60200101906001600160f81b031916908160001a905350611150565b6064881215610f9657610f49600a89614505565b610f549060306144ed565b60f81b8288610f6281614727565b995081518110610f7457610f746147c9565b60200101906001600160f81b031916908160001a905350610eee600a89614742565b6103e888121561100357610fab606489614505565b610fb69060306144ed565b60f81b8288610fc481614727565b995081518110610fd657610fd66147c9565b60200101906001600160f81b031916908160001a905350600a610ff9818a614505565b610f499190614742565b61100f6103e889614505565b61101a9060306144ed565b60f81b828861102881614727565b99508151811061103a5761103a6147c9565b60200101906001600160f81b031916908160001a905350600a61105e60648a614505565b6110689190614742565b6110739060306144ed565b60f81b828861108181614727565b995081518110611093576110936147c9565b60200101906001600160f81b031916908160001a905350600a6110b6818a614505565b6110c09190614742565b6110cb9060306144ed565b60f81b82886110d981614727565b9950815181106110eb576110eb6147c9565b60200101906001600160f81b031916908160001a90535061110d600a89614742565b6111189060306144ed565b60f81b828861112681614727565b995081518110611138576111386147c9565b60200101906001600160f81b031916908160001a9053505b8361115a81614727565b9450505b8961116881614727565b9a5050610d30565b60008767ffffffffffffffff81111561118b5761118b6147df565b6040519080825280601f01601f1916602001820160405280156111b5576020820181803683370190505b50905060009a505b878b101561121d57828b815181106111d7576111d76147c9565b602001015160f81c60f81b818c815181106111f4576111f46147c9565b60200101906001600160f81b031916908160001a9053508a61121581614727565b9b50506111bd565b9d9c50505050505050505050505050565b6001600160a01b0382163314156112875760405162461bcd60e51b815260206004820152601960248201527f4552433732313a20617070726f766520746f2063616c6c6572000000000000006044820152606401610586565b3360008181526005602090815260408083206001600160a01b03871680855290835292819020805460ff191686151590811790915590519081529192917f17307eab39ab6107e8899845ad3d59bd9653f200f220920489ca2b5937696c31910160405180910390a35050565b6112fd3383612356565b6113195760405162461bcd60e51b81526004016105869061445b565b6113258484848461264a565b50505050565b6060611335613812565b61133d613831565b611345613858565b61134d613873565b611355613873565b61135d61388e565b60008061139061136c8b61267d565b60405160200161137c9190613f25565b60405160208183030381529060405261277b565b9050600861139f600a83614533565b6113a99190614756565b6113b49060016144ed565b835260026113c3606483614533565b6113cd9190614756565b6113d89060016144ed565b602084015260256113eb61271083614533565b6113f59190614756565b6114009060016144ed565b60408401526008611414620f424083614533565b61141e9190614756565b6114299060016144ed565b6060840152600661143e6305f5e10083614533565b6114489190614756565b6114539060016144ed565b60808401526004611468633b9aca0083614533565b6114729190614756565b61147d9060016144ed565b60a084015260076114936402540be40083614533565b61149d9190614756565b6114a89060016144ed565b60c084015260046114be64174876e80083614533565b6114c89190614756565b6114d39060016144ed565b60e0840152604080518082018252601781527f37382c343434202d3439342c343738203635312c343738000000000000000000602080830191909152908752815160608101909252603080835290615413908301398560016020020181905250604051806060016040528060278152602001614b4c60279139604080870191909152805160808101909152604f808252614f246020830139606080870191909152604080519182019052602d808252614f736020830139608086015260408051606081019091526030808252615413602083013960a08601526040805160608101909152602f8082526148ab602083013960c086015260408051608081019091526060808252614e5c602083013960e0860152604080518082018252601881527f3136322c333932202d3330372c343738203633322c3437380000000000000000602080830191909152908652815180820183526000808252878301919091528251808301845281815287840152825180830184528181526060880152825180840184528281527f3939312c343130203832362c343534203631312c34373820313236322c3437388184015260808801528251808301845281815260a088015282518084018452601e81527f3135362c3433382034372c343738203332312c343738203231342c34363100008184015260c08801528251918201909252908152846007602090810291909101919091527cbf98fffb991e1362256591ffa922dbfbfd55003975004eff997affff9f89527d470000820000ff3b3bffc79926134a500070ff3b3bfffb9926134a500070908901527c95fffffc99690000c4000affc300fffc998a0000c44800ffd13bfffc996040808a01919091527c332700573b00ddfffffc9900023647006b0095fffffc990002362e008260608a01527ce0a1fff69900332700573e00e0a1fff6990033270a570000e0a1fff69960808a01527d140033002e57bdff52fff699140033002e57a473fffff699140033002e5760a08a01527dff8359fffcde3e0047870068ff9914fff699140033290087ff9914fff69960c08a01527c2330004459599cfffffcde002330004459ee59fffffcde00233000445960e08a01527dffd199f2ffff002330004459ffd199f2ffff002330004459aaff99fffcde6101008a01527c153000395999ffe9f2ffff13003044005999ffe9f2ffff1300304400596101208a01527dff0000ffc021400c16750000ff7824fff58200153000395980d4fff2ffff6101408a01527d5c0011a80000ffa100ffd15e330009610000ff0077ffc0213300096100006101608a01527dbf0000ffcc4c2d10384d0085ff66f0ffe0914a1b5c4931a8ff66f0ffd15e6101808a01527d030303242424c4c4c4ebebeb3000234a0035ff5900ffcc4c3000234a00356101a08a0152770d18331b2a610083c429ffa62d0047a754ff5200c4ff57d86101c08a015283015161190990600190614692565b611914906004614634565b91506119716301000000611929600a85614756565b61193790630100000061458c565b8a611943600a87614533565b60108110611953576119536147c9565b60200201516119629190614533565b61196c9190614756565b6127ac565b8752604083015161198490600190614692565b61198f906004614634565b61199a9060016144ed565b91506119af6301000000611929600a85614756565b602088015260408301516119c590600190614692565b6119d0906004614634565b6119db9060026144ed565b91506119f06301000000611929600a85614756565b604080890191909152830151611a0890600190614692565b611a13906004614634565b611a1e9060036144ed565b9150611a336301000000611929600a85614756565b606088015260408051610140810190915261011480825261547e6020830139865260408051606081019091526028808252614883602083013986600160200201819052506040518060a00160405280607e81526020016150fa607e9139604087015260608301518490611aa890600190614692565b60088110611ab857611ab86147c9565b6020020151604051602001611acd9190613f55565b60408051601f1981840301815291905286600360200201528460018460036020020151611afa9190614692565b60088110611b0a57611b0a6147c9565b6020020151604051602001611b1f9190614161565b60408051808303601f1901815291815260808089019290925280519182019052604f808252614834602083013960a0808801919091526040805191820190526065808252614abd602083013960c0870152604080516102408101909152610215808252614c47602083013960e08701526040805161024081019091526102178082526151fc602083013961010087015282516020840151611bc2919060006128cb565b604051602001611bd291906140fc565b60408051808303601f1901815291905261012087015282516020840151611bfb919060016128cb565b6020840151600114611c2657604051806040016040528060018152602001602d60f81b815250611c37565b604051806020016040528060008152505b611c5f60028660075b6020020151611c50906002614634565b611c5a9190614692565b612ed9565b604051602001611c71939291906141d1565b60408051808303601f19018152918152610140880191909152805160608101909152602d808252614a6d6020830139610160870152604080516060810190915260338082526155926020830139610180870152611cd16001846007611c40565b611cde6002856007611c40565b604051602001611cef929190613d46565b60408051808303601f190181529190526101a08701526080830151611d3690611d1a90600390614533565b611d259060016144ed565b60808501516103a090600290614756565b606088015160a0850151611d4990612f91565b604051602001611d5b93929190613fba565b60408051808303601f190181529181526101c088019190915280518082018252600a8082526911103837b4b73a399e9160b11b6020808401919091526101e08a01929092528251808401845290815269011179f1e17b9bb339f160b51b81830152610200890152875189820151828a01518b8501519451600095611de195909101613be9565b60408051808303601f1901815282825289820151918b015160608b0151919450611e119385939290602001613be9565b60408051808303601f19018152828252908a015160808a0151919350611e3e928492908290602001613be9565b60408051808303601f1901815282825260a08a0151918b015160c08b0151919450611e6f9385939290602001613be9565b60408051808303601f19018152828252908a015160e08a0151919350611e9c928492908290602001613be9565b60408051808303601f190181528282526101008a0151918b01516101208b0151919450611ecf9385939290602001613be9565b60408051808303601f19018152828252908a01516101408a01518b51929450611efd93859390602001613be9565b60408051808303601f190181529082905261016089015160208b8101516101808c0151939550611f31948694919201613be9565b60408051808303601f19018152908290526020808b01516101a08b015160608d0151939550611f67948694929391929101613be9565b60408051808303601f19018152908290526101c089015160608b01516101e08b0151929450611f9b93859390602001613be9565b60408051601f19818403018152919052905080611fbe8560066020020151613048565b610200890151604051611fd693929190602001613ba6565b6040516020818303038152906040529050611fef6138ad565b604051806060016040528060238152602001614a9a60239139815261201b8560005b602002015161267d565b81600160200201819052506040518060600160405280602a8152602001614b22602a9139604082015261204f856002612011565b606080830191909152604080519182019052602880825261480c6020830139608082015261207e856003612011565b60a082015260408051606081019091526026808252615969602083013960c08201526120ab856007612011565b60e082015260408051606081019091526028808252614efc60208301396101008201526120d9856006612011565b610120820152604080518082018252600481526322207d5d60e01b602080830191909152610140840191909152825181840151838501516060860151608087015160a0880151965160009761213097909101613c40565b60408051808303601f190181529082905260c084015160e085015161010086015161012087015161014088015194965061216f95879590602001613c40565b604051602081830303815290604052905060006121b48261218f8661315d565b6040516020016121a0929190613de4565b60405160208183030381529060405261315d565b9050806040516020016121c79190613ee0565b60408051601f198184030181529190529e9d5050505050505050505050505050565b600a546001600160a01b031633146122135760405162461bcd60e51b815260040161058690614426565b6001600160a01b0381166122785760405162461bcd60e51b815260206004820152602660248201527f4f776e61626c653a206e6577206f776e657220697320746865207a65726f206160448201526564647265737360d01b6064820152608401610586565b61099b816125f8565b80546001019055565b60006001600160e01b0319821663780e9d6360e01b14806104795750610479826132c3565b600081815260046020526040902080546001600160a01b0319166001600160a01b03841690811790915581906122e482610a93565b6001600160a01b03167f8c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b92560405160405180910390a45050565b600061232982846144ed565b9392505050565b60006123298284614634565b610832828260405180602001604052806000815250613313565b6000818152600260205260408120546001600160a01b03166123cf5760405162461bcd60e51b815260206004820152602c60248201527f4552433732313a206f70657261746f7220717565727920666f72206e6f6e657860448201526b34b9ba32b73a103a37b5b2b760a11b6064820152608401610586565b60006123da83610a93565b9050806001600160a01b0316846001600160a01b031614806124155750836001600160a01b031661240a84610511565b6001600160a01b0316145b8061244557506001600160a01b0380821660009081526005602090815260408083209388168352929052205460ff165b949350505050565b826001600160a01b031661246082610a93565b6001600160a01b0316146124c85760405162461bcd60e51b815260206004820152602960248201527f4552433732313a207472616e73666572206f6620746f6b656e2074686174206960448201526839903737ba1037bbb760b91b6064820152608401610586565b6001600160a01b03821661252a5760405162461bcd60e51b8152602060048201526024808201527f4552433732313a207472616e7366657220746f20746865207a65726f206164646044820152637265737360e01b6064820152608401610586565b612535838383613346565b6125406000826122af565b6001600160a01b0383166000908152600360205260408120805460019290612569908490614692565b90915550506001600160a01b03821660009081526003602052604081208054600192906125979084906144ed565b909155505060008181526002602052604080822080546001600160a01b0319166001600160a01b0386811691821790925591518493918716917fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef91a4505050565b600a80546001600160a01b038381166001600160a01b0319831681179093556040519116919082907f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e090600090a35050565b61265584848461244d565b61266184848484613351565b6113255760405162461bcd60e51b81526004016105869061437f565b6060816126a15750506040805180820190915260018152600360fc1b602082015290565b8160005b81156126cb57806126b581614727565b91506126c49050600a83614533565b91506126a5565b60008167ffffffffffffffff8111156126e6576126e66147df565b6040519080825280601f01601f191660200182016040528015612710576020820181803683370190505b5090505b841561244557612725600183614692565b9150612732600a86614756565b61273d9060306144ed565b60f81b818381518110612752576127526147c9565b60200101906001600160f81b031916908160001a905350612774600a86614533565b9450612714565b60008160405160200161278e9190613b8a565b60408051601f19818403018152919052805160209091012092915050565b6060816127d557505060408051808201909152600681526503030303030360d41b602082015290565b6040805180820190915260068082526503030303030360d41b6020830152905b811561232957600a612808601086614756565b101561285f57612819601085614756565b6128249060306144ed565b60f81b81612833600185614692565b81518110612843576128436147c9565b60200101906001600160f81b031916908160001a9053506128ac565b61286a601085614756565b6128759060376144ed565b60f81b81612884600185614692565b81518110612894576128946147c9565b60200101906001600160f81b031916908160001a9053505b6128b7601085614533565b9350816128c3816146d5565b9250506127f5565b6040805161012c80825261016082019092526060916000918291829182918291829160019183916020820181803683370190505090506129096138c8565b7b265c95854022eaa8a826db38647b201739ce4f24204af6e0aeb97230815275a8ab0f37f5ec2ffcaac252f15949a052559bd538cbe560208201527d1768e4c744107bec6966e48b6070bd558da53030af9c8422b1af5a6e4660604082015275a8a45d923ba48d952c5e519add1ee679dc58aa97fd0160608201527f0e48d317e96bb0676c73e1d91d25620a84c8154c68ff2ed985ef7adc27c1bd0c608082015275a8b483ffdd8795098e01fc17b102a5a54877a6d4f68060a08201527d1766d7abb97dd372b1290c2d7e4fd5586924479bed6a1e1763d811595a4260c08201527002e5d63388f891d6087aea8c6e0084d7b460e08201527d17624acfa651e03c326754582994d729a8f7b92398e67cc3d71b62a792f8610100820152793ed5c4b8964e9feddde8a45588adebd10888fab0bbdb9052e69b61012082015260078d1015612a785780612a5c60018f614692565b600a8110612a6c57612a6c6147c9565b60200201519450612aaa565b60009250808d600714612a8c576008612a8f565b60065b60ff16600a8110612aa257612aa26147c9565b602002015194505b60008b600114612abc576101cc612ac0565b6102305b61ffff169050600099508996508694505b60468a1015612e1a5784612ae481614727565b9550612af19060c861458c565b612afb9087614533565b9850612b0860c88a614756565b97508760961415612b5b578360011415612b2157612e1a565b60019350818e600714612b35576009612b38565b60075b60ff16600a8110612b4b57612b4b6147c9565b6020020151955060009450612e08565b612b66600286614756565b612be4578b60011415612b9257612b7f886101406144ed565b612b8b906103e8614692565b9750612b9f565b612b9c81896144ed565b97505b8915612bdf57600b60fa1b8388612bb581614727565b995081518110612bc757612bc76147c9565b60200101906001600160f81b031916908160001a9053505b612c59565b8c60011415612bff57612bf86064896144ed565b9750612c19565b612c0a8860646144ed565b612c16906103e8614692565b97505b8915612c5957600160fd1b8388612c2f81614727565b995081518110612c4157612c416147c9565b60200101906001600160f81b031916908160001a9053505b600a881015612cb457612c6d600a89614756565b612c789060306144ed565b60f81b8388612c8681614727565b995081518110612c9857612c986147c9565b60200101906001600160f81b031916908160001a905350612e08565b6064881015612d1557612cc8600a89614533565b612cd39060306144ed565b60f81b8388612ce181614727565b995081518110612cf357612cf36147c9565b60200101906001600160f81b031916908160001a905350612c6d600a89614756565b612d20606489614533565b612d2b9060306144ed565b60f81b8388612d3981614727565b995081518110612d4b57612d4b6147c9565b60200101906001600160f81b031916908160001a905350600a612d6e818a614533565b612d789190614756565b612d839060306144ed565b60f81b8388612d9181614727565b995081518110612da357612da36147c9565b60200101906001600160f81b031916908160001a905350612dc5600a89614756565b612dd09060306144ed565b60f81b8388612dde81614727565b995081518110612df057612df06147c9565b60200101906001600160f81b031916908160001a9053505b89612e1281614727565b9a5050612ad1565b60008767ffffffffffffffff811115612e3557612e356147df565b6040519080825280601f01601f191660200182016040528015612e5f576020820181803683370190505b50905060009a505b878b1015612ec757838b81518110612e8157612e816147c9565b602001015160f81c60f81b818c81518110612e9e57612e9e6147c9565b60200101906001600160f81b031916908160001a9053508a612ebf81614727565b9b5050612e67565b9e9d5050505050505050505050505050565b60607d6735464658dcf4043d5da3a3ecc8c16957e326064fbc66daeee5b6a78b598215612f2457612f0b836003614634565b612f17906103e861458c565b612f219082614533565b90505b6000612f486103e8612f39620f424085614533565b612f439190614756565b61267d565b612f586103e8612f398186614533565b612f67612f436103e886614756565b604051602001612f7993929190613cbf565b60408051601f19818403018152919052949350505050565b6060612f9b613831565b60405180610180016040528061015981526020016148da610159913981526040805161018081019091526101488082526158216020830139816001602002018190525060405180610280016040528061025c81526020016155c561025c91396040808301919091528051610180810190915261015a808252614fa0602083013960608201528061302c600185614692565b6004811061303c5761303c6147c9565b60200201519392505050565b60606130526138e7565b604080518082018252601c81527f2d33302c3839382c2d313335372c313031362c313239382c3130313600000000602080830191909152908352815160608101909252603a80835290614c0d9083013981600160200201819052506040518060600160405280603a8152602001614a33603a9139604080830191909152805160608101909152603b80825261544360208301396060820152604080516080810190915260428082526151ba602083013960808201526040805160c08101909152609a808252614b73602083013960a082015260408051608081019091526042808252615178602083013960c08201528061314d600185614692565b6007811061303c5761303c6147c9565b80516060908061317d575050604080516020810190915260008152919050565b6000600361318c8360026144ed565b6131969190614533565b6131a1906004614634565b905060006131b08260206144ed565b67ffffffffffffffff8111156131c8576131c86147df565b6040519080825280601f01601f1916602001820160405280156131f2576020820181803683370190505b5090506000604051806060016040528060408152602001614ebc604091399050600181016020830160005b8681101561327e576003818a01810151603f601282901c8116860151600c83901c8216870151600684901c831688015192909316870151600891821b60ff94851601821b92841692909201901b91160160e01b83526004909201910161321d565b50600386066001811461329857600281146132a9576132b5565b613d3d60f01b6001198301526132b5565b603d60f81b6000198301525b505050918152949350505050565b60006001600160e01b031982166380ac58cd60e01b14806132f457506001600160e01b03198216635b5e139f60e01b145b8061047957506301ffc9a760e01b6001600160e01b0319831614610479565b61331d838361345e565b61332a6000848484613351565b6106bc5760405162461bcd60e51b81526004016105869061437f565b6106bc8383836135ac565b60006001600160a01b0384163b1561345357604051630a85bd0160e11b81526001600160a01b0385169063150b7a029061339590339089908890889060040161432f565b602060405180830381600087803b1580156133af57600080fd5b505af19250505080156133df575060408051601f3d908101601f191682019092526133dc91810190613b06565b60015b613439573d80801561340d576040519150601f19603f3d011682016040523d82523d6000602084013e613412565b606091505b5080516134315760405162461bcd60e51b81526004016105869061437f565b805181602001fd5b6001600160e01b031916630a85bd0160e11b149050612445565b506001949350505050565b6001600160a01b0382166134b45760405162461bcd60e51b815260206004820181905260248201527f4552433732313a206d696e7420746f20746865207a65726f20616464726573736044820152606401610586565b6000818152600260205260409020546001600160a01b0316156135195760405162461bcd60e51b815260206004820152601c60248201527f4552433732313a20746f6b656e20616c7265616479206d696e746564000000006044820152606401610586565b61352560008383613346565b6001600160a01b038216600090815260036020526040812080546001929061354e9084906144ed565b909155505060008181526002602052604080822080546001600160a01b0319166001600160a01b03861690811790915590518392907fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef908290a45050565b6001600160a01b0383166136075761360281600880546000838152600960205260408120829055600182018355919091527ff3f7a9fe364faab93b216da50a3214154f22a0a2b415b23a84c8169e8b636ee30155565b61362a565b816001600160a01b0316836001600160a01b03161461362a5761362a8382613664565b6001600160a01b038216613641576106bc81613701565b826001600160a01b0316826001600160a01b0316146106bc576106bc82826137b0565b6000600161367184610b0a565b61367b9190614692565b6000838152600760205260409020549091508082146136ce576001600160a01b03841660009081526006602090815260408083208584528252808320548484528184208190558352600790915290208190555b5060009182526007602090815260408084208490556001600160a01b039094168352600681528383209183525290812055565b60085460009061371390600190614692565b6000838152600960205260408120546008805493945090928490811061373b5761373b6147c9565b90600052602060002001549050806008838154811061375c5761375c6147c9565b6000918252602080832090910192909255828152600990915260408082208490558582528120556008805480613794576137946147b3565b6001900381819060005260206000200160009055905550505050565b60006137bb83610b0a565b6001600160a01b039093166000908152600660209081526040808320868452825280832085905593825260079052919091209190915550565b6040518060c001604052806006906020820280368337509192915050565b6040518061020001604052806010906020820280368337509192915050565b60405180608001604052806004905b60608152602001906001900390816138405790505090565b60408051610220810190915260608152601060208201613840565b60408051610100810190915260608152600760208201613840565b6040518061010001604052806008906020820280368337509192915050565b60408051610160810190915260608152600a60208201613840565b604051806101400160405280600a906020820280368337509192915050565b6040805160e0810190915260608152600660208201613840565b80356001600160a01b038116811461391857600080fd5b919050565b60006020828403121561392f57600080fd5b61232982613901565b6000806040838503121561394b57600080fd5b61395483613901565b915061396260208401613901565b90509250929050565b60008060006060848603121561398057600080fd5b61398984613901565b925061399760208501613901565b9150604084013590509250925092565b600080600080608085870312156139bd57600080fd5b6139c685613901565b93506139d460208601613901565b925060408501359150606085013567ffffffffffffffff808211156139f857600080fd5b818701915087601f830112613a0c57600080fd5b813581811115613a1e57613a1e6147df565b604051601f8201601f19908116603f01168101908382118183101715613a4657613a466147df565b816040528281528a6020848701011115613a5f57600080fd5b82602086016020830137600060208483010152809550505050505092959194509250565b60008060408385031215613a9657600080fd5b613a9f83613901565b915060208301358015158114613ab457600080fd5b809150509250929050565b60008060408385031215613ad257600080fd5b613adb83613901565b946020939093013593505050565b600060208284031215613afb57600080fd5b8135612329816147f5565b600060208284031215613b1857600080fd5b8151612329816147f5565b600060208284031215613b3557600080fd5b5035919050565b60008060408385031215613b4f57600080fd5b50508035926020909101359150565b60008151808452613b768160208601602086016146a9565b601f01601f19169290920160200192915050565b60008251613b9c8184602087016146a9565b9190910192915050565b60008451613bb88184602089016146a9565b845190830190613bcc8183602089016146a9565b8451910190613bdf8183602088016146a9565b0195945050505050565b60008551613bfb818460208a016146a9565b855190830190613c0f818360208a016146a9565b8551910190613c228183602089016146a9565b8451910190613c358183602088016146a9565b019695505050505050565b600087516020613c538285838d016146a9565b885191840191613c668184848d016146a9565b8851920191613c788184848c016146a9565b8751920191613c8a8184848b016146a9565b8651920191613c9c8184848a016146a9565b8551920191613cae81848489016146a9565b919091019998505050505050505050565b6331bc1e9160e11b81528351600090613cdf8160048501602089016146a9565b65111031bc9e9160d11b6004918401918201528451613d0581600a8401602089016146a9565b641110391e9160d91b600a92909101918201528351613d2b81600f8401602088016146a9565b601160f91b600f929091019182015260100195945050505050565b61011160f51b81528251600090613d648160028501602088016146a9565b7f2f3e203c636972636c652066696c6c3d2275726c282353756e4772616469656e6002918401918201526303a1491160e51b60228201528351613dae8160268401602088016146a9565b7f2f3e203c673e203c706f6c79676f6e2066696c6c3d222300000000000000000060269290910191820152603d01949350505050565b7f7b226e616d65223a20224f6e436861696e2053656173696465222c202264657381527f6372697074696f6e223a202242656175746966756c2076696577732c20636f6d60208201527f706c6574656c792067656e657261746564204f6e436861696e222c226174747260408201526734b13aba32b9911d60c11b606082015260008351613e798160688501602088016146a9565b7f2c2022696d616765223a2022646174613a696d6167652f7376672b786d6c3b6260689184019182015265185cd94d8d0b60d21b60888201528351613ec581608e8401602088016146a9565b61227d60f01b608e9290910191820152609001949350505050565b7f646174613a6170706c69636174696f6e2f6a736f6e3b6261736536342c000000815260008251613f1881601d8501602087016146a9565b91909101601d0192915050565b67536561736964657360c01b815260008251613f488160088501602087016146a9565b9190910160080192915050565b6911103837b4b73a399e9160b11b81528151600090613f7b81600a8501602087016146a9565b7f222f3e203c706f6c79676f6e206f7061636974793d22302e31222066696c6c3d600a93909101928301525061222360f01b602a820152602c01919050565b6911103837b4b73a399e9160b11b81528351600090613fe081600a8501602089016146a9565b7f222f3e203c616e696d6174654d6f74696f6e20706174683d224d203020302048600a9184019182018190527f203130205a22206475723d2234732220726570656174436f756e743d22696e64602a8301527f6566696e69746522202f3e203c2f673e203c673e203c706174682066696c6c3d604a83015261222360f01b606a830152855161407681606c850160208a016146a9565b641110321e9160d91b606c9390910192830152845161409c8160718501602089016146a9565b60719201918201527f203135205a22206475723d2235732220726570656174436f756e743d22696e6460918201527f6566696e697465222f3e203c2f673e203c706f6c79676f6e2066696c6c3d222360b182015260d10195945050505050565b6911103837b4b73a399e9160b11b8152815160009061412281600a8501602087016146a9565b7f222f3e203c706f6c79676f6e206f7061636974793d22302e32222066696c6c3d600a93909101928301525061222360f01b602a820152602c01919050565b6911103837b4b73a399e9160b11b8152815160009061418781600a8501602087016146a9565b7f222f3e203c7265637420783d22302220793d2234373822206f7061636974793d600a9390910192830152506c22302e32222066696c6c3d222360981b602a820152603701919050565b6911103837b4b73a399e9160b11b815283516000906141f781600a8501602089016146a9565b7f222f3e203c616e696d6174654d6f74696f6e20706174683d226d203020302068600a91840191820152600160fd1b602a820152845161423e81602b8401602089016146a9565b7f3530303022206475723d2231353030732220726570656174436f756e743d2269602b92909101918201527f6e646566696e69746522202f3e203c2f673e203c72616469616c477261646965604b820152730373a1034b21e9129bab723b930b234b2b73a11160651b606b82015283516142bf81607f8401602088016146a9565b7f206772616469656e74556e6974733d227573657253706163654f6e557365223e607f92909101918201527f203c73746f70206f66667365743d22302e3736303422207374796c653d227374609f820152696f702d636f6c6f723a2360b01b60bf82015260c90195945050505050565b6001600160a01b038581168252841660208201526040810183905260806060820181905260009061436290830184613b5e565b9695505050505050565b6020815260006123296020830184613b5e565b60208082526032908201527f4552433732313a207472616e7366657220746f206e6f6e20455243373231526560408201527131b2b4bb32b91034b6b83632b6b2b73a32b960711b606082015260800190565b60208082526035908201527f546f6b656e73206e756d62657220746f206d696e742065786365656473206e756040820152746d626572206f66207075626c696320746f6b656e7360581b606082015260800190565b6020808252818101527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572604082015260600190565b60208082526031908201527f4552433732313a207472616e736665722063616c6c6572206973206e6f74206f6040820152701ddb995c881b9bdc88185c1c1c9bdd9959607a1b606082015260800190565b600080821280156001600160ff1b03849003851316156144ce576144ce614787565b600160ff1b83900384128116156144e7576144e7614787565b50500190565b6000821982111561450057614500614787565b500190565b6000826145145761451461479d565b600160ff1b82146000198414161561452e5761452e614787565b500590565b6000826145425761454261479d565b500490565b600181600019825b808611156145835782820483111561456957614569614787565b8086161561457657928202925b94851c949180029161454f565b50509250929050565b600061232983836000826145a257506001610479565b816145af57506000610479565b81600181146145c557600281146145cf576145eb565b6001915050610479565b60ff8411156145e0576145e0614787565b50506001821b610479565b5060208310610133831016604e8410600b841016171561460e575081810a610479565b6146188383614547565b806000190482111561462c5761462c614787565b029392505050565b600081600019048311821515161561464e5761464e614787565b500290565b60008083128015600160ff1b85018412161561467157614671614787565b6001600160ff1b038401831381161561468c5761468c614787565b50500390565b6000828210156146a4576146a4614787565b500390565b60005b838110156146c45781810151838201526020016146ac565b838111156113255750506000910152565b6000816146e4576146e4614787565b506000190190565b600181811c9082168061470057607f821691505b6020821081141561472157634e487b7160e01b600052602260045260246000fd5b50919050565b600060001982141561473b5761473b614787565b5060010190565b6000826147515761475161479d565b500790565b6000826147655761476561479d565b500690565b6000600160ff1b82141561478057614780614787565b5060000390565b634e487b7160e01b600052601160045260246000fd5b634e487b7160e01b600052601260045260246000fd5b634e487b7160e01b600052603160045260246000fd5b634e487b7160e01b600052603260045260246000fd5b634e487b7160e01b600052604160045260246000fd5b6001600160e01b03198116811461099b57600080fdfe22207d2c207b202274726169745f74797065223a202248696c6c73222c202276616c7565223a2022222077696474683d223130303022206865696768743d223733342e353331222f3e203c7265637420783d22302220793d223536332e31353622206f7061636974793d22302e33222066696c6c3d2223222f3e203c73746f70206f66667365743d223122207374796c653d2273746f702d636f6c6f723a233536342c343433203436332c343537203337352c343738203739332c343738203732312c343637203631322c3431344d2d3133372c313034432d33322c32372c3537332c32382c3736322c32363320433538322c39352c3138302c35372c32332c3837633130352d372c3539382c31372c3735372c32343420632d36302d37342d3234312d3138322d3532342d323132633234312c35352c3433352c3138312c3437352c32363520433538372c3230372c3236302c3134312c3232302c3133336331382c342c3337352c3130302c3436312c32363220433530342c3136342c31392c3132352d342c313233633134312c31352c3532342c38362c3634312c32383120433439332c3232392c3136322c3137302c3132322c313633633134312c35312c3234352c3132372c3237332c31393720433334302c3237302c36382c3133392d33392c313436633133392c35312c3234322c3132372c3236392c31393620433131372c3139352d38342c3135302d38342c3135304c2d3735382c39374c2d3133372c3130347a2d31312c3932372c3135362c3931322c3232322c3934312c3238342c3933312c313034332c3933342c313034332c313031312c2d382c31303131222f3e203c73746f70206f66667365743d22302e3938313222207374796c653d2273746f702d636f6c6f723a235b7b202274726169745f74797065223a202253686970222c202276616c7565223a2022222077696474683d223130303022206865696768743d223634392e333135222f3e203c673e203c7061746820786d6c6e733d22687474703a2f2f7777772e77332e6f72672f323030302f73766722206f7061636974793d22302e3535222066696c6c3d222322207d2c207b202274726169745f74797065223a202250616c65747465222c202276616c7565223a20223436332c343439203430332c343537203335312c343331203137372c343738203638312c3437383839342c3933332c3732322c3934382c3636362c3930302c3636362c3930302c3636342c3839302c3634362c3839302c3634352c3930302c3438322c3930302c3438312c3839302c3436322c3839302c3436312c3930302c3436312c3930302c3336342c3936372c3234372c3936312c3130322c3934362c2d392c3931322c2d31322c313030372c313033392c313030372c313034332c3933302d31312c3932372c3433312c3935322c3636332c3932312c3834352c3934352c313034332c3933342c313034332c313031312c2d382c313031312220643d224d383038372c363837632d3135382c302d3332302d332e31352d3436392d3320632d3239332c302d3631362c31302d3730312c3130632d3236312c302d3630302d31372d3830392d313720632d3131382c302d3234362c31312d3337362c3131632d3135382c302d3332302d31302d3436392d313020632d3239332c302d3337392c31302d3537342c3130632d3139352c302d3333312d31312d3534302d313120632d3131382c302d3234362c31312d3337362c3131632d3135382c302d3332302d31302d3436392d313020632d3239332c302d3631362c31372d3730312c3137632d3236312c302d3630302d31322d3830392d313220632d3131382c302d3234362c31322d3337362c3132632d3130332c302d3236332d392d3436392d3920632d39322c302d3138312c322d3236302c32632d3137312c302d3330342c302d3336322c30632d3236312c302d3333302d302d3333302d3020763532356c393035332d365636383843393033392c3638382c383231372c3638372c383038372c3638377a222f3e203c616e696d6174654d6f74696f6e20706174683d224d20302030204c202d38303530203230205a22206475723d223730732220726570656174436f756e743d22696e646566696e69746522202f3e203c2f673e203c673e203c7061746820786d6c6e733d22687474703a2f2f7777772e77332e6f72672f323030302f737667222066696c6c3d2223313031332c343230203835332c343130203733322c343431203633372c343331203630382c343436203339302c343636203234302c343537203138362c3433302039342c343437202d31362c343230202d38372c34373820313031362c3437384142434445464748494a4b4c4d4e4f505152535455565758595a6162636465666768696a6b6c6d6e6f707172737475767778797a303132333435363738392b2f22207d2c207b202274726169745f74797065223a2022436f617374222c202276616c7565223a2022313030342c343537203834382c343634203736302c343530203630382c343535203431392c343333203332302c343534203133352c343039202d382c343234202d382c34373820313030342c3437383232362c343232203137372c3431342038332c333434202d382c333637202d31322c343738203332382c3437384d313038362c3134433938322d36332c3431352d31312c3235362c323639433431322c36342c3738352d31312c3933342c3920632d39392c302d3535392c37302d3639302c3333396335302d38382c3231322d3232352c3437342d32383320632d3232312c38322d3339332c3234302d3432342c333339433431352c3139312c3731362c38392c3735332c373720632d31362c362d3334342c3134342d3431322c333334433439302c3133362c3934312c35302c3936332c343720433833322c37362c3437382c3138392c3338342c343138633132302d3231302c3432362d3330352c3436332d33313620632d3132382c36392d3232302c3136342d3234312c3234356334342d3130362c3238392d3237362c3339312d32373720632d3132362c36392d3231372c3136332d3233372c3234336339342d3137352c3237392d3234332c3237392d3234336c3433382d39394c313038362c31347a222f3e203c2f6c696e6561724772616469656e743e203c7265637420783d22302e303031222066696c6c3d2275726c2823536b794772616469656e7429222077696474683d223130303022206865696768743d223939392e383131222f3e203c706f6c79676f6e206f7061636974793d22302e3135222066696c6c3d22233830302c3935312c313031342c3932372c313033392c313030372c2d31322c313030372c2d31302c3934302c38312c3934312c3137352c3936322c3634372c3932372d31322c3936322c3133362c3934302c3233392c3936302c3634382c3936362c3935312c3933312c313034312c3933382c313034332c313031312c2d382c313031312220643d224d383039372c383436632d3135382c302d3331392d372d3437302d37632d3238352c302d3434332c32302d3635312c323020632d3137322c302d3335332d352d3434392d39632d3130312d342d3234372d32302d3431332d3230632d3131362c302d3234332c32362d3337332c323620632d3135382c302d3332302d33312d3437312d3331632d3238352c302d3335322c33362d3536302c3336632d3137322c302d3339302d33312d3535362d333120632d3131362c302d3234332c32362d3337332c3236632d3135382c302d3332302d33312d3437312d3331632d3238352c302d3434322c33352d3635302c333520632d3137322c302d3335332d352d3434392d39632d3130312d342d3234372d32302d3431332d3230632d3131362c302d3234352c32352d3337352c323520632d3135382c302d3332322d31332d3437342d3133632d3130372c302d3139372c322d3237372c33632d3133332c312d3234332c302d3337322c3020632d3137322c302d3330382d302d3330382d307633363468393035335638343643393033382c3834362c383232372c3834362c383039372c3834367a222f3e203c616e696d6174654d6f74696f6e20706174683d224d20302030204c202d38303530203430205a22206475723d223730732220726570656174436f756e743d22696e646566696e69746522202f3e203c2f673e203c673e203c706f6c79676f6e2066696c6c3d22233939392c333932203836352c343137203830362c343230203734332c343531203532392c34373820313436382c3437382d31312c3932372c3432352c3936392c3837352c3935322c313034362c3838302c313034332c3933342c313034332c313031312c2d382c313031313c7376672077696474683d2231303030707822206865696768743d22313030307078222076696577426f783d2230203020313030302031303030222076657273696f6e3d22312e312220786d6c6e733d22687474703a2f2f7777772e77332e6f72672f323030302f737667223e203c6c696e6561724772616469656e742069643d22536b794772616469656e7422206772616469656e74556e6974733d227573657253706163654f6e557365222078313d223530302e303031222079313d223939392e38313035222078323d223530302e30303039222079323d22342e383832383133652d303034223e203c73746f70206f66667365743d22302e3536303422207374796c653d2273746f702d636f6c6f723a23222f3e203c2f72616469616c4772616469656e743e203c636972636c65206f7061636974793d22302e31222066696c6c3d22234d3137332c3239396338302d37382c3238382d3134332c3532352d313536633132332d362c3233352c322c3332332c3232206331312c312c32312c322c33322c34632d38372d33302d3230352d34382d3333392d3439632d3231302d302d3430302c34332d3439352c313037206338342d37342c3239332d3132382c3532382d313237633231372c302c3339352c34372c3436362c3131346c3433352c36362063302c302d3530302d31382d3530392d3236632d3137352c32312d3332382c38312d3430322c3135316335362d36392c3139332d3133322c3335382d31363320632d31352d322d33302d342d34372d35632d32322c312d34362c322d37302c35433739362c3236322c3633362c3332342c3536302c333936206335362d37302c3139352d3133332c3336322d313634632d34342c302d39312c332d3133392c3820632d3231332c32332d3430302c38372d3439302c3136316337362d38322c3238322d3135382c3532312d313834206339322d31302c3137392d31312c3235352d35632d302d302d302d302d302d30632d38392d31342d3230342d31342d3333302c3120433532342c3234312c3333382c3330392c3235302c3338346337352d38332c3238302d3136332c3531392d313933206332312d322c34322d342c36332d36632d34352c312d39322c352d3134312c3131632d3231332c32372d3430302c39352d3438382c313730206337352d38332c3238302d3136332c3531392d3139336332362d332c35332d362c37382d38632d34322d312d38372d312d3133332c3020433435342c3137352c3236352c3233302c3137332c3239397a4d2d3134342c33303463392d3130362c3333342d3331392c3538302d31373520433233362c34312d322c3135302d36372c3233326335322d34342c3333312d3139352c3535352d343720632d37372d34352d3234312d37382d3431302d35633136322d33352c3334342c382c3431372c373020433330372c3134302c39322c3139372c36352c3230346331322d322c3236332d34322c3430372c373220632d3233362d3134342d3532302d382d3533342d314332342c3233392c3237332c3136372c3435352c33303120433237302c3139332c35372c3235372c33312c323635633130372d332c3230392c32372c3236372c383020433231342c3238342d31312c3236332d36352c333038633130362d332c3230372c32382c3236342c3830204334382c3239362d38372c3332362d38372c3332364c2d3533332c3433334c2d3134342c3330347a22207d2c207b202274726169745f74797065223a202253756e222c202276616c7565223a2022a2646970667358221220320ccb54eac3242f1a83645c7f4cf208147f0f6967d12376960be692ba6301ea64736f6c63430008070033
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.