ETH Price: $1,885.90 (-0.40%)

Transaction Decoder

Block:
6506470 at Oct-13-2018 09:10:44 AM +UTC
Transaction Fee:
0.000368968 ETH $0.70
Gas Used:
46,121 Gas / 8 Gwei

Emitted Events:

Account State Difference:

  Address   Before After State Difference Code
(DwarfPool)
300.592706547200828739 Eth300.593075515200828739 Eth0.000368968
0x2fD0D73F...940Ea922F
0.284781862 Eth
Nonce: 64
0.284412894 Eth
Nonce: 65
0.000368968
0x41bf3907...E1F1B8c51

Execution Trace

BitSongCrowdsale.approveAddress( _beneficiary=0x7B850651Fe39476d5B4511ae12687c33B55f57d8 )
pragma solidity ^0.4.24;

library SafeMath {
  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;
  }

  function div(uint256 _a, uint256 _b) internal pure returns (uint256) {
    return _a / _b;
  }

  function sub(uint256 _a, uint256 _b) internal pure returns (uint256) {
    assert(_b <= _a);
    return _a - _b;
  }

  function add(uint256 _a, uint256 _b) internal pure returns (uint256 c) {
    c = _a + _b;
    assert(c >= _a);
    return c;
  }
}

library SafeERC20 {
  function safeTransfer(ERC20 _token, address _to, uint256 _value) internal {
    require(_token.transfer(_to, _value));
  }

  function safeTransferFrom(ERC20 _token, address _from, address _to, uint256 _value) internal {
    require(_token.transferFrom(_from, _to, _value));
  }
}

interface ERC20 {
  function transferFrom(address from, address to, uint256 value) external returns (bool);
  function transfer(address _to, uint256 _value) external returns (bool);
}

contract Ownable {
  address public owner;

  event OwnershipRenounced(address indexed previousOwner);
  event OwnershipTransferred(address indexed previousOwner, address indexed newOwner);
  modifier onlyOwner() { require(msg.sender == owner); _; }

  constructor() public { owner = msg.sender; }

  function renounceOwnership() public onlyOwner() {
    emit OwnershipRenounced(owner);
    owner = address(0);
  }

  function transferOwnership(address _newOwner) public onlyOwner() {
    _transferOwnership(_newOwner);
  }

  function _transferOwnership(address _newOwner) internal {
    require(_newOwner != address(0));
    emit OwnershipTransferred(owner, _newOwner);
    owner = _newOwner;
  }
}

contract BitSongCrowdsale is Ownable{
  using SafeMath for uint256;
  using SafeERC20 for ERC20;

  ERC20 public token;
  address public wallet;
  uint256 public rate;
  uint256 public weiRaised;
  address public kycAdmin;
  uint256 public hardCap;
  uint256 public tokensAllocated;
  uint256 public openingTime;
  uint256 public closingTime;
  uint256 public duration;

  mapping(address => bool) public approvals;
  mapping(address => uint256) public balances;

  event TokenPurchase(address indexed purchaser, uint256 value, uint256 amount);
  event KycApproved(address indexed beneficiary, address indexed admin, bool status);
  event KycRefused(address indexed beneficiary, address indexed admin, bool status);

  modifier onlyKycAdmin() { require(msg.sender == kycAdmin); _; }
  modifier onlyWhileOpen { require(block.timestamp >= openingTime && block.timestamp <= closingTime); _; }

  constructor(uint256 _rate, address _wallet, uint256 _duration, uint256 _hardCap, ERC20 _tokenAddress) public {
    require(_rate > 0);
    require(_wallet != address(0));
    require(_tokenAddress != address(0));

    rate = _rate;
    wallet = _wallet;
    token = _tokenAddress;
    hardCap = _hardCap * 10**18;
    duration = _duration * 1 days;
  }

  function () external payable {
    buyTokens();
  }

  function buyTokens() public onlyWhileOpen() payable {
    require(msg.value > 0);
    require(approvals[msg.sender] == true);
    uint256 weiAmount = msg.value;
    uint256 tokenAmount = weiAmount.mul(rate);
    tokensAllocated = tokensAllocated.add(tokenAmount);
    assert(tokensAllocated <= hardCap);
    weiRaised = weiRaised.add(weiAmount);
    balances[msg.sender] = balances[msg.sender].add(tokenAmount);
    emit TokenPurchase(msg.sender, weiAmount, tokenAmount);
    wallet.transfer(msg.value);
  }

  function withdrawTokens() external {
    require(hasClosed());
    uint256 amount = balances[msg.sender];
    require(amount > 0);
    balances[msg.sender] = 0;
    token.safeTransferFrom(wallet, msg.sender, amount);
  }

  function withdrawTokensFor(address _beneficiary) external {
    require(hasClosed());
    uint256 amount = balances[_beneficiary];
    require(amount > 0);
    balances[_beneficiary] = 0;
    token.safeTransferFrom(wallet, _beneficiary, amount);
  }

  function hasClosed() public view returns (bool) {
    return block.timestamp > closingTime;
  }

  function approveAddress(address _beneficiary) external onlyKycAdmin() {
    approvals[_beneficiary] = true;
    emit KycApproved(_beneficiary, kycAdmin, true);
  }

  function refuseAddress(address _beneficiary) external onlyKycAdmin() {
    approvals[_beneficiary] = false;
    emit KycRefused(_beneficiary, kycAdmin, false);
  }

  function rewardManual(address _beneficiary, uint256 _amount) external onlyOwner() {
    require(_amount > 0);
    require(_beneficiary != address(0));
    tokensAllocated = tokensAllocated.add(_amount);
    assert(tokensAllocated <= hardCap);
    balances[_beneficiary] = balances[_beneficiary].add(_amount);
  }

  function transfer(address _beneficiary, uint256 _amount) external onlyOwner() {
    require(_amount > 0);
    require(_beneficiary != address(0));
    token.safeTransfer(_beneficiary, _amount);
  }

  function setKycAdmin(address _newAdmin) external onlyOwner() {
    kycAdmin = _newAdmin;
  }

  function startDistribution() external onlyOwner() {
    require(openingTime == 0);
    openingTime = block.timestamp;
    closingTime = openingTime.add(duration);
  }

  function setRate(uint256 _newRate) external onlyOwner() {
    rate = _newRate;
  }

  function setClosingTime(uint256 _newTime) external onlyOwner() {
    closingTime = _newTime;
  }
}