ETH Price: $2,086.61 (-3.32%)

Transaction Decoder

Block:
6276564 at Sep-05-2018 01:44:06 PM +UTC
Transaction Fee:
0.0002989728 ETH $0.62
Gas Used:
23,728 Gas / 12.6 Gwei

Emitted Events:

34 UnionPay.BareUserPay( from=[Sender] 0x081591764a13e72aeb6bed098e7da34063ea8ac4, amount=0, data=0xEB3F2427 )

Account State Difference:

  Address   Before After State Difference Code
0x08159176...063eA8aC4
0.008739978819983497 Eth
Nonce: 796
0.008441006019983497 Eth
Nonce: 797
0.0002989728
(Nanopool)
7,764.684164343458511003 Eth7,764.684463316258511003 Eth0.0002989728

Execution Trace

UnionPay.CALL( )
pragma solidity ^0.4.23;

contract UnionPay {
    event UserPay(address from,address to,uint256 amount, uint256 amountIndeed,uint256 transId);
    event BareUserPay(address from,uint256 amount,bytes data);  
    
    address public owner;  
    address public platform;
    mapping(bytes32 => uint8)  userReceipts;

    constructor() public {
      owner = msg.sender;
      platform = msg.sender;
    }
  
    modifier onlyOwner() {
      require(msg.sender == owner);
      _;
    }
  
    function transferOwnership(address newOwner) public onlyOwner {
      if (newOwner != address(0)) {
        owner = newOwner;
      }
    }

    function safePay(uint256 _transId,uint256 _feePercentage,address _to, bytes _sig) payable public returns(bool) {
        require(_feePercentage>=0 && _feePercentage<=100);
        require(_to != address(0));
        require(userReceipts[getReceiptId(msg.sender,_to,_transId)] == 0);
        require(platform!=address(0));

        bytes32 message = prefixed(keccak256(msg.sender, _to, msg.value, _feePercentage,_transId));

        require(recoverSigner(message, _sig) == platform);
        userReceipts[getReceiptId(msg.sender,_to,_transId)] = 1;
        
        if (_feePercentage == 0){
            if (msg.value > 0){
                _to.transfer(msg.value);
            }
            emit UserPay(msg.sender,_to,msg.value,msg.value,_transId);
            return true;
        }        
        uint256 val = _feePercentage * msg.value;
        assert(val/_feePercentage == msg.value);
        val = val/100;
        if (msg.value>val){
            _to.transfer(msg.value - val);
        }
        emit UserPay(msg.sender,_to,msg.value,msg.value - val,_transId);
        return true;
    }
    
    function getReceiptId(address _from,address _to, uint256 _transId) internal pure returns(bytes32){
        return keccak256(_from, _to,_transId);
    }
    
    function receiptUsed(address _from,address _to,uint256 _transId) public view returns(bool){
        return userReceipts[getReceiptId(_from,_to,_transId)] == 1;
    }
    
    function plainPay() public payable returns(bool){
        emit BareUserPay(msg.sender,msg.value,msg.data);
        return true;
    }
    
    function () public payable{
        emit BareUserPay(msg.sender,msg.value,msg.data);
    }
    
    function setPlatform(address _checker) public onlyOwner{
        require(_checker!=address(0));
        platform = _checker;
    }
    
    function withdraw() public onlyOwner{
        require(platform!=address(0));
        platform.transfer(address(this).balance);
    }
    
    function getBalance() public view returns(uint256){
        return address(this).balance;
    }


    // Signature methods

    function splitSignature(bytes sig)
    internal
    pure
    returns(uint8, bytes32, bytes32) {
        require(sig.length == 65);

        bytes32 r;
        bytes32 s;
        uint8 v;

        assembly {
            // first 32 bytes, after the length prefix
            r: = mload(add(sig, 32))
            // second 32 bytes
            s: = mload(add(sig, 64))
            // final byte (first byte of the next 32 bytes)
            v: = byte(0, mload(add(sig, 96)))
        }

        return (v, r, s);
    }

    function recoverSigner(bytes32 message, bytes sig)
    internal
    pure
    returns(address) {
        uint8 v;
        bytes32 r;
        bytes32 s;

        (v, r, s) = splitSignature(sig);

        return ecrecover(message, v, r, s);
    }

    // Builds a prefixed hash to mimic the behavior of eth_sign.
    function prefixed(bytes32 hash) internal pure returns(bytes32) {
        return keccak256("\x19Ethereum Signed Message:\n32", hash);
    }
}