ETH Price: $2,536.02 (+0.49%)

Transaction Decoder

Block:
4285782 at Sep-18-2017 12:49:39 AM +UTC
Transaction Fee:
0.0045 ETH $11.41
Gas Used:
150,000 Gas / 30 Gwei

Account State Difference:

  Address   Before After State Difference Code
(F2Pool Old)
10,035.553221811099279708 Eth10,035.557721811099279708 Eth0.0045
0xbCF0009A...382f9F91D
0.112855534 Eth
Nonce: 0
0.108355534 Eth
Nonce: 1
0.0045

Execution Trace

VEN.approveAndCall( _spender=0xBcB03C4b10F04e568Abf88Df136F579038DD0eF1, _value=2890462100000000000000, _extraData=0x )
  • Exchange.receiveApproval( _from=0xbCF0009A658d925691969e6eBaC870c382f9F91D, _value=2890462100000000000000, 0xD850942eF8811f2A866692A623011bDE52a462C1, 0x )
    • VEN.balanceOf( _owner=0xbCF0009A658d925691969e6eBaC870c382f9F91D ) => ( balance=0 )
      approveAndCall[VEN (ln:274)]
      File 1 of 2: VEN
      pragma solidity ^0.4.11;
      
      contract Owned {
      
          address public owner;
      
          function Owned() {
              owner = msg.sender;
          }
      
          modifier onlyOwner() {
              require(msg.sender == owner);
              _;
          }
      
          function setOwner(address _newOwner) onlyOwner {
              owner = _newOwner;
          }
      }
      
      /**
       * @title SafeMath
       * @dev Math operations with safety checks that throw on error
       */
      library SafeMath {
        function mul(uint256 a, uint256 b) internal constant returns (uint256) {
          uint256 c = a * b;
          assert(a == 0 || c / a == b);
          return c;
        }
      
        function div(uint256 a, uint256 b) internal constant returns (uint256) {
          // assert(b > 0); // Solidity automatically throws when dividing by 0
          uint256 c = a / b;
          // assert(a == b * c + a % b); // There is no case in which this doesn't hold
          return c;
        }
      
        function sub(uint256 a, uint256 b) internal constant returns (uint256) {
          assert(b <= a);
          return a - b;
        }
      
        function add(uint256 a, uint256 b) internal constant returns (uint256) {
          uint256 c = a + b;
          assert(c >= a);
          return c;
        }
      
        function toUINT112(uint256 a) internal constant returns(uint112) {
          assert(uint112(a) == a);
          return uint112(a);
        }
      
        function toUINT120(uint256 a) internal constant returns(uint120) {
          assert(uint120(a) == a);
          return uint120(a);
        }
      
        function toUINT128(uint256 a) internal constant returns(uint128) {
          assert(uint128(a) == a);
          return uint128(a);
        }
      }
      
      
      // Abstract contract for the full ERC 20 Token standard
      // https://github.com/ethereum/EIPs/issues/20
      
      contract Token {
          /* This is a slight change to the ERC20 base standard.
          function totalSupply() constant returns (uint256 supply);
          is replaced with:
          uint256 public totalSupply;
          This automatically creates a getter function for the totalSupply.
          This is moved to the base contract since public getter functions are not
          currently recognised as an implementation of the matching abstract
          function by the compiler.
          */
          /// total amount of tokens
          //uint256 public totalSupply;
          function totalSupply() constant returns (uint256 supply);
      
          /// @param _owner The address from which the balance will be retrieved
          /// @return The balance
          function balanceOf(address _owner) constant returns (uint256 balance);
      
          /// @notice send `_value` token to `_to` from `msg.sender`
          /// @param _to The address of the recipient
          /// @param _value The amount of token to be transferred
          /// @return Whether the transfer was successful or not
          function transfer(address _to, uint256 _value) returns (bool success);
      
          /// @notice send `_value` token to `_to` from `_from` on the condition it is approved by `_from`
          /// @param _from The address of the sender
          /// @param _to The address of the recipient
          /// @param _value The amount of token to be transferred
          /// @return Whether the transfer was successful or not
          function transferFrom(address _from, address _to, uint256 _value) returns (bool success);
      
          /// @notice `msg.sender` approves `_addr` to spend `_value` tokens
          /// @param _spender The address of the account able to transfer the tokens
          /// @param _value The amount of wei to be approved for transfer
          /// @return Whether the approval was successful or not
          function approve(address _spender, uint256 _value) returns (bool success);
      
          /// @param _owner The address of the account owning tokens
          /// @param _spender The address of the account able to transfer the tokens
          /// @return Amount of remaining tokens allowed to spent
          function allowance(address _owner, address _spender) constant returns (uint256 remaining);
      
          event Transfer(address indexed _from, address indexed _to, uint256 _value);
          event Approval(address indexed _owner, address indexed _spender, uint256 _value);
      }
      
      
      /// VEN token, ERC20 compliant
      contract VEN is Token, Owned {
          using SafeMath for uint256;
      
          string public constant name    = "VeChain Token";  //The Token's name
          uint8 public constant decimals = 18;               //Number of decimals of the smallest unit
          string public constant symbol  = "VEN";            //An identifier    
      
          // packed to 256bit to save gas usage.
          struct Supplies {
              // uint128's max value is about 3e38.
              // it's enough to present amount of tokens
              uint128 total;
              uint128 rawTokens;
          }
      
          Supplies supplies;
      
          // Packed to 256bit to save gas usage.    
          struct Account {
              // uint112's max value is about 5e33.
              // it's enough to present amount of tokens
              uint112 balance;
      
              // raw token can be transformed into balance with bonus        
              uint112 rawTokens;
      
              // safe to store timestamp
              uint32 lastMintedTimestamp;
          }
      
          // Balances for each account
          mapping(address => Account) accounts;
      
          // Owner of account approves the transfer of an amount to another account
          mapping(address => mapping(address => uint256)) allowed;
      
          // bonus that can be shared by raw tokens
          uint256 bonusOffered;
      
          // Constructor
          function VEN() {
          }
      
          function totalSupply() constant returns (uint256 supply){
              return supplies.total;
          }
      
          // Send back ether sent to me
          function () {
              revert();
          }
      
          // If sealed, transfer is enabled and mint is disabled
          function isSealed() constant returns (bool) {
              return owner == 0;
          }
      
          function lastMintedTimestamp(address _owner) constant returns(uint32) {
              return accounts[_owner].lastMintedTimestamp;
          }
      
          // Claim bonus by raw tokens
          function claimBonus(address _owner) internal{      
              require(isSealed());
              if (accounts[_owner].rawTokens != 0) {
                  uint256 realBalance = balanceOf(_owner);
                  uint256 bonus = realBalance
                      .sub(accounts[_owner].balance)
                      .sub(accounts[_owner].rawTokens);
      
                  accounts[_owner].balance = realBalance.toUINT112();
                  accounts[_owner].rawTokens = 0;
                  if(bonus > 0){
                      Transfer(this, _owner, bonus);
                  }
              }
          }
      
          // What is the balance of a particular account?
          function balanceOf(address _owner) constant returns (uint256 balance) {
              if (accounts[_owner].rawTokens == 0)
                  return accounts[_owner].balance;
      
              if (bonusOffered > 0) {
                  uint256 bonus = bonusOffered
                       .mul(accounts[_owner].rawTokens)
                       .div(supplies.rawTokens);
      
                  return bonus.add(accounts[_owner].balance)
                          .add(accounts[_owner].rawTokens);
              }
              
              return uint256(accounts[_owner].balance)
                  .add(accounts[_owner].rawTokens);
          }
      
          // Transfer the balance from owner's account to another account
          function transfer(address _to, uint256 _amount) returns (bool success) {
              require(isSealed());
      
              // implicitly claim bonus for both sender and receiver
              claimBonus(msg.sender);
              claimBonus(_to);
      
              // according to VEN's total supply, never overflow here
              if (accounts[msg.sender].balance >= _amount
                  && _amount > 0) {            
                  accounts[msg.sender].balance -= uint112(_amount);
                  accounts[_to].balance = _amount.add(accounts[_to].balance).toUINT112();
                  Transfer(msg.sender, _to, _amount);
                  return true;
              } else {
                  return false;
              }
          }
      
          // Send _value amount of tokens from address _from to address _to
          // The transferFrom method is used for a withdraw workflow, allowing contracts to send
          // tokens on your behalf, for example to "deposit" to a contract address and/or to charge
          // fees in sub-currencies; the command should fail unless the _from account has
          // deliberately authorized the sender of the message via some mechanism; we propose
          // these standardized APIs for approval:
          function transferFrom(
              address _from,
              address _to,
              uint256 _amount
          ) returns (bool success) {
              require(isSealed());
      
              // implicitly claim bonus for both sender and receiver
              claimBonus(_from);
              claimBonus(_to);
      
              // according to VEN's total supply, never overflow here
              if (accounts[_from].balance >= _amount
                  && allowed[_from][msg.sender] >= _amount
                  && _amount > 0) {
                  accounts[_from].balance -= uint112(_amount);
                  allowed[_from][msg.sender] -= _amount;
                  accounts[_to].balance = _amount.add(accounts[_to].balance).toUINT112();
                  Transfer(_from, _to, _amount);
                  return true;
              } else {
                  return false;
              }
          }
      
          // Allow _spender to withdraw from your account, multiple times, up to the _value amount.
          // If this function is called again it overwrites the current allowance with _value.
          function approve(address _spender, uint256 _amount) returns (bool success) {
              allowed[msg.sender][_spender] = _amount;
              Approval(msg.sender, _spender, _amount);
              return true;
          }
      
          /* Approves and then calls the receiving contract */
          function approveAndCall(address _spender, uint256 _value, bytes _extraData) returns (bool success) {
              allowed[msg.sender][_spender] = _value;
              Approval(msg.sender, _spender, _value);
      
              //call the receiveApproval function on the contract you want to be notified. This crafts the function signature manually so one doesn't have to include a contract in here just for this.
              //receiveApproval(address _from, uint256 _value, address _tokenContract, bytes _extraData)
              //it is assumed that when does this that the call *should* succeed, otherwise one would use vanilla approve instead.
              //if(!_spender.call(bytes4(bytes32(sha3("receiveApproval(address,uint256,address,bytes)"))), msg.sender, _value, this, _extraData)) { revert(); }
              ApprovalReceiver(_spender).receiveApproval(msg.sender, _value, this, _extraData);
              return true;
          }
      
          function allowance(address _owner, address _spender) constant returns (uint256 remaining) {
              return allowed[_owner][_spender];
          }
      
          // Mint tokens and assign to some one
          function mint(address _owner, uint256 _amount, bool _isRaw, uint32 timestamp) onlyOwner{
              if (_isRaw) {
                  accounts[_owner].rawTokens = _amount.add(accounts[_owner].rawTokens).toUINT112();
                  supplies.rawTokens = _amount.add(supplies.rawTokens).toUINT128();
              } else {
                  accounts[_owner].balance = _amount.add(accounts[_owner].balance).toUINT112();
              }
      
              accounts[_owner].lastMintedTimestamp = timestamp;
      
              supplies.total = _amount.add(supplies.total).toUINT128();
              Transfer(0, _owner, _amount);
          }
          
          // Offer bonus to raw tokens holder
          function offerBonus(uint256 _bonus) onlyOwner { 
              bonusOffered = bonusOffered.add(_bonus);
              supplies.total = _bonus.add(supplies.total).toUINT128();
              Transfer(0, this, _bonus);
          }
      
          // Set owner to zero address, to disable mint, and enable token transfer
          function seal() onlyOwner {
              setOwner(0);
          }
      }
      
      contract ApprovalReceiver {
          function receiveApproval(address _from, uint256 _value, address _tokenContract, bytes _extraData);
      }
      
      
      // Contract to sell and distribute VEN tokens
      contract VENSale is Owned{
      
          /// chart of stage transition 
          ///
          /// deploy   initialize      startTime                            endTime                 finalize
          ///                              | <-earlyStageLasts-> |             | <- closedStageLasts -> |
          ///  O-----------O---------------O---------------------O-------------O------------------------O------------>
          ///     Created     Initialized           Early             Normal             Closed            Finalized
          enum Stage {
              NotCreated,
              Created,
              Initialized,
              Early,
              Normal,
              Closed,
              Finalized
          }
      
          using SafeMath for uint256;
          
          uint256 public constant totalSupply         = (10 ** 9) * (10 ** 18); // 1 billion VEN, decimals set to 18
      
          uint256 constant privateSupply              = totalSupply * 9 / 100;  // 9% for private ICO
          uint256 constant commercialPlan             = totalSupply * 23 / 100; // 23% for commercial plan
          uint256 constant reservedForTeam            = totalSupply * 5 / 100;  // 5% for team
          uint256 constant reservedForOperations      = totalSupply * 22 / 100; // 22 for operations
      
          // 59%
          uint256 public constant nonPublicSupply     = privateSupply + commercialPlan + reservedForTeam + reservedForOperations;
          // 41%
          uint256 public constant publicSupply = totalSupply - nonPublicSupply;
      
      
          uint256 public constant officialLimit = 64371825 * (10 ** 18);
          uint256 public constant channelsLimit = publicSupply - officialLimit;
      
          // packed to 256bit
          struct SoldOut {
              uint16 placeholder; // placeholder to make struct pre-alloced
      
              // amount of tokens officially sold out.
              // max value of 120bit is about 1e36, it's enough for token amount
              uint120 official; 
      
              uint120 channels; // amount of tokens sold out via channels
          }
      
          SoldOut soldOut;
          
          uint256 constant venPerEth = 3500;  // normal exchange rate
          uint256 constant venPerEthEarlyStage = venPerEth + venPerEth * 15 / 100;  // early stage has 15% reward
      
          uint constant minBuyInterval = 30 minutes; // each account can buy once in 30 minutes
          uint constant maxBuyEthAmount = 30 ether;
         
          VEN ven; // VEN token contract follows ERC20 standard
      
          address ethVault; // the account to keep received ether
          address venVault; // the account to keep non-public offered VEN tokens
      
          uint public constant startTime = 1503057600; // time to start sale
          uint public constant endTime = 1504180800;   // tiem to close sale
          uint public constant earlyStageLasts = 3 days; // early bird stage lasts in seconds
      
          bool initialized;
          bool finalized;
      
          function VENSale() {
              soldOut.placeholder = 1;
          }    
      
          /// @notice calculte exchange rate according to current stage
          /// @return exchange rate. zero if not in sale.
          function exchangeRate() constant returns (uint256){
              if (stage() == Stage.Early) {
                  return venPerEthEarlyStage;
              }
              if (stage() == Stage.Normal) {
                  return venPerEth;
              }
              return 0;
          }
      
          /// @notice for test purpose
          function blockTime() constant returns (uint32) {
              return uint32(block.timestamp);
          }
      
          /// @notice estimate stage
          /// @return current stage
          function stage() constant returns (Stage) { 
              if (finalized) {
                  return Stage.Finalized;
              }
      
              if (!initialized) {
                  // deployed but not initialized
                  return Stage.Created;
              }
      
              if (blockTime() < startTime) {
                  // not started yet
                  return Stage.Initialized;
              }
      
              if (uint256(soldOut.official).add(soldOut.channels) >= publicSupply) {
                  // all sold out
                  return Stage.Closed;
              }
      
              if (blockTime() < endTime) {
                  // in sale            
                  if (blockTime() < startTime.add(earlyStageLasts)) {
                      // early bird stage
                      return Stage.Early;
                  }
                  // normal stage
                  return Stage.Normal;
              }
      
              // closed
              return Stage.Closed;
          }
      
          function isContract(address _addr) constant internal returns(bool) {
              uint size;
              if (_addr == 0) return false;
              assembly {
                  size := extcodesize(_addr)
              }
              return size > 0;
          }
      
          /// @notice entry to buy tokens
          function () payable {        
              buy();
          }
      
          /// @notice entry to buy tokens
          function buy() payable {
              // reject contract buyer to avoid breaking interval limit
              require(!isContract(msg.sender));
              require(msg.value >= 0.01 ether);
      
              uint256 rate = exchangeRate();
              // here don't need to check stage. rate is only valid when in sale
              require(rate > 0);
              // each account is allowed once in minBuyInterval
              require(blockTime() >= ven.lastMintedTimestamp(msg.sender) + minBuyInterval);
      
              uint256 requested;
              // and limited to maxBuyEthAmount
              if (msg.value > maxBuyEthAmount) {
                  requested = maxBuyEthAmount.mul(rate);
              } else {
                  requested = msg.value.mul(rate);
              }
      
              uint256 remained = officialLimit.sub(soldOut.official);
              if (requested > remained) {
                  //exceed remained
                  requested = remained;
              }
      
              uint256 ethCost = requested.div(rate);
              if (requested > 0) {
                  ven.mint(msg.sender, requested, true, blockTime());
                  // transfer ETH to vault
                  ethVault.transfer(ethCost);
      
                  soldOut.official = requested.add(soldOut.official).toUINT120();
                  onSold(msg.sender, requested, ethCost);        
              }
      
              uint256 toReturn = msg.value.sub(ethCost);
              if(toReturn > 0) {
                  // return over payed ETH
                  msg.sender.transfer(toReturn);
              }        
          }
      
          /// @notice returns tokens sold officially
          function officialSold() constant returns (uint256) {
              return soldOut.official;
          }
      
          /// @notice returns tokens sold via channels
          function channelsSold() constant returns (uint256) {
              return soldOut.channels;
          } 
      
          /// @notice manually offer tokens to channel
          function offerToChannel(address _channelAccount, uint256 _venAmount) onlyOwner {
              Stage stg = stage();
              // since the settlement may be delayed, so it's allowed in closed stage
              require(stg == Stage.Early || stg == Stage.Normal || stg == Stage.Closed);
      
              soldOut.channels = _venAmount.add(soldOut.channels).toUINT120();
      
              //should not exceed limit
              require(soldOut.channels <= channelsLimit);
      
              ven.mint(
                  _channelAccount,
                  _venAmount,
                  true,  // unsold tokens can be claimed by channels portion
                  blockTime()
                  );
      
              onSold(_channelAccount, _venAmount, 0);
          }
      
          /// @notice initialize to prepare for sale
          /// @param _ven The address VEN token contract following ERC20 standard
          /// @param _ethVault The place to store received ETH
          /// @param _venVault The place to store non-publicly supplied VEN tokens
          function initialize(
              VEN _ven,
              address _ethVault,
              address _venVault) onlyOwner {
              require(stage() == Stage.Created);
      
              // ownership of token contract should already be this
              require(_ven.owner() == address(this));
      
              require(address(_ethVault) != 0);
              require(address(_venVault) != 0);      
      
              ven = _ven;
              
              ethVault = _ethVault;
              venVault = _venVault;    
              
              ven.mint(
                  venVault,
                  reservedForTeam.add(reservedForOperations),
                  false, // team and operations reserved portion can't share unsold tokens
                  blockTime()
              );
      
              ven.mint(
                  venVault,
                  privateSupply.add(commercialPlan),
                  true, // private ICO and commercial plan can share unsold tokens
                  blockTime()
              );
      
              initialized = true;
              onInitialized();
          }
      
          /// @notice finalize
          function finalize() onlyOwner {
              // only after closed stage
              require(stage() == Stage.Closed);       
      
              uint256 unsold = publicSupply.sub(soldOut.official).sub(soldOut.channels);
      
              if (unsold > 0) {
                  // unsold VEN as bonus
                  ven.offerBonus(unsold);        
              }
              ven.seal();
      
              finalized = true;
              onFinalized();
          }
      
          event onInitialized();
          event onFinalized();
      
          event onSold(address indexed buyer, uint256 venAmount, uint256 ethCost);
      }

      File 2 of 2: Exchange
      pragma solidity ^0.4.11;
      
      contract Owned {
      
          address public owner;
      
          function Owned() {
              owner = msg.sender;
          }
      
          modifier onlyOwner() {
              require(msg.sender == owner);
              _;
          }
      
          function setOwner(address _newOwner) onlyOwner {
              owner = _newOwner;
          }
      }
      
      
      /**
       * @title SafeMath
       * @dev Math operations with safety checks that throw on error
       */
      library SafeMath {
        function mul(uint256 a, uint256 b) internal constant returns (uint256) {
          uint256 c = a * b;
          assert(a == 0 || c / a == b);
          return c;
        }
      
        function div(uint256 a, uint256 b) internal constant returns (uint256) {
          // assert(b > 0); // Solidity automatically throws when dividing by 0
          uint256 c = a / b;
          // assert(a == b * c + a % b); // There is no case in which this doesn't hold
          return c;
        }
      
        function sub(uint256 a, uint256 b) internal constant returns (uint256) {
          assert(b <= a);
          return a - b;
        }
      
        function add(uint256 a, uint256 b) internal constant returns (uint256) {
          uint256 c = a + b;
          assert(c >= a);
          return c;
        }
      
        function toUINT112(uint256 a) internal constant returns(uint112) {
          assert(uint112(a) == a);
          return uint112(a);
        }
      
        function toUINT120(uint256 a) internal constant returns(uint120) {
          assert(uint120(a) == a);
          return uint120(a);
        }
      
        function toUINT128(uint256 a) internal constant returns(uint128) {
          assert(uint128(a) == a);
          return uint128(a);
        }
      }
      
      
      // Abstract contract for the full ERC 20 Token standard
      // https://github.com/ethereum/EIPs/issues/20
      
      contract Token {
          /* This is a slight change to the ERC20 base standard.
          function totalSupply() constant returns (uint256 supply);
          is replaced with:
          uint256 public totalSupply;
          This automatically creates a getter function for the totalSupply.
          This is moved to the base contract since public getter functions are not
          currently recognised as an implementation of the matching abstract
          function by the compiler.
          */
          /// total amount of tokens
          //uint256 public totalSupply;
          function totalSupply() constant returns (uint256 supply);
      
          /// @param _owner The address from which the balance will be retrieved
          /// @return The balance
          function balanceOf(address _owner) constant returns (uint256 balance);
      
          /// @notice send `_value` token to `_to` from `msg.sender`
          /// @param _to The address of the recipient
          /// @param _value The amount of token to be transferred
          /// @return Whether the transfer was successful or not
          function transfer(address _to, uint256 _value) returns (bool success);
      
          /// @notice send `_value` token to `_to` from `_from` on the condition it is approved by `_from`
          /// @param _from The address of the sender
          /// @param _to The address of the recipient
          /// @param _value The amount of token to be transferred
          /// @return Whether the transfer was successful or not
          function transferFrom(address _from, address _to, uint256 _value) returns (bool success);
      
          /// @notice `msg.sender` approves `_addr` to spend `_value` tokens
          /// @param _spender The address of the account able to transfer the tokens
          /// @param _value The amount of wei to be approved for transfer
          /// @return Whether the approval was successful or not
          function approve(address _spender, uint256 _value) returns (bool success);
      
          /// @param _owner The address of the account owning tokens
          /// @param _spender The address of the account able to transfer the tokens
          /// @return Amount of remaining tokens allowed to spent
          function allowance(address _owner, address _spender) constant returns (uint256 remaining);
      
          event Transfer(address indexed _from, address indexed _to, uint256 _value);
          event Approval(address indexed _owner, address indexed _spender, uint256 _value);
      }
      
      
      contract Exchange is Owned {
      
          event onExchangeTokenToEther(address who, uint256 tokenAmount, uint256 etherAmount);
      
          using SafeMath for uint256;
      
          Token public token = Token(0xD850942eF8811f2A866692A623011bDE52a462C1);
      
          // 1 ether = ? tokens
          uint256 public rate = 4025;
      
          // quota of token for every account that can be exchanged to ether
          uint256 public tokenQuota = 402500 ether;
      
          // quota of ether for every account that can be exchanged to token
          // uint256 public etherQuota = 100 ether;
      
          bool public tokenToEtherAllowed = true;
          // bool public etherToTokenAllowed = false;
      
          // uint256 public totalReturnedCredit;             //returned ven  
      
      
          // struct QuotaUsed {
          //     uint128 tokens;
          //     uint128 ethers;
          // }
          mapping(address => uint256) accountQuotaUsed;
      
          function Exchange() {
          }
      
          function () payable {
          }
      
      
          function withdrawEther(address _address,uint256 _amount) onlyOwner {
              require(_address != 0);
              _address.transfer(_amount);
          }
      
          function withdrawToken(address _address, uint256 _amount) onlyOwner {
              require(_address != 0);
              token.transfer(_address, _amount);
          }
      
          function quotaUsed(address _account) constant returns(uint256 ) {
              return accountQuotaUsed[_account];
          }
      
          //tested
          function setRate(uint256 _rate) onlyOwner {
              rate = _rate;
          }
      
          //tested
          function setTokenQuota(uint256 _quota) onlyOwner {
              tokenQuota = _quota;
          }
      
          // function setEtherQuota(uint256 _quota) onlyOwner {
          //     etherQuota = _quota;
          // }
      
          //tested    
          function setTokenToEtherAllowed(bool _allowed) onlyOwner {
              tokenToEtherAllowed = _allowed;
          }
      
          // function setEtherToTokenAllowed(bool _allowed) onlyOwner {
          //     etherToTokenAllowed = _allowed;
          // }
      
          function receiveApproval(address _from, uint256 _value, address /*_tokenContract*/, bytes /*_extraData*/) {
              exchangeTokenToEther(_from, _value);
          }
      
          function exchangeTokenToEther(address _from, uint256 _tokenAmount) internal {
              require(tokenToEtherAllowed);
              require(msg.sender == address(token));
              require(!isContract(_from));
      
              uint256 quota = tokenQuota.sub(accountQuotaUsed[_from]);                
      
              if (_tokenAmount > quota)
                  _tokenAmount = quota;
              
              uint256 balance = token.balanceOf(_from);
              if (_tokenAmount > balance)
                  _tokenAmount = balance;
      
              require(_tokenAmount>0);    //require the token should be above 0
      
              //require(_tokenAmount > 0.01 ether);
              require(token.transferFrom(_from, this, _tokenAmount));        
      
              accountQuotaUsed[_from] = _tokenAmount.add(accountQuotaUsed[_from]);
              
              uint256 etherAmount = _tokenAmount / rate;
              require(etherAmount > 0);
              _from.transfer(etherAmount);
      
              // totalReturnedCredit+=_tokenAmount;
      
              onExchangeTokenToEther(_from, _tokenAmount, etherAmount);
          }
      
      
          //exchange EtherToToken放到fallback函数中
          //TokenToEther
          //    function exchangeEtherToToken() payable {
          //       require(etherToTokenAllowed);
          //        require(!isContract(msg.sender));
          //
          //        uint256 quota = etherQuota.sub(accountQuotaUsed[msg.sender].ethers);
      
          //        uint256 etherAmount = msg.value;
          //        require(etherAmount >= 0.01 ether && etherAmount <= quota);
          //        
          //        uint256 tokenAmount = etherAmount * rate;
      
          //        accountQuotaUsed[msg.sender].ethers = etherAmount.add(accountQuotaUsed[msg.sender].ethers).toUINT128();
      
          //        require(token.transfer(msg.sender, tokenAmount));
      
          //        onExchangeEtherToToken(msg.sender, tokenAmount, etherAmount);                                                        
          //    }
      
          function isContract(address _addr) constant internal returns(bool) {
              uint size;
              if (_addr == 0)
                  return false;
              assembly {
                  size := extcodesize(_addr)
              }
              return size > 0;
          }
      }