Transaction Hash:
Block:
18907145 at Dec-31-2023 06:08:23 PM +UTC
Transaction Fee:
0.000781576099148259 ETH
$1.99
Gas Used:
46,611 Gas / 16.768061169 Gwei
Emitted Events:
326 |
Dubbz.Approval( owner=[Sender] 0xaa2b4a88424dbda9b4c45995dfb47a2403ab2b5c, spender=0x00000000...43aC78BA3, value=115792089237316195423570985008687907853269984665640564039457584007913129639935 )
|
Account State Difference:
Address | Before | After | State Difference | ||
---|---|---|---|---|---|
0x38029C62...b21DbDa17 | |||||
0x4838B106...B0BAD5f97
Miner
| (Titan Builder) | 18.389577276649642487 Eth | 18.389577742759642487 Eth | 0.00000046611 | |
0xAA2B4a88...403ab2B5C |
0.142698213123576147 Eth
Nonce: 98
|
0.141916637024427888 Eth
Nonce: 99
| 0.000781576099148259 |
Execution Trace
Dubbz.approve( spender=0x000000000022D473030F116dDEE9F6B43aC78BA3, amount=115792089237316195423570985008687907853269984665640564039457584007913129639935 ) => ( True )
// SPDX-License-Identifier: MIT pragma solidity 0.8.17; import '../common/Ownable.sol'; import '../common/ERC20.sol'; import '../common/IDexRouter.sol'; import '../common/TokenHandler.sol'; import '../common/DividendTracker.sol'; import '../common/IDexFactory.sol'; import '../common/ILpPair.sol'; import '../common/IERC20.sol'; import '../common/Context.sol'; import '../common/EnumerableSet.sol'; contract Dubbz is ERC20, Ownable { uint256 public maxBuyAmount; uint256 public maxSellAmount; uint256 public maxWallet; IDexRouter public immutable dexRouter; address public immutable lpPair; address public immutable lpPairEth; bool public lpToEth; IERC20 public immutable STABLECOIN; bool private swapping; uint256 public swapTokensAtAmount; // Anti-bot and anti-whale mappings and variables mapping(address => uint256) private _holderLastTransferTimestamp; // to hold last Transfers temporarily during launch bool public transferDelayEnabled = true; // must be used with Stablecoin TokenHandler private immutable tokenHandler; DividendTracker public immutable dividendTracker; address public projectAddress; address public operationsAddress; address public futureOwnerAddress; bool public projectGetsTokens; uint256 public tradingActiveBlock = 0; // 0 means trading is not active mapping (address => bool) public restrictedWallets; uint256 public blockForPenaltyEnd; bool public limitsInEffect = true; bool public tradingActive = false; bool public swapEnabled = false; uint256 public buyTotalFees; uint256 public buyLiquidityFee; uint256 public buyProjectFee; uint256 public buyOperationsFee; uint256 public sellTotalFees; uint256 public sellProjectFee; uint256 public sellLiquidityFee; uint256 public sellOperationsFee; uint256 constant FEE_DIVISOR = 10000; uint256 public tokensForProject; uint256 public tokensForOperations; uint256 public tokensForLiquidity; mapping (address => bool) private _isExcludedFromFees; mapping (address => bool) public _isExcludedMaxTransactionAmount; mapping (address => bool) public automatedMarketMakerPairs; mapping (address => bool) public addressVerified; address private verificationAddress; bool private verificationRequired; // Events event VestingTokens(address indexed wallet, uint256 amount); event SetAutomatedMarketMakerPair(address indexed pair, bool indexed value); event EnabledTrading(); event RemovedLimits(); event ExcludeFromFees(address indexed account, bool isExcluded); event UpdatedMaxBuyAmount(uint256 newAmount); event UpdatedMaxSellAmount(uint256 newAmount); event UpdatedMaxWalletAmount(uint256 newAmount); event UpdatedBuyFee(uint256 newAmount); event UpdatedSellFee(uint256 newAmount); event UpdatedProjectAddress(address indexed newWallet); event UpdatedLiquidityAddress(address indexed newWallet); event UpdatedOperationsAddress(address indexed newWallet); event MaxTransactionExclusion(address _address, bool excluded); event OwnerForcedSwapBack(uint256 timestamp); event CaughtEarlyBuyer(address sniper); event TransferForeignToken(address token, uint256 amount); event IncludeInDividends(address indexed wallet); event ExcludeFromDividends(address indexed wallet); constructor(bool _lpIsEth) ERC20("Dubbz", "Dubbz") { lpToEth = _lpIsEth; address stablecoinAddress; address _dexRouter; // automatically detect router/desired stablecoin if(block.chainid == 1){ stablecoinAddress = 0xA0b86991c6218b36c1d19D4a2e9Eb0cE3606eB48; // USDC _dexRouter = 0x7a250d5630B4cF539739dF2C5dAcb4c659F2488D; // ETH: Uniswap V2 } else if(block.chainid == 5){ stablecoinAddress = 0x07865c6E87B9F70255377e024ace6630C1Eaa37F; // Görli Testnet USDC _dexRouter = 0x7a250d5630B4cF539739dF2C5dAcb4c659F2488D; // ETH: Uniswap V2 } else if(block.chainid == 56){ stablecoinAddress = 0xe9e7CEA3DedcA5984780Bafc599bD69ADd087D56; // BUSD _dexRouter = 0x10ED43C718714eb63d5aA57B78B54704E256024E; // BNB Chain: PCS V2 } else if(block.chainid == 97){ stablecoinAddress = 0x78867BbEeF44f2326bF8DDd1941a4439382EF2A7; // BSC Testnet BUSD _dexRouter = 0xD99D1c33F9fC3444f8101754aBC46c52416550D1; // BNB Chain: PCS V2 } else if(block.chainid == 137){ stablecoinAddress = 0x2791Bca1f2de4661ED88A30C99A7a9449Aa84174; // USDC _dexRouter = 0xa5E0829CaCEd8fFDD4De3c43696c57F7D7A678ff; // Polygon: Quickswap } else if(block.chainid == 80001){ stablecoinAddress = 0x0FA8781a83E46826621b3BC094Ea2A0212e71B23; // Mumbai USDC _dexRouter = 0xa5E0829CaCEd8fFDD4De3c43696c57F7D7A678ff; // Mumbai Polygon: Quickswap } else { revert("Chain not configured"); } dividendTracker = new DividendTracker(stablecoinAddress); // stablecoin is what is distributed from dividend tracker STABLECOIN = IERC20(stablecoinAddress); require(STABLECOIN.decimals() > 0 , "Incorrect liquidity token"); address newOwner = msg.sender; // can leave alone if owner is deployer. dexRouter = IDexRouter(_dexRouter); // create pair lpPair = IDexFactory(dexRouter.factory()).createPair(address(this), address(STABLECOIN)); setAutomatedMarketMakerPair(address(lpPair), true); lpPairEth = IDexFactory(dexRouter.factory()).createPair(address(this), dexRouter.WETH()); setAutomatedMarketMakerPair(address(lpPairEth), true); uint256 totalSupply = 10 * 1e6 * 1e18; maxBuyAmount = totalSupply * 25 / 10000; // 0.25% maxSellAmount = totalSupply * 25 / 10000; // 0.25% maxWallet = totalSupply * 5 / 1000; // 0.5% swapTokensAtAmount = totalSupply * 1 / 10000; // 0.01% tokenHandler = new TokenHandler(); buyProjectFee = 0; buyLiquidityFee = 400; buyOperationsFee = 300; buyTotalFees = buyProjectFee + buyLiquidityFee + buyOperationsFee; sellProjectFee = 0; sellLiquidityFee = 100; sellOperationsFee = 600; sellTotalFees = sellProjectFee + sellLiquidityFee + sellOperationsFee; // update these! projectAddress = address(newOwner); futureOwnerAddress = address(newOwner); operationsAddress = address(newOwner); _excludeFromMaxTransaction(newOwner, true); _excludeFromMaxTransaction(futureOwnerAddress, true); _excludeFromMaxTransaction(address(this), true); _excludeFromMaxTransaction(address(0xdead), true); _excludeFromMaxTransaction(address(projectAddress), true); _excludeFromMaxTransaction(address(operationsAddress), true); _excludeFromMaxTransaction(address(dexRouter), true); // exclude from receiving dividends dividendTracker.excludeFromDividends(address(dividendTracker)); dividendTracker.excludeFromDividends(address(this)); dividendTracker.excludeFromDividends(newOwner); dividendTracker.excludeFromDividends(address(dexRouter)); dividendTracker.excludeFromDividends(address(0xdead)); dividendTracker.excludeFromDividends(address(futureOwnerAddress)); dividendTracker.excludeFromDividends(lpPair); dividendTracker.excludeFromDividends(lpPairEth); excludeFromFees(newOwner, true); excludeFromFees(futureOwnerAddress, true); excludeFromFees(address(this), true); excludeFromFees(address(0xdead), true); excludeFromFees(address(projectAddress), true); excludeFromFees(address(operationsAddress), true); excludeFromFees(address(dexRouter), true); _createInitialSupply(address(newOwner), totalSupply); transferOwnership(newOwner); } receive() external payable {} // Owner Functions function setVerificationRules(address _verificationAddress, bool _verificationRequired) external onlyOwner { require(_verificationAddress != address(0), "invalid address"); verificationAddress = _verificationAddress; verificationRequired = _verificationRequired; } function updateLpToEth(bool _lpToEth) external onlyOwner { if(_lpToEth){ require(balanceOf(address(lpPairEth))>0, "Must have tokens in ETH pair to set as default LP pair"); } else { require(balanceOf(address(lpPair))>0, "Must have tokens in STABLECOIN pair to set as default LP pair"); } lpToEth = _lpToEth; } function enableTrading(uint256 blocksForPenalty) external onlyOwner { require(tradingActiveBlock == 0, "Trading is already active, cannot relaunch."); require(blocksForPenalty <= 10, "Cannot make penalty blocks more than 10"); tradingActive = true; swapEnabled = true; tradingActiveBlock = block.number; blockForPenaltyEnd = tradingActiveBlock + blocksForPenalty; emit EnabledTrading(); } function pauseTrading() external onlyOwner { require(tradingActiveBlock > 0, "Cannot pause until token has launched"); require(tradingActive, "Trading is already paused"); tradingActive = false; } function unpauseTrading() external onlyOwner { require(tradingActiveBlock > 0, "Cannot unpause until token has launched"); require(!tradingActive, "Trading is already unpaused"); tradingActive = true; } // disable Transfer delay - cannot be reenabled function disableTransferDelay() external onlyOwner { transferDelayEnabled = false; } function manageRestrictedWallets(address[] calldata wallets, bool restricted) external onlyOwner { for(uint256 i = 0; i < wallets.length; i++){ restrictedWallets[wallets[i]] = restricted; } } function removeLimits() external onlyOwner { limitsInEffect = false; maxBuyAmount = totalSupply(); maxSellAmount = totalSupply(); transferDelayEnabled = false; emit RemovedLimits(); } function updateMaxBuyAmount(uint256 newNum) external onlyOwner { require(newNum >= (totalSupply() * 1 / 1000)/1e18, "Cannot set max buy amount lower than 0.1%"); maxBuyAmount = newNum * (10**18); emit UpdatedMaxBuyAmount(maxBuyAmount); } function updateMaxSellAmount(uint256 newNum) external onlyOwner { require(newNum >= (totalSupply() * 1 / 1000)/1e18, "Cannot set max sell amount lower than 0.1%"); maxSellAmount = newNum * (10**18); emit UpdatedMaxSellAmount(maxSellAmount); } function updateMaxWalletAmount(uint256 newNum) external onlyOwner { require(newNum >= (totalSupply() * 1 / 100) / (10 ** decimals()), "Cannot set max sell amount lower than 1%"); maxWallet = newNum * (10 ** decimals()); emit UpdatedMaxWalletAmount(maxWallet); } // change the minimum amount of tokens to sell from fees function updateSwapTokensAtAmount(uint256 newAmount) external onlyOwner { \t require(newAmount >= totalSupply() * 1 / 10000000, "Swap amount cannot be lower than 0.00001% total supply."); \t require(newAmount <= totalSupply() * 1 / 1000, "Swap amount cannot be higher than 0.1% total supply."); \t swapTokensAtAmount = newAmount; \t} function transferForeignToken(address _token, address _to) external onlyOwner returns (bool _sent) { require(_token != address(0), "_token address cannot be 0"); require(_token != address(this) || !tradingActive, "Can't withdraw native tokens while trading is active"); uint256 _contractBalance = IERC20(_token).balanceOf(address(this)); _sent = IERC20(_token).transfer(_to, _contractBalance); emit TransferForeignToken(_token, _contractBalance); } function setProjectAddress(address _projectAddress) external onlyOwner { require(_projectAddress != address(0), "address cannot be 0"); projectAddress = payable(_projectAddress); emit UpdatedProjectAddress(_projectAddress); } function setOperationsAddress(address _operationsAddress) external onlyOwner { require(_operationsAddress != address(0), "address cannot be 0"); operationsAddress = payable(_operationsAddress); emit UpdatedOperationsAddress(_operationsAddress); } function forceSwapBack(bool inEth) external onlyOwner { require(balanceOf(address(this)) >= swapTokensAtAmount, "Can only swap when token amount is at or higher than restriction"); swapping = true; if(inEth){ swapBackEth(); } else { swapBack(); } swapping = false; emit OwnerForcedSwapBack(block.timestamp); } function airdropToWallets(address[] memory wallets, uint256[] memory amountsInTokens) external onlyOwner { require(wallets.length == amountsInTokens.length, "arrays must be the same length"); require(wallets.length < 200, "Can only airdrop 200 wallets per txn due to gas limits"); for(uint256 i = 0; i < wallets.length; i++){ address wallet = wallets[i]; uint256 amount = amountsInTokens[i]; super._transfer(msg.sender, wallet, amount); } } function excludeFromMaxTransaction(address updAds, bool isEx) external onlyOwner { if(!isEx){ require(updAds != lpPair, "Cannot remove uniswap pair from max txn"); } _isExcludedMaxTransactionAmount[updAds] = isEx; } function setAutomatedMarketMakerPair(address pair, bool value) public onlyOwner { require(pair != lpPair || value, "The pair cannot be removed from automatedMarketMakerPairs"); automatedMarketMakerPairs[pair] = value; _excludeFromMaxTransaction(pair, value); emit SetAutomatedMarketMakerPair(pair, value); } function updateBuyFees(uint256 _projectFee, uint256 _liquidityFee, uint256 _operationsFee) external onlyOwner { buyProjectFee = _projectFee; buyLiquidityFee = _liquidityFee; buyOperationsFee = _operationsFee; buyTotalFees = buyProjectFee + buyLiquidityFee + buyOperationsFee; require(buyTotalFees <= 800, "Must keep fees at 8% or less"); emit UpdatedBuyFee(buyTotalFees); } function updateSellFees(uint256 _projectFee, uint256 _liquidityFee, uint256 _operationsFee) external onlyOwner { sellProjectFee = _projectFee; sellLiquidityFee = _liquidityFee; sellOperationsFee = _operationsFee; sellTotalFees = sellProjectFee + sellLiquidityFee + sellOperationsFee; require(sellTotalFees <= 1400, "Must keep fees at 14% or less"); emit UpdatedSellFee(sellTotalFees); } function excludeFromFees(address account, bool excluded) public onlyOwner { _isExcludedFromFees[account] = excluded; emit ExcludeFromFees(account, excluded); } function updateClaimWait(uint256 claimWait) external onlyOwner { dividendTracker.updateClaimWait(claimWait); } function setProjectGetsTokens(bool getsTokens) external onlyOwner { projectGetsTokens = getsTokens; } // excludes wallets and contracts from dividends (such as CEX hotwallets, etc.) function excludeFromDividends(address account) external onlyOwner { dividendTracker.excludeFromDividends(account); emit ExcludeFromDividends(account); } // removes exclusion on wallets and contracts from dividends (such as CEX hotwallets, etc.) function includeInDividends(address account) external onlyOwner { dividendTracker.includeInDividends(account); emit IncludeInDividends(account); } // external functions function claim() external { distributeDividends(); \t\tdividendTracker.processAccount(payable(msg.sender), false); } function distributeDividends() public { dividendTracker.distributeTokenDividends(); } function verificationToBuy(uint8 _v, bytes32 _r, bytes32 _s) public { // anti-bot / snipe method to restrict buyers at launch. require(tradingActive, "trading not active"); bytes memory prefix = "\\x19Ethereum Signed Message:\ 32"; bytes32 hash = keccak256(abi.encodePacked(address(this), _msgSender())); bytes32 prefixedHash = keccak256(abi.encodePacked(prefix, hash)); address signer = ecrecover(prefixedHash, _v, _r, _s); if (signer == verificationAddress) { addressVerified[_msgSender()] = true; } } // private / internal functions function _transfer(address from, address to, uint256 amount) internal override { require(from != address(0), "ERC20: transfer from the zero address"); require(to != address(0), "ERC20: transfer to the zero address"); // transfer of 0 is allowed, but triggers no logic. In case of staking where a staking pool is paying out 0 rewards. if(amount == 0){ super._transfer(from, to, 0); return; } if(_isExcludedFromFees[from] || _isExcludedFromFees[to]){ dividendTracker.distributeTokenDividends(); super._transfer(from, to, amount); dividendTracker.setBalance(payable(from), balanceOf(from)); dividendTracker.setBalance(payable(to), balanceOf(to)); return; } if(!tradingActive){ require(_isExcludedFromFees[from] || _isExcludedFromFees[to], "Trading is not active."); } if(!earlyBuyPenaltyInEffect() && blockForPenaltyEnd > 0){ require(!restrictedWallets[from] || to == owner() || to == address(0xdead), "Bots cannot transfer tokens in or out except to owner or dead address."); } if(limitsInEffect){ if (from != owner() && to != owner() && to != address(0) && to != address(0xdead) && !_isExcludedFromFees[from] && !_isExcludedFromFees[to]){ // at launch if the transfer delay is enabled, ensure the block timestamps for purchasers is set -- during launch. if (transferDelayEnabled){ if (to != address(dexRouter) && to != address(lpPair)){ require(_holderLastTransferTimestamp[tx.origin] + 4 <= block.number, "_transfer:: Transfer Delay enabled. Only one purchase per 3 blocks allowed."); _holderLastTransferTimestamp[tx.origin] = block.number; } } //on buy if (automatedMarketMakerPairs[from] && !_isExcludedMaxTransactionAmount[to]) { if (verificationRequired) { require(addressVerified[to] == true,"Verify you are human first"); } require(amount <= maxBuyAmount, "Buy transfer amount exceeds the max buy."); require(amount + balanceOf(to) <= maxWallet, "Max wallet exceeded"); } //on sell else if (automatedMarketMakerPairs[to] && !_isExcludedMaxTransactionAmount[from]) { require(amount <= maxSellAmount, "Sell transfer amount exceeds the max sell."); } else if (!_isExcludedMaxTransactionAmount[to]){ require(amount + balanceOf(to) <= maxWallet, "Max wallet exceeded"); } } } uint256 contractTokenBalance = balanceOf(address(this)); bool canSwap = contractTokenBalance >= swapTokensAtAmount; if(canSwap && swapEnabled && !swapping && automatedMarketMakerPairs[to]) { swapping = true; if(lpToEth){ swapBackEth(); } else { swapBack(); } swapping = false; } bool takeFee = true; // if any account belongs to _isExcludedFromFee account then remove the fee if(_isExcludedFromFees[from] || _isExcludedFromFees[to]) { takeFee = false; } uint256 fees = 0; // only take fees on buys/sells, do not take on wallet transfers if(takeFee){ // bot/sniper penalty. if(earlyBuyPenaltyInEffect() && automatedMarketMakerPairs[from] && !automatedMarketMakerPairs[to] && buyTotalFees > 0){ if(!restrictedWallets[to]){ restrictedWallets[to] = true; } fees = amount * buyTotalFees / FEE_DIVISOR; \t tokensForProject += fees * buyProjectFee / buyTotalFees; \t tokensForLiquidity += fees * buyLiquidityFee / buyTotalFees; tokensForOperations += fees * buyOperationsFee / buyTotalFees; } // on sell else if (automatedMarketMakerPairs[to] && sellTotalFees > 0){ fees = amount * sellTotalFees / FEE_DIVISOR; tokensForLiquidity += fees * sellLiquidityFee / sellTotalFees; tokensForProject += fees * sellProjectFee / sellTotalFees; tokensForOperations += fees * sellOperationsFee / sellTotalFees; } // on buy else if(automatedMarketMakerPairs[from] && buyTotalFees > 0) { \t fees = amount * buyTotalFees / FEE_DIVISOR; \t tokensForProject += fees * buyProjectFee / buyTotalFees; \t tokensForLiquidity += fees * buyLiquidityFee / buyTotalFees; tokensForOperations += fees * buyOperationsFee / buyTotalFees; } if(fees > 0){ super._transfer(from, address(this), fees); } \t \tamount -= fees; } dividendTracker.distributeTokenDividends(); super._transfer(from, to, amount); dividendTracker.setBalance(payable(from), balanceOf(from)); dividendTracker.setBalance(payable(to), balanceOf(to)); } // if LP pair in use is STABLECOIN, this function will be used to handle fee distribution. function swapTokensForSTABLECOIN(uint256 tokenAmount) private { address[] memory path = new address[](2); path[0] = address(this); path[1] = address(STABLECOIN); _approve(address(this), address(dexRouter), tokenAmount); dexRouter.swapExactTokensForTokensSupportingFeeOnTransferTokens(tokenAmount, 0, path, address(tokenHandler), block.timestamp); } function swapBack() private { if(projectGetsTokens && tokensForProject > 0 && balanceOf(address(this)) >= tokensForProject) { super._transfer(address(this), address(projectAddress), tokensForProject); tokensForProject = 0; } uint256 contractBalance = balanceOf(address(this)); uint256 totalTokensToSwap = tokensForProject + tokensForOperations + tokensForLiquidity; if(contractBalance == 0 || totalTokensToSwap == 0) {return;} if(contractBalance > swapTokensAtAmount * 60){ contractBalance = swapTokensAtAmount * 60; } if(tokensForLiquidity > 0){ uint256 liquidityTokens = contractBalance * tokensForLiquidity / totalTokensToSwap; super._transfer(address(this), lpPair, liquidityTokens); try ILpPair(lpPair).sync(){} catch {} contractBalance -= liquidityTokens; totalTokensToSwap -= tokensForLiquidity; } swapTokensForSTABLECOIN(contractBalance); tokenHandler.sendTokenToOwner(address(STABLECOIN)); uint256 stablecoinBalance = STABLECOIN.balanceOf(address(this)); uint256 stablecoinForOperations = stablecoinBalance * tokensForOperations / totalTokensToSwap; tokensForProject = 0; tokensForOperations = 0; tokensForLiquidity = 0; if(stablecoinForOperations > 0){ STABLECOIN.transfer(operationsAddress, stablecoinForOperations); } if(STABLECOIN.balanceOf(address(this)) > 0){ STABLECOIN.transfer(projectAddress, STABLECOIN.balanceOf(address(this))); } } // if LP pair in use is ETH, this function will be used to handle fee distribution. function swapBackEth() private { if(projectGetsTokens && tokensForProject > 0 && balanceOf(address(this)) >= tokensForProject) { super._transfer(address(this), address(projectAddress), tokensForProject); tokensForProject = 0; } bool success; uint256 contractBalance = balanceOf(address(this)); uint256 totalTokensToSwap = tokensForProject + tokensForOperations + tokensForLiquidity; if(contractBalance == 0 || totalTokensToSwap == 0) {return;} if(contractBalance > swapTokensAtAmount * 60){ contractBalance = swapTokensAtAmount * 60; } if(tokensForLiquidity > 0){ uint256 liquidityTokens = contractBalance * tokensForLiquidity / totalTokensToSwap; super._transfer(address(this), lpPairEth, liquidityTokens); try ILpPair(lpPairEth).sync(){} catch {} contractBalance -= liquidityTokens; totalTokensToSwap -= tokensForLiquidity; } swapTokensForEth(contractBalance); uint256 ethBalance = address(this).balance; uint256 ethForOperations = ethBalance * tokensForOperations / totalTokensToSwap; tokensForProject = 0; tokensForOperations = 0; tokensForLiquidity = 0; if(ethForOperations > 0){ (success, ) = operationsAddress.call{value: ethForOperations}(""); } if(address(this).balance > 0){ (success, ) = projectAddress.call{value: address(this).balance}(""); } } function swapTokensForEth(uint256 tokenAmount) private { // generate the uniswap pair path of token -> weth address[] memory path = new address[](2); path[0] = address(this); path[1] = dexRouter.WETH(); _approve(address(this), address(dexRouter), tokenAmount); // make the swap dexRouter.swapExactTokensForETHSupportingFeeOnTransferTokens(tokenAmount, 0, path, address(this), block.timestamp); } function _excludeFromMaxTransaction(address updAds, bool isExcluded) private { _isExcludedMaxTransactionAmount[updAds] = isExcluded; emit MaxTransactionExclusion(updAds, isExcluded); } //views function earlyBuyPenaltyInEffect() private view returns (bool){ return block.number < blockForPenaltyEnd; } function getBlockNumber() external view returns (uint256){ return block.number; } function getClaimWait() external view returns(uint256) { return dividendTracker.claimWait(); } function getTotalDividendsDistributed() external view returns (uint256) { return dividendTracker.totalDividendsDistributed(); } function isExcludedFromFees(address account) public view returns(bool) { return _isExcludedFromFees[account]; } function withdrawableDividendOf(address account) public view returns(uint256) { \treturn dividendTracker.withdrawableDividendOf(account); \t} \tfunction dividendTokenBalanceOf(address account) public view returns (uint256) { \t\treturn dividendTracker.holderBalance(account); \t} function getAccountDividendsInfo(address account) external view returns ( address, int256, int256, uint256, uint256, uint256, uint256, uint256) { return dividendTracker.getAccount(account); } function getLastProcessedIndex() external view returns(uint256) { \treturn dividendTracker.getLastProcessedIndex(); } function getNumberOfDividendTokenHolders() external view returns(uint256) { return dividendTracker.getNumberOfTokenHolders(); } function getNumberOfDividends() external view returns(uint256) { return dividendTracker.totalBalance(); } }// SPDX-License-Identifier: MIT pragma solidity 0.8.17; import './Ownable.sol'; import './IERC20.sol'; contract TokenHandler is Ownable { function sendTokenToOwner(address token) external onlyOwner { if(IERC20(token).balanceOf(address(this)) > 0){ IERC20(token).transfer(owner(), IERC20(token).balanceOf(address(this))); } } } // SPDX-License-Identifier: MIT pragma solidity 0.8.17; library SafeMathUint { function toInt256Safe(uint256 a) internal pure returns (int256) { int256 b = int256(a); require(b >= 0); return b; } }// SPDX-License-Identifier: MIT pragma solidity 0.8.17; library SafeMathInt { int256 private constant MIN_INT256 = int256(1) << 255; int256 private constant MAX_INT256 = ~(int256(1) << 255); /** * @dev Multiplies two int256 variables and fails on overflow. */ function mul(int256 a, int256 b) internal pure returns (int256) { int256 c = a * b; // Detect overflow when multiplying MIN_INT256 with -1 require(c != MIN_INT256 || (a & MIN_INT256) != (b & MIN_INT256)); require((b == 0) || (c / b == a)); return c; } /** * @dev Division of two int256 variables and fails on overflow. */ function div(int256 a, int256 b) internal pure returns (int256) { // Prevent overflow when dividing MIN_INT256 by -1 require(b != -1 || a != MIN_INT256); // Solidity already throws when dividing by 0. return a / b; } /** * @dev Subtracts two int256 variables and fails on overflow. */ function sub(int256 a, int256 b) internal pure returns (int256) { int256 c = a - b; require((b >= 0 && c <= a) || (b < 0 && c > a)); return c; } /** * @dev Adds two int256 variables and fails on overflow. */ function add(int256 a, int256 b) internal pure returns (int256) { int256 c = a + b; require((b >= 0 && c >= a) || (b < 0 && c < a)); return c; } /** * @dev Converts to absolute value, and fails on overflow. */ function abs(int256 a) internal pure returns (int256) { require(a != MIN_INT256); return a < 0 ? -a : a; } function toUint256Safe(int256 a) internal pure returns (uint256) { require(a >= 0); return uint256(a); } }// SPDX-License-Identifier: MIT pragma solidity 0.8.17; library SafeMath { /** * @dev Returns the addition of two unsigned integers, reverting on * overflow. * * Counterpart to Solidity's `+` operator. * * Requirements: * * - Addition cannot overflow. */ function add(uint256 a, uint256 b) internal pure returns (uint256) { uint256 c = a + b; require(c >= a, "SafeMath: addition overflow"); return c; } /** * @dev Returns the subtraction of two unsigned integers, reverting on * overflow (when the result is negative). * * Counterpart to Solidity's `-` operator. * * Requirements: * * - Subtraction cannot overflow. */ function sub(uint256 a, uint256 b) internal pure returns (uint256) { return sub(a, b, "SafeMath: subtraction overflow"); } /** * @dev Returns the subtraction of two unsigned integers, reverting with custom message on * overflow (when the result is negative). * * Counterpart to Solidity's `-` operator. * * Requirements: * * - Subtraction cannot overflow. */ function sub(uint256 a, uint256 b, string memory errorMessage) internal pure returns (uint256) { require(b <= a, errorMessage); uint256 c = a - b; return c; } /** * @dev Returns the multiplication of two unsigned integers, reverting on * overflow. * * Counterpart to Solidity's `*` operator. * * Requirements: * * - Multiplication cannot overflow. */ function mul(uint256 a, uint256 b) internal pure returns (uint256) { // Gas optimization: this is cheadar 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; } /** * @dev Returns the integer division of two unsigned integers. Reverts on * division by zero. The result is rounded towards zero. * * Counterpart to Solidity's `/` operator. Note: this function uses a * `revert` opcode (which leaves remaining gas untouched) while Solidity * uses an invalid opcode to revert (consuming all remaining gas). * * Requirements: * * - The divisor cannot be zero. */ function div(uint256 a, uint256 b) internal pure returns (uint256) { return div(a, b, "SafeMath: division by zero"); } /** * @dev Returns the integer division of two unsigned integers. Reverts with custom message on * division by zero. The result is rounded towards zero. * * Counterpart to Solidity's `/` operator. Note: this function uses a * `revert` opcode (which leaves remaining gas untouched) while Solidity * uses an invalid opcode to revert (consuming all remaining gas). * * Requirements: * * - The divisor cannot be zero. */ function div(uint256 a, uint256 b, string memory errorMessage) internal pure returns (uint256) { 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; } /** * @dev Returns the remainder of dividing two unsigned integers. (unsigned integer modulo), * Reverts when dividing by zero. * * Counterpart to Solidity's `%` operator. This function uses a `revert` * opcode (which leaves remaining gas untouched) while Solidity uses an * invalid opcode to revert (consuming all remaining gas). * * Requirements: * * - The divisor cannot be zero. */ function mod(uint256 a, uint256 b) internal pure returns (uint256) { return mod(a, b, "SafeMath: modulo by zero"); } /** * @dev Returns the remainder of dividing two unsigned integers. (unsigned integer modulo), * Reverts with custom message when dividing by zero. * * Counterpart to Solidity's `%` operator. This function uses a `revert` * opcode (which leaves remaining gas untouched) while Solidity uses an * invalid opcode to revert (consuming all remaining gas). * * Requirements: * * - The divisor cannot be zero. */ function mod(uint256 a, uint256 b, string memory errorMessage) internal pure returns (uint256) { require(b != 0, errorMessage); return a % b; } }// SPDX-License-Identifier: MIT pragma solidity 0.8.17; import './Context.sol'; contract Ownable is Context { address private _owner; event OwnershipTransferred(address indexed previousOwner, address indexed newOwner); constructor () { address msgSender = _msgSender(); _owner = msgSender; emit OwnershipTransferred(address(0), msgSender); } function owner() public view returns (address) { return _owner; } modifier onlyOwner() { require(_owner == _msgSender(), "Ownable: caller is not the owner"); _; } function renounceOwnership() external virtual onlyOwner { emit OwnershipTransferred(_owner, address(0)); _owner = address(0); } function transferOwnership(address newOwner) public virtual onlyOwner { require(newOwner != address(0), "Ownable: new owner is the zero address"); emit OwnershipTransferred(_owner, newOwner); _owner = newOwner; } }// SPDX-License-Identifier: MIT pragma solidity 0.8.17; interface ILpPair { function sync() external; }// SPDX-License-Identifier: MIT pragma solidity 0.8.17; interface IERC20 { function totalSupply() external view returns (uint256); function balanceOf(address account) external view returns (uint256); function transfer(address recipient, uint256 amount) external returns (bool); function allowance(address owner, address spender) external view returns (uint256); function approve(address spender, uint256 amount) external returns (bool); function transferFrom(address sender, address recipient, uint256 amount) external returns (bool); function name() external view returns (string memory); function symbol() external view returns (string memory); function decimals() external view returns (uint8); event Transfer(address indexed from, address indexed to, uint256 value); event Approval(address indexed owner, address indexed spender, uint256 value); } // SPDX-License-Identifier: MIT pragma solidity 0.8.17; interface IDexRouter { function factory() external pure returns (address); function WETH() external pure returns (address); function swapExactTokensForETHSupportingFeeOnTransferTokens(uint amountIn, uint amountOutMin, address[] calldata path, address to, uint deadline) external; function swapExactETHForTokensSupportingFeeOnTransferTokens(uint amountOutMin, address[] calldata path, address to, uint deadline) external payable; function swapExactTokensForTokensSupportingFeeOnTransferTokens(uint amountIn, uint amountOutMin, address[] calldata path, address to, uint deadline) external; function addLiquidityETH(address token, uint256 amountTokenDesired, uint256 amountTokenMin, uint256 amountETHMin, address to, uint256 deadline) external payable returns (uint256 amountToken, uint256 amountETH, uint256 liquidity); function addLiquidity(address tokenA, address tokenB, uint amountADesired, uint amountBDesired, uint amountAMin, uint amountBMin, address to, uint deadline) external returns (uint amountA, uint amountB, uint liquidity); function getAmountsOut(uint amountIn, address[] calldata path) external view returns (uint[] memory amounts); } // SPDX-License-Identifier: MIT pragma solidity 0.8.17; interface IDexFactory { function createPair(address tokenA, address tokenB) external returns (address pair); } // SPDX-License-Identifier: MIT pragma solidity 0.8.17; library EnumerableSet { struct Set { bytes32[] _values; mapping(bytes32 => uint256) _indexes; } function _add(Set storage set, bytes32 value) private returns (bool) { if (!_contains(set, value)) { set._values.push(value); set._indexes[value] = set._values.length; return true; } else { return false; } } function _remove(Set storage set, bytes32 value) private returns (bool) { uint256 valueIndex = set._indexes[value]; if (valueIndex != 0) { uint256 toDeleteIndex = valueIndex - 1; uint256 lastIndex = set._values.length - 1; if (lastIndex != toDeleteIndex) { bytes32 lastValue = set._values[lastIndex]; set._values[toDeleteIndex] = lastValue; set._indexes[lastValue] = valueIndex; } set._values.pop(); delete set._indexes[value]; return true; } else { return false; } } function _contains(Set storage set, bytes32 value) private view returns (bool) { return set._indexes[value] != 0; } function _length(Set storage set) private view returns (uint256) { return set._values.length; } function _at(Set storage set, uint256 index) private view returns (bytes32) { return set._values[index]; } function _values(Set storage set) private view returns (bytes32[] memory) { return set._values; } // AddressSet struct AddressSet { Set _inner; } function add(AddressSet storage set, address value) internal returns (bool) { return _add(set._inner, bytes32(uint256(uint160(value)))); } function remove(AddressSet storage set, address value) internal returns (bool) { return _remove(set._inner, bytes32(uint256(uint160(value)))); } /** * @dev Returns true if the value is in the set. O(1). */ function contains(AddressSet storage set, address value) internal view returns (bool) { return _contains(set._inner, bytes32(uint256(uint160(value)))); } /** * @dev Returns the number of values in the set. O(1). */ function length(AddressSet storage set) internal view returns (uint256) { return _length(set._inner); } function at(AddressSet storage set, uint256 index) internal view returns (address) { return address(uint160(uint256(_at(set._inner, index)))); } function values(AddressSet storage set) internal view returns (address[] memory) { bytes32[] memory store = _values(set._inner); address[] memory result; /// @solidity memory-safe-assembly assembly { result := store } return result; } }// SPDX-License-Identifier: MIT pragma solidity 0.8.17; import './IERC20.sol'; import './Context.sol'; contract ERC20 is Context, IERC20 { mapping(address => uint256) private _balances; mapping(address => mapping(address => uint256)) private _allowances; uint256 private _totalSupply; string private _name; string private _symbol; constructor(string memory name_, string memory symbol_) { _name = name_; _symbol = symbol_; } function name() public view virtual override returns (string memory) { return _name; } function symbol() public view virtual override returns (string memory) { return _symbol; } function decimals() public view virtual override returns (uint8) { return 18; } function totalSupply() public view virtual override returns (uint256) { return _totalSupply; } function balanceOf(address account) public view virtual override returns (uint256) { return _balances[account]; } function transfer(address recipient, uint256 amount) public virtual override returns (bool) { _transfer(_msgSender(), recipient, amount); return true; } function allowance(address owner, address spender) public view virtual override returns (uint256) { return _allowances[owner][spender]; } function approve(address spender, uint256 amount) public virtual override returns (bool) { _approve(_msgSender(), spender, amount); return true; } function transferFrom(address sender, address recipient, uint256 amount) public virtual override returns (bool) { _transfer(sender, recipient, amount); uint256 currentAllowance = _allowances[sender][_msgSender()]; require(currentAllowance >= amount, "ERC20: transfer amount exceeds allowance"); unchecked { _approve(sender, _msgSender(), currentAllowance - amount); } return true; } function increaseAllowance(address spender, uint256 addedValue) public virtual returns (bool) { _approve(_msgSender(), spender, _allowances[_msgSender()][spender] + addedValue); return true; } function decreaseAllowance(address spender, uint256 subtractedValue) public virtual returns (bool) { uint256 currentAllowance = _allowances[_msgSender()][spender]; require(currentAllowance >= subtractedValue, "ERC20: decreased allowance below zero"); unchecked { _approve(_msgSender(), spender, currentAllowance - subtractedValue); } return true; } function _transfer(address sender, address recipient, uint256 amount) internal virtual { require(sender != address(0), "ERC20: transfer from the zero address"); require(recipient != address(0), "ERC20: transfer to the zero address"); uint256 senderBalance = _balances[sender]; require(senderBalance >= amount, "ERC20: transfer amount exceeds balance"); unchecked { _balances[sender] = senderBalance - amount; } _balances[recipient] += amount; emit Transfer(sender, recipient, amount); } function _createInitialSupply(address account, uint256 amount) internal virtual { require(account != address(0), "ERC20: mint to the zero address"); _totalSupply += amount; _balances[account] += amount; emit Transfer(address(0), account, amount); } function _approve(address owner, address spender, uint256 amount) internal virtual { require(owner != address(0), "ERC20: approve from the zero address"); require(spender != address(0), "ERC20: approve to the zero address"); _allowances[owner][spender] = amount; emit Approval(owner, spender, amount); } }// SPDX-License-Identifier: MIT pragma solidity 0.8.17; import './DividendPayingToken.sol'; contract DividendTracker is DividendPayingToken { using SafeMath for uint256; using SafeMathInt for int256; Map private tokenHoldersMap; uint256 public lastProcessedIndex; mapping (address => bool) public excludedFromDividends; mapping (address => uint256) public lastClaimTimes; uint256 public claimWait; uint256 public immutable minimumTokenBalanceForDividends; event ExcludeFromDividends(address indexed account); event IncludeInDividends(address indexed account); event ClaimWaitUpdated(uint256 indexed newValue, uint256 indexed oldValue); event Claim(address indexed account, uint256 amount, bool indexed automatic); constructor(address _token) { \tclaimWait = 1200; minimumTokenBalanceForDividends = 1; token = _token; } struct Map { address[] keys; mapping(address => uint) values; mapping(address => uint) indexOf; mapping(address => bool) inserted; } function get(address key) private view returns (uint) { return tokenHoldersMap.values[key]; } function getIndexOfKey(address key) private view returns (int) { if(!tokenHoldersMap.inserted[key]) { return -1; } return int(tokenHoldersMap.indexOf[key]); } function getKeyAtIndex(uint index) private view returns (address) { return tokenHoldersMap.keys[index]; } function size() private view returns (uint) { return tokenHoldersMap.keys.length; } function set(address key, uint val) private { if (tokenHoldersMap.inserted[key]) { tokenHoldersMap.values[key] = val; } else { tokenHoldersMap.inserted[key] = true; tokenHoldersMap.values[key] = val; tokenHoldersMap.indexOf[key] = tokenHoldersMap.keys.length; tokenHoldersMap.keys.push(key); } } function remove(address key) private { if (!tokenHoldersMap.inserted[key]) { return; } delete tokenHoldersMap.inserted[key]; delete tokenHoldersMap.values[key]; uint index = tokenHoldersMap.indexOf[key]; uint lastIndex = tokenHoldersMap.keys.length - 1; address lastKey = tokenHoldersMap.keys[lastIndex]; tokenHoldersMap.indexOf[lastKey] = index; delete tokenHoldersMap.indexOf[key]; tokenHoldersMap.keys[index] = lastKey; tokenHoldersMap.keys.pop(); } function excludeFromDividends(address account) external onlyOwner { \texcludedFromDividends[account] = true; \t_setBalance(account, 0); \tremove(account); \temit ExcludeFromDividends(account); } function includeInDividends(address account) external onlyOwner { \trequire(excludedFromDividends[account]); \texcludedFromDividends[account] = false; \temit IncludeInDividends(account); } function updateClaimWait(uint256 newClaimWait) external onlyOwner { require(newClaimWait >= 1200 && newClaimWait <= 86400); require(newClaimWait != claimWait); emit ClaimWaitUpdated(newClaimWait, claimWait); claimWait = newClaimWait; } function getLastProcessedIndex() external view returns(uint256) { \treturn lastProcessedIndex; } function getNumberOfTokenHolders() external view returns(uint256) { return tokenHoldersMap.keys.length; } function getAccount(address _account) public view returns ( address account, int256 index, int256 iterationsUntilProcessed, uint256 withdrawableDividends, uint256 totalDividends, uint256 lastClaimTime, uint256 nextClaimTime, uint256 secondsUntilAutoClaimAvailable) { account = _account; index = getIndexOfKey(account); iterationsUntilProcessed = -1; if(index >= 0) { if(uint256(index) > lastProcessedIndex) { iterationsUntilProcessed = index.sub(int256(lastProcessedIndex)); } else { uint256 processesUntilEndOfArray = tokenHoldersMap.keys.length > lastProcessedIndex ? tokenHoldersMap.keys.length.sub(lastProcessedIndex) : 0; iterationsUntilProcessed = index.add(int256(processesUntilEndOfArray)); } } withdrawableDividends = withdrawableDividendOf(account); totalDividends = accumulativeDividendOf(account); lastClaimTime = lastClaimTimes[account]; nextClaimTime = lastClaimTime > 0 ? lastClaimTime.add(claimWait) : 0; secondsUntilAutoClaimAvailable = nextClaimTime > block.timestamp ? nextClaimTime.sub(block.timestamp) : 0; } function canAutoClaim(uint256 lastClaimTime) private view returns (bool) { \tif(lastClaimTime > block.timestamp) { \t\treturn false; \t} \treturn block.timestamp.sub(lastClaimTime) >= claimWait; } function setBalance(address payable account, uint256 newBalance) external onlyOwner { \tif(excludedFromDividends[account]) { \t\treturn; \t} \tif(newBalance >= minimumTokenBalanceForDividends) { _setBalance(account, newBalance); \t\tset(account, newBalance); \t} \telse { _setBalance(account, 0); \t\tremove(account); \t} \tprocessAccount(account, true); } function process(uint256 gas) public returns (uint256, uint256, uint256) { \tuint256 numberOfTokenHolders = tokenHoldersMap.keys.length; \tif(numberOfTokenHolders == 0) { \t\treturn (0, 0, lastProcessedIndex); \t} \tuint256 _lastProcessedIndex = lastProcessedIndex; \tuint256 gasUsed = 0; \tuint256 gasLeft = gasleft(); \tuint256 iterations = 0; \tuint256 claims = 0; \twhile(gasUsed < gas && iterations < numberOfTokenHolders) { \t\t_lastProcessedIndex++; \t\tif(_lastProcessedIndex >= tokenHoldersMap.keys.length) { \t\t\t_lastProcessedIndex = 0; \t\t} \t\taddress account = tokenHoldersMap.keys[_lastProcessedIndex]; \t\tif(canAutoClaim(lastClaimTimes[account])) { \t\t\tif(processAccount(payable(account), true)) { \t\t\t\tclaims++; \t\t\t} \t\t} \t\titerations++; \t\tuint256 newGasLeft = gasleft(); \t\tif(gasLeft > newGasLeft) { \t\t\tgasUsed = gasUsed.add(gasLeft.sub(newGasLeft)); \t\t} \t\tgasLeft = newGasLeft; \t} \tlastProcessedIndex = _lastProcessedIndex; \treturn (iterations, claims, lastProcessedIndex); } function processAccount(address payable account, bool automatic) public onlyOwner returns (bool) { uint256 amount = _withdrawDividendOfUser(account); \tif(amount > 0) { \t\tlastClaimTimes[account] = block.timestamp; emit Claim(account, amount, automatic); \t\treturn true; \t} \treturn false; } }// SPDX-License-Identifier: MIT pragma solidity 0.8.17; interface DividendPayingTokenOptionalInterface { /// @notice View the amount of dividend in wei that an address can withdraw. /// @param _owner The address of a token holder. /// @return The amount of dividend in wei that `_owner` can withdraw. function withdrawableDividendOf(address _owner) external view returns(uint256); /// @notice View the amount of dividend in wei that an address has withdrawn. /// @param _owner The address of a token holder. /// @return The amount of dividend in wei that `_owner` has withdrawn. function withdrawnDividendOf(address _owner) external view returns(uint256); /// @notice View the amount of dividend in wei that an address has earned in total. /// @dev accumulativeDividendOf(_owner) = withdrawableDividendOf(_owner) + withdrawnDividendOf(_owner) /// @param _owner The address of a token holder. /// @return The amount of dividend in wei that `_owner` has earned in total. function accumulativeDividendOf(address _owner) external view returns(uint256); } // SPDX-License-Identifier: MIT pragma solidity 0.8.17; interface DividendPayingTokenInterface { /// @notice View the amount of dividend in wei that an address can withdraw. /// @param _owner The address of a token holder. /// @return The amount of dividend in wei that `_owner` can withdraw. function dividendOf(address _owner) external view returns(uint256); /// @notice Distributes ether to token holders as dividends. /// @dev SHOULD distribute the paid ether to token holders as dividends. /// SHOULD NOT directly transfer ether to token holders in this function. /// MUST emit a `DividendsDistributed` event when the amount of distributed ether is greater than 0. function distributeDividends() external payable; /// @notice Withdraws the ether distributed to the sender. /// @dev SHOULD transfer `dividendOf(msg.sender)` wei to `msg.sender`, and `dividendOf(msg.sender)` SHOULD be 0 after the transfer. /// MUST emit a `DividendWithdrawn` event if the amount of ether transferred is greater than 0. function withdrawDividend() external; /// @dev This event MUST emit when ether is distributed to token holders. /// @param from The address which sends ether to this contract. /// @param weiAmount The amount of distributed ether in wei. event DividendsDistributed( address indexed from, uint256 weiAmount ); /// @dev This event MUST emit when an address withdraws their dividend. /// @param to The address which withdraws ether from this contract. /// @param weiAmount The amount of withdrawn ether in wei. event DividendWithdrawn( address indexed to, uint256 weiAmount ); }// SPDX-License-Identifier: MIT pragma solidity 0.8.17; import './DividendPayingTokenInterface.sol'; import './DividendPayingTokenOptionalInterface.sol'; import './Ownable.sol'; import './SafeMath.sol'; import './SafeMathUint.sol'; import './SafeMathInt.sol'; import './ERC20.sol'; contract DividendPayingToken is DividendPayingTokenInterface, DividendPayingTokenOptionalInterface, Ownable { using SafeMath for uint256; using SafeMathUint for uint256; using SafeMathInt for int256; // With `magnitude`, we can properly distribute dividends even if the amount of received ether is small. // For more discussion about choosing the value of `magnitude`, // see https://github.com/ethereum/EIPs/issues/1726#issuecomment-472352728 uint256 constant internal magnitude = 2**128; uint256 internal magnifiedDividendPerShare; address public token; // About dividendCorrection: // If the token balance of a `_user` is never changed, the dividend of `_user` can be computed with: // `dividendOf(_user) = dividendPerShare * balanceOf(_user)`. // When `balanceOf(_user)` is changed (via minting/burning/transferring tokens), // `dividendOf(_user)` should not be changed, // but the computed value of `dividendPerShare * balanceOf(_user)` is changed. // To keep the `dividendOf(_user)` unchanged, we add a correction term: // `dividendOf(_user) = dividendPerShare * balanceOf(_user) + dividendCorrectionOf(_user)`, // where `dividendCorrectionOf(_user)` is updated whenever `balanceOf(_user)` is changed: // `dividendCorrectionOf(_user) = dividendPerShare * (old balanceOf(_user)) - (new balanceOf(_user))`. // So now `dividendOf(_user)` returns the same value before and after `balanceOf(_user)` is changed. mapping(address => int256) internal magnifiedDividendCorrections; mapping(address => uint256) internal withdrawnDividends; mapping (address => uint256) public holderBalance; uint256 public totalBalance; uint256 public totalDividendsDistributed; uint256 public totalDividendsWaitingToSend; /// @dev Distributes dividends whenever ether is paid to this contract. receive() external payable { distributeDividends(); } /// @notice Distributes ether to token holders as dividends. /// @dev It reverts if the total supply of tokens is 0. /// It emits the `DividendsDistributed` event if the amount of received ether is greater than 0. /// About undistributed ether: /// In each distribution, there is a small amount of ether not distributed, /// the magnified amount of which is /// `(msg.value * magnitude) % totalSupply()`. /// With a well-chosen `magnitude`, the amount of undistributed ether /// (de-magnified) in a distribution can be less than 1 wei. /// We can actually keep track of the undistributed ether in a distribution /// and try to distribute it in the next distribution, /// but keeping track of such data on-chain costs much more than /// the saved ether, so we don't do that. function distributeDividends() public override payable { require(false); // } function distributeTokenDividends() public { if(totalBalance > 0){ uint256 tokensToAdd; uint256 balance = IERC20(token).balanceOf(address(this)); if(totalDividendsWaitingToSend < balance){ tokensToAdd = balance - totalDividendsWaitingToSend; } else { tokensToAdd = 0; } if (tokensToAdd > 0) { magnifiedDividendPerShare = magnifiedDividendPerShare.add( (tokensToAdd).mul(magnitude) / totalBalance ); emit DividendsDistributed(msg.sender, tokensToAdd); totalDividendsDistributed = totalDividendsDistributed.add(tokensToAdd); totalDividendsWaitingToSend = totalDividendsWaitingToSend.add(tokensToAdd); } } } /// @notice Withdraws the ether distributed to the sender. /// @dev It emits a `DividendWithdrawn` event if the amount of withdrawn ether is greater than 0. function withdrawDividend() public virtual override { _withdrawDividendOfUser(payable(msg.sender)); } /// @notice Withdraws the ether distributed to the sender. /// @dev It emits a `DividendWithdrawn` event if the amount of withdrawn ether is greater than 0. function _withdrawDividendOfUser(address payable user) internal returns (uint256) { uint256 _withdrawableDividend = withdrawableDividendOf(user); if (_withdrawableDividend > 0) { withdrawnDividends[user] = withdrawnDividends[user].add(_withdrawableDividend); if(totalDividendsWaitingToSend >= _withdrawableDividend){ totalDividendsWaitingToSend -= _withdrawableDividend; } else { totalDividendsWaitingToSend = 0; } emit DividendWithdrawn(user, _withdrawableDividend); bool success = IERC20(token).transfer(user, _withdrawableDividend); if(!success) { withdrawnDividends[user] = withdrawnDividends[user].sub(_withdrawableDividend); return 0; } return _withdrawableDividend; } return 0; } /// @notice View the amount of dividend in wei that an address can withdraw. /// @param _owner The address of a token holder. /// @return The amount of dividend in wei that `_owner` can withdraw. function dividendOf(address _owner) public view override returns(uint256) { return withdrawableDividendOf(_owner); } /// @notice View the amount of dividend in wei that an address can withdraw. /// @param _owner The address of a token holder. /// @return The amount of dividend in wei that `_owner` can withdraw. function withdrawableDividendOf(address _owner) public view override returns(uint256) { return accumulativeDividendOf(_owner).sub(withdrawnDividends[_owner]); } /// @notice View the amount of dividend in wei that an address has withdrawn. /// @param _owner The address of a token holder. /// @return The amount of dividend in wei that `_owner` has withdrawn. function withdrawnDividendOf(address _owner) public view override returns(uint256) { return withdrawnDividends[_owner]; } /// @notice View the amount of dividend in wei that an address has earned in total. /// @dev accumulativeDividendOf(_owner) = withdrawableDividendOf(_owner) + withdrawnDividendOf(_owner) /// = (magnifiedDividendPerShare * balanceOf(_owner) + magnifiedDividendCorrections[_owner]) / magnitude /// @param _owner The address of a token holder. /// @return The amount of dividend in wei that `_owner` has earned in total. function accumulativeDividendOf(address _owner) public view override returns(uint256) { return magnifiedDividendPerShare.mul(holderBalance[_owner]).toInt256Safe() .add(magnifiedDividendCorrections[_owner]).toUint256Safe() / magnitude; } /// @dev Internal function that increases tokens to an account. /// Update magnifiedDividendCorrections to keep dividends unchanged. /// @param account The account that will receive the created tokens. /// @param value The amount that will be created. function _increase(address account, uint256 value) internal { magnifiedDividendCorrections[account] = magnifiedDividendCorrections[account] .sub( (magnifiedDividendPerShare.mul(value)).toInt256Safe() ); } /// @dev Internal function that reduces an amount of the token of a given account. /// Update magnifiedDividendCorrections to keep dividends unchanged. /// @param account The account whose tokens will be burnt. /// @param value The amount that will be burnt. function _reduce(address account, uint256 value) internal { magnifiedDividendCorrections[account] = magnifiedDividendCorrections[account] .add( (magnifiedDividendPerShare.mul(value)).toInt256Safe() ); } function _setBalance(address account, uint256 newBalance) internal { uint256 currentBalance = holderBalance[account]; holderBalance[account] = newBalance; if(newBalance > currentBalance) { uint256 increaseAmount = newBalance.sub(currentBalance); _increase(account, increaseAmount); totalBalance += increaseAmount; } else if(newBalance < currentBalance) { uint256 reduceAmount = currentBalance.sub(newBalance); _reduce(account, reduceAmount); totalBalance -= reduceAmount; } } }// SPDX-License-Identifier: MIT pragma solidity 0.8.17; abstract contract Context { function _msgSender() internal view virtual returns (address) { return msg.sender; } function _msgData() internal view virtual returns (bytes calldata) { this; // silence state mutability warning without generating bytecode - see https://github.com/ethereum/solidity/issues/2691 return msg.data; } }