Skip to content

Instantly share code, notes, and snippets.

@yuriy77k
Created November 7, 2018 13:22
Show Gist options
  • Save yuriy77k/96fb54f0d8da5aa7ffa90e2b84c0b416 to your computer and use it in GitHub Desktop.
Save yuriy77k/96fb54f0d8da5aa7ffa90e2b84c0b416 to your computer and use it in GitHub Desktop.
HyipProfit
pragma solidity ^0.4.24;
library SafeMath {
function mul(uint256 a, uint256 b) internal pure returns (uint256) {
uint256 c = a * b;
assert(a == 0 || c / a == b);
return c;
}
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 c;
}
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) {
uint256 c = a + b;
assert(c >= a);
return c;
}
}
contract ERC20 {
function totalSupply() public constant returns (uint256 supply);
function balanceOf(address who) public constant returns (uint256 value);
function allowance(address owner, address spender) public constant returns (uint256 permitted);
function transfer(address to, uint256 value) public returns (bool ok);
function transferFrom(address from, address to, uint256 value) public returns (bool ok);
function approve(address spender, uint256 value) public returns (bool ok);
event Transfer(address indexed from, address indexed to, uint256 value);
event Approval(address indexed owner, address indexed spender, uint256 value);
}
contract HyipProfit is ERC20 {
using SafeMath for uint256;
string public constant name = "HYIP Profit";
string public constant symbol = "HYIP";
uint8 public constant decimals = 8;
uint256 initialSupply = 450000000000000;
uint256 constant preSaleSoftCap = 31250000000000;
uint256 public preSaleFund = 0;
uint256 public spentFunds = 0;
uint256 public IcoFund = 0;
uint256 public soldTokens = 0; //reduces when somebody returns money
mapping (address => uint256) tokenBalances; //amount of tokens each address holds
mapping (address => uint256) preSaleWeiBalances; //amount of Wei, paid for tokens on preSale. Used only before project completed.
mapping (address => uint256) weiBalances; //amount of Wei, paid for tokens that smb holds. Used only before project completed.
uint256 public currentStage = 0;
tokenAddressGetter tg;
bool public isICOfinalized = false;
address public HyipProfitTokenTeamAddress;
address public utilityTokenAddress = 0x0;
modifier onlyTeam {
if (msg.sender == HyipProfitTokenTeamAddress) {
_;
}
}
mapping (address => mapping (address => uint256)) allowed;
mapping (uint256 => address) teamAddresses;
uint256 currentDividendsRound;
mapping (uint256 => uint256) dividendsPerTokenPerRound;
mapping (uint256 => mapping (address => uint256)) poolBalances;
mapping (address => uint256) lastWithdrawal;
event dividendsReceived (uint256 round, uint256 totalValue, uint256 dividendsPerToken);
event dividendsWithdraw (address tokenHolder, uint256 valueInWei);
event tokensReceived (address from, uint256 tokensAmount);
event tokensWithdrawal (address to, uint256 tokensAmount);
event StageSubmittedAndEtherPassedToTheTeam(uint256 stage, uint256 when, uint256 weiAmount);
event etherWithdrawFromTheContract(address tokenHolder, uint256 numberOfTokensSoldBack, uint256 weiValue);
event Burned(address indexed from, uint256 amount);
// ERC20 interface implementation
function totalSupply() public constant returns (uint256) {
return initialSupply;
}
function balanceOf(address tokenHolder) public view returns (uint256 balance) {
return tokenBalances[tokenHolder];
}
function allowance(address owner, address spender) public constant returns (uint256) {
return allowed[owner][spender];
}
function transfer(address to, uint256 value) public returns (bool success) {
if (tokenBalances[msg.sender] >= value && value > 0) {
if (to == address(this)) {
if (!isICOfinalized) {
returnAllAvailableFunds();
return true;
}
else {
passTokensToTheDividendsPool(value);
return true;
}
}
else {
return transferTokensAndEtherValue(msg.sender, to, value, getHoldersAverageTokenPrice(msg.sender).mul(value) , getUsersPreSalePercentage(msg.sender));
}
} else return false;
}
function transferFrom(address from, address to, uint256 value) public returns (bool success) {
if (tokenBalances[from]>=value && allowed[from][to] >= value && value > 0) {
if (transferTokensAndEtherValue(from, to, value, getHoldersAverageTokenPrice(from).mul(value), getUsersPreSalePercentage(from))){
allowed[from][to] = allowed[from][to].sub(value);
if (from == address(this) && poolBalanceOf(to) >= value) {
if (withdrawDividends(to)) {
poolBalances[currentDividendsRound][to] = poolBalances[currentDividendsRound][to].sub(value);
}
}
return true;
}
return false;
}
return false;
}
function approve(address spender, uint256 value) public returns (bool success) {
if ((value != 0) && (tokenBalances[msg.sender] >= value)){
allowed[msg.sender][spender] = value;
emit Approval (msg.sender, spender, value);
return true;
} else{
return false;
}
}
// Constructor, fallback, return funds
constructor () public {
HyipProfitTokenTeamAddress = msg.sender;
currentDividendsRound = 0;
tokenBalances[address(this)] = initialSupply;
teamAddresses[0] = HyipProfitTokenTeamAddress;
teamAddresses[1] = 0xcC6bCF304d0Ada4Bc7B00Aa1c2c463FBEc263B7e;
teamAddresses[2] = 0x1F16BE21574FA46846fCfeae5ef587c29200f93e;
teamAddresses[3] = 0x93A10f35Bc5439E419fdDcE04Ea44779B0E1017C;
teamAddresses[4] = 0x71bAfdD5bd44D3e1038fE4c0Bc486fb4BB67b806;
}
function () public payable {
if (!isICOfinalized) {
uint256 currentPrice = getCurrentSellPrice();
uint256 valueInWei = 0;
uint256 tokensToPass = 0;
uint256 preSalePercent = 0;
require (msg.value >= currentPrice);
tokensToPass = msg.value.div(currentPrice);
require (tokenBalances[address(this)]>= tokensToPass);
valueInWei = tokensToPass.mul(currentPrice);
soldTokens = soldTokens.add(tokensToPass);
if (currentStage == 0) {
preSaleWeiBalances [address(this)] = preSaleWeiBalances [address(this)].add(valueInWei);
preSalePercent = 100;
preSaleFund = preSaleFund.add(msg.value);
}
else {
weiBalances[address(this)] = weiBalances[address(this)].add(valueInWei);
preSalePercent = 0;
IcoFund = IcoFund.add(msg.value);
}
transferTokensAndEtherValue(address(this), msg.sender, tokensToPass, valueInWei, preSalePercent);
}
else {
require (msg.value >= 10**18);
topUpDividends();
}
}
function returnAllAvailableFunds() public {
require (tokenBalances[msg.sender]>0); //you need to be a tokenHolder
require (!isICOfinalized); //you can not return tokens after project is completed
uint256 preSaleWei = getPreSaleWeiToReturn(msg.sender);
uint256 IcoWei = getIcoWeiToReturn(msg.sender);
uint256 weiToReturn = preSaleWei.add(IcoWei);
uint256 amountOfTokensToReturn = tokenBalances[msg.sender];
require (amountOfTokensToReturn>0);
uint256 preSalePercentage = getUsersPreSalePercentage(msg.sender);
transferTokensAndEtherValue(msg.sender, address(this), amountOfTokensToReturn, weiToReturn, preSalePercentage);
emit etherWithdrawFromTheContract(msg.sender, amountOfTokensToReturn, IcoWei.add(preSaleWei));
preSaleWeiBalances[address(this)] = preSaleWeiBalances[address(this)].sub(preSaleWei);
weiBalances[address(this)] = weiBalances[address(this)].sub(IcoWei);
soldTokens = soldTokens.sub(amountOfTokensToReturn);
msg.sender.transfer(weiToReturn);
preSaleFund = preSaleFund.sub(preSaleWei);
IcoFund = IcoFund.sub(IcoWei);
}
function passTokensToTheDividendsPool(uint256 amount) internal {
if (tokenBalances[msg.sender] >= amount) {
tokenBalances[address(this)] = tokenBalances[address(this)].add(amount);
tokenBalances[msg.sender] = tokenBalances[msg.sender].sub(amount);
emit Transfer(msg.sender, address(this), amount);
allowed[address(this)][msg.sender] = allowed[address(this)][msg.sender].add(amount);
emit Approval (address(this), msg.sender, amount);
if (poolBalanceOf(msg.sender) == 0) lastWithdrawal[msg.sender] = currentDividendsRound;
poolBalances[currentDividendsRound][msg.sender] = poolBalances[currentDividendsRound][msg.sender].add(amount);
emit tokensReceived(msg.sender, amount);
}
}
function topUpDividends() public payable {
require (msg.value >= 10**18);
uint256 dividends = msg.value;
uint256 tokensInPool = balanceOf(address(this));
dividendsPerTokenPerRound[currentDividendsRound] = dividends.div(tokensInPool);
emit dividendsReceived (currentDividendsRound, dividends, dividendsPerTokenPerRound[currentDividendsRound]);
currentDividendsRound = currentDividendsRound.add(1);
}
function withdrawDividends(address holder) public returns (bool success) {
require (poolBalanceOf(holder) > 0);
uint256 totalDividendsForHolder = dividendsOf(holder);
if (totalDividendsForHolder == 0) return true;
uint256 holdersTotalTokensInPool = 0;
for (uint256 i = lastWithdrawal[holder]; i < currentDividendsRound; i = i.add(1)) {
holdersTotalTokensInPool = holdersTotalTokensInPool.add(poolBalances[i][holder]);
poolBalances[i][holder] = 0;
}
holder.transfer(totalDividendsForHolder);
emit dividendsWithdraw (holder, totalDividendsForHolder);
poolBalances[currentDividendsRound][holder] = holdersTotalTokensInPool;
lastWithdrawal[holder] = currentDividendsRound;
return true;
} //AnyBody can call
// View functions
function dividendsOf(address holder) public view returns (uint256 dividendsAmount) {
uint256 dividends = 0;
for (uint256 i = lastWithdrawal[holder]; i < currentDividendsRound; i = i.add(1)) {
for(uint256 j = lastWithdrawal[holder]; j <= i; j = j.add(1)) {
if (poolBalances[j][holder]>0 && dividendsPerTokenPerRound[i] > 0)
dividends = dividends.add(poolBalances[j][holder].mul(dividendsPerTokenPerRound[i]));
}
}
return dividends;
}
function icoFinalized() public view returns (bool) {
return isICOfinalized;
}
function poolBalanceOf(address holder) public view returns (uint256 balance){
uint256 holdersTotalTokensInThePool = 0;
for (uint256 i = lastWithdrawal[msg.sender]; i <= currentDividendsRound; i = i.add(1)) {
holdersTotalTokensInThePool = holdersTotalTokensInThePool.add(poolBalances[i][holder]);
}
return holdersTotalTokensInThePool;
}
function getWeiBalance(address a) public view returns (uint256 weiBalance) {
return weiBalances[a];
}
function getUsersPreSalePercentage(address a) public view returns (uint256 preSaleTokensPercent) {
if (!isICOfinalized && (preSaleWeiBalances[a].add(weiBalances[a]) > 0)) {
uint256 result = (preSaleWeiBalances[a].mul(100)).div((preSaleWeiBalances[a].add(weiBalances[a])));
require (result<=100);
return result;
}
return 0;
}
function getTotalWeiAvailableToReturn(address a) public view returns (uint256 amount) {
return getPreSaleWeiToReturn(a).add(getIcoWeiToReturn(a));
}
function getPreSaleWeiToReturn (address holder) public view returns (uint256 amount) {
if (currentStage == 0) return preSaleWeiBalances[holder];
if (currentStage == 1) return preSaleWeiBalances[holder].mul(7).div(10);
if (currentStage == 2) return preSaleWeiBalances[holder].mul(4).div(10);
return 0;
}
function getIcoWeiToReturn (address holder) public view returns (uint256 amount) {
if (currentStage <= 3) return weiBalances[holder];
if (currentStage == 4) return weiBalances[holder].mul(7).div(10);
if (currentStage == 5) return weiBalances[holder].mul(4).div(10);
return 0;
}
function getHoldersAverageTokenPrice(address holder) public view returns (uint256 avPriceInWei) {
if (!isICOfinalized)
return (weiBalances[holder].add(preSaleWeiBalances[holder])).div(tokenBalances[holder]);
return 0;
}
function getCurrentSellPrice() public view returns (uint256 priceInWei) {
if (isICOfinalized) return 0;
if (currentStage == 0) return 10**6 * 8 ; //this is equal to 0.0008 ETH for 1 token
if (currentStage == 1) return 10**6 * 16;
if (currentStage == 2) return 10**6 * 24;
if (currentStage == 3) return 10**6 * 32;
return 0;
}
function getAvailableFundsForTheTeam() public view returns (uint256 amount) {
if (currentStage == 1) return preSaleFund.mul(3).div(10);
if (currentStage == 2) return (preSaleFund.sub(spentFunds)).div(2);
if (currentStage == 3) return preSaleFund.sub(spentFunds);
if (currentStage == 4) return IcoFund.mul(3).div(10);
if (currentStage == 5) return (IcoFund.sub(spentFunds)).div(2);
if (currentStage == 6) return address(this).balance;
}
function checkIfMissionCompleted() public view returns (bool success) {
if (currentStage == 0 && soldTokens >= preSaleSoftCap) return true;
if (currentStage == 1 && preSaleFund.mul(3).div(5) <= IcoFund) return true;
if (currentStage == 2 && preSaleFund.mul(6).div(5) <= IcoFund) return true;
if (currentStage>=3 &&
(utilityTokenAddress == 0x0 || tg.getBeneficiaryAddress() != address(this))) return false;
if (currentStage == 3 && preSaleFund.mul(2) <= IcoFund) return true;
if (currentStage == 4 && utilityTokenAddress.balance >= IcoFund.mul(3).div(5)) return true;
if (currentStage == 5 && utilityTokenAddress.balance >= IcoFund.mul(6).div(5)) return true;
if (currentStage == 6 && utilityTokenAddress.balance >= IcoFund.mul(2)) return true;
return false;
}
// Team functions
function setUtilityTokenAddressOnce(address a) public onlyTeam {
if (utilityTokenAddress == 0x0) {
utilityTokenAddress = a;
tg = tokenAddressGetter(a);
}
}
function finalizeICO() internal onlyTeam {
require(!isICOfinalized); // this function can be called only once
passTokensToTheTeam();
burnUndistributedTokens(); // undistributed tokens are destroyed
isICOfinalized = true;
}
function passTokensToTheTeam() internal returns (uint256 tokenAmount) { //This function passes tokens to the team without weiValue, so the team can not withdraw ether by returning tokens to the contract
uint256 tokensForEachMember = soldTokens.div(20); // 4% for each team member
uint256 tokensToPass = tokensForEachMember.mul(5);
for (uint256 i = 0; i< 5; i = i.add(1)) {
address teamMember = teamAddresses[i];
tokenBalances[teamMember] = tokenBalances[teamMember].add(tokensForEachMember);
emit Transfer(address(this), teamMember, tokensForEachMember);
}
soldTokens = soldTokens.add(tokensToPass);
return tokensToPass;
}
function submitNextStage() public onlyTeam returns (bool success) {
if (!checkIfMissionCompleted()) return false;
if (currentStage==3) spentFunds = 0;
if (currentStage == 6) finalizeICO();
currentStage = currentStage.add(1);
passEtherToTheTeam();
return true;
}
function passEtherToTheTeam() internal returns (bool success) {
uint256 weiAmount = getAvailableFundsForTheTeam();
HyipProfitTokenTeamAddress.transfer(weiAmount);
spentFunds = spentFunds.add(weiAmount);
emit StageSubmittedAndEtherPassedToTheTeam(currentStage, now, weiAmount);
return true;
}
function transferTokensAndEtherValue(address from, address to, uint256 value, uint256 weiValue, uint256 preSalePercent) internal returns (bool success){
if (tokenBalances[from] >= value) {
tokenBalances[to] = tokenBalances[to].add(value);
tokenBalances[from] = tokenBalances[from].sub(value);
if (!isICOfinalized) {
preSaleWeiBalances[from] = preSaleWeiBalances[from].sub(weiValue.mul(preSalePercent).div(100));
preSaleWeiBalances[to] = preSaleWeiBalances[to].add(weiValue.mul(preSalePercent).div(100));
require (preSalePercent<=100);
weiBalances[from] = weiBalances[from].sub(weiValue.mul(100 - preSalePercent).div(100));
weiBalances[to] = weiBalances[to].add(weiValue.mul(100 - preSalePercent).div(100));
}
emit Transfer(from, to, value);
return true;
}
return false;
}
function burnUndistributedTokens() internal {
uint256 toBurn = initialSupply.sub(soldTokens);
initialSupply = initialSupply.sub(toBurn);
tokenBalances[address(this)] = 0;
emit Burned(address(this), toBurn);
}
}
contract tokenAddressGetter {
function getBeneficiaryAddress() public view returns (address);
}
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment