Transaction Hash:
Block:
5126634 at Feb-20-2018 09:43:40 PM +UTC
Transaction Fee:
0.0014257 ETH
$2.73
Gas Used:
71,285 Gas / 20 Gwei
Emitted Events:
61 |
EtherWaifus.Transfer( from=0xff37ae8ebd211f5574cee13b6b02653767350d2d, to=[Sender] 0x0effcec157f88e562eedc76a5c9022ac21f18feb, tokenId=17 )
|
62 |
EtherWaifus.TokenSold( tokenId=17, oldPrice=24386526444139608, newPrice=54192280986976906, prevOwner=0xff37ae8ebd211f5574cee13b6b02653767350d2d, winner=[Sender] 0x0effcec157f88e562eedc76a5c9022ac21f18feb, name=Rem )
|
Account State Difference:
Address | Before | After | State Difference | ||
---|---|---|---|---|---|
0x0EffCec1...c21F18FEB |
2.196682047751291759 Eth
Nonce: 175
|
2.170869821307152151 Eth
Nonce: 176
| 0.025812226444139608 | ||
0x5A0b54D5...D3E029c4c
Miner
| (Spark Pool) | 3,450.681071359242622538 Eth | 3,450.682497059242622538 Eth | 0.0014257 | |
0xe43F953a...aDc3d9f6A | 11.060352783420189822 Eth | 11.062303705535720991 Eth | 0.001950922115531169 | ||
0xFf37aE8E...767350D2d | 0.232075285745230256 Eth | 0.254510890073838695 Eth | 0.022435604328608439 |
Execution Trace
ETH 1
EtherWaifus.purchase( _tokenId=17 )

- ETH 0.022435604328608439
0xff37ae8ebd211f5574cee13b6b02653767350d2d.CALL( )
- ETH 0.975613473555860392
0x0effcec157f88e562eedc76a5c9022ac21f18feb.CALL( )
purchase[EtherWaifus (ln:176)]
_addressNotNull[EtherWaifus (ln:186)]
div[EtherWaifus (ln:191)]
mul[EtherWaifus (ln:191)]
sub[EtherWaifus (ln:192)]
div[EtherWaifus (ln:197)]
mul[EtherWaifus (ln:197)]
div[EtherWaifus (ln:200)]
mul[EtherWaifus (ln:200)]
div[EtherWaifus (ln:203)]
mul[EtherWaifus (ln:203)]
_transfer[EtherWaifus (ln:206)]
Transfer[EtherWaifus (ln:382)]
transfer[EtherWaifus (ln:210)]
TokenSold[EtherWaifus (ln:213)]
transfer[EtherWaifus (ln:215)]
pragma solidity ^0.4.18; // solhint-disable-line /// @title Interface for contracts conforming to ERC-721: Non-Fungible Tokens /// @author Dieter Shirley <[email protected]> (https://github.com/dete) contract ERC721 { // Required methods function approve(address _to, uint256 _tokenId) public; function balanceOf(address _owner) public view returns (uint256 balance); function implementsERC721() public pure returns (bool); function ownerOf(uint256 _tokenId) public view returns (address addr); function takeOwnership(uint256 _tokenId) public; function totalSupply() public view returns (uint256 total); function transferFrom(address _from, address _to, uint256 _tokenId) public; function transfer(address _to, uint256 _tokenId) public; event Transfer(address indexed from, address indexed to, uint256 tokenId); event Approval(address indexed owner, address indexed approved, uint256 tokenId); // Optional // function name() public view returns (string name); // function symbol() public view returns (string symbol); // function tokenOfOwnerByIndex(address _owner, uint256 _index) external view returns (uint256 tokenId); // function tokenMetadata(uint256 _tokenId) public view returns (string infoUrl); } contract EtherWaifus is ERC721 { /*** EVENTS ***/ /// @dev The Birth event is fired whenever a new waifu comes into existence. event Birth(uint256 tokenId, string name, address owner); /// @dev The TokenSold event is fired whenever a token is sold. event TokenSold(uint256 tokenId, uint256 oldPrice, uint256 newPrice, address prevOwner, address winner, string name); /// @dev Transfer event as defined in current draft of ERC721. /// ownership is assigned, including births. event Transfer(address from, address to, uint256 tokenId); /*** CONSTANTS ***/ /// @notice Name and symbol of the non fungible token, as defined in ERC721. string public constant NAME = "EtherWaifus"; // solhint-disable-line string public constant SYMBOL = "WaifuToken"; // solhint-disable-line uint256 private startingPrice = 0.001 ether; uint256 private firstStepLimit = 0.053613 ether; uint256 private secondStepLimit = 0.564957 ether; /*** STORAGE ***/ /// @dev A mapping from waifu IDs to the address that owns them. All waifus have /// some valid owner address. mapping (uint256 => address) public waifuIndexToOwner; // @dev A mapping from owner address to count of tokens that address owns. // Used internally inside balanceOf() to resolve ownership count. mapping (address => uint256) private ownershipTokenCount; /// @dev A mapping from WaifuIDs to an address that has been approved to call /// transferFrom(). Each Waifu can only have one approved address for transfer /// at any time. A zero value means no approval is outstanding. mapping (uint256 => address) public waifuIndexToApproved; // @dev A mapping from WaifuIDs to the price of the token. mapping (uint256 => uint256) private waifuIndexToPrice; // The addresses of the accounts (or contracts) that can execute actions within each roles. address public ceoAddress; address public cooAddress; /*** DATATYPES ***/ struct Waifu { string name; } Waifu[] private waifus; /*** ACCESS MODIFIERS ***/ /// @dev Access modifier for CEO-only functionality modifier onlyCEO() { require(msg.sender == ceoAddress); _; } /// @dev Access modifier for COO-only functionality modifier onlyCOO() { require(msg.sender == cooAddress); _; } /// Access modifier for contract owner only functionality modifier onlyCLevel() { require( msg.sender == ceoAddress || msg.sender == cooAddress ); _; } /*** CONSTRUCTOR ***/ function EtherWaifus() public { ceoAddress = msg.sender; cooAddress = msg.sender; } /*** PUBLIC FUNCTIONS ***/ /// @notice Grant another address the right to transfer token via takeOwnership() and transferFrom(). /// @param _to The address to be granted transfer approval. Pass address(0) to /// clear all approvals. /// @param _tokenId The ID of the Token that can be transferred if this call succeeds. /// @dev Required for ERC-721 compliance. function approve( address _to, uint256 _tokenId ) public { // Caller must own token. require(_owns(msg.sender, _tokenId)); waifuIndexToApproved[_tokenId] = _to; Approval(msg.sender, _to, _tokenId); } /// For querying balance of a particular account /// @param _owner The address for balance query /// @dev Required for ERC-721 compliance. function balanceOf(address _owner) public view returns (uint256 balance) { return ownershipTokenCount[_owner]; } /// @dev Creates a new Waifu with the given name. function createContractWaifu(string _name) public onlyCLevel { _createWaifu(_name, address(this), startingPrice); } /// @notice Returns all the relevant information about a specific waifu. /// @param _tokenId The tokenId of the waifu of interest. function getWaifu(uint256 _tokenId) public view returns ( string waifuName, uint256 sellingPrice, address owner ) { Waifu storage waifu = waifus[_tokenId]; waifuName = waifu.name; sellingPrice = waifuIndexToPrice[_tokenId]; owner = waifuIndexToOwner[_tokenId]; } function implementsERC721() public pure returns (bool) { return true; } /// @dev Required for ERC-721 compliance. function name() public pure returns (string) { return NAME; } /// For querying owner of token /// @param _tokenId The tokenID for owner inquiry /// @dev Required for ERC-721 compliance. function ownerOf(uint256 _tokenId) public view returns (address owner) { owner = waifuIndexToOwner[_tokenId]; require(owner != address(0)); } function payout(address _to) public onlyCLevel { _payout(_to); } // Allows someone to send ether and obtain the token function purchase(uint256 _tokenId) public payable { address oldOwner = waifuIndexToOwner[_tokenId]; address newOwner = msg.sender; uint256 sellingPrice = waifuIndexToPrice[_tokenId]; // Making sure token owner is not sending to self require(oldOwner != newOwner); // Safety check to prevent against an unexpected 0x0 default. require(_addressNotNull(newOwner)); // Making sure sent amount is greater than or equal to the sellingPrice require(msg.value >= sellingPrice); uint256 payment = uint256(SafeMath.div(SafeMath.mul(sellingPrice, 92), 100)); uint256 purchaseExcess = SafeMath.sub(msg.value, sellingPrice); // Update prices if (sellingPrice < firstStepLimit) { // first stage waifuIndexToPrice[_tokenId] = SafeMath.div(SafeMath.mul(sellingPrice, 200), 90); } else if (sellingPrice < secondStepLimit) { // second stage waifuIndexToPrice[_tokenId] = SafeMath.div(SafeMath.mul(sellingPrice, 120), 90); } else { // third stage waifuIndexToPrice[_tokenId] = SafeMath.div(SafeMath.mul(sellingPrice, 115), 90); } _transfer(oldOwner, newOwner, _tokenId); // Pay previous tokenOwner if owner is not contract if (oldOwner != address(this)) { oldOwner.transfer(payment); //(1-0.08) } TokenSold(_tokenId, sellingPrice, waifuIndexToPrice[_tokenId], oldOwner, newOwner, waifus[_tokenId].name); msg.sender.transfer(purchaseExcess); } function priceOf(uint256 _tokenId) public view returns (uint256 price) { return waifuIndexToPrice[_tokenId]; } /// @dev Assigns a new address to act as the CEO. Only available to the current CEO. /// @param _newCEO The address of the new CEO function setCEO(address _newCEO) public onlyCEO { require(_newCEO != address(0)); ceoAddress = _newCEO; } /// @dev Assigns a new address to act as the COO. Only available to the current COO. /// @param _newCOO The address of the new COO function setCOO(address _newCOO) public onlyCEO { require(_newCOO != address(0)); cooAddress = _newCOO; } /// @dev Required for ERC-721 compliance. function symbol() public pure returns (string) { return SYMBOL; } /// @notice Allow pre-approved user to take ownership of a token /// @param _tokenId The ID of the Token that can be transferred if this call succeeds. /// @dev Required for ERC-721 compliance. function takeOwnership(uint256 _tokenId) public { address newOwner = msg.sender; address oldOwner = waifuIndexToOwner[_tokenId]; // Safety check to prevent against an unexpected 0x0 default. require(_addressNotNull(newOwner)); // Making sure transfer is approved require(_approved(newOwner, _tokenId)); _transfer(oldOwner, newOwner, _tokenId); } /// @param _owner The owner whose celebrity tokens we are interested in. /// @dev This method MUST NEVER be called by smart contract code. First, it's fairly /// expensive (it walks the entire Waifus array looking for waifus belonging to owner), /// but it also returns a dynamic array, which is only supported for web3 calls, and /// not contract-to-contract calls. function tokensOfOwner(address _owner) public view returns(uint256[] ownerTokens) { uint256 tokenCount = balanceOf(_owner); if (tokenCount == 0) { // Return an empty array return new uint256[](0); } else { uint256[] memory result = new uint256[](tokenCount); uint256 totalWaifus = totalSupply(); uint256 resultIndex = 0; uint256 waifuId; for (waifuId = 0; waifuId <= totalWaifus; waifuId++) { if (waifuIndexToOwner[waifuId] == _owner) { result[resultIndex] = waifuId; resultIndex++; } } return result; } } /// For querying totalSupply of token /// @dev Required for ERC-721 compliance. function totalSupply() public view returns (uint256 total) { return waifus.length; } /// Owner initates the transfer of the token to another account /// @param _to The address for the token to be transferred to. /// @param _tokenId The ID of the Token that can be transferred if this call succeeds. /// @dev Required for ERC-721 compliance. function transfer( address _to, uint256 _tokenId ) public { require(_owns(msg.sender, _tokenId)); require(_addressNotNull(_to)); _transfer(msg.sender, _to, _tokenId); } /// Third-party initiates transfer of token from address _from to address _to /// @param _from The address for the token to be transferred from. /// @param _to The address for the token to be transferred to. /// @param _tokenId The ID of the Token that can be transferred if this call succeeds. /// @dev Required for ERC-721 compliance. function transferFrom( address _from, address _to, uint256 _tokenId ) public { require(_owns(_from, _tokenId)); require(_approved(_to, _tokenId)); require(_addressNotNull(_to)); _transfer(_from, _to, _tokenId); } /*** PRIVATE FUNCTIONS ***/ /// Safety check on _to address to prevent against an unexpected 0x0 default. function _addressNotNull(address _to) private pure returns (bool) { return _to != address(0); } /// For checking approval of transfer for address _to function _approved(address _to, uint256 _tokenId) private view returns (bool) { return waifuIndexToApproved[_tokenId] == _to; } /// For creating Waifus function _createWaifu(string _name, address _owner, uint256 _price) private { Waifu memory _waifu = Waifu({ name: _name }); uint256 newWaifuId = waifus.push(_waifu) - 1; // It's probably never going to happen, 4 billion tokens are A LOT, but // let's just be 100% sure we never let this happen. require(newWaifuId == uint256(uint32(newWaifuId))); Birth(newWaifuId, _name, _owner); waifuIndexToPrice[newWaifuId] = _price; // This will assign ownership, and also emit the Transfer event as // per ERC721 draft _transfer(address(0), _owner, newWaifuId); } /// Check for token ownership function _owns(address claimant, uint256 _tokenId) private view returns (bool) { return claimant == waifuIndexToOwner[_tokenId]; } /// For paying out balance on contract function _payout(address _to) private { if (_to == address(0)) { ceoAddress.transfer(this.balance); } else { _to.transfer(this.balance); } } /// @dev Assigns ownership of a specific Waifu to an address. function _transfer(address _from, address _to, uint256 _tokenId) private { // Since the number of waifus is capped to 2^32 we can't overflow this ownershipTokenCount[_to]++; //transfer ownership waifuIndexToOwner[_tokenId] = _to; // When creating new waifus _from is 0x0, but we can't account that address. if (_from != address(0)) { ownershipTokenCount[_from]--; // clear any previously approved ownership exchange delete waifuIndexToApproved[_tokenId]; } // Emit the transfer event. Transfer(_from, _to, _tokenId); } } library SafeMath { /** * @dev Multiplies two numbers, throws on overflow. */ function mul(uint256 a, uint256 b) internal pure returns (uint256) { if (a == 0) { return 0; } uint256 c = a * b; assert(c / a == b); return c; } /** * @dev Integer division of two numbers, truncating the quotient. */ function div(uint256 a, uint256 b) internal pure returns (uint256) { // assert(b > 0); // Solidity automatically throws when dividing by 0 uint256 c = a / b; // assert(a == b * c + a % b); // There is no case in which this doesn't hold return c; } /** * @dev Substracts two numbers, throws on overflow (i.e. if subtrahend is greater than minuend). */ function sub(uint256 a, uint256 b) internal pure returns (uint256) { assert(b <= a); return a - b; } /** * @dev Adds two numbers, throws on overflow. */ function add(uint256 a, uint256 b) internal pure returns (uint256) { uint256 c = a + b; assert(c >= a); return c; } }