ETH Price: $3,392.96 (-1.26%)
Gas: 2 Gwei

Contract

0x0f614fbD7e17edfaC5cA3C59e72a1EC3c5b71624
 

Overview

ETH Balance

0 ETH

Eth Value

$0.00

Multichain Info

No addresses found
Transaction Hash
Method
Block
From
To
Value
0x6080604076463012019-04-27 0:35:481890 days ago1556325348IN
 Create: TrueCAD
0 ETH0.003510481

View more zero value Internal Transactions in Advanced View mode

Advanced mode:
Loading...
Loading

Contract Source Code Verified (Exact Match)

Contract Name:
TrueCAD

Compiler Version
v0.4.23+commit.124ca40d

Optimization Enabled:
Yes with 20000 runs

Other Settings:
default evmVersion

Contract Source Code (Solidity)

/**
 *Submitted for verification at Etherscan.io on 2019-04-27
*/

pragma solidity ^0.4.23;

// File: contracts/TrueCoinReceiver.sol

contract TrueCoinReceiver {
    function tokenFallback( address from, uint256 value ) external;
}

// File: openzeppelin-solidity/contracts/token/ERC20/ERC20Basic.sol

/**
 * @title ERC20Basic
 * @dev Simpler version of ERC20 interface
 * @dev see https://github.com/ethereum/EIPs/issues/179
 */
contract ERC20Basic {
  function totalSupply() public view returns (uint256);
  function balanceOf(address who) public view returns (uint256);
  function transfer(address to, uint256 value) public returns (bool);
  event Transfer(address indexed from, address indexed to, uint256 value);
}

// File: openzeppelin-solidity/contracts/token/ERC20/ERC20.sol

/**
 * @title ERC20 interface
 * @dev see https://github.com/ethereum/EIPs/issues/20
 */
contract ERC20 is ERC20Basic {
  function allowance(address owner, address spender) public view returns (uint256);
  function transferFrom(address from, address to, uint256 value) public returns (bool);
  function approve(address spender, uint256 value) public returns (bool);
  event Approval(address indexed owner, address indexed spender, uint256 value);
}

// File: registry/contracts/Registry.sol

interface RegistryClone {
    function syncAttributeValue(address _who, bytes32 _attribute, uint256 _value) external;
}

contract Registry {
    struct AttributeData {
        uint256 value;
        bytes32 notes;
        address adminAddr;
        uint256 timestamp;
    }
    
    // never remove any storage variables
    address public owner;
    address public pendingOwner;
    bool initialized;

    // Stores arbitrary attributes for users. An example use case is an ERC20
    // token that requires its users to go through a KYC/AML check - in this case
    // a validator can set an account's "hasPassedKYC/AML" attribute to 1 to indicate
    // that account can use the token. This mapping stores that value (1, in the
    // example) as well as which validator last set the value and at what time,
    // so that e.g. the check can be renewed at appropriate intervals.
    mapping(address => mapping(bytes32 => AttributeData)) attributes;
    // The logic governing who is allowed to set what attributes is abstracted as
    // this accessManager, so that it may be replaced by the owner as needed
    bytes32 constant WRITE_PERMISSION = keccak256("canWriteTo-");
    mapping(bytes32 => RegistryClone[]) subscribers;

    event OwnershipTransferred(
        address indexed previousOwner,
        address indexed newOwner
    );
    event SetAttribute(address indexed who, bytes32 attribute, uint256 value, bytes32 notes, address indexed adminAddr);
    event SetManager(address indexed oldManager, address indexed newManager);
    event StartSubscription(bytes32 indexed attribute, RegistryClone indexed subscriber);
    event StopSubscription(bytes32 indexed attribute, RegistryClone indexed subscriber);

    // Allows a write if either a) the writer is that Registry's owner, or
    // b) the writer is writing to attribute foo and that writer already has
    // the canWriteTo-foo attribute set (in that same Registry)
    function confirmWrite(bytes32 _attribute, address _admin) internal view returns (bool) {
        return (_admin == owner || hasAttribute(_admin, keccak256(WRITE_PERMISSION ^ _attribute)));
    }

    // Writes are allowed only if the accessManager approves
    function setAttribute(address _who, bytes32 _attribute, uint256 _value, bytes32 _notes) public {
        require(confirmWrite(_attribute, msg.sender));
        attributes[_who][_attribute] = AttributeData(_value, _notes, msg.sender, block.timestamp);
        emit SetAttribute(_who, _attribute, _value, _notes, msg.sender);

        RegistryClone[] storage targets = subscribers[_attribute];
        uint256 index = targets.length;
        while (index --> 0) {
            targets[index].syncAttributeValue(_who, _attribute, _value);
        }
    }

    function subscribe(bytes32 _attribute, RegistryClone _syncer) external onlyOwner {
        subscribers[_attribute].push(_syncer);
        emit StartSubscription(_attribute, _syncer);
    }

    function unsubscribe(bytes32 _attribute, uint256 _index) external onlyOwner {
        uint256 length = subscribers[_attribute].length;
        require(_index < length);
        emit StopSubscription(_attribute, subscribers[_attribute][_index]);
        subscribers[_attribute][_index] = subscribers[_attribute][length - 1];
        subscribers[_attribute].length = length - 1;
    }

    function subscriberCount(bytes32 _attribute) public view returns (uint256) {
        return subscribers[_attribute].length;
    }

    function setAttributeValue(address _who, bytes32 _attribute, uint256 _value) public {
        require(confirmWrite(_attribute, msg.sender));
        attributes[_who][_attribute] = AttributeData(_value, "", msg.sender, block.timestamp);
        emit SetAttribute(_who, _attribute, _value, "", msg.sender);
        RegistryClone[] storage targets = subscribers[_attribute];
        uint256 index = targets.length;
        while (index --> 0) {
            targets[index].syncAttributeValue(_who, _attribute, _value);
        }
    }

    // Returns true if the uint256 value stored for this attribute is non-zero
    function hasAttribute(address _who, bytes32 _attribute) public view returns (bool) {
        return attributes[_who][_attribute].value != 0;
    }


    // Returns the exact value of the attribute, as well as its metadata
    function getAttribute(address _who, bytes32 _attribute) public view returns (uint256, bytes32, address, uint256) {
        AttributeData memory data = attributes[_who][_attribute];
        return (data.value, data.notes, data.adminAddr, data.timestamp);
    }

    function getAttributeValue(address _who, bytes32 _attribute) public view returns (uint256) {
        return attributes[_who][_attribute].value;
    }

    function getAttributeAdminAddr(address _who, bytes32 _attribute) public view returns (address) {
        return attributes[_who][_attribute].adminAddr;
    }

    function getAttributeTimestamp(address _who, bytes32 _attribute) public view returns (uint256) {
        return attributes[_who][_attribute].timestamp;
    }

    function syncAttribute(bytes32 _attribute, uint256 _startIndex, address[] _addresses) external {
        RegistryClone[] storage targets = subscribers[_attribute];
        uint256 index = targets.length;
        while (index --> _startIndex) {
            RegistryClone target = targets[index];
            for (uint256 i = _addresses.length; i --> 0; ) {
                address who = _addresses[i];
                target.syncAttributeValue(who, _attribute, attributes[who][_attribute].value);
            }
        }
    }

    function reclaimEther(address _to) external onlyOwner {
        _to.transfer(address(this).balance);
    }

    function reclaimToken(ERC20 token, address _to) external onlyOwner {
        uint256 balance = token.balanceOf(this);
        token.transfer(_to, balance);
    }

   /**
    * @dev Throws if called by any account other than the owner.
    */
    modifier onlyOwner() {
        require(msg.sender == owner, "only Owner");
        _;
    }

    /**
    * @dev Modifier throws if called by any account other than the pendingOwner.
    */
    modifier onlyPendingOwner() {
        require(msg.sender == pendingOwner);
        _;
    }

    /**
    * @dev Allows the current owner to set the pendingOwner address.
    * @param newOwner The address to transfer ownership to.
    */
    function transferOwnership(address newOwner) public onlyOwner {
        pendingOwner = newOwner;
    }

    /**
    * @dev Allows the pendingOwner address to finalize the transfer.
    */
    function claimOwnership() public onlyPendingOwner {
        emit OwnershipTransferred(owner, pendingOwner);
        owner = pendingOwner;
        pendingOwner = address(0);
    }
}

// File: openzeppelin-solidity/contracts/ownership/Ownable.sol

/**
 * @title Ownable
 * @dev The Ownable contract has an owner address, and provides basic authorization control
 * functions, this simplifies the implementation of "user permissions".
 */
contract Ownable {
  address public owner;


  event OwnershipTransferred(address indexed previousOwner, address indexed newOwner);


  /**
   * @dev The Ownable constructor sets the original `owner` of the contract to the sender
   * account.
   */
  function Ownable() public {
    owner = msg.sender;
  }

  /**
   * @dev Throws if called by any account other than the owner.
   */
  modifier onlyOwner() {
    require(msg.sender == owner);
    _;
  }

  /**
   * @dev Allows the current owner to transfer control of the contract to a newOwner.
   * @param newOwner The address to transfer ownership to.
   */
  function transferOwnership(address newOwner) public onlyOwner {
    require(newOwner != address(0));
    emit OwnershipTransferred(owner, newOwner);
    owner = newOwner;
  }

}

// File: openzeppelin-solidity/contracts/ownership/Claimable.sol

/**
 * @title Claimable
 * @dev Extension for the Ownable contract, where the ownership needs to be claimed.
 * This allows the new owner to accept the transfer.
 */
contract Claimable is Ownable {
  address public pendingOwner;

  /**
   * @dev Modifier throws if called by any account other than the pendingOwner.
   */
  modifier onlyPendingOwner() {
    require(msg.sender == pendingOwner);
    _;
  }

  /**
   * @dev Allows the current owner to set the pendingOwner address.
   * @param newOwner The address to transfer ownership to.
   */
  function transferOwnership(address newOwner) onlyOwner public {
    pendingOwner = newOwner;
  }

  /**
   * @dev Allows the pendingOwner address to finalize the transfer.
   */
  function claimOwnership() onlyPendingOwner public {
    emit OwnershipTransferred(owner, pendingOwner);
    owner = pendingOwner;
    pendingOwner = address(0);
  }
}

// File: openzeppelin-solidity/contracts/math/SafeMath.sol

/**
 * @title SafeMath
 * @dev Math operations with safety checks that throw on error
 */
library SafeMath {

  /**
  * @dev Multiplies two numbers, throws on overflow.
  */
  function mul(uint256 a, uint256 b) internal pure returns (uint256 c) {
    if (a == 0) {
      return 0;
    }
    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 a / b;
  }

  /**
  * @dev Subtracts 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 c) {
    c = a + b;
    assert(c >= a);
    return c;
  }
}

// File: contracts/modularERC20/BalanceSheet.sol

// A wrapper around the balanceOf mapping.
contract BalanceSheet is Claimable {
    using SafeMath for uint256;

    mapping (address => uint256) public balanceOf;

    function addBalance(address _addr, uint256 _value) public onlyOwner {
        balanceOf[_addr] = balanceOf[_addr].add(_value);
    }

    function subBalance(address _addr, uint256 _value) public onlyOwner {
        balanceOf[_addr] = balanceOf[_addr].sub(_value);
    }

    function setBalance(address _addr, uint256 _value) public onlyOwner {
        balanceOf[_addr] = _value;
    }
}

// File: contracts/modularERC20/AllowanceSheet.sol

// A wrapper around the allowanceOf mapping.
contract AllowanceSheet is Claimable {
    using SafeMath for uint256;

    mapping (address => mapping (address => uint256)) public allowanceOf;

    function addAllowance(address _tokenHolder, address _spender, uint256 _value) public onlyOwner {
        allowanceOf[_tokenHolder][_spender] = allowanceOf[_tokenHolder][_spender].add(_value);
    }

    function subAllowance(address _tokenHolder, address _spender, uint256 _value) public onlyOwner {
        allowanceOf[_tokenHolder][_spender] = allowanceOf[_tokenHolder][_spender].sub(_value);
    }

    function setAllowance(address _tokenHolder, address _spender, uint256 _value) public onlyOwner {
        allowanceOf[_tokenHolder][_spender] = _value;
    }
}

// File: contracts/ProxyStorage.sol

/*
Defines the storage layout of the token implementaiton contract. Any newly declared
state variables in future upgrades should be appened to the bottom. Never remove state variables
from this list
 */
contract ProxyStorage {
    address public owner;
    address public pendingOwner;

    bool initialized;
    
    BalanceSheet balances_Deprecated;
    AllowanceSheet allowances_Deprecated;

    uint256 totalSupply_;
    
    bool private paused_Deprecated = false;
    address private globalPause_Deprecated;

    uint256 public burnMin = 0;
    uint256 public burnMax = 0;

    Registry public registry;

    string name_Deprecated;
    string symbol_Deprecated;

    uint[] gasRefundPool_Deprecated;
    uint256 private redemptionAddressCount_Deprecated;
    uint256 public minimumGasPriceForFutureRefunds;

    mapping (address => uint256) _balanceOf;
    mapping (address => mapping (address => uint256)) _allowance;
    mapping (bytes32 => mapping (address => uint256)) attributes;


    /* Additionally, we have several keccak-based storage locations.
     * If you add more keccak-based storage mappings, such as mappings, you must document them here.
     * If the length of the keccak input is the same as an existing mapping, it is possible there could be a preimage collision.
     * A preimage collision can be used to attack the contract by treating one storage location as another,
     * which would always be a critical issue.
     * Carefully examine future keccak-based storage to ensure there can be no preimage collisions.
     *******************************************************************************************************
     ** length     input                                                         usage
     *******************************************************************************************************
     ** 19         "trueXXX.proxy.owner"                                         Proxy Owner
     ** 27         "trueXXX.pending.proxy.owner"                                 Pending Proxy Owner
     ** 28         "trueXXX.proxy.implementation"                                Proxy Implementation
     ** 32         uint256(11)                                                   gasRefundPool_Deprecated
     ** 64         uint256(address),uint256(14)                                  balanceOf
     ** 64         uint256(address),keccak256(uint256(address),uint256(15))      allowance
     ** 64         uint256(address),keccak256(bytes32,uint256(16))               attributes
    **/
}

// File: contracts/HasOwner.sol

/**
 * @title HasOwner
 * @dev The HasOwner contract is a copy of Claimable Contract by Zeppelin. 
 and provides basic authorization control functions. Inherits storage layout of 
 ProxyStorage.
 */
contract HasOwner is ProxyStorage {

    event OwnershipTransferred(
        address indexed previousOwner,
        address indexed newOwner
    );

    /**
    * @dev sets the original `owner` of the contract to the sender
    * at construction. Must then be reinitialized 
    */
    constructor() public {
        owner = msg.sender;
        emit OwnershipTransferred(address(0), owner);
    }

    /**
    * @dev Throws if called by any account other than the owner.
    */
    modifier onlyOwner() {
        require(msg.sender == owner, "only Owner");
        _;
    }

    /**
    * @dev Modifier throws if called by any account other than the pendingOwner.
    */
    modifier onlyPendingOwner() {
        require(msg.sender == pendingOwner);
        _;
    }

    /**
    * @dev Allows the current owner to set the pendingOwner address.
    * @param newOwner The address to transfer ownership to.
    */
    function transferOwnership(address newOwner) public onlyOwner {
        pendingOwner = newOwner;
    }

    /**
    * @dev Allows the pendingOwner address to finalize the transfer.
    */
    function claimOwnership() public onlyPendingOwner {
        emit OwnershipTransferred(owner, pendingOwner);
        owner = pendingOwner;
        pendingOwner = address(0);
    }
}

// File: contracts/ReclaimerToken.sol

contract ReclaimerToken is HasOwner {
    /**  
    *@dev send all eth balance in the contract to another address
    */
    function reclaimEther(address _to) external onlyOwner {
        _to.transfer(address(this).balance);
    }

    /**  
    *@dev send all token balance of an arbitary erc20 token
    in the contract to another address
    */
    function reclaimToken(ERC20 token, address _to) external onlyOwner {
        uint256 balance = token.balanceOf(this);
        token.transfer(_to, balance);
    }

    /**  
    *@dev allows owner of the contract to gain ownership of any contract that the contract currently owns
    */
    function reclaimContract(Ownable _ownable) external onlyOwner {
        _ownable.transferOwnership(owner);
    }

}

// File: contracts/modularERC20/ModularBasicToken.sol

// Fork of OpenZeppelin's BasicToken
/**
 * @title Basic token
 * @dev Basic version of StandardToken, with no allowances.
 */
contract ModularBasicToken is HasOwner {
    using SafeMath for uint256;

    event Transfer(address indexed from, address indexed to, uint256 value);

    /**
    * @dev total number of tokens in existence
    */
    function totalSupply() public view returns (uint256) {
        return totalSupply_;
    }

    function balanceOf(address _who) public view returns (uint256) {
        return _getBalance(_who);
    }

    function _getBalance(address _who) internal view returns (uint256) {
        return _balanceOf[_who];
    }

    function _addBalance(address _who, uint256 _value) internal returns (uint256 priorBalance) {
        priorBalance = _balanceOf[_who];
        _balanceOf[_who] = priorBalance.add(_value);
    }

    function _subBalance(address _who, uint256 _value) internal returns (uint256 result) {
        result = _balanceOf[_who].sub(_value);
        _balanceOf[_who] = result;
    }

    function _setBalance(address _who, uint256 _value) internal {
        _balanceOf[_who] = _value;
    }
}

// File: contracts/modularERC20/ModularStandardToken.sol

/**
 * @title Standard ERC20 token
 *
 * @dev Implementation of the basic standard token.
 * @dev https://github.com/ethereum/EIPs/issues/20
 * @dev Based on code by FirstBlood: https://github.com/Firstbloodio/token/blob/master/smart_contract/FirstBloodToken.sol
 */
contract ModularStandardToken is ModularBasicToken {
    using SafeMath for uint256;
    
    event Approval(address indexed owner, address indexed spender, uint256 value);
    
    /**
     * @dev Approve the passed address to spend the specified amount of tokens on behalf of msg.sender.
     *
     * Beware that changing an allowance with this method brings the risk that someone may use both the old
     * and the new allowance by unfortunate transaction ordering. One possible solution to mitigate this
     * race condition is to first reduce the spender's allowance to 0 and set the desired value afterwards:
     * https://github.com/ethereum/EIPs/issues/20#issuecomment-263524729
     * @param _spender The address which will spend the funds.
     * @param _value The amount of tokens to be spent.
     */
    function approve(address _spender, uint256 _value) public returns (bool) {
        _approveAllArgs(_spender, _value, msg.sender);
        return true;
    }

    function _approveAllArgs(address _spender, uint256 _value, address _tokenHolder) internal {
        _setAllowance(_tokenHolder, _spender, _value);
        emit Approval(_tokenHolder, _spender, _value);
    }

    /**
     * @dev Increase the amount of tokens that an owner allowed to a spender.
     *
     * approve should be called when allowed[_spender] == 0. To increment
     * allowed value is better to use this function to avoid 2 calls (and wait until
     * the first transaction is mined)
     * From MonolithDAO Token.sol
     * @param _spender The address which will spend the funds.
     * @param _addedValue The amount of tokens to increase the allowance by.
     */
    function increaseApproval(address _spender, uint _addedValue) public returns (bool) {
        _increaseApprovalAllArgs(_spender, _addedValue, msg.sender);
        return true;
    }

    function _increaseApprovalAllArgs(address _spender, uint256 _addedValue, address _tokenHolder) internal {
        _addAllowance(_tokenHolder, _spender, _addedValue);
        emit Approval(_tokenHolder, _spender, _getAllowance(_tokenHolder, _spender));
    }

    /**
     * @dev Decrease the amount of tokens that an owner allowed to a spender.
     *
     * approve should be called when allowed[_spender] == 0. To decrement
     * allowed value is better to use this function to avoid 2 calls (and wait until
     * the first transaction is mined)
     * From MonolithDAO Token.sol
     * @param _spender The address which will spend the funds.
     * @param _subtractedValue The amount of tokens to decrease the allowance by.
     */
    function decreaseApproval(address _spender, uint _subtractedValue) public returns (bool) {
        _decreaseApprovalAllArgs(_spender, _subtractedValue, msg.sender);
        return true;
    }

    function _decreaseApprovalAllArgs(address _spender, uint256 _subtractedValue, address _tokenHolder) internal {
        uint256 oldValue = _getAllowance(_tokenHolder, _spender);
        uint256 newValue;
        if (_subtractedValue > oldValue) {
            newValue = 0;
        } else {
            newValue = oldValue - _subtractedValue;
        }
        _setAllowance(_tokenHolder, _spender, newValue);
        emit Approval(_tokenHolder,_spender, newValue);
    }

    function allowance(address _who, address _spender) public view returns (uint256) {
        return _getAllowance(_who, _spender);
    }

    function _getAllowance(address _who, address _spender) internal view returns (uint256 value) {
        return _allowance[_who][_spender];
    }

    function _addAllowance(address _who, address _spender, uint256 _value) internal {
        _allowance[_who][_spender] = _allowance[_who][_spender].add(_value);
    }

    function _subAllowance(address _who, address _spender, uint256 _value) internal returns (uint256 newAllowance){
        newAllowance = _allowance[_who][_spender].sub(_value);
        _allowance[_who][_spender] = newAllowance;
    }

    function _setAllowance(address _who, address _spender, uint256 _value) internal {
        _allowance[_who][_spender] = _value;
    }
}

// File: contracts/modularERC20/ModularBurnableToken.sol

/**
 * @title Burnable Token
 * @dev Token that can be irreversibly burned (destroyed).
 */
contract ModularBurnableToken is ModularStandardToken {
    event Burn(address indexed burner, uint256 value);
    event Mint(address indexed to, uint256 value);
    uint256 constant CENT = 10 ** 16;

    function burn(uint256 _value) external {
        _burnAllArgs(msg.sender, _value - _value % CENT);
    }

    function _burnAllArgs(address _from, uint256 _value) internal {
        // no need to require value <= totalSupply, since that would imply the
        // sender's balance is greater than the totalSupply, which *should* be an assertion failure
        _subBalance(_from, _value);
        totalSupply_ = totalSupply_.sub(_value);
        emit Burn(_from, _value);
        emit Transfer(_from, address(0), _value);
    }
}

// File: contracts/BurnableTokenWithBounds.sol

/**
 * @title Burnable Token WithBounds
 * @dev Burning functions as redeeming money from the system. The platform will keep track of who burns coins,
 * and will send them back the equivalent amount of money (rounded down to the nearest cent).
 */
contract BurnableTokenWithBounds is ModularBurnableToken {

    event SetBurnBounds(uint256 newMin, uint256 newMax);

    function _burnAllArgs(address _burner, uint256 _value) internal {
        require(_value >= burnMin, "below min burn bound");
        require(_value <= burnMax, "exceeds max burn bound");
        super._burnAllArgs(_burner, _value);
    }

    //Change the minimum and maximum amount that can be burned at once. Burning
    //may be disabled by setting both to 0 (this will not be done under normal
    //operation, but we can't add checks to disallow it without losing a lot of
    //flexibility since burning could also be as good as disabled
    //by setting the minimum extremely high, and we don't want to lock
    //in any particular cap for the minimum)
    function setBurnBounds(uint256 _min, uint256 _max) external onlyOwner {
        require(_min <= _max, "min > max");
        burnMin = _min;
        burnMax = _max;
        emit SetBurnBounds(_min, _max);
    }
}

// File: contracts/GasRefundToken.sol

/**  
@title Gas Refund Token
Allow any user to sponsor gas refunds for transfer and mints. Utilitzes the gas refund mechanism in EVM
Each time an non-empty storage slot is set to 0, evm refund 15,000 to the sender
of the transaction.
*/
contract GasRefundToken is ProxyStorage {

    /**
      A buffer of "Sheep" runs from 0xffff...fffe down
      They suicide when you call them, if you are their parent
    */

    function sponsorGas2() external {
        /**
        Deploy (9 bytes)
          PC Assembly       Opcodes                                       Stack
          00 PUSH1(31)      60 1f                                         1f
          02 DUP1           80                                            1f 1f
          03 PUSH1(9)       60 09                                         1f 1f 09
          05 RETURNDATASIZE 3d                                            1f 1f 09 00
          06 CODECOPY       39                                            1f
          07 RETURNDATASIZE 3d                                            1f 00
          08 RETURN         f3
        Sheep (31 bytes = 3 + 20 + 8)
          PC Assembly       Opcodes                                       Stack
          00 RETURNDATASIZE 3d                                            0
          01 CALLER         33                                            0 caller
          02 PUSH20(me)     73 memememememememememememememememememememe   0 caller me
          17 EQ             14                                            0 valid
          18 PUSH1(1d)      60 1d                                         0 valid 1d
          1a JUMPI          57                                            0
          1b DUP1           80                                            0 0
          1c REVERT         fd
          1d JUMPDEST       5b                                            0
          1e SELFDESTRUCT   ff
        */
        assembly {
            mstore(0, or(0x601f8060093d393df33d33730000000000000000000000000000000000000000, address))
            mstore(32,   0x14601d5780fd5bff000000000000000000000000000000000000000000000000)
            let sheep := create(0, 0, 0x28)
            let offset := sload(0xffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff)
            let location := sub(0xfffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe, offset)
            sstore(location, sheep)
            sheep := create(0, 0, 0x28)
            sstore(sub(location, 1), sheep)
            sheep := create(0, 0, 0x28)
            sstore(sub(location, 2), sheep)
            sstore(0xffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff, add(offset, 3))
        }
    }

    /**
    @dev refund 39,000 gas
    @dev costs slightly more than 16,100 gas
    */
    function gasRefund39() internal {
        assembly {
            let offset := sload(0xffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff)
            if gt(offset, 0) {
              let location := sub(0xffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff,offset)
              sstore(0xffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff, sub(offset, 1))
              let sheep := sload(location)
              pop(call(gas, sheep, 0, 0, 0, 0, 0))
              sstore(location, 0)
            }
        }
    }

    function sponsorGas() external {
        uint256 refundPrice = minimumGasPriceForFutureRefunds;
        require(refundPrice > 0);
        assembly {
            let offset := sload(0xfffff)
            let result := add(offset, 9)
            sstore(0xfffff, result)
            let position := add(offset, 0x100000)
            sstore(position, refundPrice)
            position := add(position, 1)
            sstore(position, refundPrice)
            position := add(position, 1)
            sstore(position, refundPrice)
            position := add(position, 1)
            sstore(position, refundPrice)
            position := add(position, 1)
            sstore(position, refundPrice)
            position := add(position, 1)
            sstore(position, refundPrice)
            position := add(position, 1)
            sstore(position, refundPrice)
            position := add(position, 1)
            sstore(position, refundPrice)
            position := add(position, 1)
            sstore(position, refundPrice)
        }
    }

    function minimumGasPriceForRefund() public view returns (uint256 result) {
        assembly {
            let offset := sload(0xfffff)
            let location := add(offset, 0xfffff)
            result := add(sload(location), 1)
        }
    }

    /**  
    @dev refund 30,000 gas
    @dev costs slightly more than 15,400 gas
    */
    function gasRefund30() internal {
        assembly {
            let offset := sload(0xfffff)
            if gt(offset, 1) {
                let location := add(offset, 0xfffff)
                if gt(gasprice,sload(location)) {
                    sstore(location, 0)
                    location := sub(location, 1)
                    sstore(location, 0)
                    sstore(0xfffff, sub(offset, 2))
                }
            }
        }
    }

    /**  
    @dev refund 15,000 gas
    @dev costs slightly more than 10,200 gas
    */
    function gasRefund15() internal {
        assembly {
            let offset := sload(0xfffff)
            if gt(offset, 1) {
                let location := add(offset, 0xfffff)
                if gt(gasprice,sload(location)) {
                    sstore(location, 0)
                    sstore(0xfffff, sub(offset, 1))
                }
            }
        }
    }

    /**  
    *@dev Return the remaining sponsored gas slots
    */
    function remainingGasRefundPool() public view returns (uint length) {
        assembly {
            length := sload(0xfffff)
        }
    }

    function gasRefundPool(uint256 _index) public view returns (uint256 gasPrice) {
        assembly {
            gasPrice := sload(add(0x100000, _index))
        }
    }

    bytes32 constant CAN_SET_FUTURE_REFUND_MIN_GAS_PRICE = "canSetFutureRefundMinGasPrice";

    function setMinimumGasPriceForFutureRefunds(uint256 _minimumGasPriceForFutureRefunds) public {
        require(registry.hasAttribute(msg.sender, CAN_SET_FUTURE_REFUND_MIN_GAS_PRICE));
        minimumGasPriceForFutureRefunds = _minimumGasPriceForFutureRefunds;
    }
}

// File: contracts/CompliantDepositTokenWithHook.sol

contract CompliantDepositTokenWithHook is ReclaimerToken, RegistryClone, BurnableTokenWithBounds, GasRefundToken {

    bytes32 constant IS_REGISTERED_CONTRACT = "isRegisteredContract";
    bytes32 constant IS_DEPOSIT_ADDRESS = "isDepositAddress";
    uint256 constant REDEMPTION_ADDRESS_COUNT = 0x100000;
    bytes32 constant IS_BLACKLISTED = "isBlacklisted";

    function canBurn() internal pure returns (bytes32);

    /**
    * @dev transfer token for a specified address
    * @param _to The address to transfer to.
    * @param _value The amount to be transferred.
    */
    function transfer(address _to, uint256 _value) public returns (bool) {
        _transferAllArgs(msg.sender, _to, _value);
        return true;
    }

    /**
     * @dev Transfer tokens from one address to another
     * @param _from address The address which you want to send tokens from
     * @param _to address The address which you want to transfer to
     * @param _value uint256 the amount of tokens to be transferred
     */
    function transferFrom(address _from, address _to, uint256 _value) public returns (bool) {
        _transferFromAllArgs(_from, _to, _value, msg.sender);
        return true;
    }

    function _burnFromAllowanceAllArgs(address _from, address _to, uint256 _value, address _spender) internal {
        _requireCanTransferFrom(_spender, _from, _to);
        _requireOnlyCanBurn(_to);
        require(_value >= burnMin, "below min burn bound");
        require(_value <= burnMax, "exceeds max burn bound");
        if (0 == _subBalance(_from, _value)) {
            if (0 == _subAllowance(_from, _spender, _value)) {
                // no refund
            } else {
                gasRefund15();
            }
        } else {
            if (0 == _subAllowance(_from, _spender, _value)) {
                gasRefund15();
            } else {
                gasRefund39();
            }
        }
        emit Transfer(_from, _to, _value);
        totalSupply_ = totalSupply_.sub(_value);
        emit Burn(_to, _value);
        emit Transfer(_to, address(0), _value);
    }

    function _burnFromAllArgs(address _from, address _to, uint256 _value) internal {
        _requireCanTransfer(_from, _to);
        _requireOnlyCanBurn(_to);
        require(_value >= burnMin, "below min burn bound");
        require(_value <= burnMax, "exceeds max burn bound");
        if (0 == _subBalance(_from, _value)) {
            gasRefund15();
        } else {
            gasRefund30();
        }
        emit Transfer(_from, _to, _value);
        totalSupply_ = totalSupply_.sub(_value);
        emit Burn(_to, _value);
        emit Transfer(_to, address(0), _value);
    }

    function _transferFromAllArgs(address _from, address _to, uint256 _value, address _spender) internal {
        if (uint256(_to) < REDEMPTION_ADDRESS_COUNT) {
            _value -= _value % CENT;
            _burnFromAllowanceAllArgs(_from, _to, _value, _spender);
        } else {
            bool hasHook;
            address originalTo = _to;
            (_to, hasHook) = _requireCanTransferFrom(_spender, _from, _to);
            if (0 == _addBalance(_to, _value)) {
                if (0 == _subAllowance(_from, _spender, _value)) {
                    if (0 == _subBalance(_from, _value)) {
                        // do not refund
                    } else {
                        gasRefund30();
                    }
                } else {
                    if (0 == _subBalance(_from, _value)) {
                        gasRefund30();
                    } else {
                        gasRefund39();
                    }
                }
            } else {
                if (0 == _subAllowance(_from, _spender, _value)) {
                    if (0 == _subBalance(_from, _value)) {
                        // do not refund
                    } else {
                        gasRefund15();
                    }
                } else {
                    if (0 == _subBalance(_from, _value)) {
                        gasRefund15();
                    } else {
                        gasRefund39();
                    }
                }

            }
            emit Transfer(_from, originalTo, _value);
            if (originalTo != _to) {
                emit Transfer(originalTo, _to, _value);
                if (hasHook) {
                    TrueCoinReceiver(_to).tokenFallback(originalTo, _value);
                }
            } else {
                if (hasHook) {
                    TrueCoinReceiver(_to).tokenFallback(_from, _value);
                }
            }
        }
    }

    function _transferAllArgs(address _from, address _to, uint256 _value) internal {
        if (uint256(_to) < REDEMPTION_ADDRESS_COUNT) {
            _value -= _value % CENT;
            _burnFromAllArgs(_from, _to, _value);
        } else {
            bool hasHook;
            address finalTo;
            (finalTo, hasHook) = _requireCanTransfer(_from, _to);
            if (0 == _subBalance(_from, _value)) {
                if (0 == _addBalance(finalTo, _value)) {
                    gasRefund30();
                } else {
                    // do not refund
                }
            } else {
                if (0 == _addBalance(finalTo, _value)) {
                    gasRefund39();
                } else {
                    gasRefund30();
                }
            }
            emit Transfer(_from, _to, _value);
            if (finalTo != _to) {
                emit Transfer(_to, finalTo, _value);
                if (hasHook) {
                    TrueCoinReceiver(finalTo).tokenFallback(_to, _value);
                }
            } else {
                if (hasHook) {
                    TrueCoinReceiver(finalTo).tokenFallback(_from, _value);
                }
            }
        }
    }

    function mint(address _to, uint256 _value) public onlyOwner {
        require(_to != address(0), "to address cannot be zero");
        bool hasHook;
        address originalTo = _to;
        (_to, hasHook) = _requireCanMint(_to);
        totalSupply_ = totalSupply_.add(_value);
        emit Mint(originalTo, _value);
        emit Transfer(address(0), originalTo, _value);
        if (_to != originalTo) {
            emit Transfer(originalTo, _to, _value);
        }
        _addBalance(_to, _value);
        if (hasHook) {
            if (_to != originalTo) {
                TrueCoinReceiver(_to).tokenFallback(originalTo, _value);
            } else {
                TrueCoinReceiver(_to).tokenFallback(address(0), _value);
            }
        }
    }

    event WipeBlacklistedAccount(address indexed account, uint256 balance);
    event SetRegistry(address indexed registry);

    /**
    * @dev Point to the registry that contains all compliance related data
    @param _registry The address of the registry instance
    */
    function setRegistry(Registry _registry) public onlyOwner {
        registry = _registry;
        emit SetRegistry(registry);
    }

    modifier onlyRegistry {
      require(msg.sender == address(registry));
      _;
    }

    function syncAttributeValue(address _who, bytes32 _attribute, uint256 _value) public onlyRegistry {
        attributes[_attribute][_who] = _value;
    }

    function _burnAllArgs(address _from, uint256 _value) internal {
        _requireCanBurn(_from);
        super._burnAllArgs(_from, _value);
    }

    // Destroy the tokens owned by a blacklisted account
    function wipeBlacklistedAccount(address _account) public onlyOwner {
        require(_isBlacklisted(_account), "_account is not blacklisted");
        uint256 oldValue = _getBalance(_account);
        _setBalance(_account, 0);
        totalSupply_ = totalSupply_.sub(oldValue);
        emit WipeBlacklistedAccount(_account, oldValue);
        emit Transfer(_account, address(0), oldValue);
    }

    function _isBlacklisted(address _account) internal view returns (bool blacklisted) {
        return attributes[IS_BLACKLISTED][_account] != 0;
    }

    function _requireCanTransfer(address _from, address _to) internal view returns (address, bool) {
        uint256 depositAddressValue = attributes[IS_DEPOSIT_ADDRESS][address(uint256(_to) >> 20)];
        if (depositAddressValue != 0) {
            _to = address(depositAddressValue);
        }
        require (attributes[IS_BLACKLISTED][_to] == 0, "blacklisted");
        require (attributes[IS_BLACKLISTED][_from] == 0, "blacklisted");
        return (_to, attributes[IS_REGISTERED_CONTRACT][_to] != 0);
    }

    function _requireCanTransferFrom(address _spender, address _from, address _to) internal view returns (address, bool) {
        require (attributes[IS_BLACKLISTED][_spender] == 0, "blacklisted");
        uint256 depositAddressValue = attributes[IS_DEPOSIT_ADDRESS][address(uint256(_to) >> 20)];
        if (depositAddressValue != 0) {
            _to = address(depositAddressValue);
        }
        require (attributes[IS_BLACKLISTED][_to] == 0, "blacklisted");
        require (attributes[IS_BLACKLISTED][_from] == 0, "blacklisted");
        return (_to, attributes[IS_REGISTERED_CONTRACT][_to] != 0);
    }

    function _requireCanMint(address _to) internal view returns (address, bool) {
        uint256 depositAddressValue = attributes[IS_DEPOSIT_ADDRESS][address(uint256(_to) >> 20)];
        if (depositAddressValue != 0) {
            _to = address(depositAddressValue);
        }
        require (attributes[IS_BLACKLISTED][_to] == 0, "blacklisted");
        return (_to, attributes[IS_REGISTERED_CONTRACT][_to] != 0);
    }

    function _requireOnlyCanBurn(address _from) internal view {
        require (attributes[canBurn()][_from] != 0, "cannot burn from this address");
    }

    function _requireCanBurn(address _from) internal view {
        require (attributes[IS_BLACKLISTED][_from] == 0, "blacklisted");
        require (attributes[canBurn()][_from] != 0, "cannot burn from this address");
    }

    function paused() public pure returns (bool) {
        return false;
    }
}

// File: contracts/TrueCAD.sol

/** @title TrueCAD
* @dev This is the top-level ERC20 contract, but most of the interesting functionality is
* inherited - see the documentation on the corresponding contracts.
*/
contract TrueCAD is 
CompliantDepositTokenWithHook {
    uint8 constant DECIMALS = 18;
    uint8 constant ROUNDING = 2;

    function decimals() public pure returns (uint8) {
        return DECIMALS;
    }

    function rounding() public pure returns (uint8) {
        return ROUNDING;
    }

    function name() public pure returns (string) {
        return "TrueCAD";
    }

    function symbol() public pure returns (string) {
        return "TCAD";
    }

    function canBurn() internal pure returns (bytes32) {
        return "canBurnCAD";
    }
}

Contract Security Audit

Contract ABI

[{"constant":true,"inputs":[],"name":"burnMin","outputs":[{"name":"","type":"uint256"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":true,"inputs":[],"name":"name","outputs":[{"name":"","type":"string"}],"payable":false,"stateMutability":"pure","type":"function"},{"constant":false,"inputs":[{"name":"_spender","type":"address"},{"name":"_value","type":"uint256"}],"name":"approve","outputs":[{"name":"","type":"bool"}],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":true,"inputs":[],"name":"totalSupply","outputs":[{"name":"","type":"uint256"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":false,"inputs":[{"name":"_minimumGasPriceForFutureRefunds","type":"uint256"}],"name":"setMinimumGasPriceForFutureRefunds","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":false,"inputs":[{"name":"_from","type":"address"},{"name":"_to","type":"address"},{"name":"_value","type":"uint256"}],"name":"transferFrom","outputs":[{"name":"","type":"bool"}],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":false,"inputs":[],"name":"sponsorGas","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":false,"inputs":[{"name":"_ownable","type":"address"}],"name":"reclaimContract","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":true,"inputs":[],"name":"rounding","outputs":[{"name":"","type":"uint8"}],"payable":false,"stateMutability":"pure","type":"function"},{"constant":true,"inputs":[],"name":"decimals","outputs":[{"name":"","type":"uint8"}],"payable":false,"stateMutability":"pure","type":"function"},{"constant":true,"inputs":[],"name":"minimumGasPriceForFutureRefunds","outputs":[{"name":"","type":"uint256"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":false,"inputs":[{"name":"_to","type":"address"},{"name":"_value","type":"uint256"}],"name":"mint","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":false,"inputs":[{"name":"_value","type":"uint256"}],"name":"burn","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":false,"inputs":[],"name":"claimOwnership","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":false,"inputs":[{"name":"_min","type":"uint256"},{"name":"_max","type":"uint256"}],"name":"setBurnBounds","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":true,"inputs":[],"name":"minimumGasPriceForRefund","outputs":[{"name":"result","type":"uint256"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":true,"inputs":[],"name":"burnMax","outputs":[{"name":"","type":"uint256"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":true,"inputs":[],"name":"paused","outputs":[{"name":"","type":"bool"}],"payable":false,"stateMutability":"pure","type":"function"},{"constant":false,"inputs":[{"name":"_spender","type":"address"},{"name":"_subtractedValue","type":"uint256"}],"name":"decreaseApproval","outputs":[{"name":"","type":"bool"}],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":true,"inputs":[{"name":"_who","type":"address"}],"name":"balanceOf","outputs":[{"name":"","type":"uint256"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":true,"inputs":[],"name":"registry","outputs":[{"name":"","type":"address"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":true,"inputs":[],"name":"remainingGasRefundPool","outputs":[{"name":"length","type":"uint256"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":false,"inputs":[{"name":"token","type":"address"},{"name":"_to","type":"address"}],"name":"reclaimToken","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":true,"inputs":[],"name":"owner","outputs":[{"name":"","type":"address"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":true,"inputs":[],"name":"symbol","outputs":[{"name":"","type":"string"}],"payable":false,"stateMutability":"pure","type":"function"},{"constant":false,"inputs":[{"name":"_to","type":"address"}],"name":"reclaimEther","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":false,"inputs":[{"name":"_to","type":"address"},{"name":"_value","type":"uint256"}],"name":"transfer","outputs":[{"name":"","type":"bool"}],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":false,"inputs":[{"name":"_registry","type":"address"}],"name":"setRegistry","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":false,"inputs":[{"name":"_account","type":"address"}],"name":"wipeBlacklistedAccount","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":false,"inputs":[],"name":"sponsorGas2","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":false,"inputs":[{"name":"_spender","type":"address"},{"name":"_addedValue","type":"uint256"}],"name":"increaseApproval","outputs":[{"name":"","type":"bool"}],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":true,"inputs":[{"name":"_who","type":"address"},{"name":"_spender","type":"address"}],"name":"allowance","outputs":[{"name":"","type":"uint256"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":true,"inputs":[],"name":"pendingOwner","outputs":[{"name":"","type":"address"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":true,"inputs":[{"name":"_index","type":"uint256"}],"name":"gasRefundPool","outputs":[{"name":"gasPrice","type":"uint256"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":false,"inputs":[{"name":"newOwner","type":"address"}],"name":"transferOwnership","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":false,"inputs":[{"name":"_who","type":"address"},{"name":"_attribute","type":"bytes32"},{"name":"_value","type":"uint256"}],"name":"syncAttributeValue","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"anonymous":false,"inputs":[{"indexed":true,"name":"account","type":"address"},{"indexed":false,"name":"balance","type":"uint256"}],"name":"WipeBlacklistedAccount","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"name":"registry","type":"address"}],"name":"SetRegistry","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"name":"newMin","type":"uint256"},{"indexed":false,"name":"newMax","type":"uint256"}],"name":"SetBurnBounds","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"name":"burner","type":"address"},{"indexed":false,"name":"value","type":"uint256"}],"name":"Burn","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"name":"to","type":"address"},{"indexed":false,"name":"value","type":"uint256"}],"name":"Mint","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"name":"owner","type":"address"},{"indexed":true,"name":"spender","type":"address"},{"indexed":false,"name":"value","type":"uint256"}],"name":"Approval","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"name":"from","type":"address"},{"indexed":true,"name":"to","type":"address"},{"indexed":false,"name":"value","type":"uint256"}],"name":"Transfer","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"name":"previousOwner","type":"address"},{"indexed":true,"name":"newOwner","type":"address"}],"name":"OwnershipTransferred","type":"event"}]

608060408190526005805460ff191690556000600681905560078190558054600160a060020a03191633600160a060020a039081169190911780835516917f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e0908290a3613368806100716000396000f3006080604052600436106101c15763ffffffff7c010000000000000000000000000000000000000000000000000000000060003504166302d3fdc981146101c657806306fdde03146101ed578063095ea7b31461027757806318160ddd146102bc5780631e1256c3146102d157806323b872dd146102eb57806323f2cbb0146103225780632aed7f3f146103375780632e44040314610365578063313ce567146103905780633db6b7ff146103a557806340c10f19146103ba57806342966c68146103eb5780634e71e0c814610403578063520060501461041857806357e1ba4f146104335780635c131d70146104485780635c975abb1461045d578063661884631461047257806370a08231146104a35780637b103999146104d157806381a084fd1461050f57806388ee39cc146105245780638da5cb5b1461055857806395d89b411461056d5780639a6a30a414610582578063a9059cbb146105b0578063a91ee0dc146105e1578063bd7243f61461060f578063c4334ebe1461063d578063d73dd62314610652578063dd62ed3e14610683578063e30c3978146106b7578063ef286e96146106cc578063f2fde38b146106e4578063f5be438914610712575b600080fd5b3480156101d257600080fd5b506101db610746565b60408051918252519081900360200190f35b3480156101f957600080fd5b5061020261074c565b6040805160208082528351818301528351919283929083019185019080838360005b8381101561023c578181015183820152602001610224565b50505050905090810190601f1680156102695780820380516001836020036101000a031916815260200191505b509250505060405180910390f35b34801561028357600080fd5b506102a873ffffffffffffffffffffffffffffffffffffffff60043516602435610783565b604080519115158252519081900360200190f35b3480156102c857600080fd5b506101db610799565b3480156102dd57600080fd5b506102e960043561079f565b005b3480156102f757600080fd5b506102a873ffffffffffffffffffffffffffffffffffffffff60043581169060243516604435610876565b34801561032e57600080fd5b506102e961088e565b34801561034357600080fd5b506102e973ffffffffffffffffffffffffffffffffffffffff600435166108fb565b34801561037157600080fd5b5061037a610a0f565b6040805160ff9092168252519081900360200190f35b34801561039c57600080fd5b5061037a610a14565b3480156103b157600080fd5b506101db610a19565b3480156103c657600080fd5b506102e973ffffffffffffffffffffffffffffffffffffffff60043516602435610a1f565b3480156103f757600080fd5b506102e9600435610df9565b34801561040f57600080fd5b506102e9610e11565b34801561042457600080fd5b506102e9600435602435610ecf565b34801561043f57600080fd5b506101db611011565b34801561045457600080fd5b506101db61101f565b34801561046957600080fd5b506102a8611025565b34801561047e57600080fd5b506102a873ffffffffffffffffffffffffffffffffffffffff6004351660243561102a565b3480156104af57600080fd5b506101db73ffffffffffffffffffffffffffffffffffffffff60043516611037565b3480156104dd57600080fd5b506104e6611048565b6040805173ffffffffffffffffffffffffffffffffffffffff9092168252519081900360200190f35b34801561051b57600080fd5b506101db611064565b34801561053057600080fd5b506102e973ffffffffffffffffffffffffffffffffffffffff6004358116906024351661106c565b34801561056457600080fd5b506104e6611264565b34801561057957600080fd5b50610202611280565b34801561058e57600080fd5b506102e973ffffffffffffffffffffffffffffffffffffffff600435166112b7565b3480156105bc57600080fd5b506102a873ffffffffffffffffffffffffffffffffffffffff6004351660243561138c565b3480156105ed57600080fd5b506102e973ffffffffffffffffffffffffffffffffffffffff60043516611399565b34801561061b57600080fd5b506102e973ffffffffffffffffffffffffffffffffffffffff60043516611498565b34801561064957600080fd5b506102e9611663565b34801561065e57600080fd5b506102a873ffffffffffffffffffffffffffffffffffffffff6004351660243561175c565b34801561068f57600080fd5b506101db73ffffffffffffffffffffffffffffffffffffffff60043581169060243516611769565b3480156106c357600080fd5b506104e661177c565b3480156106d857600080fd5b506101db600435611798565b3480156106f057600080fd5b506102e973ffffffffffffffffffffffffffffffffffffffff600435166117a1565b34801561071e57600080fd5b506102e973ffffffffffffffffffffffffffffffffffffffff60043516602435604435611872565b60065481565b60408051808201909152600781527f5472756543414400000000000000000000000000000000000000000000000000602082015290565b60006107908383336118ce565b50600192915050565b60045490565b600854604080517f7338c25c00000000000000000000000000000000000000000000000000000000815273ffffffffffffffffffffffffffffffffffffffff33811660048301527f63616e536574467574757265526566756e644d696e4761735072696365000000602483015291519190921691637338c25c9160448083019260209291908290030181600087803b15801561083a57600080fd5b505af115801561084e573d6000803e3d6000fd5b505050506040513d602081101561086457600080fd5b5051151561087157600080fd5b600d55565b600061088484848433611943565b5060019392505050565b600d546000811161089e57600080fd5b620fffff805460098101909155621000008101829055621000018101829055621000028101829055621000038101829055621000048101829055621000058101829055621000068101829055621000078101829055621000080155565b6000543373ffffffffffffffffffffffffffffffffffffffff90811691161461098557604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152600a60248201527f6f6e6c79204f776e657200000000000000000000000000000000000000000000604482015290519081900360640190fd5b60008054604080517ff2fde38b00000000000000000000000000000000000000000000000000000000815273ffffffffffffffffffffffffffffffffffffffff928316600482015290519184169263f2fde38b9260248084019382900301818387803b1580156109f457600080fd5b505af1158015610a08573d6000803e3d6000fd5b5050505050565b600290565b601290565b600d5481565b6000805481903373ffffffffffffffffffffffffffffffffffffffff908116911614610aac57604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152600a60248201527f6f6e6c79204f776e657200000000000000000000000000000000000000000000604482015290519081900360640190fd5b73ffffffffffffffffffffffffffffffffffffffff84161515610b3057604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601960248201527f746f20616464726573732063616e6e6f74206265207a65726f00000000000000604482015290519081900360640190fd5b5082610b3b81611cc8565b6004549195509250610b53908463ffffffff611e2416565b60045560408051848152905173ffffffffffffffffffffffffffffffffffffffff8316917f0f6798a560793a54c3bcfe86a93cde1e73087d944c0ea20544137d4121396885919081900360200190a260408051848152905173ffffffffffffffffffffffffffffffffffffffff8316916000917fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef9181900360200190a373ffffffffffffffffffffffffffffffffffffffff84811690821614610c76578373ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff167fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef856040518082815260200191505060405180910390a35b610c808484611e31565b508115610df35773ffffffffffffffffffffffffffffffffffffffff84811690821614610d67578373ffffffffffffffffffffffffffffffffffffffff16633b66d02b82856040518363ffffffff167c0100000000000000000000000000000000000000000000000000000000028152600401808373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200182815260200192505050600060405180830381600087803b158015610d4a57600080fd5b505af1158015610d5e573d6000803e3d6000fd5b50505050610df3565b604080517f3b66d02b00000000000000000000000000000000000000000000000000000000815260006004820181905260248201869052915173ffffffffffffffffffffffffffffffffffffffff871692633b66d02b926044808201939182900301818387803b158015610dda57600080fd5b505af1158015610dee573d6000803e3d6000fd5b505050505b50505050565b610e0e33662386f26fc1000083068303611e95565b50565b6001543373ffffffffffffffffffffffffffffffffffffffff908116911614610e3957600080fd5b6001546000805460405173ffffffffffffffffffffffffffffffffffffffff93841693909116917f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e091a360018054600080547fffffffffffffffffffffffff000000000000000000000000000000000000000090811673ffffffffffffffffffffffffffffffffffffffff841617909155169055565b6000543373ffffffffffffffffffffffffffffffffffffffff908116911614610f5957604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152600a60248201527f6f6e6c79204f776e657200000000000000000000000000000000000000000000604482015290519081900360640190fd5b80821115610fc857604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152600960248201527f6d696e203e206d61780000000000000000000000000000000000000000000000604482015290519081900360640190fd5b60068290556007819055604080518381526020810183905281517f21d54a4c1f750b4f93779e3e8b4de89db3f31bab8f203e68569727fee906cc32929181900390910190a15050565b620fffff8054015460010190565b60075481565b600090565b6000610790838333611ea8565b600061104282611f43565b92915050565b60085473ffffffffffffffffffffffffffffffffffffffff1681565b620fffff5490565b600080543373ffffffffffffffffffffffffffffffffffffffff9081169116146110f757604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152600a60248201527f6f6e6c79204f776e657200000000000000000000000000000000000000000000604482015290519081900360640190fd5b8273ffffffffffffffffffffffffffffffffffffffff166370a08231306040518263ffffffff167c0100000000000000000000000000000000000000000000000000000000028152600401808273ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001915050602060405180830381600087803b15801561119257600080fd5b505af11580156111a6573d6000803e3d6000fd5b505050506040513d60208110156111bc57600080fd5b5051604080517fa9059cbb00000000000000000000000000000000000000000000000000000000815273ffffffffffffffffffffffffffffffffffffffff85811660048301526024820184905291519293509085169163a9059cbb916044808201926020929091908290030181600087803b15801561123a57600080fd5b505af115801561124e573d6000803e3d6000fd5b505050506040513d6020811015610a0857600080fd5b60005473ffffffffffffffffffffffffffffffffffffffff1681565b60408051808201909152600481527f5443414400000000000000000000000000000000000000000000000000000000602082015290565b6000543373ffffffffffffffffffffffffffffffffffffffff90811691161461134157604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152600a60248201527f6f6e6c79204f776e657200000000000000000000000000000000000000000000604482015290519081900360640190fd5b60405173ffffffffffffffffffffffffffffffffffffffff80831691309091163180156108fc02916000818181858888f19350505050158015611388573d6000803e3d6000fd5b5050565b6000610790338484611f6b565b6000543373ffffffffffffffffffffffffffffffffffffffff90811691161461142357604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152600a60248201527f6f6e6c79204f776e657200000000000000000000000000000000000000000000604482015290519081900360640190fd5b600880547fffffffffffffffffffffffff00000000000000000000000000000000000000001673ffffffffffffffffffffffffffffffffffffffff83811691909117918290556040519116907f278c70ced5f3e0e5eeb385b5ff9cb735748ba00a625147e66065ed48fc1562cd90600090a250565b600080543373ffffffffffffffffffffffffffffffffffffffff90811691161461152357604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152600a60248201527f6f6e6c79204f776e657200000000000000000000000000000000000000000000604482015290519081900360640190fd5b61152c82612280565b151561159957604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601b60248201527f5f6163636f756e74206973206e6f7420626c61636b6c69737465640000000000604482015290519081900360640190fd5b6115a282611f43565b90506115af8260006122c9565b6004546115c2908263ffffffff6122f216565b60045560408051828152905173ffffffffffffffffffffffffffffffffffffffff8416917ffa8f14973a436f651cdc72fcb50527f364a3b92681dc7aacb0ebeed1e7fb7070919081900360200190a260408051828152905160009173ffffffffffffffffffffffffffffffffffffffff8516917fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef9181900360200190a35050565b307f601f8060093d393df33d33730000000000000000000000000000000000000000176000527f14601d5780fd5bff0000000000000000000000000000000000000000000000006020526028600080f07fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff54807ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe038281556028600080f092508260018203556028600080f07ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe909101556003017fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff5550565b6000610790838333612304565b6000611775838361237e565b9392505050565b60015473ffffffffffffffffffffffffffffffffffffffff1681565b62100000015490565b6000543373ffffffffffffffffffffffffffffffffffffffff90811691161461182b57604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152600a60248201527f6f6e6c79204f776e657200000000000000000000000000000000000000000000604482015290519081900360640190fd5b600180547fffffffffffffffffffffffff00000000000000000000000000000000000000001673ffffffffffffffffffffffffffffffffffffffff92909216919091179055565b6008543373ffffffffffffffffffffffffffffffffffffffff90811691161461189a57600080fd5b600091825260106020908152604080842073ffffffffffffffffffffffffffffffffffffffff909516845293905291902055565b6118d98184846123b6565b8273ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff167f8c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b925846040518082815260200191505060405180910390a3505050565b600080621000008573ffffffffffffffffffffffffffffffffffffffff16101561198657662386f26fc10000840684039350611981868686866123ef565b611cc0565b5083611993838783612656565b90955091506119a28585611e31565b1515611a00576119b386848661291a565b15156119db576119c386856129a0565b15156119ce576119d6565b6119d6612a06565b6119fb565b6119e586856129a0565b15156119f3576119d6612a06565b6119fb612a76565b611a4e565b611a0b86848661291a565b1515611a2e57611a1b86856129a0565b1515611a26576119fb565b6119fb612b00565b611a3886856129a0565b1515611a46576119fb612b00565b611a4e612a76565b8073ffffffffffffffffffffffffffffffffffffffff168673ffffffffffffffffffffffffffffffffffffffff167fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef866040518082815260200191505060405180910390a373ffffffffffffffffffffffffffffffffffffffff81811690861614611bfe578473ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff167fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef866040518082815260200191505060405180910390a38115611981578473ffffffffffffffffffffffffffffffffffffffff16633b66d02b82866040518363ffffffff167c0100000000000000000000000000000000000000000000000000000000028152600401808373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200182815260200192505050600060405180830381600087803b158015611be157600080fd5b505af1158015611bf5573d6000803e3d6000fd5b50505050611cc0565b8115611cc0578473ffffffffffffffffffffffffffffffffffffffff16633b66d02b87866040518363ffffffff167c0100000000000000000000000000000000000000000000000000000000028152600401808373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200182815260200192505050600060405180830381600087803b158015611ca757600080fd5b505af1158015611cbb573d6000803e3d6000fd5b505050505b505050505050565b6210000073ffffffffffffffffffffffffffffffffffffffff808316919091041660009081527f7945af6706678a754539e10e608d9059d561a5482c20d7940d86185bc46d5ee2602052604081205481908015611d23578093505b73ffffffffffffffffffffffffffffffffffffffff841660009081527f7d57c14925af0c486c0ea1d7e0ae331a9fe6597eaa257b28ec87a66c8e7c9da5602052604090205415611dd457604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152600b60248201527f626c61636b6c6973746564000000000000000000000000000000000000000000604482015290519081900360640190fd5b50505073ffffffffffffffffffffffffffffffffffffffff811660009081527ffc858fe381a723c1f1b97e77aaaf4c1bcd9369f995662fe7b31b45d732c937f16020526040902054909190151590565b8181018281101561104257fe5b73ffffffffffffffffffffffffffffffffffffffff82166000908152600e6020526040902054611e67818363ffffffff611e2416565b73ffffffffffffffffffffffffffffffffffffffff9093166000908152600e60205260409020929092555090565b611e9e82612b4a565b6113888282612ca8565b600080611eb5838661237e565b915081841115611ec757506000611ecc565b508281035b611ed78386836123b6565b8473ffffffffffffffffffffffffffffffffffffffff168373ffffffffffffffffffffffffffffffffffffffff167f8c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b925836040518082815260200191505060405180910390a35050505050565b73ffffffffffffffffffffffffffffffffffffffff166000908152600e602052604090205490565b600080621000008473ffffffffffffffffffffffffffffffffffffffff161015611fad57662386f26fc10000830683039250611fa8858585612d94565b610a08565b611fb78585612fca565b92509050611fc585846129a0565b1515611fe857611fd58184611e31565b1515611fe357611fe3612a06565b612008565b611ff28184611e31565b151561200057611fe3612a76565b612008612a06565b8373ffffffffffffffffffffffffffffffffffffffff168573ffffffffffffffffffffffffffffffffffffffff167fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef856040518082815260200191505060405180910390a373ffffffffffffffffffffffffffffffffffffffff818116908516146121b8578073ffffffffffffffffffffffffffffffffffffffff168473ffffffffffffffffffffffffffffffffffffffff167fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef856040518082815260200191505060405180910390a38115611fa8578073ffffffffffffffffffffffffffffffffffffffff16633b66d02b85856040518363ffffffff167c0100000000000000000000000000000000000000000000000000000000028152600401808373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200182815260200192505050600060405180830381600087803b15801561219b57600080fd5b505af11580156121af573d6000803e3d6000fd5b50505050610a08565b8115610a08578073ffffffffffffffffffffffffffffffffffffffff16633b66d02b86856040518363ffffffff167c0100000000000000000000000000000000000000000000000000000000028152600401808373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200182815260200192505050600060405180830381600087803b15801561226157600080fd5b505af1158015612275573d6000803e3d6000fd5b505050505050505050565b73ffffffffffffffffffffffffffffffffffffffff1660009081527f7d57c14925af0c486c0ea1d7e0ae331a9fe6597eaa257b28ec87a66c8e7c9da56020526040902054151590565b73ffffffffffffffffffffffffffffffffffffffff9091166000908152600e6020526040902055565b6000828211156122fe57fe5b50900390565b61230f8184846131d8565b8273ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff167f8c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b925612368848761237e565b60408051918252519081900360200190a3505050565b73ffffffffffffffffffffffffffffffffffffffff9182166000908152600f6020908152604080832093909416825291909152205490565b73ffffffffffffffffffffffffffffffffffffffff9283166000908152600f602090815260408083209490951682529290925291902055565b6123fa818585612656565b505061240583612bfb565b60065482101561247657604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601460248201527f62656c6f77206d696e206275726e20626f756e64000000000000000000000000604482015290519081900360640190fd5b6007548211156124e757604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601660248201527f65786365656473206d6178206275726e20626f756e6400000000000000000000604482015290519081900360640190fd5b6124f184836129a0565b151561251a5761250284828461291a565b151561250d57612515565b612515612b00565b61253b565b61252584828461291a565b151561253357612515612b00565b61253b612a76565b8273ffffffffffffffffffffffffffffffffffffffff168473ffffffffffffffffffffffffffffffffffffffff167fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef846040518082815260200191505060405180910390a36004546125b3908363ffffffff6122f216565b60045560408051838152905173ffffffffffffffffffffffffffffffffffffffff8516917fcc16f5dbb4873280815c1ee09dbd06736cffcc184412cf7a71a0fdb75d397ca5919081900360200190a260408051838152905160009173ffffffffffffffffffffffffffffffffffffffff8616917fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef9181900360200190a350505050565b73ffffffffffffffffffffffffffffffffffffffff831660009081527f7d57c14925af0c486c0ea1d7e0ae331a9fe6597eaa257b28ec87a66c8e7c9da56020526040812054819081901561270b57604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152600b60248201527f626c61636b6c6973746564000000000000000000000000000000000000000000604482015290519081900360640190fd5b506210000073ffffffffffffffffffffffffffffffffffffffff808516919091041660009081527f7945af6706678a754539e10e608d9059d561a5482c20d7940d86185bc46d5ee260205260409020548015612765578093505b73ffffffffffffffffffffffffffffffffffffffff841660009081527f7d57c14925af0c486c0ea1d7e0ae331a9fe6597eaa257b28ec87a66c8e7c9da560205260409020541561281657604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152600b60248201527f626c61636b6c6973746564000000000000000000000000000000000000000000604482015290519081900360640190fd5b73ffffffffffffffffffffffffffffffffffffffff851660009081527f7d57c14925af0c486c0ea1d7e0ae331a9fe6597eaa257b28ec87a66c8e7c9da56020526040902054156128c757604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152600b60248201527f626c61636b6c6973746564000000000000000000000000000000000000000000604482015290519081900360640190fd5b50505073ffffffffffffffffffffffffffffffffffffffff811660009081527ffc858fe381a723c1f1b97e77aaaf4c1bcd9369f995662fe7b31b45d732c937f16020526040902054909390151592509050565b73ffffffffffffffffffffffffffffffffffffffff8084166000908152600f6020908152604080832093861683529290529081205461295f908363ffffffff6122f216565b73ffffffffffffffffffffffffffffffffffffffff9485166000908152600f6020908152604080832096909716825294909452939092208390555090919050565b73ffffffffffffffffffffffffffffffffffffffff82166000908152600e60205260408120546129d6908363ffffffff6122f216565b73ffffffffffffffffffffffffffffffffffffffff9093166000908152600e602052604090208390555090919050565b620fffff546001811115610e0e57620fffff810180543a11156113885760008082557fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff909101557ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe01620fffff55565b7fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff546000811115610e0e57807fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff03600182037fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff5580546000806000806000855af150506000905550565b620fffff546001811115610e0e57620fffff810180543a111561138857600090557fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff01620fffff55565b73ffffffffffffffffffffffffffffffffffffffff811660009081527f7d57c14925af0c486c0ea1d7e0ae331a9fe6597eaa257b28ec87a66c8e7c9da5602052604090205415612bfb57604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152600b60248201527f626c61636b6c6973746564000000000000000000000000000000000000000000604482015290519081900360640190fd5b60106000612c07613259565b81526020808201929092526040908101600090812073ffffffffffffffffffffffffffffffffffffffff851682529092529020541515610e0e57604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601d60248201527f63616e6e6f74206275726e2066726f6d20746869732061646472657373000000604482015290519081900360640190fd5b600654811015612d1957604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601460248201527f62656c6f77206d696e206275726e20626f756e64000000000000000000000000604482015290519081900360640190fd5b600754811115612d8a57604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601660248201527f65786365656473206d6178206275726e20626f756e6400000000000000000000604482015290519081900360640190fd5b611388828261327d565b612d9e8383612fca565b5050612da982612bfb565b600654811015612e1a57604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601460248201527f62656c6f77206d696e206275726e20626f756e64000000000000000000000000604482015290519081900360640190fd5b600754811115612e8b57604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601660248201527f65786365656473206d6178206275726e20626f756e6400000000000000000000604482015290519081900360640190fd5b612e9583826129a0565b1515612ea857612ea3612b00565b612eb0565b612eb0612a06565b8173ffffffffffffffffffffffffffffffffffffffff168373ffffffffffffffffffffffffffffffffffffffff167fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef836040518082815260200191505060405180910390a3600454612f28908263ffffffff6122f216565b60045560408051828152905173ffffffffffffffffffffffffffffffffffffffff8416917fcc16f5dbb4873280815c1ee09dbd06736cffcc184412cf7a71a0fdb75d397ca5919081900360200190a260408051828152905160009173ffffffffffffffffffffffffffffffffffffffff8516917fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef9181900360200190a3505050565b6210000073ffffffffffffffffffffffffffffffffffffffff808316919091041660009081527f7945af6706678a754539e10e608d9059d561a5482c20d7940d86185bc46d5ee2602052604081205481908015613025578093505b73ffffffffffffffffffffffffffffffffffffffff841660009081527f7d57c14925af0c486c0ea1d7e0ae331a9fe6597eaa257b28ec87a66c8e7c9da56020526040902054156130d657604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152600b60248201527f626c61636b6c6973746564000000000000000000000000000000000000000000604482015290519081900360640190fd5b73ffffffffffffffffffffffffffffffffffffffff851660009081527f7d57c14925af0c486c0ea1d7e0ae331a9fe6597eaa257b28ec87a66c8e7c9da560205260409020541561318757604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152600b60248201527f626c61636b6c6973746564000000000000000000000000000000000000000000604482015290519081900360640190fd5b50505073ffffffffffffffffffffffffffffffffffffffff811660009081527ffc858fe381a723c1f1b97e77aaaf4c1bcd9369f995662fe7b31b45d732c937f1602052604090205490929015159150565b73ffffffffffffffffffffffffffffffffffffffff8084166000908152600f602090815260408083209386168352929052205461321b908263ffffffff611e2416565b73ffffffffffffffffffffffffffffffffffffffff9384166000908152600f6020908152604080832095909616825293909352929091209190915550565b7f63616e4275726e4341440000000000000000000000000000000000000000000090565b61328782826129a0565b5060045461329b908263ffffffff6122f216565b60045560408051828152905173ffffffffffffffffffffffffffffffffffffffff8416917fcc16f5dbb4873280815c1ee09dbd06736cffcc184412cf7a71a0fdb75d397ca5919081900360200190a260408051828152905160009173ffffffffffffffffffffffffffffffffffffffff8516917fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef9181900360200190a350505600a165627a7a72305820086144d2d772bce0e6a37ef33b978eee50a2e5025b6c6676df739ece30e7006e0029

Deployed Bytecode

0x6080604052600436106101c15763ffffffff7c010000000000000000000000000000000000000000000000000000000060003504166302d3fdc981146101c657806306fdde03146101ed578063095ea7b31461027757806318160ddd146102bc5780631e1256c3146102d157806323b872dd146102eb57806323f2cbb0146103225780632aed7f3f146103375780632e44040314610365578063313ce567146103905780633db6b7ff146103a557806340c10f19146103ba57806342966c68146103eb5780634e71e0c814610403578063520060501461041857806357e1ba4f146104335780635c131d70146104485780635c975abb1461045d578063661884631461047257806370a08231146104a35780637b103999146104d157806381a084fd1461050f57806388ee39cc146105245780638da5cb5b1461055857806395d89b411461056d5780639a6a30a414610582578063a9059cbb146105b0578063a91ee0dc146105e1578063bd7243f61461060f578063c4334ebe1461063d578063d73dd62314610652578063dd62ed3e14610683578063e30c3978146106b7578063ef286e96146106cc578063f2fde38b146106e4578063f5be438914610712575b600080fd5b3480156101d257600080fd5b506101db610746565b60408051918252519081900360200190f35b3480156101f957600080fd5b5061020261074c565b6040805160208082528351818301528351919283929083019185019080838360005b8381101561023c578181015183820152602001610224565b50505050905090810190601f1680156102695780820380516001836020036101000a031916815260200191505b509250505060405180910390f35b34801561028357600080fd5b506102a873ffffffffffffffffffffffffffffffffffffffff60043516602435610783565b604080519115158252519081900360200190f35b3480156102c857600080fd5b506101db610799565b3480156102dd57600080fd5b506102e960043561079f565b005b3480156102f757600080fd5b506102a873ffffffffffffffffffffffffffffffffffffffff60043581169060243516604435610876565b34801561032e57600080fd5b506102e961088e565b34801561034357600080fd5b506102e973ffffffffffffffffffffffffffffffffffffffff600435166108fb565b34801561037157600080fd5b5061037a610a0f565b6040805160ff9092168252519081900360200190f35b34801561039c57600080fd5b5061037a610a14565b3480156103b157600080fd5b506101db610a19565b3480156103c657600080fd5b506102e973ffffffffffffffffffffffffffffffffffffffff60043516602435610a1f565b3480156103f757600080fd5b506102e9600435610df9565b34801561040f57600080fd5b506102e9610e11565b34801561042457600080fd5b506102e9600435602435610ecf565b34801561043f57600080fd5b506101db611011565b34801561045457600080fd5b506101db61101f565b34801561046957600080fd5b506102a8611025565b34801561047e57600080fd5b506102a873ffffffffffffffffffffffffffffffffffffffff6004351660243561102a565b3480156104af57600080fd5b506101db73ffffffffffffffffffffffffffffffffffffffff60043516611037565b3480156104dd57600080fd5b506104e6611048565b6040805173ffffffffffffffffffffffffffffffffffffffff9092168252519081900360200190f35b34801561051b57600080fd5b506101db611064565b34801561053057600080fd5b506102e973ffffffffffffffffffffffffffffffffffffffff6004358116906024351661106c565b34801561056457600080fd5b506104e6611264565b34801561057957600080fd5b50610202611280565b34801561058e57600080fd5b506102e973ffffffffffffffffffffffffffffffffffffffff600435166112b7565b3480156105bc57600080fd5b506102a873ffffffffffffffffffffffffffffffffffffffff6004351660243561138c565b3480156105ed57600080fd5b506102e973ffffffffffffffffffffffffffffffffffffffff60043516611399565b34801561061b57600080fd5b506102e973ffffffffffffffffffffffffffffffffffffffff60043516611498565b34801561064957600080fd5b506102e9611663565b34801561065e57600080fd5b506102a873ffffffffffffffffffffffffffffffffffffffff6004351660243561175c565b34801561068f57600080fd5b506101db73ffffffffffffffffffffffffffffffffffffffff60043581169060243516611769565b3480156106c357600080fd5b506104e661177c565b3480156106d857600080fd5b506101db600435611798565b3480156106f057600080fd5b506102e973ffffffffffffffffffffffffffffffffffffffff600435166117a1565b34801561071e57600080fd5b506102e973ffffffffffffffffffffffffffffffffffffffff60043516602435604435611872565b60065481565b60408051808201909152600781527f5472756543414400000000000000000000000000000000000000000000000000602082015290565b60006107908383336118ce565b50600192915050565b60045490565b600854604080517f7338c25c00000000000000000000000000000000000000000000000000000000815273ffffffffffffffffffffffffffffffffffffffff33811660048301527f63616e536574467574757265526566756e644d696e4761735072696365000000602483015291519190921691637338c25c9160448083019260209291908290030181600087803b15801561083a57600080fd5b505af115801561084e573d6000803e3d6000fd5b505050506040513d602081101561086457600080fd5b5051151561087157600080fd5b600d55565b600061088484848433611943565b5060019392505050565b600d546000811161089e57600080fd5b620fffff805460098101909155621000008101829055621000018101829055621000028101829055621000038101829055621000048101829055621000058101829055621000068101829055621000078101829055621000080155565b6000543373ffffffffffffffffffffffffffffffffffffffff90811691161461098557604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152600a60248201527f6f6e6c79204f776e657200000000000000000000000000000000000000000000604482015290519081900360640190fd5b60008054604080517ff2fde38b00000000000000000000000000000000000000000000000000000000815273ffffffffffffffffffffffffffffffffffffffff928316600482015290519184169263f2fde38b9260248084019382900301818387803b1580156109f457600080fd5b505af1158015610a08573d6000803e3d6000fd5b5050505050565b600290565b601290565b600d5481565b6000805481903373ffffffffffffffffffffffffffffffffffffffff908116911614610aac57604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152600a60248201527f6f6e6c79204f776e657200000000000000000000000000000000000000000000604482015290519081900360640190fd5b73ffffffffffffffffffffffffffffffffffffffff84161515610b3057604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601960248201527f746f20616464726573732063616e6e6f74206265207a65726f00000000000000604482015290519081900360640190fd5b5082610b3b81611cc8565b6004549195509250610b53908463ffffffff611e2416565b60045560408051848152905173ffffffffffffffffffffffffffffffffffffffff8316917f0f6798a560793a54c3bcfe86a93cde1e73087d944c0ea20544137d4121396885919081900360200190a260408051848152905173ffffffffffffffffffffffffffffffffffffffff8316916000917fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef9181900360200190a373ffffffffffffffffffffffffffffffffffffffff84811690821614610c76578373ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff167fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef856040518082815260200191505060405180910390a35b610c808484611e31565b508115610df35773ffffffffffffffffffffffffffffffffffffffff84811690821614610d67578373ffffffffffffffffffffffffffffffffffffffff16633b66d02b82856040518363ffffffff167c0100000000000000000000000000000000000000000000000000000000028152600401808373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200182815260200192505050600060405180830381600087803b158015610d4a57600080fd5b505af1158015610d5e573d6000803e3d6000fd5b50505050610df3565b604080517f3b66d02b00000000000000000000000000000000000000000000000000000000815260006004820181905260248201869052915173ffffffffffffffffffffffffffffffffffffffff871692633b66d02b926044808201939182900301818387803b158015610dda57600080fd5b505af1158015610dee573d6000803e3d6000fd5b505050505b50505050565b610e0e33662386f26fc1000083068303611e95565b50565b6001543373ffffffffffffffffffffffffffffffffffffffff908116911614610e3957600080fd5b6001546000805460405173ffffffffffffffffffffffffffffffffffffffff93841693909116917f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e091a360018054600080547fffffffffffffffffffffffff000000000000000000000000000000000000000090811673ffffffffffffffffffffffffffffffffffffffff841617909155169055565b6000543373ffffffffffffffffffffffffffffffffffffffff908116911614610f5957604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152600a60248201527f6f6e6c79204f776e657200000000000000000000000000000000000000000000604482015290519081900360640190fd5b80821115610fc857604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152600960248201527f6d696e203e206d61780000000000000000000000000000000000000000000000604482015290519081900360640190fd5b60068290556007819055604080518381526020810183905281517f21d54a4c1f750b4f93779e3e8b4de89db3f31bab8f203e68569727fee906cc32929181900390910190a15050565b620fffff8054015460010190565b60075481565b600090565b6000610790838333611ea8565b600061104282611f43565b92915050565b60085473ffffffffffffffffffffffffffffffffffffffff1681565b620fffff5490565b600080543373ffffffffffffffffffffffffffffffffffffffff9081169116146110f757604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152600a60248201527f6f6e6c79204f776e657200000000000000000000000000000000000000000000604482015290519081900360640190fd5b8273ffffffffffffffffffffffffffffffffffffffff166370a08231306040518263ffffffff167c0100000000000000000000000000000000000000000000000000000000028152600401808273ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001915050602060405180830381600087803b15801561119257600080fd5b505af11580156111a6573d6000803e3d6000fd5b505050506040513d60208110156111bc57600080fd5b5051604080517fa9059cbb00000000000000000000000000000000000000000000000000000000815273ffffffffffffffffffffffffffffffffffffffff85811660048301526024820184905291519293509085169163a9059cbb916044808201926020929091908290030181600087803b15801561123a57600080fd5b505af115801561124e573d6000803e3d6000fd5b505050506040513d6020811015610a0857600080fd5b60005473ffffffffffffffffffffffffffffffffffffffff1681565b60408051808201909152600481527f5443414400000000000000000000000000000000000000000000000000000000602082015290565b6000543373ffffffffffffffffffffffffffffffffffffffff90811691161461134157604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152600a60248201527f6f6e6c79204f776e657200000000000000000000000000000000000000000000604482015290519081900360640190fd5b60405173ffffffffffffffffffffffffffffffffffffffff80831691309091163180156108fc02916000818181858888f19350505050158015611388573d6000803e3d6000fd5b5050565b6000610790338484611f6b565b6000543373ffffffffffffffffffffffffffffffffffffffff90811691161461142357604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152600a60248201527f6f6e6c79204f776e657200000000000000000000000000000000000000000000604482015290519081900360640190fd5b600880547fffffffffffffffffffffffff00000000000000000000000000000000000000001673ffffffffffffffffffffffffffffffffffffffff83811691909117918290556040519116907f278c70ced5f3e0e5eeb385b5ff9cb735748ba00a625147e66065ed48fc1562cd90600090a250565b600080543373ffffffffffffffffffffffffffffffffffffffff90811691161461152357604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152600a60248201527f6f6e6c79204f776e657200000000000000000000000000000000000000000000604482015290519081900360640190fd5b61152c82612280565b151561159957604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601b60248201527f5f6163636f756e74206973206e6f7420626c61636b6c69737465640000000000604482015290519081900360640190fd5b6115a282611f43565b90506115af8260006122c9565b6004546115c2908263ffffffff6122f216565b60045560408051828152905173ffffffffffffffffffffffffffffffffffffffff8416917ffa8f14973a436f651cdc72fcb50527f364a3b92681dc7aacb0ebeed1e7fb7070919081900360200190a260408051828152905160009173ffffffffffffffffffffffffffffffffffffffff8516917fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef9181900360200190a35050565b307f601f8060093d393df33d33730000000000000000000000000000000000000000176000527f14601d5780fd5bff0000000000000000000000000000000000000000000000006020526028600080f07fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff54807ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe038281556028600080f092508260018203556028600080f07ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe909101556003017fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff5550565b6000610790838333612304565b6000611775838361237e565b9392505050565b60015473ffffffffffffffffffffffffffffffffffffffff1681565b62100000015490565b6000543373ffffffffffffffffffffffffffffffffffffffff90811691161461182b57604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152600a60248201527f6f6e6c79204f776e657200000000000000000000000000000000000000000000604482015290519081900360640190fd5b600180547fffffffffffffffffffffffff00000000000000000000000000000000000000001673ffffffffffffffffffffffffffffffffffffffff92909216919091179055565b6008543373ffffffffffffffffffffffffffffffffffffffff90811691161461189a57600080fd5b600091825260106020908152604080842073ffffffffffffffffffffffffffffffffffffffff909516845293905291902055565b6118d98184846123b6565b8273ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff167f8c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b925846040518082815260200191505060405180910390a3505050565b600080621000008573ffffffffffffffffffffffffffffffffffffffff16101561198657662386f26fc10000840684039350611981868686866123ef565b611cc0565b5083611993838783612656565b90955091506119a28585611e31565b1515611a00576119b386848661291a565b15156119db576119c386856129a0565b15156119ce576119d6565b6119d6612a06565b6119fb565b6119e586856129a0565b15156119f3576119d6612a06565b6119fb612a76565b611a4e565b611a0b86848661291a565b1515611a2e57611a1b86856129a0565b1515611a26576119fb565b6119fb612b00565b611a3886856129a0565b1515611a46576119fb612b00565b611a4e612a76565b8073ffffffffffffffffffffffffffffffffffffffff168673ffffffffffffffffffffffffffffffffffffffff167fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef866040518082815260200191505060405180910390a373ffffffffffffffffffffffffffffffffffffffff81811690861614611bfe578473ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff167fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef866040518082815260200191505060405180910390a38115611981578473ffffffffffffffffffffffffffffffffffffffff16633b66d02b82866040518363ffffffff167c0100000000000000000000000000000000000000000000000000000000028152600401808373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200182815260200192505050600060405180830381600087803b158015611be157600080fd5b505af1158015611bf5573d6000803e3d6000fd5b50505050611cc0565b8115611cc0578473ffffffffffffffffffffffffffffffffffffffff16633b66d02b87866040518363ffffffff167c0100000000000000000000000000000000000000000000000000000000028152600401808373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200182815260200192505050600060405180830381600087803b158015611ca757600080fd5b505af1158015611cbb573d6000803e3d6000fd5b505050505b505050505050565b6210000073ffffffffffffffffffffffffffffffffffffffff808316919091041660009081527f7945af6706678a754539e10e608d9059d561a5482c20d7940d86185bc46d5ee2602052604081205481908015611d23578093505b73ffffffffffffffffffffffffffffffffffffffff841660009081527f7d57c14925af0c486c0ea1d7e0ae331a9fe6597eaa257b28ec87a66c8e7c9da5602052604090205415611dd457604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152600b60248201527f626c61636b6c6973746564000000000000000000000000000000000000000000604482015290519081900360640190fd5b50505073ffffffffffffffffffffffffffffffffffffffff811660009081527ffc858fe381a723c1f1b97e77aaaf4c1bcd9369f995662fe7b31b45d732c937f16020526040902054909190151590565b8181018281101561104257fe5b73ffffffffffffffffffffffffffffffffffffffff82166000908152600e6020526040902054611e67818363ffffffff611e2416565b73ffffffffffffffffffffffffffffffffffffffff9093166000908152600e60205260409020929092555090565b611e9e82612b4a565b6113888282612ca8565b600080611eb5838661237e565b915081841115611ec757506000611ecc565b508281035b611ed78386836123b6565b8473ffffffffffffffffffffffffffffffffffffffff168373ffffffffffffffffffffffffffffffffffffffff167f8c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b925836040518082815260200191505060405180910390a35050505050565b73ffffffffffffffffffffffffffffffffffffffff166000908152600e602052604090205490565b600080621000008473ffffffffffffffffffffffffffffffffffffffff161015611fad57662386f26fc10000830683039250611fa8858585612d94565b610a08565b611fb78585612fca565b92509050611fc585846129a0565b1515611fe857611fd58184611e31565b1515611fe357611fe3612a06565b612008565b611ff28184611e31565b151561200057611fe3612a76565b612008612a06565b8373ffffffffffffffffffffffffffffffffffffffff168573ffffffffffffffffffffffffffffffffffffffff167fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef856040518082815260200191505060405180910390a373ffffffffffffffffffffffffffffffffffffffff818116908516146121b8578073ffffffffffffffffffffffffffffffffffffffff168473ffffffffffffffffffffffffffffffffffffffff167fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef856040518082815260200191505060405180910390a38115611fa8578073ffffffffffffffffffffffffffffffffffffffff16633b66d02b85856040518363ffffffff167c0100000000000000000000000000000000000000000000000000000000028152600401808373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200182815260200192505050600060405180830381600087803b15801561219b57600080fd5b505af11580156121af573d6000803e3d6000fd5b50505050610a08565b8115610a08578073ffffffffffffffffffffffffffffffffffffffff16633b66d02b86856040518363ffffffff167c0100000000000000000000000000000000000000000000000000000000028152600401808373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200182815260200192505050600060405180830381600087803b15801561226157600080fd5b505af1158015612275573d6000803e3d6000fd5b505050505050505050565b73ffffffffffffffffffffffffffffffffffffffff1660009081527f7d57c14925af0c486c0ea1d7e0ae331a9fe6597eaa257b28ec87a66c8e7c9da56020526040902054151590565b73ffffffffffffffffffffffffffffffffffffffff9091166000908152600e6020526040902055565b6000828211156122fe57fe5b50900390565b61230f8184846131d8565b8273ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff167f8c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b925612368848761237e565b60408051918252519081900360200190a3505050565b73ffffffffffffffffffffffffffffffffffffffff9182166000908152600f6020908152604080832093909416825291909152205490565b73ffffffffffffffffffffffffffffffffffffffff9283166000908152600f602090815260408083209490951682529290925291902055565b6123fa818585612656565b505061240583612bfb565b60065482101561247657604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601460248201527f62656c6f77206d696e206275726e20626f756e64000000000000000000000000604482015290519081900360640190fd5b6007548211156124e757604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601660248201527f65786365656473206d6178206275726e20626f756e6400000000000000000000604482015290519081900360640190fd5b6124f184836129a0565b151561251a5761250284828461291a565b151561250d57612515565b612515612b00565b61253b565b61252584828461291a565b151561253357612515612b00565b61253b612a76565b8273ffffffffffffffffffffffffffffffffffffffff168473ffffffffffffffffffffffffffffffffffffffff167fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef846040518082815260200191505060405180910390a36004546125b3908363ffffffff6122f216565b60045560408051838152905173ffffffffffffffffffffffffffffffffffffffff8516917fcc16f5dbb4873280815c1ee09dbd06736cffcc184412cf7a71a0fdb75d397ca5919081900360200190a260408051838152905160009173ffffffffffffffffffffffffffffffffffffffff8616917fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef9181900360200190a350505050565b73ffffffffffffffffffffffffffffffffffffffff831660009081527f7d57c14925af0c486c0ea1d7e0ae331a9fe6597eaa257b28ec87a66c8e7c9da56020526040812054819081901561270b57604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152600b60248201527f626c61636b6c6973746564000000000000000000000000000000000000000000604482015290519081900360640190fd5b506210000073ffffffffffffffffffffffffffffffffffffffff808516919091041660009081527f7945af6706678a754539e10e608d9059d561a5482c20d7940d86185bc46d5ee260205260409020548015612765578093505b73ffffffffffffffffffffffffffffffffffffffff841660009081527f7d57c14925af0c486c0ea1d7e0ae331a9fe6597eaa257b28ec87a66c8e7c9da560205260409020541561281657604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152600b60248201527f626c61636b6c6973746564000000000000000000000000000000000000000000604482015290519081900360640190fd5b73ffffffffffffffffffffffffffffffffffffffff851660009081527f7d57c14925af0c486c0ea1d7e0ae331a9fe6597eaa257b28ec87a66c8e7c9da56020526040902054156128c757604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152600b60248201527f626c61636b6c6973746564000000000000000000000000000000000000000000604482015290519081900360640190fd5b50505073ffffffffffffffffffffffffffffffffffffffff811660009081527ffc858fe381a723c1f1b97e77aaaf4c1bcd9369f995662fe7b31b45d732c937f16020526040902054909390151592509050565b73ffffffffffffffffffffffffffffffffffffffff8084166000908152600f6020908152604080832093861683529290529081205461295f908363ffffffff6122f216565b73ffffffffffffffffffffffffffffffffffffffff9485166000908152600f6020908152604080832096909716825294909452939092208390555090919050565b73ffffffffffffffffffffffffffffffffffffffff82166000908152600e60205260408120546129d6908363ffffffff6122f216565b73ffffffffffffffffffffffffffffffffffffffff9093166000908152600e602052604090208390555090919050565b620fffff546001811115610e0e57620fffff810180543a11156113885760008082557fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff909101557ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe01620fffff55565b7fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff546000811115610e0e57807fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff03600182037fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff5580546000806000806000855af150506000905550565b620fffff546001811115610e0e57620fffff810180543a111561138857600090557fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff01620fffff55565b73ffffffffffffffffffffffffffffffffffffffff811660009081527f7d57c14925af0c486c0ea1d7e0ae331a9fe6597eaa257b28ec87a66c8e7c9da5602052604090205415612bfb57604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152600b60248201527f626c61636b6c6973746564000000000000000000000000000000000000000000604482015290519081900360640190fd5b60106000612c07613259565b81526020808201929092526040908101600090812073ffffffffffffffffffffffffffffffffffffffff851682529092529020541515610e0e57604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601d60248201527f63616e6e6f74206275726e2066726f6d20746869732061646472657373000000604482015290519081900360640190fd5b600654811015612d1957604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601460248201527f62656c6f77206d696e206275726e20626f756e64000000000000000000000000604482015290519081900360640190fd5b600754811115612d8a57604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601660248201527f65786365656473206d6178206275726e20626f756e6400000000000000000000604482015290519081900360640190fd5b611388828261327d565b612d9e8383612fca565b5050612da982612bfb565b600654811015612e1a57604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601460248201527f62656c6f77206d696e206275726e20626f756e64000000000000000000000000604482015290519081900360640190fd5b600754811115612e8b57604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601660248201527f65786365656473206d6178206275726e20626f756e6400000000000000000000604482015290519081900360640190fd5b612e9583826129a0565b1515612ea857612ea3612b00565b612eb0565b612eb0612a06565b8173ffffffffffffffffffffffffffffffffffffffff168373ffffffffffffffffffffffffffffffffffffffff167fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef836040518082815260200191505060405180910390a3600454612f28908263ffffffff6122f216565b60045560408051828152905173ffffffffffffffffffffffffffffffffffffffff8416917fcc16f5dbb4873280815c1ee09dbd06736cffcc184412cf7a71a0fdb75d397ca5919081900360200190a260408051828152905160009173ffffffffffffffffffffffffffffffffffffffff8516917fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef9181900360200190a3505050565b6210000073ffffffffffffffffffffffffffffffffffffffff808316919091041660009081527f7945af6706678a754539e10e608d9059d561a5482c20d7940d86185bc46d5ee2602052604081205481908015613025578093505b73ffffffffffffffffffffffffffffffffffffffff841660009081527f7d57c14925af0c486c0ea1d7e0ae331a9fe6597eaa257b28ec87a66c8e7c9da56020526040902054156130d657604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152600b60248201527f626c61636b6c6973746564000000000000000000000000000000000000000000604482015290519081900360640190fd5b73ffffffffffffffffffffffffffffffffffffffff851660009081527f7d57c14925af0c486c0ea1d7e0ae331a9fe6597eaa257b28ec87a66c8e7c9da560205260409020541561318757604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152600b60248201527f626c61636b6c6973746564000000000000000000000000000000000000000000604482015290519081900360640190fd5b50505073ffffffffffffffffffffffffffffffffffffffff811660009081527ffc858fe381a723c1f1b97e77aaaf4c1bcd9369f995662fe7b31b45d732c937f1602052604090205490929015159150565b73ffffffffffffffffffffffffffffffffffffffff8084166000908152600f602090815260408083209386168352929052205461321b908263ffffffff611e2416565b73ffffffffffffffffffffffffffffffffffffffff9384166000908152600f6020908152604080832095909616825293909352929091209190915550565b7f63616e4275726e4341440000000000000000000000000000000000000000000090565b61328782826129a0565b5060045461329b908263ffffffff6122f216565b60045560408051828152905173ffffffffffffffffffffffffffffffffffffffff8416917fcc16f5dbb4873280815c1ee09dbd06736cffcc184412cf7a71a0fdb75d397ca5919081900360200190a260408051828152905160009173ffffffffffffffffffffffffffffffffffffffff8516917fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef9181900360200190a350505600a165627a7a72305820086144d2d772bce0e6a37ef33b978eee50a2e5025b6c6676df739ece30e7006e0029

Swarm Source

bzzr://086144d2d772bce0e6a37ef33b978eee50a2e5025b6c6676df739ece30e7006e

Block Transaction Difficulty Gas Used Reward
View All Blocks Produced

Block Uncle Number Difficulty Gas Used Reward
View All Uncles
Loading...
Loading
Loading...
Loading

Validator Index Block Amount
View All Withdrawals

Transaction Hash Block Value Eth2 PubKey Valid
View All Deposits
Loading...
Loading
[ 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.