账户
0x63...e45f
0x63...e45f

0x63...e45f

$500
此合同的源代码已经过验证!
合同元数据
编译器
0.5.17+commit.d19bba13
语言
Solidity
合同源代码
文件 1 的 1:MatrixErc20.sol
pragma solidity ^0.5.17;


library SafeMath {
    function add(uint256 a, uint256 b) internal pure returns (uint256) {
        uint256 c = a + b;
        require(c >= a, "SafeMath: addition overflow");

        return c;
    }

    function sub(uint256 a, uint256 b) internal pure returns (uint256) {
        return sub(a, b, "SafeMath: subtraction overflow");
    }

    function sub(uint256 a, uint256 b, string memory errorMessage) internal pure returns (uint256) {
        require(b <= a, errorMessage);
        uint256 c = a - b;

        return c;
    }

    function mul(uint256 a, uint256 b) internal pure returns (uint256) {
        // Gas optimization: this is cheaper than requiring 'a' not being zero, but the
        // benefit is lost if 'b' is also tested.
        // See: https://github.com/OpenZeppelin/openzeppelin-contracts/pull/522
        if (a == 0) {
            return 0;
        }

        uint256 c = a * b;
        require(c / a == b, "SafeMath: multiplication overflow");

        return c;
    }

    function div(uint256 a, uint256 b) internal pure returns (uint256) {
        return div(a, b, "SafeMath: division by zero");
    }

    function div(uint256 a, uint256 b, string memory errorMessage) internal pure returns (uint256) {
        // Solidity only automatically asserts when dividing by 0
        require(b > 0, errorMessage);
        uint256 c = a / b;
        // assert(a == b * c + a % b); // There is no case in which this doesn't hold

        return c;
    }

    function mod(uint256 a, uint256 b) internal pure returns (uint256) {
        return mod(a, b, "SafeMath: modulo by zero");
    }

    function mod(uint256 a, uint256 b, string memory errorMessage) internal pure returns (uint256) {
        require(b != 0, errorMessage);
        return a % b;
    }
}

interface IERC20 {
    function transfer(address to, uint256 value) external returns (bool);

    function approve(address spender, uint256 value) external returns (bool);

    function transferFrom(address from, address to, uint256 value) external returns (bool);

    function totalSupply() external view returns (uint256);

    function balanceOf(address who) external view returns (uint256);

    function allowance(address owner, address spender) external view returns (uint256);

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

    event Approval(address indexed owner, address indexed spender, uint256 value);
}

contract MatrixErc20 {
    using SafeMath for uint;
    address public ownerWallet;

    struct UserStruct {
        bool isExist;
        uint id;
        mapping (address => uint256) tokenRewards;
        mapping (address => uint) referrerID; //Token address to Referrer ID
        mapping (address => address[]) referral; //Token address to list of addresses
        mapping(address => mapping(uint => uint)) levelExpired; //Token address to level number to expiration date
    }

    uint REFERRER_1_LEVEL_LIMIT = 5;
    uint PERIOD_LENGTH = 180 days;
    uint ADMIN_FEE_PERCENTAGE = 10;
    mapping(address => mapping(uint => uint)) public LEVEL_PRICE; //Token address to level number to price
    mapping (address => UserStruct) public users;
    mapping (uint => address) public userList;
    uint public currUserID = 0;
    
    mapping (address => bool) public tokens;
    
    mapping (address => uint256) public ownerFees;

    event regLevelEvent(address indexed _user, address indexed _referrer, uint _time, address _token);
    event buyLevelEvent(address indexed _user, uint _level, uint _time, address _token);
    event prolongateLevelEvent(address indexed _user, uint _level, uint _time, address _token);
    event getMoneyForLevelEvent(address indexed _user, address indexed _referral, uint _level, address _token);
    event lostMoneyForLevelEvent(address indexed _user, address indexed _referral, uint _level, uint _time, address _token);
    
    function setPeriodLength(uint _periodLength) public onlyOwner {
        PERIOD_LENGTH = _periodLength;
    }

    function setAdminFeePercentage(uint _adminFeePercentage) public onlyOwner {
        require (_adminFeePercentage >= 0 && _adminFeePercentage <= 100, "Fee must be between 0 and 100");
        
        ADMIN_FEE_PERCENTAGE = _adminFeePercentage;
    }
    
    function toggleToken(address _token, bool _enabled) public onlyOwner {
        tokens[_token] = _enabled;
        
        if (_enabled) {
            for(uint i = 1; i <= 10; i++) {
                users[ownerWallet].levelExpired[_token][i] = 55555555555;
            }
            
            users[ownerWallet].referrerID[_token] = 0;
        }
    }
    
    function setTokenPriceAtLevel(address _token, uint _level, uint _price) public onlyOwner {
        require(_level > 0 && _level <= 10, "Invalid level");

        LEVEL_PRICE[_token][_level] = _price;
    }
    
    function setTokenPrice(address _token, uint _price1, uint _price2, uint _price3, uint _price4, uint _price5, uint _price6, uint _price7, uint _price8, uint _price9, uint _price10) public onlyOwner {
        LEVEL_PRICE[_token][1] = _price1;
        LEVEL_PRICE[_token][2] = _price2;
        LEVEL_PRICE[_token][3] = _price3;
        LEVEL_PRICE[_token][4] = _price4;
        LEVEL_PRICE[_token][5] = _price5;
        LEVEL_PRICE[_token][6] = _price6;
        LEVEL_PRICE[_token][7] = _price7;
        LEVEL_PRICE[_token][8] = _price8;
        LEVEL_PRICE[_token][9] = _price9;
        LEVEL_PRICE[_token][10] = _price10;
        
        if (!tokens[_token]) {
            toggleToken(_token, true);
        }
    }

    constructor() public {
        ownerWallet = msg.sender;

        UserStruct memory userStruct;
        currUserID++;
        
        userStruct = UserStruct({
            isExist: true,
            id: currUserID
        });
        users[ownerWallet] = userStruct;
        userList[currUserID] = ownerWallet;
    }
    
    modifier onlyOwner() {
        require(msg.sender == ownerWallet, 'caller must be the owner');
        _;
    }
    
    //Use this before going live to avoid issues
    function transferOwnership(address newOwner) public onlyOwner {
        require(newOwner != address(0), 'new owner is the zero address');
        require(!users[newOwner].isExist, 'new owner needs to be a new address');
        
        UserStruct memory userStruct = UserStruct({
            isExist: true,
            id: 1
        });
        
        users[newOwner] = userStruct;
        userList[1] = newOwner;

        delete users[ownerWallet];
        ownerWallet = newOwner;
    }

    function regUser(address _token, uint _referrerID) public payable {
        require(tokens[_token], "Token is not enabled");
        require(!users[msg.sender].isExist, 'User exist');
        require(_referrerID > 0 && _referrerID <= currUserID, 'Incorrect referrer Id');
        
        IERC20 token = IERC20(_token);
        require(token.transferFrom(msg.sender, address(this), LEVEL_PRICE[_token][1]), "Couldn't take the tokens from the sender");

        if(users[userList[_referrerID]].referral[_token].length >= REFERRER_1_LEVEL_LIMIT) _referrerID = users[findFreeReferrer(_token, userList[_referrerID])].id;

        UserStruct memory userStruct;
        currUserID++;

        userStruct = UserStruct({
            isExist: true,
            id: currUserID
        });
        
        users[msg.sender] = userStruct;
        users[msg.sender].referrerID[_token] = _referrerID;
        userList[currUserID] = msg.sender;

        users[msg.sender].levelExpired[_token][1] = now + PERIOD_LENGTH;

        users[userList[_referrerID]].referral[_token].push(msg.sender);

        payForLevel(_token, 1, msg.sender);

        emit regLevelEvent(msg.sender, userList[_referrerID], now, _token);
    }

    function buyLevel(address _token, uint _level) public payable {
        require(users[msg.sender].isExist, 'User not exist'); 
        require(_level > 0 && _level <= 10, 'Incorrect level');
        require(tokens[_token], "Token is not enabled");
        
        IERC20 token = IERC20(_token);
        require(token.transferFrom(msg.sender, address(this), LEVEL_PRICE[_token][_level]), "Couldn't take the tokens from the sender");

        if(_level == 1) {
            users[msg.sender].levelExpired[_token][1] += PERIOD_LENGTH;
        }
        else {
            for(uint l =_level - 1; l > 0; l--) require(users[msg.sender].levelExpired[_token][l] >= now, 'Buy the previous level');

            if(users[msg.sender].levelExpired[_token][_level] == 0) users[msg.sender].levelExpired[_token][_level] = now + PERIOD_LENGTH;
            else users[msg.sender].levelExpired[_token][_level] += PERIOD_LENGTH;
        }

        payForLevel(_token, _level, msg.sender);

        emit buyLevelEvent(msg.sender, _level, now, _token);
    }
    
    function getRefererInTree(address _token, uint _level, address _user) internal view returns(address) {
        address referer;
        address referer1;
        address referer2;
        address referer3;
        address referer4;

        if(_level == 1 || _level == 6) {
            referer = userList[users[_user].referrerID[_token]];
        }
        else if(_level == 2 || _level == 7) {
            referer1 = userList[users[_user].referrerID[_token]];
            referer = userList[users[referer1].referrerID[_token]];
        }
        else if(_level == 3 || _level == 8) {
            referer1 = userList[users[_user].referrerID[_token]];
            referer2 = userList[users[referer1].referrerID[_token]];
            referer = userList[users[referer2].referrerID[_token]];
        }
        else if(_level == 4 || _level == 9) {
            referer1 = userList[users[_user].referrerID[_token]];
            referer2 = userList[users[referer1].referrerID[_token]];
            referer3 = userList[users[referer2].referrerID[_token]];
            referer = userList[users[referer3].referrerID[_token]];
        }
        else if(_level == 5 || _level == 10) {
            referer1 = userList[users[_user].referrerID[_token]];
            referer2 = userList[users[referer1].referrerID[_token]];
            referer3 = userList[users[referer2].referrerID[_token]];
            referer4 = userList[users[referer3].referrerID[_token]];
            referer = userList[users[referer4].referrerID[_token]];
        }

        if(!users[referer].isExist) referer = userList[1];
        
        return referer;
    }
    
    function payForLevel(address _token, uint _level, address _user) internal {
        address referer = getRefererInTree(_token, _level, _user);

        if(users[referer].levelExpired[_token][_level] >= now) {
            uint levelPrice = LEVEL_PRICE[_token][_level];
            uint payToOwner = levelPrice.mul(ADMIN_FEE_PERCENTAGE).div(100);
            uint payToReferrer = levelPrice.sub(payToOwner);
            users[address(uint160(referer))].tokenRewards[_token] = users[address(uint160(referer))].tokenRewards[_token].add(payToReferrer);
            ownerFees[_token] = ownerFees[_token].add(payToOwner);

            emit getMoneyForLevelEvent(referer, msg.sender, _level, _token);
        }
        else {
            emit lostMoneyForLevelEvent(referer, msg.sender, _level, now, _token);

            payForLevel(_token, _level, referer);
        }
    }

    function findFreeReferrer(address _token, address _user) public view returns(address) {
        if(users[_user].referral[_token].length < REFERRER_1_LEVEL_LIMIT) return _user;

        address[] memory referrals = new address[](315);
        referrals[0] = users[_user].referral[_token][0];
        referrals[1] = users[_user].referral[_token][1];
        referrals[2] = users[_user].referral[_token][2];
        referrals[3] = users[_user].referral[_token][3];
        referrals[4] = users[_user].referral[_token][4];

        address freeReferrer;
        bool noFreeReferrer = true;

        for(uint i = 0; i < 126; i++) {
            if(users[referrals[i]].referral[_token].length == REFERRER_1_LEVEL_LIMIT) {
                if(i < 155) {
                    referrals[(i+1)*2] = users[referrals[i]].referral[_token][0];
                    referrals[(i+1)*2+1] = users[referrals[i]].referral[_token][2];
                    referrals[(i+1)*2+2] = users[referrals[i]].referral[_token][3];
                    referrals[(i+1)*2+3] = users[referrals[i]].referral[_token][4];
                    referrals[(i+1)*2+4] = users[referrals[i]].referral[_token][5];
                }
            }
            else {
                noFreeReferrer = false;
                freeReferrer = referrals[i];
                break;
            }
        }

        require(!noFreeReferrer, 'No Free Referrer');

        return freeReferrer;
    }
    
    function withdraw(address _token) public {
        uint256 total = users[msg.sender].tokenRewards[_token];
        require(total > 0, "Nothing to withdraw");
        users[msg.sender].tokenRewards[_token] = 0;
        IERC20 token = IERC20(_token);
        require(token.transfer(msg.sender, total), "Couldn't send the tokens");
    }
    
    function _withdrawFees(address _token) public onlyOwner {
        uint256 total = ownerFees[_token];
        require(total > 0, "Nothing to withdraw");
        ownerFees[_token] = 0;
        IERC20 token = IERC20(_token);
        require(token.transfer(msg.sender, total), "Couldn't send the tokens");
    }

    function viewUserReferral(address _token, address _user) public view returns(address[] memory) {
        return users[_user].referral[_token];
    }

    function viewUserReferrer(address _token, address _user) public view returns(uint256) {
        return users[_user].referrerID[_token];
    }

    function viewUserLevelExpired(address _token, address _user, uint _level) public view returns(uint) {
        return users[_user].levelExpired[_token][_level];
    }

    function viewUserIsExist(address _user) public view returns(bool) {
        return users[_user].isExist;
    }

    function viewUserRewards(address _user, address _token) public view returns(uint256) {
        return users[_user].tokenRewards[_token];
    }

    function bytesToAddress(bytes memory bys) private pure returns (address addr) {
        assembly {
            addr := mload(add(bys, 20))
        }
    }

    function _close(address payable _to) public onlyOwner { 
        selfdestruct(_to);  
    }
}
设置
{
  "compilationTarget": {
    "MatrixErc20.sol": "MatrixErc20"
  },
  "evmVersion": "istanbul",
  "libraries": {},
  "optimizer": {
    "enabled": true,
    "runs": 200
  },
  "remappings": []
}
ABI
[{"inputs":[],"payable":false,"stateMutability":"nonpayable","type":"constructor"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"_user","type":"address"},{"indexed":false,"internalType":"uint256","name":"_level","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"_time","type":"uint256"},{"indexed":false,"internalType":"address","name":"_token","type":"address"}],"name":"buyLevelEvent","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"_user","type":"address"},{"indexed":true,"internalType":"address","name":"_referral","type":"address"},{"indexed":false,"internalType":"uint256","name":"_level","type":"uint256"},{"indexed":false,"internalType":"address","name":"_token","type":"address"}],"name":"getMoneyForLevelEvent","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"_user","type":"address"},{"indexed":true,"internalType":"address","name":"_referral","type":"address"},{"indexed":false,"internalType":"uint256","name":"_level","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"_time","type":"uint256"},{"indexed":false,"internalType":"address","name":"_token","type":"address"}],"name":"lostMoneyForLevelEvent","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"_user","type":"address"},{"indexed":false,"internalType":"uint256","name":"_level","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"_time","type":"uint256"},{"indexed":false,"internalType":"address","name":"_token","type":"address"}],"name":"prolongateLevelEvent","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"_user","type":"address"},{"indexed":true,"internalType":"address","name":"_referrer","type":"address"},{"indexed":false,"internalType":"uint256","name":"_time","type":"uint256"},{"indexed":false,"internalType":"address","name":"_token","type":"address"}],"name":"regLevelEvent","type":"event"},{"constant":true,"inputs":[{"internalType":"address","name":"","type":"address"},{"internalType":"uint256","name":"","type":"uint256"}],"name":"LEVEL_PRICE","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":false,"inputs":[{"internalType":"address payable","name":"_to","type":"address"}],"name":"_close","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":false,"inputs":[{"internalType":"address","name":"_token","type":"address"}],"name":"_withdrawFees","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":false,"inputs":[{"internalType":"address","name":"_token","type":"address"},{"internalType":"uint256","name":"_level","type":"uint256"}],"name":"buyLevel","outputs":[],"payable":true,"stateMutability":"payable","type":"function"},{"constant":true,"inputs":[],"name":"currUserID","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":true,"inputs":[{"internalType":"address","name":"_token","type":"address"},{"internalType":"address","name":"_user","type":"address"}],"name":"findFreeReferrer","outputs":[{"internalType":"address","name":"","type":"address"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":true,"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"ownerFees","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":true,"inputs":[],"name":"ownerWallet","outputs":[{"internalType":"address","name":"","type":"address"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":false,"inputs":[{"internalType":"address","name":"_token","type":"address"},{"internalType":"uint256","name":"_referrerID","type":"uint256"}],"name":"regUser","outputs":[],"payable":true,"stateMutability":"payable","type":"function"},{"constant":false,"inputs":[{"internalType":"uint256","name":"_adminFeePercentage","type":"uint256"}],"name":"setAdminFeePercentage","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":false,"inputs":[{"internalType":"uint256","name":"_periodLength","type":"uint256"}],"name":"setPeriodLength","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":false,"inputs":[{"internalType":"address","name":"_token","type":"address"},{"internalType":"uint256","name":"_price1","type":"uint256"},{"internalType":"uint256","name":"_price2","type":"uint256"},{"internalType":"uint256","name":"_price3","type":"uint256"},{"internalType":"uint256","name":"_price4","type":"uint256"},{"internalType":"uint256","name":"_price5","type":"uint256"},{"internalType":"uint256","name":"_price6","type":"uint256"},{"internalType":"uint256","name":"_price7","type":"uint256"},{"internalType":"uint256","name":"_price8","type":"uint256"},{"internalType":"uint256","name":"_price9","type":"uint256"},{"internalType":"uint256","name":"_price10","type":"uint256"}],"name":"setTokenPrice","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":false,"inputs":[{"internalType":"address","name":"_token","type":"address"},{"internalType":"uint256","name":"_level","type":"uint256"},{"internalType":"uint256","name":"_price","type":"uint256"}],"name":"setTokenPriceAtLevel","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":false,"inputs":[{"internalType":"address","name":"_token","type":"address"},{"internalType":"bool","name":"_enabled","type":"bool"}],"name":"toggleToken","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":true,"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"tokens","outputs":[{"internalType":"bool","name":"","type":"bool"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":false,"inputs":[{"internalType":"address","name":"newOwner","type":"address"}],"name":"transferOwnership","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":true,"inputs":[{"internalType":"uint256","name":"","type":"uint256"}],"name":"userList","outputs":[{"internalType":"address","name":"","type":"address"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":true,"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"users","outputs":[{"internalType":"bool","name":"isExist","type":"bool"},{"internalType":"uint256","name":"id","type":"uint256"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":true,"inputs":[{"internalType":"address","name":"_user","type":"address"}],"name":"viewUserIsExist","outputs":[{"internalType":"bool","name":"","type":"bool"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":true,"inputs":[{"internalType":"address","name":"_token","type":"address"},{"internalType":"address","name":"_user","type":"address"},{"internalType":"uint256","name":"_level","type":"uint256"}],"name":"viewUserLevelExpired","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":true,"inputs":[{"internalType":"address","name":"_token","type":"address"},{"internalType":"address","name":"_user","type":"address"}],"name":"viewUserReferral","outputs":[{"internalType":"address[]","name":"","type":"address[]"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":true,"inputs":[{"internalType":"address","name":"_token","type":"address"},{"internalType":"address","name":"_user","type":"address"}],"name":"viewUserReferrer","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":true,"inputs":[{"internalType":"address","name":"_user","type":"address"},{"internalType":"address","name":"_token","type":"address"}],"name":"viewUserRewards","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":false,"inputs":[{"internalType":"address","name":"_token","type":"address"}],"name":"withdraw","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"}]