ETH Price: $2,091.93 (+0.67%)

Transaction Decoder

Block:
10401584 at Jul-05-2020 08:39:24 PM +UTC
Transaction Fee:
0.007700049 ETH $16.11
Gas Used:
285,187 Gas / 27 Gwei

Emitted Events:

59 Sync.Upline( addr=[Sender] 0xaa374b59f1feaca484cbee7a616f315efc8d8a81, upline=0xB49c295b...F441ae376 )
60 Sync.NewDeposit( addr=[Sender] 0xaa374b59f1feaca484cbee7a616f315efc8d8a81, amount=500000000000000000 )
61 Sync.DirectPayout( addr=0xB49c295b...F441ae376, from=[Sender] 0xaa374b59f1feaca484cbee7a616f315efc8d8a81, amount=50000000000000000 )

Account State Difference:

  Address   Before After State Difference Code
(Spark Pool)
30.371787121259627565 Eth30.379487170259627565 Eth0.007700049
0xAA374B59...eFc8d8a81
0.6475814 Eth
Nonce: 0
0.139881351 Eth
Nonce: 1
0.507700049
0xFa85069E...6b1E4430A 7,372.307244265753466134 Eth7,372.802244265753466134 Eth0.495
0xfBDe41c7...2dAd5E9CC 29.02289863303016132 Eth29.02789863303016132 Eth0.005

Execution Trace

ETH 0.5 Sync.deposit( _upline=0xB49c295b2665fe0b943760D61aA505fF441ae376 )
  • ETH 0.005 0xfbde41c75d9a504258acd7e1885404f2dad5e9cc.CALL( )
    /*! ether.chain3.sol | (c) 2020 Develop by BelovITLab LLC (smartcontract.ru), author @stupidlovejoy | SPDX-License-Identifier: MIT License */
    
    pragma solidity 0.6.8;
    
    
    /*
     * @dev Provides information about the current execution context, including the
     * sender of the transaction and its data. While these are generally available
     * via msg.sender and msg.data, they should not be accessed in such a direct
     * manner, since when dealing with GSN meta-transactions the account sending and
     * paying for execution may not be the actual sender (as far as an application
     * is concerned).
     *
     * This contract is only required for intermediate, library-like contracts.
     */
    abstract contract Context {
        function _msgSender() internal view virtual returns (address payable) {
            return msg.sender;
        }
    
        function _msgData() internal view virtual returns (bytes memory) {
            this; // silence state mutability warning without generating bytecode - see https://github.com/ethereum/solidity/issues/2691
            return msg.data;
        }
    }
    
    
    /**
     * @dev Contract module which provides a basic access control mechanism, where
     * there is an account (an owner) that can be granted exclusive access to
     * specific functions.
     *
     * By default, the owner account will be the one that deploys the contract. This
     * can later be changed with {transferOwnership}.
     *
     * This module is used through inheritance. It will make available the modifier
     * `onlyOwner`, which can be applied to your functions to restrict their use to
     * the owner.
     */
    contract Ownable is Context {
        address private _owner;
    
        event OwnershipTransferred(address indexed previousOwner, address indexed newOwner);
    
        /**
         * @dev Initializes the contract setting the deployer as the initial owner.
         */
        constructor () internal {
            address msgSender = _msgSender();
            _owner = msgSender;
            emit OwnershipTransferred(address(0), msgSender);
        }
    
        /**
         * @dev Returns the address of the current owner.
         */
        function owner() public view returns (address) {
            return _owner;
        }
    
        /**
         * @dev Throws if called by any account other than the owner.
         */
        modifier onlyOwner() {
            require(_owner == _msgSender(), "Ownable: caller is not the owner");
            _;
        }
    
        /**
         * @dev Leaves the contract without owner. It will not be possible to call
         * `onlyOwner` functions anymore. Can only be called by the current owner.
         *
         * NOTE: Renouncing ownership will leave the contract without an owner,
         * thereby removing any functionality that is only available to the owner.
         */
        function renounceOwnership() public virtual onlyOwner {
            emit OwnershipTransferred(_owner, address(0));
            _owner = address(0);
        }
    
        /**
         * @dev Transfers ownership of the contract to a new account (`newOwner`).
         * Can only be called by the current owner.
         */
        function transferOwnership(address newOwner) public virtual onlyOwner {
            require(newOwner != address(0), "Ownable: new owner is the zero address");
            emit OwnershipTransferred(_owner, newOwner);
            _owner = newOwner;
        }
    }
    
    
    /**
     * @dev Contract module which allows children to implement an emergency stop
     * mechanism that can be triggered by an authorized account.
     *
     * This module is used through inheritance. It will make available the
     * modifiers `whenNotPaused` and `whenPaused`, which can be applied to
     * the functions of your contract. Note that they will not be pausable by
     * simply including this module, only once the modifiers are put in place.
     */
    contract Pausable is Context {
        /**
         * @dev Emitted when the pause is triggered by `account`.
         */
        event Paused(address account);
    
        /**
         * @dev Emitted when the pause is lifted by `account`.
         */
        event Unpaused(address account);
    
        bool private _paused;
    
        /**
         * @dev Initializes the contract in unpaused state.
         */
        constructor () internal {
            _paused = false;
        }
    
        /**
         * @dev Returns true if the contract is paused, and false otherwise.
         */
        function paused() public view returns (bool) {
            return _paused;
        }
    
        /**
         * @dev Modifier to make a function callable only when the contract is not paused.
         *
         * Requirements:
         *
         * - The contract must not be paused.
         */
        modifier whenNotPaused() {
            require(!_paused, "Pausable: paused");
            _;
        }
    
        /**
         * @dev Modifier to make a function callable only when the contract is paused.
         *
         * Requirements:
         *
         * - The contract must be paused.
         */
        modifier whenPaused() {
            require(_paused, "Pausable: not paused");
            _;
        }
    
        /**
         * @dev Triggers stopped state.
         *
         * Requirements:
         *
         * - The contract must not be paused.
         */
        function _pause() internal virtual whenNotPaused {
            _paused = true;
            emit Paused(_msgSender());
        }
    
        /**
         * @dev Returns to normal state.
         *
         * Requirements:
         *
         * - The contract must be paused.
         */
        function _unpause() internal virtual whenPaused {
            _paused = false;
            emit Unpaused(_msgSender());
        }
    }
    
    
    contract Destructible {
        address payable public grand_owner;
    
        event GrandOwnershipTransferred(address indexed previous_owner, address indexed new_owner);
    
        constructor() public {
            grand_owner = msg.sender;
        }
    
        function transferGrandOwnership(address payable _to) external {
            require(msg.sender == grand_owner, "Access denied (only grand owner)");
            
            grand_owner = _to;
        }
    
        function destruct() external {
            require(msg.sender == grand_owner, "Access denied (only grand owner)");
    
            selfdestruct(grand_owner);
        }
    }
    
    contract EtherChain is Ownable, Destructible, Pausable {
        struct User {
            uint256 cycle;
            address upline;
            uint256 referrals;
            uint256 payouts;
            uint256 direct_bonus;
            uint256 pool_bonus;
            uint256 match_bonus;
            uint256 deposit_amount;
            uint256 deposit_payouts;
            uint40 deposit_time;
            uint256 total_deposits;
            uint256 total_payouts;
            uint256 total_structure;
        }
    
        mapping(address => User) public users;
    
        uint256[] public cycles;                        // ether
        uint8[] public ref_bonuses;                     // 1 => 1%
    
        uint8[] public pool_bonuses;                    // 1 => 1%
        uint40 public pool_last_draw = uint40(block.timestamp);
        uint256 public pool_cycle;
        uint256 public pool_balance;
        mapping(uint256 => mapping(address => uint256)) public pool_users_refs_deposits_sum;
        mapping(uint8 => address) public pool_top;
    
        uint256 public total_withdraw;
        
        event Upline(address indexed addr, address indexed upline);
        event NewDeposit(address indexed addr, uint256 amount);
        event DirectPayout(address indexed addr, address indexed from, uint256 amount);
        event MatchPayout(address indexed addr, address indexed from, uint256 amount);
        event PoolPayout(address indexed addr, uint256 amount);
        event Withdraw(address indexed addr, uint256 amount);
        event LimitReached(address indexed addr, uint256 amount);
    
        constructor() public {
            ref_bonuses.push(30);
            ref_bonuses.push(10);
            ref_bonuses.push(10);
            ref_bonuses.push(10);
            ref_bonuses.push(10);
            ref_bonuses.push(8);
            ref_bonuses.push(8);
            ref_bonuses.push(8);
            ref_bonuses.push(8);
            ref_bonuses.push(8);
            ref_bonuses.push(5);
            ref_bonuses.push(5);
            ref_bonuses.push(5);
            ref_bonuses.push(5);
            ref_bonuses.push(5);
    
            pool_bonuses.push(40);
            pool_bonuses.push(30);
            pool_bonuses.push(20);
            pool_bonuses.push(10);
    
            cycles.push(10 ether);
            cycles.push(30 ether);
            cycles.push(90 ether);
            cycles.push(200 ether);
        }
    
        receive() payable external whenNotPaused {
            _deposit(msg.sender, msg.value);
        }
    
        function _setUpline(address _addr, address _upline) private {
            if(users[_addr].upline == address(0) && _upline != _addr && (users[_upline].deposit_time > 0 || _upline == owner())) {
                users[_addr].upline = _upline;
                users[_upline].referrals++;
    
                emit Upline(_addr, _upline);
    
                for(uint8 i = 0; i < ref_bonuses.length; i++) {
                    if(_upline == address(0)) break;
    
                    users[_upline].total_structure++;
    
                    _upline = users[_upline].upline;
                }
            }
        }
    
        function _deposit(address _addr, uint256 _amount) private {
            require(users[_addr].upline != address(0) || _addr == owner(), "No upline");
    
            if(users[_addr].deposit_time > 0) {
                users[_addr].cycle++;
                
                require(users[_addr].payouts >= this.maxPayoutOf(users[_addr].deposit_amount), "Deposit already exists");
                require(_amount >= users[_addr].deposit_amount && _amount <= cycles[users[_addr].cycle > cycles.length - 1 ? cycles.length - 1 : users[_addr].cycle], "Bad amount");
            }
            else require(_amount >= 0.1 ether && _amount <= cycles[0], "Bad amount");
            
            users[_addr].payouts = 0;
            users[_addr].deposit_amount = _amount;
            users[_addr].deposit_payouts = 0;
            users[_addr].deposit_time = uint40(block.timestamp);
            users[_addr].total_deposits += _amount;
            
            emit NewDeposit(_addr, _amount);
    
            if(users[_addr].upline != address(0)) {
                users[users[_addr].upline].direct_bonus += _amount / 10;
    
                emit DirectPayout(users[_addr].upline, _addr, _amount / 10);
            }
    
            _pollDeposits(_addr, _amount);
    
            if(pool_last_draw + 1 days < block.timestamp) {
                _drawPool();
            }
    
            payable(owner()).transfer(_amount / 100);
        }
    
        function _pollDeposits(address _addr, uint256 _amount) private {
            pool_balance += _amount / 20;
    
            address upline = users[_addr].upline;
    
            if(upline == address(0)) return;
            
            pool_users_refs_deposits_sum[pool_cycle][upline] += _amount;
    
            for(uint8 i = 0; i < pool_bonuses.length; i++) {
                if(pool_top[i] == upline) break;
    
                if(pool_top[i] == address(0)) {
                    pool_top[i] = upline;
                    break;
                }
    
                if(pool_users_refs_deposits_sum[pool_cycle][upline] > pool_users_refs_deposits_sum[pool_cycle][pool_top[i]]) {
                    for(uint8 j = i + 1; j < pool_bonuses.length; j++) {
                        if(pool_top[j] == upline) {
                            for(uint8 k = j; k <= pool_bonuses.length; k++) {
                                pool_top[k] = pool_top[k + 1];
                            }
                            break;
                        }
                    }
    
                    for(uint8 j = uint8(pool_bonuses.length - 1); j > i; j--) {
                        pool_top[j] = pool_top[j - 1];
                    }
    
                    pool_top[i] = upline;
    
                    break;
                }
            }
        }
    
        function _refPayout(address _addr, uint256 _amount) private {
            address up = users[_addr].upline;
    
            for(uint8 i = 0; i < ref_bonuses.length; i++) {
                if(up == address(0)) break;
                
                if(users[up].referrals >= i + 1) {
                    uint256 bonus = _amount * ref_bonuses[i] / 100;
                    
                    users[up].match_bonus += bonus;
    
                    emit MatchPayout(up, _addr, bonus);
                }
    
                up = users[up].upline;
            }
        }
    
        function _drawPool() private {
            pool_last_draw = uint40(block.timestamp);
            pool_cycle++;
    
            uint256 draw_amount = pool_balance / 10;
    
            for(uint8 i = 0; i < pool_bonuses.length; i++) {
                if(pool_top[i] == address(0)) break;
    
                uint256 win = draw_amount * pool_bonuses[i] / 100;
    
                users[pool_top[i]].pool_bonus += win;
                pool_balance -= win;
    
                emit PoolPayout(pool_top[i], win);
            }
            
            for(uint8 i = 0; i < pool_bonuses.length; i++) {
                pool_top[i] = address(0);
            }
        }
    
        function deposit(address _upline) payable external whenNotPaused {
            _setUpline(msg.sender, _upline);
            _deposit(msg.sender, msg.value);
        }
    
        function withdraw() external whenNotPaused {
            (uint256 to_payout, uint256 max_payout) = this.payoutOf(msg.sender);
            
            require(users[msg.sender].payouts < max_payout, "Full payouts");
    
            // Deposit payout
            if(to_payout > 0) {
                if(users[msg.sender].payouts + to_payout > max_payout) {
                    to_payout = max_payout - users[msg.sender].payouts;
                }
    
                users[msg.sender].deposit_payouts += to_payout;
                users[msg.sender].payouts += to_payout;
    
                _refPayout(msg.sender, to_payout);
            }
            
            // Direct payout
            if(users[msg.sender].payouts < max_payout && users[msg.sender].direct_bonus > 0) {
                uint256 direct_bonus = users[msg.sender].direct_bonus;
    
                if(users[msg.sender].payouts + direct_bonus > max_payout) {
                    direct_bonus = max_payout - users[msg.sender].payouts;
                }
    
                users[msg.sender].direct_bonus -= direct_bonus;
                users[msg.sender].payouts += direct_bonus;
                to_payout += direct_bonus;
            }
            
            // Pool payout
            if(users[msg.sender].payouts < max_payout && users[msg.sender].pool_bonus > 0) {
                uint256 pool_bonus = users[msg.sender].pool_bonus;
    
                if(users[msg.sender].payouts + pool_bonus > max_payout) {
                    pool_bonus = max_payout - users[msg.sender].payouts;
                }
    
                users[msg.sender].pool_bonus -= pool_bonus;
                users[msg.sender].payouts += pool_bonus;
                to_payout += pool_bonus;
            }
    
            // Match payout
            if(users[msg.sender].payouts < max_payout && users[msg.sender].match_bonus > 0) {
                uint256 match_bonus = users[msg.sender].match_bonus;
    
                if(users[msg.sender].payouts + match_bonus > max_payout) {
                    match_bonus = max_payout - users[msg.sender].payouts;
                }
    
                users[msg.sender].match_bonus -= match_bonus;
                users[msg.sender].payouts += match_bonus;
                to_payout += match_bonus;
            }
    
            require(to_payout > 0, "Zero payout");
            
            users[msg.sender].total_payouts += to_payout;
            total_withdraw += to_payout;
    
            payable(msg.sender).transfer(to_payout);
    
            emit Withdraw(msg.sender, to_payout);
    
            if(users[msg.sender].payouts >= max_payout) {
                emit LimitReached(msg.sender, users[msg.sender].payouts);
            }
        }
        
        function drawPool() external onlyOwner {
            _drawPool();
        }
        
        function pause() external onlyOwner {
            _pause();
        }
        
        function unpause() external onlyOwner {
            _unpause();
        }
    
        function maxPayoutOf(uint256 _amount) pure external returns(uint256) {
            return _amount * 31 / 10;
        }
    
        function payoutOf(address _addr) view external returns(uint256 payout, uint256 max_payout) {
            max_payout = this.maxPayoutOf(users[_addr].deposit_amount);
    
            if(users[_addr].deposit_payouts < max_payout) {
                payout = (users[_addr].deposit_amount * ((block.timestamp - users[_addr].deposit_time) / 1 days) / 100) - users[_addr].deposit_payouts;
                
                if(users[_addr].deposit_payouts + payout > max_payout) {
                    payout = max_payout - users[_addr].deposit_payouts;
                }
            }
        }
    
        /*
            Only external call
        */
        function userInfo(address _addr) view external returns(address upline, uint40 deposit_time, uint256 deposit_amount, uint256 payouts, uint256 direct_bonus, uint256 pool_bonus, uint256 match_bonus) {
            return (users[_addr].upline, users[_addr].deposit_time, users[_addr].deposit_amount, users[_addr].payouts, users[_addr].direct_bonus, users[_addr].pool_bonus, users[_addr].match_bonus);
        }
    
        function userInfoTotals(address _addr) view external returns(uint256 referrals, uint256 total_deposits, uint256 total_payouts, uint256 total_structure) {
            return (users[_addr].referrals, users[_addr].total_deposits, users[_addr].total_payouts, users[_addr].total_structure);
        }
    
        function contractInfo() view external returns(uint256 _total_withdraw, uint40 _pool_last_draw, uint256 _pool_balance, uint256 _pool_lider) {
            return (total_withdraw, pool_last_draw, pool_balance, pool_users_refs_deposits_sum[pool_cycle][pool_top[0]]);
        }
    
        function poolTopInfo() view external returns(address[4] memory addrs, uint256[4] memory deps) {
            for(uint8 i = 0; i < pool_bonuses.length; i++) {
                if(pool_top[i] == address(0)) break;
    
                addrs[i] = pool_top[i];
                deps[i] = pool_users_refs_deposits_sum[pool_cycle][pool_top[i]];
            }
        }
    }
    
    contract Sync is EtherChain {
        bool public sync_close = false;
    
        function sync(address[] calldata _users, address[] calldata _uplines, uint256[] calldata _data) external onlyOwner {
            require(!sync_close, "Sync already close");
    
            for(uint256 i = 0; i < _users.length; i++) {
                address addr = _users[i];
                uint256 q = i * 12;
    
                //require(users[_uplines[i]].total_deposits > 0, "No upline");
    
                if(users[addr].total_deposits == 0) {
                    emit Upline(addr, _uplines[i]);
                }
    
                users[addr].cycle = _data[q];
                users[addr].upline = _uplines[i];
                users[addr].referrals = _data[q + 1];
                users[addr].payouts = _data[q + 2];
                users[addr].direct_bonus = _data[q + 3];
                users[addr].pool_bonus = _data[q + 4];
                users[addr].match_bonus = _data[q + 5];
                users[addr].deposit_amount = _data[q + 6];
                users[addr].deposit_payouts = _data[q + 7];
                users[addr].deposit_time = uint40(_data[q + 8]);
                users[addr].total_deposits = _data[q + 9];
                users[addr].total_payouts = _data[q + 10];
                users[addr].total_structure = _data[q + 11];
            }
        }
    
        function syncGlobal(uint40 _pool_last_draw, uint256 _pool_cycle, uint256 _pool_balance, uint256 _total_withdraw, address[] calldata _pool_top) external onlyOwner {
            require(!sync_close, "Sync already close");
    
            pool_last_draw = _pool_last_draw;
            pool_cycle = _pool_cycle;
            pool_balance = _pool_balance;
            total_withdraw = _total_withdraw;
    
            for(uint8 i = 0; i < pool_bonuses.length; i++) {
                pool_top[i] = _pool_top[i];
            }
        }
    
        function syncUp() external payable {}
    
        function syncClose() external onlyOwner {
            require(!sync_close, "Sync already close");
    
            sync_close = true;
        }
    }