More Info
Private Name Tags
ContractCreator
Multichain Info
No addresses found
Latest 25 from a total of 368 transactions
Transaction Hash |
Method
|
Block
|
From
|
To
|
|||||
---|---|---|---|---|---|---|---|---|---|
Fill Bid | 18105781 | 564 days ago | IN | 0 ETH | 0.00098933 | ||||
Fill Bid | 15640462 | 910 days ago | IN | 0 ETH | 0.00176172 | ||||
Fill Bid | 15241996 | 971 days ago | IN | 0 ETH | 0.00065589 | ||||
Revoke Bid | 15018320 | 1007 days ago | IN | 0 ETH | 0.00113456 | ||||
Place Bid | 15016213 | 1007 days ago | IN | 0.75 ETH | 0.00244286 | ||||
Fill Bid | 14486092 | 1093 days ago | IN | 0 ETH | 0.00334528 | ||||
Fill Bid | 14381247 | 1109 days ago | IN | 0 ETH | 0.00493769 | ||||
Revoke Bid | 14115919 | 1151 days ago | IN | 0 ETH | 0.00726725 | ||||
Fill Bid | 13945545 | 1177 days ago | IN | 0 ETH | 0.00982634 | ||||
Fill Bid | 13945397 | 1177 days ago | IN | 0 ETH | 0.01033741 | ||||
Place Bid | 13938030 | 1178 days ago | IN | 1.3 ETH | 0.01064512 | ||||
Revoke Bid | 13745735 | 1208 days ago | IN | 0 ETH | 0.00325904 | ||||
Revoke Bid | 13739647 | 1209 days ago | IN | 0 ETH | 0.00366474 | ||||
Revoke Bid | 13739638 | 1209 days ago | IN | 0 ETH | 0.00407933 | ||||
Revoke Bid | 13739481 | 1209 days ago | IN | 0 ETH | 0.00458149 | ||||
Place Bid | 13738483 | 1209 days ago | IN | 0.66 ETH | 0.02359901 | ||||
Place Bid | 13737015 | 1209 days ago | IN | 0.55 ETH | 0.01029897 | ||||
Place Bid | 13729240 | 1211 days ago | IN | 0.55 ETH | 0.01354652 | ||||
Revoke Bid | 13727071 | 1211 days ago | IN | 0 ETH | 0.00288729 | ||||
Place Bid | 13727046 | 1211 days ago | IN | 0.67 ETH | 0.00802014 | ||||
Place Bid | 13723254 | 1212 days ago | IN | 0.38 ETH | 0.01463768 | ||||
Revoke Bid | 13723197 | 1212 days ago | IN | 0 ETH | 0.00574089 | ||||
Place Bid | 13716829 | 1213 days ago | IN | 0.38 ETH | 0.02160914 | ||||
Revoke Bid | 13708334 | 1214 days ago | IN | 0 ETH | 0.00378854 | ||||
Revoke Bid | 13708333 | 1214 days ago | IN | 0 ETH | 0.00390644 |
Latest 25 internal transactions (View All)
Advanced mode:
Parent Transaction Hash | Method | Block |
From
|
To
|
|||
---|---|---|---|---|---|---|---|
Transfer | 18105781 | 564 days ago | 0.021 ETH | ||||
Transfer | 18105781 | 564 days ago | 0.399 ETH | ||||
Transfer | 15640462 | 910 days ago | 0.025 ETH | ||||
Transfer | 15640462 | 910 days ago | 0.475 ETH | ||||
Transfer | 15241996 | 971 days ago | 0.0275 ETH | ||||
Transfer | 15241996 | 971 days ago | 0.5225 ETH | ||||
Transfer | 15018320 | 1007 days ago | 0.75 ETH | ||||
- | 14486092 | 1093 days ago | 0.021 ETH | ||||
- | 14486092 | 1093 days ago | 0.399 ETH | ||||
- | 14381247 | 1109 days ago | 0.025 ETH | ||||
- | 14381247 | 1109 days ago | 0.475 ETH | ||||
- | 14115919 | 1151 days ago | 0.21 ETH | ||||
- | 13945545 | 1177 days ago | 0.0375 ETH | ||||
- | 13945545 | 1177 days ago | 0.7125 ETH | ||||
- | 13945397 | 1177 days ago | 0.065 ETH | ||||
- | 13945397 | 1177 days ago | 1.235 ETH | ||||
- | 13745735 | 1208 days ago | 0.38 ETH | ||||
- | 13739647 | 1209 days ago | 0.66 ETH | ||||
- | 13739638 | 1209 days ago | 0.45 ETH | ||||
- | 13739481 | 1209 days ago | 0.55 ETH | ||||
- | 13727071 | 1211 days ago | 0.67 ETH | ||||
- | 13723197 | 1212 days ago | 0.38 ETH | ||||
- | 13708334 | 1214 days ago | 0.38 ETH | ||||
- | 13708333 | 1214 days ago | 0.541 ETH | ||||
- | 13700914 | 1215 days ago | 0.4 ETH |
Loading...
Loading
Contract Name:
TulipBidding
Compiler Version
v0.8.7+commit.e28d00a7
Optimization Enabled:
No with 200 runs
Other Settings:
default evmVersion
Contract Source Code (Solidity Standard Json-Input format)
// SPDX-License-Identifier: MIT pragma solidity ^0.8.0; import {EtherTulip} from "./EtherTulip.sol"; import {IERC721} from "@openzeppelin/contracts/token/ERC721/IERC721.sol"; import {EnumerableSet} from "@openzeppelin/contracts/utils/structs/EnumerableSet.sol"; contract TulipBidding { using EnumerableSet for EnumerableSet.AddressSet; mapping(uint256 => EnumerableSet.AddressSet) private bidderSet; mapping(address => mapping(uint256 => uint256)) public bids; address public immutable feeRecipient; uint256 public immutable feeBps; address public immutable etherTulip; event BidPlaced(uint256 tulipNumber, address buyer, uint256 price); event BidRevoked(uint256 tulipNumber, address buyer); event BidClaimed(uint256 tulipNumber, address buyer, address seller); constructor( address _feeRecipient, uint256 _feeBps, address _etherTulip ) { feeRecipient = _feeRecipient; feeBps = _feeBps; etherTulip = _etherTulip; } function getBidderCount(uint256 tulipNumber) external view returns (uint256 count) { return bidderSet[tulipNumber].length(); } function getBidderAt(uint256 tulipNumber, uint256 bidderIndex) external view returns (address bidder) { return bidderSet[tulipNumber].at(bidderIndex); } function getBidders(uint256 tulipNumber) external view returns (address[] memory bidders) { return bidderSet[tulipNumber].values(); } function placeBid(uint256 tulipNumber) external payable { // increase bid bids[msg.sender][tulipNumber] += msg.value; // add to bidder set bidderSet[tulipNumber].add(msg.sender); // emit event emit BidPlaced(tulipNumber, msg.sender, bids[msg.sender][tulipNumber]); } function revokeBid(uint256 tulipNumber) external { uint256 bid = bids[msg.sender][tulipNumber]; // clear bid delete bids[msg.sender][tulipNumber]; // remove from bidder set require(bidderSet[tulipNumber].remove(msg.sender), "!bidder"); // return funds payable(msg.sender).transfer(bid); // emit event emit BidRevoked(tulipNumber, msg.sender); } function arbBid( uint256 tulipNumber, address buyer, uint256 botFee ) external { uint256 value = bids[buyer][tulipNumber]; uint256 marketFee = (value * feeBps) / 10000; uint256 price = value - botFee - marketFee; // clear bid delete bids[buyer][tulipNumber]; // remove from bidder set require(bidderSet[tulipNumber].remove(buyer), "!bidder"); // perform purchase EtherTulip(etherTulip).buyTulip{value: price}(tulipNumber); // transfer tulip to buyer EtherTulip(etherTulip).giftTulip(tulipNumber, buyer); // pay the fees payable(msg.sender).transfer(botFee); payable(feeRecipient).transfer(marketFee); // emit event emit BidClaimed(tulipNumber, buyer, msg.sender); } function fillBid(uint256 tulipNumber, address buyer) external { uint256 value = bids[buyer][tulipNumber]; uint256 marketFee = (value * feeBps) / 10000; uint256 price = value - marketFee; // clear bid delete bids[buyer][tulipNumber]; // remove from bidder set require(bidderSet[tulipNumber].remove(buyer), "!bidder"); // transfer tulip to buyer IERC721(etherTulip).transferFrom(msg.sender, buyer, tulipNumber); // pay the fees payable(msg.sender).transfer(price); payable(feeRecipient).transfer(marketFee); // emit event emit BidClaimed(tulipNumber, buyer, msg.sender); } }
// SPDX-License-Identifier: MIT pragma solidity ^0.8.0; import {ERC721} from "@openzeppelin/contracts/token/ERC721/ERC721.sol"; // This is a revised version of the revised version of the original EtherRock contract 0x41f28833Be34e6EDe3c58D1f597bef429861c4E2 with all the rocks removed and rock properties replaced by tulips. // The original contract at 0x41f28833Be34e6EDe3c58D1f597bef429861c4E2 had a simple mistake in the buyRock() function where it would mint a rock and not a tulip. The line: // require(rocks[rockNumber].currentlyForSale == true); // Had to check for the existance of a tulip, as follows: // require(tulips[tulipNumber].currentlyForSale == true); // Therefore in the original contract, anyone could buy anyone elses rock whereas they should have been buying a tulip (regardless of whether the owner chose to sell it or not) contract EtherTulip is ERC721("EtherTulip", unicode"🌷") { struct Tulip { uint256 listingTime; uint256 price; uint256 timesSold; } mapping(uint256 => Tulip) public tulips; uint256 public latestNewTulipForSale; address public immutable feeRecipient; event TulipForSale(uint256 tulipNumber, address owner, uint256 price); event TulipNotForSale(uint256 tulipNumber, address owner); event TulipSold(uint256 tulipNumber, address buyer, uint256 price); constructor(address _feeRecipient) { // set fee recipient feeRecipient = _feeRecipient; // mint founder tulip to yours and only ERC721._mint(address(0x777B0884f97Fd361c55e472530272Be61cEb87c8), 0); // initialize auction for second tulip latestNewTulipForSale = 1; tulips[latestNewTulipForSale].listingTime = block.timestamp; } // Dutch-ish Auction function currentPrice(uint256 tulipNumber) public view returns (uint256 price) { if (tulipNumber == latestNewTulipForSale) { // if currently in auction uint256 initialPrice = 1000 ether; uint256 decayPeriod = 1 days; // price = initial_price - initial_price * (current_time - start_time) / decay_period uint256 elapsedTime = block.timestamp - tulips[tulipNumber].listingTime; if (elapsedTime >= decayPeriod) return 0; return initialPrice - ((initialPrice * elapsedTime) / decayPeriod); } else { // if not in auction return tulips[tulipNumber].price; } } // ERC721 function _baseURI() internal pure override returns (string memory) { return "ipfs://"; } function tokenURI(uint256 tokenId) public view override returns (string memory) { string memory bulbURI = "bafkreiamkokggzkchkosx5jvz6fduzwcn7qugkp7pytmk327sblrtwa4ze"; string[100] memory tulipURIs = [ "bafkreiajlq3nd3nc7xu245eqwnoqkh4fz25rdkinw7wkjtqp54xcbxdski", "bafkreickj5f3dw6abk4ezj6vvephroacnizpt474jxjffccklci4bm7ulu", "bafkreid3ckzlcv3p6zh3o3cwbryno3keqvzyagptnxtoer3awuvyeydz4u", "bafkreid43dy6zwtnvwo6skonl4tcechlhnu7rihp2gnvoaydpxn3qw6yva", "bafkreicnjnnu6w3crbzxkoa2g625avmnecrh4ptmuvj2q5izygeqhttjny", "bafkreiaefcjmmploftqi6ao3dzrpjttnrhay2hqit5pxtzpwkismviwi7e", "bafkreid2jrk7rquxayxdfvyg3w447wqssoutoybqeihuunmau4n5ourj4i", "bafkreicvpenpqezcdehpcd2snsgj4ysl2la44fzqdolblr6evpps3uujcm", "bafkreihuxku32udn2eolrdivdugr2dd5sot2u7cvth3qpsjoqbkllbwkgy", "bafkreidumlyzcwbuaajaeeycq6e6y2zcr4jwq2amy35nsh7laphiecbyvi", "bafkreigjkkhwb3rtdgvbhjc6aoarrlqmyfe7f4d67fvtxprqns3r5zu6jm", "bafkreigmgqiafi4obtmtzzsocylrfcctjtemm3tdvecr3lobo6sqz73o3u", "bafkreig5b6rx3mdxwyrtnumyehvuetzrs3dwkxv76wtqhfayammh2hlsve", "bafkreicvhumzmjl7bo4uqhngwxfpk6upry4waxm3csk7wgkkyzjejgcwai", "bafkreiehn3tovb6wa2nt6nsouavmy6vrijc4z775hk4hjzdn7b5y5lkiya", "bafkreid6lxzktumy535tvzedks3acaqzrqirim4migkbxrp5erx5g6mfeu", "bafkreihk3mimxxqimwkowm2arpfaewz7m5o76wcvw4acyixhtun673az34", "bafkreidrrmeujlvad7oigztu45yeqxwcb6wpo464vwxqst4qo3itwk6ld4", "bafkreihl6hl2ufborow5cfiypdtkk2segp3vxfetupapsjnjvzwxlq2che", "bafkreidnv2t75e3kx6ugajs23l4736assxkzrggsocz2dvygsjlvhvntza", "bafkreih2hhoeruntbfy4ufngko3wbdp4m6gz7otu3hntuvntlflyehkejq", "bafkreieiqmc2t62t2pgabenfkeiclb4gsifcwgqvmkcwqc7lkaxm4vqvai", "bafkreieqi56ie7iyl4rne2r7ypnvrbcpvxxqp346c74smgd3vypgspdwdi", "bafkreidfw4bmdl3rcoehnak43beoeoucbbvtkxlagisvnuchug4foprhpq", "bafkreiasg674cri4woi2nmf2be27i24gdegxhc5hgahpygd7vqcrqvz4va", "bafkreidn3ocz3m6lzgu7whiooq74mtczbget5fie7dldzwhafdkty5yje4", "bafkreieg474qoeg5mykld7dpego4oy4tvjorlgtpjacw3cjrl7oogsfs6a", "bafkreiclbj3ujgxgh5i3plai5t24p7g2aavfzahu2gxq4mavena5ojxfsm", "bafkreia2vekumcnyswh2w62htqu7vedtksepokixpqn4qwzxhefj2bdniq", "bafkreia5456ribmisd555cwt6vzu57oqqkonugozywjuozsoxlo7ffoqxi", "bafkreiekctvro262pk4uz67jpqivxtniebqk3gyfqkclw2vqy3m3iocw6e", "bafkreigtxjoh4rx62uzlx4t23txqk3owvn3ckhy766mjiwgvy3fny2ozza", "bafkreifui4jzaffl74zakibwmt2js2w2qosm7yzgimecwvqpqxpz2psnmu", "bafkreif7426cefprxiupxwggnkjbtht3f3uiaeci2qc3irlppuy7kqoenq", "bafkreigdimqwjrkoklbfiigbkfpvmrniealdjr2bv52jb72sfbk6aqorqy", "bafkreidy2eohxrecrk3si4coolxawlcdmqrvhskhkal4d6bvxjfmds2m24", "bafkreie3qfepdl4pkse6uu3iqwfb2fqusfp3qsp2rh7ynvptps3bbnqzdm", "bafkreih2hv3vjipj2kibpynrmulnnoxj5737cbeg2bjl42zc3bhcfdzf2u", "bafkreiba3227uclzobdglmkvyz4b5da6gp3p3ehtfqznuprlgu3lt7j6nq", "bafkreiakbrgj7y2hzb7dhrpub6c6i72i3cp6ygsrof55t5ueb7wtg33bae", "bafkreig5x3r3dqllt2ndhxpahni5fjeeo2obq4ffls4hkbi7a2cfqtappi", "bafkreihfiplwjo4rdszuhnvjynx7iesij3borjd3rf5onlfgkwlqqjpbou", "bafkreicpnd4dysoaov3ra3jxsnoanyfyg3d56ntuzljro77a2bpgbzo6nu", "bafkreicdlambpulyaytsiffl2yra55ejhdcchpzlx3trutkoc4tinkiomq", "bafkreia57yvsl5esloc7fsn7uiw5q6opkk477f7qgh2sd2mibl66dwgz4m", "bafkreigpxggeedsw2y7rd5dqzhhf4qxq3hw3y5qqnmzgmbg5225z32ke6e", "bafkreihnezodqrdgshpkx5mkkebiz472y6r3xjnhjtrea75td6jxobf7ha", "bafkreiabrwr6bj62rwayp2nvaa7welt24h3rfrsiilj6f6qav4ryjceb4q", "bafkreiddvsn24iswrmxt3ceo7pngsvm44qzzknqto2hz4meytsbqw2q2yy", "bafkreibu53ggwhoojqkqws3l6nampmzxyczkymw5ozfz6q4t7tpwimha4y", "bafkreigwpa25qyx7f23v7t2nbdzywfigwp324dez2kvfrwyibeckl7kmry", "bafkreidjlqycv7hkvn6zmsnwbjy7l563pya2fwtnj5mmiqzosjzyuvv7ay", "bafkreibphjjkaarzd6kknwwmjtooccup7p3wyuvahmaemcyfjnbdbjci24", "bafkreie3unp25d6yx7yvi7bohox2mftpq5rirhl444ylep5ivijovybs5u", "bafkreifwh5ih76ed44cy545vyjovzre2esgk7a6m2njeb2qo5ofzus6q34", "bafkreie3h3ltvvnokylzyzcpultd7a5ba5ldp4ui4ejlhr7lbucklj4zfu", "bafkreiho2ufocna6dgavys6w3hzjudcppni7fnmo5pdes3cn3m7ndiu3b4", "bafkreigdwveaxxfl4sfdnyedx4nnnfv7fzspjmmr3j4eozjfv7tjsovwma", "bafkreidfixu2x5svom4aivbzbwyszrt7etpy7vm4deiepxgfyzshzn2ypi", "bafkreialq7tz66rwqift3hwfhs7mkpjraqp24lk4kk2amlnh535zzkizzy", "bafkreieay36cd2bsrn2tipb4zijvkylnfj7dinsaz342gnope7wsrrypjq", "bafkreieo6nbwxyahaafswfbe4sahqby3h7ru2hfitzru6iqyxqgch7nrzq", "bafkreicsv6fe4ykonns6hxctcjht7bqx5646cbpgbsug4xr6zg5rbthdj4", "bafkreihsgdzvuirts3jt3g26lto73qqu4bg7n4qu72dgmkujjkenee7e7a", "bafkreigkjwzzhv4cndw3ehsmsbmixzrxxvrpqj2e2yemxnnlwkew5f6gs4", "bafkreic5k7u2m4g3yufqpin7ofnb7hgl5vdtmto36l33bw6sjw5hktmo5a", "bafkreihxcdefq5k4vomblxkuj75d3b3ixasxvhuobag4whkrwyahfyfidq", "bafkreicjwz7p6dpvcqs7ordhzwslabcwuj3y3hehk4pl63jktebjmncjfy", "bafkreiakfuy3pabipys3ese6uerduczphy5rdkccguhyhsgyprmn7qz3ke", "bafkreialjw6gr3smsmhmrvcdvlsxxvgroyubz7fk2wrvpcrmomx7pdnjoe", "bafkreieqchumy7zibxq4mqppdqcdvfawq6ot7dxpbz3v5fue2bwyp5ivsm", "bafkreicev7nx7s75yqv2jpttw4rrukztwtr4kubxrs3wv3v4qlkz4nq4uy", "bafkreid2itljfobnxpvjfxnsas2fqbly5dyn7tkogvcojjxmb7muqiejta", "bafkreibmiuznmb5lk7gsg6376b4z6orbn2b535u6wnimiq5hwe5scref6u", "bafkreigxvddcurbaxgubrenqeahen47dktm4om535kaedeqenx7p6zodty", "bafkreibtwpwhksgskwqshwuqehwfm772pu7rfxjcxpijpu5ikh2cwvczvq", "bafkreic6j5qpgw326wdt6wyaf6efmbzzcw4rl5zfjiken3uxhu4s65hc2e", "bafkreibzl22el47bco2hi6wj6zovosteowcdkzifegbg3xncdhd3d3pjnu", "bafkreiew3gjv2dgldfzw5kfsginkvqz5rzktd5d4n6a2fnmuqjjk7hhcim", "bafkreigbjzymbt3fmlmfta5oror25swedugp2drdvsjb3juahn6e33r6l4", "bafkreicpzyumn7fo7negxdwb3anljwf2lqyseqeh7yogcoiyi75kfbtqky", "bafkreihsznkosxzsgfkutuuzhh442k7esjduylhda5ausfbautruuazzmm", "bafkreidbg24dnbt5to5vl6m5ektfwstwhp2djhsllf5mtciizlgg4lse6y", "bafkreidefsxxb5i4nvdblya2mh7h4uo47dwuthb5jqmartgg4svwrvx2xe", "bafkreic3qsb5mn22xy46gkg3whiqhvb2lqjmucbquiq27fyf76b7gpf3kq", "bafkreiaej6mkm6jjixeqtheomqtbpc4xp3yrtp66vnmcywxler5qlgidw4", "bafkreidhgkhkuwp5qqxzdmga7uh52qzeveu53zt4a2ahoauur6gaorf3sq", "bafkreibbw4oyakigmurscj3aknyq4tq4b43jjk2wsoszo4672shsog4ssu", "bafkreibbhcuhxye2ga3yblf4vmxsq4n3aofxckbyl66axugczb3e6dansu", "bafkreibxvv6oipwxyaejwvf4vnftsd6n3bf6ixnvubyus4q5nqud4plcve", "bafkreiahwdvpwtzcn2t45rswol5gka42yd2zkn37tn4mscq7mesuwngqyq", "bafkreig3mw4xebzxvrj7w5yvkg6o5a55tf2nlenbpruf7oyo3gauqjhtim", "bafkreihkaovv3fj2usy3kkzp4hspkisaqymki56pljvqm7yrv2ijta3pbq", "bafkreicytcw572hqzksn3zrdut22xo7wjxkf4ovjijokhpkx6vnc3nax7i", "bafkreidsfwztyha6q7i5temqm44mlhslcxerjhasj2kdfqhkssl5yfrgnu", "bafkreibsvhu3jxeyohswwic7u3xtzwwzbgssqrmmwzyj2fwhnykwpuu7eu", "bafkreibexexusac5lrtzmk26f4gc7pg2hzdf4upmd7ks37rteazol76ccy", "bafkreiguleybtirhrwdfp3bmmkm7xwq63so3waxgbwyh277uh62dbh55n4", "bafkreig7bdme4w26it2vhpdu2ahxbtl2svul5i3h5xhlpbdrxnmrnomqze", "bafkreicjehn5krr3lcnyl65eyh3njrfw5atepdzfmjaga44fmangdv5t6q" ]; require(tokenId < 100, "Enter a tokenId from 0 to 99. Only 100 tulips."); if (tokenId >= latestNewTulipForSale) { return string(abi.encodePacked(_baseURI(), bulbURI)); } else { return string(abi.encodePacked(_baseURI(), tulipURIs[tokenId])); } } function _beforeTokenTransfer( address, address, uint256 tokenId ) internal override { // unlist tulip tulips[tokenId].listingTime = 0; // emit event emit TulipNotForSale(tokenId, msg.sender); } // ETHERROCK function getTulipInfo(uint256 tulipNumber) public view returns ( address owner, uint256 listingTime, uint256 price, uint256 timesSold ) { return ( ERC721.ownerOf(tulipNumber), tulips[tulipNumber].listingTime, currentPrice(tulipNumber), tulips[tulipNumber].timesSold ); } function buyTulip(uint256 tulipNumber) public payable { // check sellable require(tulips[tulipNumber].listingTime != 0); require(tulipNumber < 100, "Enter a tokenId from 0 to 99. Only 100 tulips."); // check for sufficient payment require(msg.value >= currentPrice(tulipNumber)); // unlist and update metadata tulips[tulipNumber].listingTime = 0; tulips[tulipNumber].timesSold++; // swap ownership for payment if (tulipNumber >= latestNewTulipForSale) { // if new, _mint() uint256 _latestNewTulipForSale = latestNewTulipForSale; // update auction if (latestNewTulipForSale < 99) { latestNewTulipForSale++; tulips[latestNewTulipForSale].listingTime = block.timestamp; } else { latestNewTulipForSale++; } // mint and transfer payment ERC721._mint(msg.sender, _latestNewTulipForSale); payable(feeRecipient).transfer(msg.value); } else { // if old, _transfer() address seller = ERC721.ownerOf(tulipNumber); ERC721._transfer(seller, msg.sender, tulipNumber); payable(seller).transfer(msg.value); } // emit event emit TulipSold(tulipNumber, msg.sender, msg.value); } function sellTulip(uint256 tulipNumber, uint256 price) public { require(msg.sender == ERC721.ownerOf(tulipNumber)); require(price > 0); tulips[tulipNumber].price = price; tulips[tulipNumber].listingTime = block.timestamp; // emit event emit TulipForSale(tulipNumber, msg.sender, price); } function dontSellTulip(uint256 tulipNumber) public { require(msg.sender == ERC721.ownerOf(tulipNumber)); tulips[tulipNumber].listingTime = 0; // emit event emit TulipNotForSale(tulipNumber, msg.sender); } function giftTulip(uint256 tulipNumber, address receiver) public { ERC721.transferFrom(msg.sender, receiver, tulipNumber); } }
// 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 Library for managing * https://en.wikipedia.org/wiki/Set_(abstract_data_type)[sets] of primitive * types. * * Sets have the following properties: * * - Elements are added, removed, and checked for existence in constant time * (O(1)). * - Elements are enumerated in O(n). No guarantees are made on the ordering. * * ``` * contract Example { * // Add the library methods * using EnumerableSet for EnumerableSet.AddressSet; * * // Declare a set state variable * EnumerableSet.AddressSet private mySet; * } * ``` * * As of v3.3.0, sets of type `bytes32` (`Bytes32Set`), `address` (`AddressSet`) * and `uint256` (`UintSet`) are supported. */ library EnumerableSet { // To implement this library for multiple types with as little code // repetition as possible, we write it in terms of a generic Set type with // bytes32 values. // The Set implementation uses private functions, and user-facing // implementations (such as AddressSet) are just wrappers around the // underlying Set. // This means that we can only create new EnumerableSets for types that fit // in bytes32. struct Set { // Storage of set values bytes32[] _values; // Position of the value in the `values` array, plus 1 because index 0 // means a value is not in the set. mapping(bytes32 => uint256) _indexes; } /** * @dev Add a value to a set. O(1). * * Returns true if the value was added to the set, that is if it was not * already present. */ function _add(Set storage set, bytes32 value) private returns (bool) { if (!_contains(set, value)) { set._values.push(value); // The value is stored at length-1, but we add 1 to all indexes // and use 0 as a sentinel value set._indexes[value] = set._values.length; return true; } else { return false; } } /** * @dev Removes a value from a set. O(1). * * Returns true if the value was removed from the set, that is if it was * present. */ function _remove(Set storage set, bytes32 value) private returns (bool) { // We read and store the value's index to prevent multiple reads from the same storage slot uint256 valueIndex = set._indexes[value]; if (valueIndex != 0) { // Equivalent to contains(set, value) // To delete an element from the _values array in O(1), we swap the element to delete with the last one in // the array, and then remove the last element (sometimes called as 'swap and pop'). // This modifies the order of the array, as noted in {at}. uint256 toDeleteIndex = valueIndex - 1; uint256 lastIndex = set._values.length - 1; if (lastIndex != toDeleteIndex) { bytes32 lastvalue = set._values[lastIndex]; // Move the last value to the index where the value to delete is set._values[toDeleteIndex] = lastvalue; // Update the index for the moved value set._indexes[lastvalue] = valueIndex; // Replace lastvalue's index to valueIndex } // Delete the slot where the moved value was stored set._values.pop(); // Delete the index for the deleted slot delete set._indexes[value]; return true; } else { return false; } } /** * @dev Returns true if the value is in the set. O(1). */ function _contains(Set storage set, bytes32 value) private view returns (bool) { return set._indexes[value] != 0; } /** * @dev Returns the number of values on the set. O(1). */ function _length(Set storage set) private view returns (uint256) { return set._values.length; } /** * @dev Returns the value stored at position `index` in the set. O(1). * * Note that there are no guarantees on the ordering of values inside the * array, and it may change when more values are added or removed. * * Requirements: * * - `index` must be strictly less than {length}. */ function _at(Set storage set, uint256 index) private view returns (bytes32) { return set._values[index]; } /** * @dev Return the entire set in an array * * WARNING: This operation will copy the entire storage to memory, which can be quite expensive. This is designed * to mostly be used by view accessors that are queried without any gas fees. Developers should keep in mind that * this function has an unbounded cost, and using it as part of a state-changing function may render the function * uncallable if the set grows to a point where copying to memory consumes too much gas to fit in a block. */ function _values(Set storage set) private view returns (bytes32[] memory) { return set._values; } // Bytes32Set struct Bytes32Set { Set _inner; } /** * @dev Add a value to a set. O(1). * * Returns true if the value was added to the set, that is if it was not * already present. */ function add(Bytes32Set storage set, bytes32 value) internal returns (bool) { return _add(set._inner, value); } /** * @dev Removes a value from a set. O(1). * * Returns true if the value was removed from the set, that is if it was * present. */ function remove(Bytes32Set storage set, bytes32 value) internal returns (bool) { return _remove(set._inner, value); } /** * @dev Returns true if the value is in the set. O(1). */ function contains(Bytes32Set storage set, bytes32 value) internal view returns (bool) { return _contains(set._inner, value); } /** * @dev Returns the number of values in the set. O(1). */ function length(Bytes32Set storage set) internal view returns (uint256) { return _length(set._inner); } /** * @dev Returns the value stored at position `index` in the set. O(1). * * Note that there are no guarantees on the ordering of values inside the * array, and it may change when more values are added or removed. * * Requirements: * * - `index` must be strictly less than {length}. */ function at(Bytes32Set storage set, uint256 index) internal view returns (bytes32) { return _at(set._inner, index); } /** * @dev Return the entire set in an array * * WARNING: This operation will copy the entire storage to memory, which can be quite expensive. This is designed * to mostly be used by view accessors that are queried without any gas fees. Developers should keep in mind that * this function has an unbounded cost, and using it as part of a state-changing function may render the function * uncallable if the set grows to a point where copying to memory consumes too much gas to fit in a block. */ function values(Bytes32Set storage set) internal view returns (bytes32[] memory) { return _values(set._inner); } // AddressSet struct AddressSet { Set _inner; } /** * @dev Add a value to a set. O(1). * * Returns true if the value was added to the set, that is if it was not * already present. */ function add(AddressSet storage set, address value) internal returns (bool) { return _add(set._inner, bytes32(uint256(uint160(value)))); } /** * @dev Removes a value from a set. O(1). * * Returns true if the value was removed from the set, that is if it was * present. */ function remove(AddressSet storage set, address value) internal returns (bool) { return _remove(set._inner, bytes32(uint256(uint160(value)))); } /** * @dev Returns true if the value is in the set. O(1). */ function contains(AddressSet storage set, address value) internal view returns (bool) { return _contains(set._inner, bytes32(uint256(uint160(value)))); } /** * @dev Returns the number of values in the set. O(1). */ function length(AddressSet storage set) internal view returns (uint256) { return _length(set._inner); } /** * @dev Returns the value stored at position `index` in the set. O(1). * * Note that there are no guarantees on the ordering of values inside the * array, and it may change when more values are added or removed. * * Requirements: * * - `index` must be strictly less than {length}. */ function at(AddressSet storage set, uint256 index) internal view returns (address) { return address(uint160(uint256(_at(set._inner, index)))); } /** * @dev Return the entire set in an array * * WARNING: This operation will copy the entire storage to memory, which can be quite expensive. This is designed * to mostly be used by view accessors that are queried without any gas fees. Developers should keep in mind that * this function has an unbounded cost, and using it as part of a state-changing function may render the function * uncallable if the set grows to a point where copying to memory consumes too much gas to fit in a block. */ function values(AddressSet storage set) internal view returns (address[] memory) { bytes32[] memory store = _values(set._inner); address[] memory result; assembly { result := store } return result; } // UintSet struct UintSet { Set _inner; } /** * @dev Add a value to a set. O(1). * * Returns true if the value was added to the set, that is if it was not * already present. */ function add(UintSet storage set, uint256 value) internal returns (bool) { return _add(set._inner, bytes32(value)); } /** * @dev Removes a value from a set. O(1). * * Returns true if the value was removed from the set, that is if it was * present. */ function remove(UintSet storage set, uint256 value) internal returns (bool) { return _remove(set._inner, bytes32(value)); } /** * @dev Returns true if the value is in the set. O(1). */ function contains(UintSet storage set, uint256 value) internal view returns (bool) { return _contains(set._inner, bytes32(value)); } /** * @dev Returns the number of values on the set. O(1). */ function length(UintSet storage set) internal view returns (uint256) { return _length(set._inner); } /** * @dev Returns the value stored at position `index` in the set. O(1). * * Note that there are no guarantees on the ordering of values inside the * array, and it may change when more values are added or removed. * * Requirements: * * - `index` must be strictly less than {length}. */ function at(UintSet storage set, uint256 index) internal view returns (uint256) { return uint256(_at(set._inner, index)); } /** * @dev Return the entire set in an array * * WARNING: This operation will copy the entire storage to memory, which can be quite expensive. This is designed * to mostly be used by view accessors that are queried without any gas fees. Developers should keep in mind that * this function has an unbounded cost, and using it as part of a state-changing function may render the function * uncallable if the set grows to a point where copying to memory consumes too much gas to fit in a block. */ function values(UintSet storage set) internal view returns (uint256[] memory) { bytes32[] memory store = _values(set._inner); uint256[] memory result; assembly { result := store } return result; } }
// SPDX-License-Identifier: MIT 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; /** * @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 "../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; /** * @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; /** * @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 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; 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 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": false, "runs": 200 }, "outputSelection": { "*": { "*": [ "evm.bytecode", "evm.deployedBytecode", "abi" ] } }, "libraries": {} }
Contract Security Audit
- No Contract Security Audit Submitted- Submit Audit Here
Contract ABI
API[{"inputs":[{"internalType":"address","name":"_feeRecipient","type":"address"},{"internalType":"uint256","name":"_feeBps","type":"uint256"},{"internalType":"address","name":"_etherTulip","type":"address"}],"stateMutability":"nonpayable","type":"constructor"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"tulipNumber","type":"uint256"},{"indexed":false,"internalType":"address","name":"buyer","type":"address"},{"indexed":false,"internalType":"address","name":"seller","type":"address"}],"name":"BidClaimed","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"tulipNumber","type":"uint256"},{"indexed":false,"internalType":"address","name":"buyer","type":"address"},{"indexed":false,"internalType":"uint256","name":"price","type":"uint256"}],"name":"BidPlaced","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"tulipNumber","type":"uint256"},{"indexed":false,"internalType":"address","name":"buyer","type":"address"}],"name":"BidRevoked","type":"event"},{"inputs":[{"internalType":"uint256","name":"tulipNumber","type":"uint256"},{"internalType":"address","name":"buyer","type":"address"},{"internalType":"uint256","name":"botFee","type":"uint256"}],"name":"arbBid","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"},{"internalType":"uint256","name":"","type":"uint256"}],"name":"bids","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"etherTulip","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"feeBps","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"feeRecipient","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"tulipNumber","type":"uint256"},{"internalType":"address","name":"buyer","type":"address"}],"name":"fillBid","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"tulipNumber","type":"uint256"},{"internalType":"uint256","name":"bidderIndex","type":"uint256"}],"name":"getBidderAt","outputs":[{"internalType":"address","name":"bidder","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"tulipNumber","type":"uint256"}],"name":"getBidderCount","outputs":[{"internalType":"uint256","name":"count","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"tulipNumber","type":"uint256"}],"name":"getBidders","outputs":[{"internalType":"address[]","name":"bidders","type":"address[]"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"tulipNumber","type":"uint256"}],"name":"placeBid","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[{"internalType":"uint256","name":"tulipNumber","type":"uint256"}],"name":"revokeBid","outputs":[],"stateMutability":"nonpayable","type":"function"}]
Contract Creation Code
60e06040523480156200001157600080fd5b506040516200180e3803806200180e8339818101604052810190620000379190620000e4565b8273ffffffffffffffffffffffffffffffffffffffff1660808173ffffffffffffffffffffffffffffffffffffffff1660601b815250508160a081815250508073ffffffffffffffffffffffffffffffffffffffff1660c08173ffffffffffffffffffffffffffffffffffffffff1660601b81525050505050620001b7565b600081519050620000c78162000183565b92915050565b600081519050620000de816200019d565b92915050565b6000806000606084860312156200010057620000ff6200017e565b5b60006200011086828701620000b6565b93505060206200012386828701620000cd565b92505060406200013686828701620000b6565b9150509250925092565b60006200014d8262000154565b9050919050565b600073ffffffffffffffffffffffffffffffffffffffff82169050919050565b6000819050919050565b600080fd5b6200018e8162000140565b81146200019a57600080fd5b50565b620001a88162000174565b8114620001b457600080fd5b50565b60805160601c60a05160c05160601c6115f06200021e600039600081816102f8015281816104df01528181610a6c0152610af80152600081816102d4015281816103e0015261096201526000818161031c015281816105b50152610bcc01526115f06000f3fe60806040526004361061009c5760003560e01c806358fff20e1161006457806358fff20e1461019c5780638154f218146101d95780639979ef4514610202578063d1a1f09a1461021e578063db89c5e914610247578063fae09be5146102705761009c565b806301495c1c146100a157806324a9d853146100de5780632dddcb621461010957806346904840146101345780634fa7b5071461015f575b600080fd5b3480156100ad57600080fd5b506100c860048036038101906100c39190610fbb565b6102ad565b6040516100d59190611255565b60405180910390f35b3480156100ea57600080fd5b506100f36102d2565b6040516101009190611255565b60405180910390f35b34801561011557600080fd5b5061011e6102f6565b60405161012b91906111c1565b60405180910390f35b34801561014057600080fd5b5061014961031a565b60405161015691906111c1565b60405180910390f35b34801561016b57600080fd5b5061018660048036038101906101819190610ffb565b61033e565b6040516101939190611213565b60405180910390f35b3480156101a857600080fd5b506101c360048036038101906101be9190610ffb565b610361565b6040516101d09190611255565b60405180910390f35b3480156101e557600080fd5b5061020060048036038101906101fb9190611028565b610384565b005b61021c60048036038101906102179190610ffb565b61065c565b005b34801561022a57600080fd5b5061024560048036038101906102409190610ffb565b610776565b005b34801561025357600080fd5b5061026e60048036038101906102699190611068565b610906565b005b34801561027c57600080fd5b50610297600480360381019061029291906110bb565b610c74565b6040516102a491906111c1565b60405180910390f35b6001602052816000526040600020602052806000526040600020600091509150505481565b7f000000000000000000000000000000000000000000000000000000000000000081565b7f000000000000000000000000000000000000000000000000000000000000000081565b7f000000000000000000000000000000000000000000000000000000000000000081565b606061035a600080848152602001908152602001600020610ca2565b9050919050565b600061037d600080848152602001908152602001600020610cc3565b9050919050565b6000600160008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600084815260200190815260200160002054905060006127107f00000000000000000000000000000000000000000000000000000000000000008361040a91906113d8565b61041491906113a7565b9050600081836104249190611432565b9050600160008573ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060008681526020019081526020016000206000905561049e84600080888152602001908152602001600020610cd890919063ffffffff16565b6104dd576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016104d490611235565b60405180910390fd5b7f000000000000000000000000000000000000000000000000000000000000000073ffffffffffffffffffffffffffffffffffffffff166323b872dd3386886040518463ffffffff1660e01b815260040161053a939291906111dc565b600060405180830381600087803b15801561055457600080fd5b505af1158015610568573d6000803e3d6000fd5b505050503373ffffffffffffffffffffffffffffffffffffffff166108fc829081150290604051600060405180830381858888f193505050501580156105b2573d6000803e3d6000fd5b507f000000000000000000000000000000000000000000000000000000000000000073ffffffffffffffffffffffffffffffffffffffff166108fc839081150290604051600060405180830381858888f19350505050158015610619573d6000803e3d6000fd5b507ffd9c0b9a91ed824caef4b611bb4ee6570f6ee555940d834fb0c69708c37a235685853360405161064d93929190611299565b60405180910390a15050505050565b34600160003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600083815260200190815260200160002060008282546106bc9190611351565b925050819055506106e733600080848152602001908152602001600020610d0890919063ffffffff16565b507f0e54eff26401bf69b81b26f60bd85ef47f5d85275c1d268d84f68d6897431c478133600160003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060008581526020019081526020016000205460405161076b939291906112d0565b60405180910390a150565b6000600160003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000206000838152602001908152602001600020549050600160003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060008381526020019081526020016000206000905561084333600080858152602001908152602001600020610cd890919063ffffffff16565b610882576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161087990611235565b60405180910390fd5b3373ffffffffffffffffffffffffffffffffffffffff166108fc829081150290604051600060405180830381858888f193505050501580156108c8573d6000803e3d6000fd5b507f9edd4863b9e24d3cdc33236f8db27d8af64fa502af04ceb36548e382803ab7f882336040516108fa929190611270565b60405180910390a15050565b6000600160008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600085815260200190815260200160002054905060006127107f00000000000000000000000000000000000000000000000000000000000000008361098c91906113d8565b61099691906113a7565b905060008184846109a79190611432565b6109b19190611432565b9050600160008673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600087815260200190815260200160002060009055610a2b85600080898152602001908152602001600020610cd890919063ffffffff16565b610a6a576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610a6190611235565b60405180910390fd5b7f000000000000000000000000000000000000000000000000000000000000000073ffffffffffffffffffffffffffffffffffffffff16630603849082886040518363ffffffff1660e01b8152600401610ac49190611255565b6000604051808303818588803b158015610add57600080fd5b505af1158015610af1573d6000803e3d6000fd5b50505050507f000000000000000000000000000000000000000000000000000000000000000073ffffffffffffffffffffffffffffffffffffffff166327a664e987876040518363ffffffff1660e01b8152600401610b51929190611270565b600060405180830381600087803b158015610b6b57600080fd5b505af1158015610b7f573d6000803e3d6000fd5b505050503373ffffffffffffffffffffffffffffffffffffffff166108fc859081150290604051600060405180830381858888f19350505050158015610bc9573d6000803e3d6000fd5b507f000000000000000000000000000000000000000000000000000000000000000073ffffffffffffffffffffffffffffffffffffffff166108fc839081150290604051600060405180830381858888f19350505050158015610c30573d6000803e3d6000fd5b507ffd9c0b9a91ed824caef4b611bb4ee6570f6ee555940d834fb0c69708c37a2356868633604051610c6493929190611299565b60405180910390a1505050505050565b6000610c9a82600080868152602001908152602001600020610d3890919063ffffffff16565b905092915050565b60606000610cb283600001610d52565b905060608190508092505050919050565b6000610cd182600001610dae565b9050919050565b6000610d00836000018373ffffffffffffffffffffffffffffffffffffffff1660001b610dbf565b905092915050565b6000610d30836000018373ffffffffffffffffffffffffffffffffffffffff1660001b610ed3565b905092915050565b6000610d478360000183610f43565b60001c905092915050565b606081600001805480602002602001604051908101604052809291908181526020018280548015610da257602002820191906000526020600020905b815481526020019060010190808311610d8e575b50505050509050919050565b600081600001805490509050919050565b60008083600101600084815260200190815260200160002054905060008114610ec7576000600182610df19190611432565b9050600060018660000180549050610e099190611432565b9050818114610e78576000866000018281548110610e2a57610e2961152f565b5b9060005260206000200154905080876000018481548110610e4e57610e4d61152f565b5b90600052602060002001819055508387600101600083815260200190815260200160002081905550505b85600001805480610e8c57610e8b611500565b5b600190038181906000526020600020016000905590558560010160008681526020019081526020016000206000905560019350505050610ecd565b60009150505b92915050565b6000610edf8383610f6e565b610f38578260000182908060018154018082558091505060019003906000526020600020016000909190919091505582600001805490508360010160008481526020019081526020016000208190555060019050610f3d565b600090505b92915050565b6000826000018281548110610f5b57610f5a61152f565b5b9060005260206000200154905092915050565b600080836001016000848152602001908152602001600020541415905092915050565b600081359050610fa08161158c565b92915050565b600081359050610fb5816115a3565b92915050565b60008060408385031215610fd257610fd161155e565b5b6000610fe085828601610f91565b9250506020610ff185828601610fa6565b9150509250929050565b6000602082840312156110115761101061155e565b5b600061101f84828501610fa6565b91505092915050565b6000806040838503121561103f5761103e61155e565b5b600061104d85828601610fa6565b925050602061105e85828601610f91565b9150509250929050565b6000806000606084860312156110815761108061155e565b5b600061108f86828701610fa6565b93505060206110a086828701610f91565b92505060406110b186828701610fa6565b9150509250925092565b600080604083850312156110d2576110d161155e565b5b60006110e085828601610fa6565b92505060206110f185828601610fa6565b9150509250929050565b60006111078383611113565b60208301905092915050565b61111c81611466565b82525050565b61112b81611466565b82525050565b600061113c82611317565b611146818561132f565b935061115183611307565b8060005b8381101561118257815161116988826110fb565b975061117483611322565b925050600181019050611155565b5085935050505092915050565b600061119c600783611340565b91506111a782611563565b602082019050919050565b6111bb81611498565b82525050565b60006020820190506111d66000830184611122565b92915050565b60006060820190506111f16000830186611122565b6111fe6020830185611122565b61120b60408301846111b2565b949350505050565b6000602082019050818103600083015261122d8184611131565b905092915050565b6000602082019050818103600083015261124e8161118f565b9050919050565b600060208201905061126a60008301846111b2565b92915050565b600060408201905061128560008301856111b2565b6112926020830184611122565b9392505050565b60006060820190506112ae60008301866111b2565b6112bb6020830185611122565b6112c86040830184611122565b949350505050565b60006060820190506112e560008301866111b2565b6112f26020830185611122565b6112ff60408301846111b2565b949350505050565b6000819050602082019050919050565b600081519050919050565b6000602082019050919050565b600082825260208201905092915050565b600082825260208201905092915050565b600061135c82611498565b915061136783611498565b9250827fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff0382111561139c5761139b6114a2565b5b828201905092915050565b60006113b282611498565b91506113bd83611498565b9250826113cd576113cc6114d1565b5b828204905092915050565b60006113e382611498565b91506113ee83611498565b9250817fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff0483118215151615611427576114266114a2565b5b828202905092915050565b600061143d82611498565b915061144883611498565b92508282101561145b5761145a6114a2565b5b828203905092915050565b600061147182611478565b9050919050565b600073ffffffffffffffffffffffffffffffffffffffff82169050919050565b6000819050919050565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052601160045260246000fd5b7f4e487b7100000000000000000000000000000000000000000000000000000000600052601260045260246000fd5b7f4e487b7100000000000000000000000000000000000000000000000000000000600052603160045260246000fd5b7f4e487b7100000000000000000000000000000000000000000000000000000000600052603260045260246000fd5b600080fd5b7f2162696464657200000000000000000000000000000000000000000000000000600082015250565b61159581611466565b81146115a057600080fd5b50565b6115ac81611498565b81146115b757600080fd5b5056fea2646970667358221220574cbdb6f8626032bb8d2022a46d06ddf0c4c72fc2dfb21ed48cc11b7483ef9864736f6c63430008070033000000000000000000000000360059bbd6df9ae032e93a8e5fa7900bbd10363a00000000000000000000000000000000000000000000000000000000000001f4000000000000000000000000d5fbd81cef9aba7464c5f17e529444918a8ecc57
Deployed Bytecode
0x60806040526004361061009c5760003560e01c806358fff20e1161006457806358fff20e1461019c5780638154f218146101d95780639979ef4514610202578063d1a1f09a1461021e578063db89c5e914610247578063fae09be5146102705761009c565b806301495c1c146100a157806324a9d853146100de5780632dddcb621461010957806346904840146101345780634fa7b5071461015f575b600080fd5b3480156100ad57600080fd5b506100c860048036038101906100c39190610fbb565b6102ad565b6040516100d59190611255565b60405180910390f35b3480156100ea57600080fd5b506100f36102d2565b6040516101009190611255565b60405180910390f35b34801561011557600080fd5b5061011e6102f6565b60405161012b91906111c1565b60405180910390f35b34801561014057600080fd5b5061014961031a565b60405161015691906111c1565b60405180910390f35b34801561016b57600080fd5b5061018660048036038101906101819190610ffb565b61033e565b6040516101939190611213565b60405180910390f35b3480156101a857600080fd5b506101c360048036038101906101be9190610ffb565b610361565b6040516101d09190611255565b60405180910390f35b3480156101e557600080fd5b5061020060048036038101906101fb9190611028565b610384565b005b61021c60048036038101906102179190610ffb565b61065c565b005b34801561022a57600080fd5b5061024560048036038101906102409190610ffb565b610776565b005b34801561025357600080fd5b5061026e60048036038101906102699190611068565b610906565b005b34801561027c57600080fd5b50610297600480360381019061029291906110bb565b610c74565b6040516102a491906111c1565b60405180910390f35b6001602052816000526040600020602052806000526040600020600091509150505481565b7f00000000000000000000000000000000000000000000000000000000000001f481565b7f000000000000000000000000d5fbd81cef9aba7464c5f17e529444918a8ecc5781565b7f000000000000000000000000360059bbd6df9ae032e93a8e5fa7900bbd10363a81565b606061035a600080848152602001908152602001600020610ca2565b9050919050565b600061037d600080848152602001908152602001600020610cc3565b9050919050565b6000600160008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600084815260200190815260200160002054905060006127107f00000000000000000000000000000000000000000000000000000000000001f48361040a91906113d8565b61041491906113a7565b9050600081836104249190611432565b9050600160008573ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060008681526020019081526020016000206000905561049e84600080888152602001908152602001600020610cd890919063ffffffff16565b6104dd576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016104d490611235565b60405180910390fd5b7f000000000000000000000000d5fbd81cef9aba7464c5f17e529444918a8ecc5773ffffffffffffffffffffffffffffffffffffffff166323b872dd3386886040518463ffffffff1660e01b815260040161053a939291906111dc565b600060405180830381600087803b15801561055457600080fd5b505af1158015610568573d6000803e3d6000fd5b505050503373ffffffffffffffffffffffffffffffffffffffff166108fc829081150290604051600060405180830381858888f193505050501580156105b2573d6000803e3d6000fd5b507f000000000000000000000000360059bbd6df9ae032e93a8e5fa7900bbd10363a73ffffffffffffffffffffffffffffffffffffffff166108fc839081150290604051600060405180830381858888f19350505050158015610619573d6000803e3d6000fd5b507ffd9c0b9a91ed824caef4b611bb4ee6570f6ee555940d834fb0c69708c37a235685853360405161064d93929190611299565b60405180910390a15050505050565b34600160003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600083815260200190815260200160002060008282546106bc9190611351565b925050819055506106e733600080848152602001908152602001600020610d0890919063ffffffff16565b507f0e54eff26401bf69b81b26f60bd85ef47f5d85275c1d268d84f68d6897431c478133600160003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060008581526020019081526020016000205460405161076b939291906112d0565b60405180910390a150565b6000600160003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000206000838152602001908152602001600020549050600160003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060008381526020019081526020016000206000905561084333600080858152602001908152602001600020610cd890919063ffffffff16565b610882576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161087990611235565b60405180910390fd5b3373ffffffffffffffffffffffffffffffffffffffff166108fc829081150290604051600060405180830381858888f193505050501580156108c8573d6000803e3d6000fd5b507f9edd4863b9e24d3cdc33236f8db27d8af64fa502af04ceb36548e382803ab7f882336040516108fa929190611270565b60405180910390a15050565b6000600160008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600085815260200190815260200160002054905060006127107f00000000000000000000000000000000000000000000000000000000000001f48361098c91906113d8565b61099691906113a7565b905060008184846109a79190611432565b6109b19190611432565b9050600160008673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600087815260200190815260200160002060009055610a2b85600080898152602001908152602001600020610cd890919063ffffffff16565b610a6a576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610a6190611235565b60405180910390fd5b7f000000000000000000000000d5fbd81cef9aba7464c5f17e529444918a8ecc5773ffffffffffffffffffffffffffffffffffffffff16630603849082886040518363ffffffff1660e01b8152600401610ac49190611255565b6000604051808303818588803b158015610add57600080fd5b505af1158015610af1573d6000803e3d6000fd5b50505050507f000000000000000000000000d5fbd81cef9aba7464c5f17e529444918a8ecc5773ffffffffffffffffffffffffffffffffffffffff166327a664e987876040518363ffffffff1660e01b8152600401610b51929190611270565b600060405180830381600087803b158015610b6b57600080fd5b505af1158015610b7f573d6000803e3d6000fd5b505050503373ffffffffffffffffffffffffffffffffffffffff166108fc859081150290604051600060405180830381858888f19350505050158015610bc9573d6000803e3d6000fd5b507f000000000000000000000000360059bbd6df9ae032e93a8e5fa7900bbd10363a73ffffffffffffffffffffffffffffffffffffffff166108fc839081150290604051600060405180830381858888f19350505050158015610c30573d6000803e3d6000fd5b507ffd9c0b9a91ed824caef4b611bb4ee6570f6ee555940d834fb0c69708c37a2356868633604051610c6493929190611299565b60405180910390a1505050505050565b6000610c9a82600080868152602001908152602001600020610d3890919063ffffffff16565b905092915050565b60606000610cb283600001610d52565b905060608190508092505050919050565b6000610cd182600001610dae565b9050919050565b6000610d00836000018373ffffffffffffffffffffffffffffffffffffffff1660001b610dbf565b905092915050565b6000610d30836000018373ffffffffffffffffffffffffffffffffffffffff1660001b610ed3565b905092915050565b6000610d478360000183610f43565b60001c905092915050565b606081600001805480602002602001604051908101604052809291908181526020018280548015610da257602002820191906000526020600020905b815481526020019060010190808311610d8e575b50505050509050919050565b600081600001805490509050919050565b60008083600101600084815260200190815260200160002054905060008114610ec7576000600182610df19190611432565b9050600060018660000180549050610e099190611432565b9050818114610e78576000866000018281548110610e2a57610e2961152f565b5b9060005260206000200154905080876000018481548110610e4e57610e4d61152f565b5b90600052602060002001819055508387600101600083815260200190815260200160002081905550505b85600001805480610e8c57610e8b611500565b5b600190038181906000526020600020016000905590558560010160008681526020019081526020016000206000905560019350505050610ecd565b60009150505b92915050565b6000610edf8383610f6e565b610f38578260000182908060018154018082558091505060019003906000526020600020016000909190919091505582600001805490508360010160008481526020019081526020016000208190555060019050610f3d565b600090505b92915050565b6000826000018281548110610f5b57610f5a61152f565b5b9060005260206000200154905092915050565b600080836001016000848152602001908152602001600020541415905092915050565b600081359050610fa08161158c565b92915050565b600081359050610fb5816115a3565b92915050565b60008060408385031215610fd257610fd161155e565b5b6000610fe085828601610f91565b9250506020610ff185828601610fa6565b9150509250929050565b6000602082840312156110115761101061155e565b5b600061101f84828501610fa6565b91505092915050565b6000806040838503121561103f5761103e61155e565b5b600061104d85828601610fa6565b925050602061105e85828601610f91565b9150509250929050565b6000806000606084860312156110815761108061155e565b5b600061108f86828701610fa6565b93505060206110a086828701610f91565b92505060406110b186828701610fa6565b9150509250925092565b600080604083850312156110d2576110d161155e565b5b60006110e085828601610fa6565b92505060206110f185828601610fa6565b9150509250929050565b60006111078383611113565b60208301905092915050565b61111c81611466565b82525050565b61112b81611466565b82525050565b600061113c82611317565b611146818561132f565b935061115183611307565b8060005b8381101561118257815161116988826110fb565b975061117483611322565b925050600181019050611155565b5085935050505092915050565b600061119c600783611340565b91506111a782611563565b602082019050919050565b6111bb81611498565b82525050565b60006020820190506111d66000830184611122565b92915050565b60006060820190506111f16000830186611122565b6111fe6020830185611122565b61120b60408301846111b2565b949350505050565b6000602082019050818103600083015261122d8184611131565b905092915050565b6000602082019050818103600083015261124e8161118f565b9050919050565b600060208201905061126a60008301846111b2565b92915050565b600060408201905061128560008301856111b2565b6112926020830184611122565b9392505050565b60006060820190506112ae60008301866111b2565b6112bb6020830185611122565b6112c86040830184611122565b949350505050565b60006060820190506112e560008301866111b2565b6112f26020830185611122565b6112ff60408301846111b2565b949350505050565b6000819050602082019050919050565b600081519050919050565b6000602082019050919050565b600082825260208201905092915050565b600082825260208201905092915050565b600061135c82611498565b915061136783611498565b9250827fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff0382111561139c5761139b6114a2565b5b828201905092915050565b60006113b282611498565b91506113bd83611498565b9250826113cd576113cc6114d1565b5b828204905092915050565b60006113e382611498565b91506113ee83611498565b9250817fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff0483118215151615611427576114266114a2565b5b828202905092915050565b600061143d82611498565b915061144883611498565b92508282101561145b5761145a6114a2565b5b828203905092915050565b600061147182611478565b9050919050565b600073ffffffffffffffffffffffffffffffffffffffff82169050919050565b6000819050919050565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052601160045260246000fd5b7f4e487b7100000000000000000000000000000000000000000000000000000000600052601260045260246000fd5b7f4e487b7100000000000000000000000000000000000000000000000000000000600052603160045260246000fd5b7f4e487b7100000000000000000000000000000000000000000000000000000000600052603260045260246000fd5b600080fd5b7f2162696464657200000000000000000000000000000000000000000000000000600082015250565b61159581611466565b81146115a057600080fd5b50565b6115ac81611498565b81146115b757600080fd5b5056fea2646970667358221220574cbdb6f8626032bb8d2022a46d06ddf0c4c72fc2dfb21ed48cc11b7483ef9864736f6c63430008070033
Constructor Arguments (ABI-Encoded and is the last bytes of the Contract Creation Code above)
000000000000000000000000360059bbd6df9ae032e93a8e5fa7900bbd10363a00000000000000000000000000000000000000000000000000000000000001f4000000000000000000000000d5fbd81cef9aba7464c5f17e529444918a8ecc57
-----Decoded View---------------
Arg [0] : _feeRecipient (address): 0x360059bBD6Df9AE032e93A8E5Fa7900BBd10363A
Arg [1] : _feeBps (uint256): 500
Arg [2] : _etherTulip (address): 0xd5Fbd81Cef9aBA7464C5f17E529444918a8EcC57
-----Encoded View---------------
3 Constructor Arguments found :
Arg [0] : 000000000000000000000000360059bbd6df9ae032e93a8e5fa7900bbd10363a
Arg [1] : 00000000000000000000000000000000000000000000000000000000000001f4
Arg [2] : 000000000000000000000000d5fbd81cef9aba7464c5f17e529444918a8ecc57
Loading...
Loading
Loading...
Loading
Loading...
Loading
Loading...
Loading
Loading...
Loading
[ Download: CSV Export ]
[ Download: CSV Export ]
A contract address hosts a smart contract, which is a set of code stored on the blockchain that runs when predetermined conditions are met. Learn more about addresses in our Knowledge Base.