ETH Price: $1,912.12 (+0.04%)

Transaction Decoder

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 Code
0x0EffCec1...c21F18FEB
2.196682047751291759 Eth
Nonce: 175
2.170869821307152151 Eth
Nonce: 176
0.025812226444139608
(Spark Pool)
3,450.681071359242622538 Eth3,450.682497059242622538 Eth0.0014257
0xe43F953a...aDc3d9f6A 11.060352783420189822 Eth11.062303705535720991 Eth0.001950922115531169
0xFf37aE8E...767350D2d 0.232075285745230256 Eth0.254510890073838695 Eth0.022435604328608439

Execution Trace

ETH 1 EtherWaifus.purchase( _tokenId=17 )
  • ETH 0.022435604328608439 0xff37ae8ebd211f5574cee13b6b02653767350d2d.CALL( )
  • ETH 0.975613473555860392 0x0effcec157f88e562eedc76a5c9022ac21f18feb.CALL( )
    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;
      }
    }