ETH Price: $2,162.69 (-10.05%)

Transaction Decoder

Block:
21325857 at Dec-04-2024 01:28:47 AM +UTC
Transaction Fee:
0.00138987364552613 ETH $3.01
Gas Used:
66,730 Gas / 20.828317781 Gwei

Emitted Events:

382 pepeCoin.Transfer( from=[Receiver] BasedAIBridge, to=[Sender] 0x7be5820c4751e5efe96e3206c882bf34094cbb71, value=4308000000000000000000 )
383 BasedAIBridge.Withdrawn( user=[Sender] 0x7be5820c4751e5efe96e3206c882bf34094cbb71, amount=4308000000000000000000 )

Account State Difference:

  Address   Before After State Difference Code
0x40359B38...5b393bd72
(Pepe Coin: Pre-Bridge v2 Staking)
0x7BE5820C...4094CbB71
0.012191470984499243 Eth
Nonce: 272
0.010801597338973113 Eth
Nonce: 273
0.00138987364552613
(beaverbuild)
7.532247714097060968 Eth7.532254387097060968 Eth0.000006673
0xA9E8aCf0...e41a9489A

Execution Trace

BasedAIBridge.CALL( )
  • pepeCoin.transfer( to=0x7BE5820C4751E5EFE96e3206c882bF34094CbB71, amount=4308000000000000000000 ) => ( True )
    File 1 of 2: BasedAIBridge
    // SPDX-License-Identifier: MIT
    /*
    #########################################################
    # ____    _    ____  _____ ____    _    ___    __   __  #
    #| __ )  / \\  / ___|| ____|  _ \\  / \\  |_ _|  / /___\\ \\ #
    #|  _ \\ / _ \\ \\___ \\|  _| | | | |/ _ \\  | |  / /_____\\ \\#
    #| |_) / ___ \\ ___) | |___| |_| / ___ \\ | |  \\ \\_____/ /#
    #|____/_/___\\_\\____/|_____|____/_/_  \\_\\___|  \\_\\   /_/ #
    #| __ )|  _ \\|_ _|  _ \\ / ___| ____|                    #
    #|  _ \\| |_) || || | | | |  _|  _|                      #
    #| |_) |  _ < | || |_| | |_| | |___                     #
    #|____/|_| \\_\\___|____/ \\____|_____|                    #
    #########################################################   
    # BRIDGE REWARDS - Bridge.sol - www.getbased.ai         #
    #########################################################
    */
    pragma solidity ^0.8.0;
    import "@openzeppelin/contracts/security/ReentrancyGuard.sol";
    import "@openzeppelin/contracts/security/Pausable.sol";
    interface IERC20 {
        function transferFrom(address sender, address recipient, uint256 amount) external returns (bool);
        function transfer(address recipient, uint256 amount) external returns (bool);
        function balanceOf(address account) external view returns (uint256);
    }
    interface IERC721 {
        function transferFrom(address from, address to, uint256 tokenId) external;
        function ownerOf(uint256 tokenId) external view returns (address);
    }
    contract BasedAIBridge is ReentrancyGuard, Pausable {
        IERC20 public pepeCoin;
        IERC721 public brainNFT;
        address public owner;
        address public pepeCoinAddress;
        bool public mainnetLive;
        uint256 public startTime;
        struct Stake {
            address tokenAddress;
            uint256 amount;
            uint256 timestamp;
            uint256 rate;
            uint256[] brainIds;
        }
        struct TokenConfig {
            uint256 initialRate;
            uint256 rateIncreaseAmount;
            uint256 rateIncreaseInterval;
            bool isSupported;
        }
        mapping(address => Stake[]) public stakes;
        mapping(address => uint256) public credits;
        mapping(address => bool) public hasStaked;
        mapping(address => uint256) public lastKnownCredits;
        address[] public stakers;
        mapping(address => TokenConfig) public tokenConfigs;
        mapping(address => uint256) public finalScores; 
        event Staked(address indexed user, address tokenAddress, uint256 amount, uint256 timestamp, uint256 rate);
        event BrainStaked(address indexed user, uint256 tokenId, uint256 timestamp, uint256 rate);
        event MainnetActivated();
        event Withdrawn(address indexed user, uint256 amount);
        event BrainWithdrawn(address indexed user, uint256 tokenId);
        event CreditsUpdated(address indexed user, uint256 credits);
        event FinalScoreRecorded(address indexed user, uint256 finalScore);
        modifier onlyOwner() {
            require(msg.sender == owner, "Only owner can call this function.");
            _;
        }
        constructor() {
            pepeCoinAddress = 0xA9E8aCf069C58aEc8825542845Fd754e41a9489A;
            pepeCoin = IERC20(0xA9E8aCf069C58aEc8825542845Fd754e41a9489A);
            brainNFT = IERC721(0xA9E8aCf069C58aEc8825542845Fd754e41a9489A);
            owner = msg.sender;
            mainnetLive = false;
            startTime = block.timestamp;
            tokenConfigs[0xA9E8aCf069C58aEc8825542845Fd754e41a9489A] = TokenConfig({
                initialRate: 500,
                rateIncreaseAmount: 0,
                rateIncreaseInterval: 30 days,
                isSupported: true
            });
        }
        // For user with TFT Enforcer
        function getCurrentRate(address tokenAddress) public view returns (uint256) {
            TokenConfig storage config = tokenConfigs[tokenAddress];
            uint256 timeElapsed = block.timestamp - startTime;
            uint256 periods = timeElapsed / config.rateIncreaseInterval;
            return config.initialRate + (config.rateIncreaseAmount * periods);
        }
        function setBasedBrainNFT(address tokenAddress) external onlyOwner {
            brainNFT = IERC721(tokenAddress);
        }
        function addOrUpdateToken(address tokenAddress, uint256 _initialRate, uint256 _rateIncreaseAmount, uint256 _rateIncreaseInterval) external onlyOwner {
            tokenConfigs[tokenAddress] = TokenConfig({
                initialRate: _initialRate, // 500 for Pepecoin, 5000 for Brain Specific Token, 5 for Brain Credits, 5800 for $BASED, 1000 for FHE-ORDERBOOK Brain Token
                rateIncreaseAmount: _rateIncreaseAmount,
                rateIncreaseInterval: _rateIncreaseInterval,
                isSupported: true
            });
        }
        function removeToken(address tokenAddress) external onlyOwner {
            tokenConfigs[tokenAddress].isSupported = false;
        }
        function stake(address tokenAddress, uint256 _amount) external whenNotPaused nonReentrant {
            require(_amount > 0, "Amount must be greater than zero");
            require(tokenConfigs[tokenAddress].isSupported, "Token is not supported for staking");
            require(!mainnetLive, "Mainnet is live!");
            IERC20(tokenAddress).transferFrom(msg.sender, address(this), _amount);
            uint256 currentRate = tokenConfigs[tokenAddress].initialRate;
            // Empty brain array
            uint256[] memory brainIds = new uint256[](0);
            
            // Recover all credits if in the first 30 days
            if (block.timestamp - startTime <= 30 days) {
                credits[msg.sender] += lastKnownCredits[msg.sender];
                lastKnownCredits[msg.sender] = 0;
            }
            
            _addStake(msg.sender, tokenAddress, _amount, brainIds, currentRate);
        }
        function _addStake(address _staker, address _tokenAddress, uint256 _amount, uint256[] memory _brainIds, uint256 _rate) private {
            if (!hasStaked[_staker]) {
                hasStaked[_staker] = true;
                stakers.push(_staker);
            }
            stakes[_staker].push(Stake({
                tokenAddress: _tokenAddress,
                amount: _amount,
                timestamp: block.timestamp,
                rate: _rate,
                brainIds: _brainIds
            }));
            emit Staked(_staker, _tokenAddress, _amount, block.timestamp, _rate);
        }
        function recoverERC20(address tokenAddress, uint256 tokenAmount) external onlyOwner {
            require(tokenAddress != pepeCoinAddress, "Unable to remove prebriged PepeCoin");
            IERC20(tokenAddress).transfer(owner, tokenAmount);
        }
        function recoverERC721(address tokenAddress, uint256 tokenId) external onlyOwner {
            IERC721(tokenAddress).transferFrom(address(this), owner, tokenId);
        }
        function stakeBrain(uint256 _tokenId) external whenNotPaused nonReentrant {
            require(brainNFT.ownerOf(_tokenId) == msg.sender, "Not the owner of the Brain");
            require(!mainnetLive, "Mainnet is live!");
            brainNFT.transferFrom(msg.sender, address(this), _tokenId);
            uint256 currentRate = tokenConfigs[address(pepeCoin)].initialRate;
            uint256[] memory brainIds = new uint256[](1);
            brainIds[0] = _tokenId;
            _addStake(msg.sender, address(pepeCoin), 100000 * (10 ** 18), brainIds, currentRate);
            emit BrainStaked(msg.sender, _tokenId, block.timestamp, currentRate);
        }
        function withdraw() external whenNotPaused nonReentrant {
            uint256 totalStaked = 0;
            uint256 stakeCount = stakes[msg.sender].length;
            
            // make sure the users credits are calculated.
            if (block.timestamp - startTime <= 30 days) {
                // Store any previous credit balances
                lastKnownCredits[msg.sender] = credits[msg.sender];
                // Update the credit table to the latest
                updateCredits(msg.sender);
                // Combine the old and new updated credits
                lastKnownCredits[msg.sender] += credits[msg.sender];
            }
            for (uint i = stakeCount; i > 0; i--) {
                uint index = i - 1;
                Stake storage stake = stakes[msg.sender][index];
                totalStaked += stake.amount;
                // If it is a Brain they can only withdraw the Brain
                if (stake.brainIds.length == 0) {
                    IERC20(stake.tokenAddress).transfer(msg.sender, stake.amount);
                    emit Withdrawn(msg.sender, stake.amount);
                }
                
                // Transfer any Brain NFTs back to the user
                for (uint j = 0; j < stake.brainIds.length; j++) {
                    brainNFT.transferFrom(address(this), msg.sender, stake.brainIds[j]);
                    emit BrainWithdrawn(msg.sender, stake.brainIds[j]);
                }
                
                stakes[msg.sender][index] = stakes[msg.sender][stakes[msg.sender].length - 1];
                stakes[msg.sender].pop();
            }
            require(totalStaked > 0, "Nothing to remove from BasedAI bridge");
            credits[msg.sender] = 0;
        }
        function triggerMainnetLive() external onlyOwner {
            mainnetLive = true;
            for (uint i = 0; i < stakers.length; i++) {
                finalScores[stakers[i]] = getCredits(stakers[i]) + credits[stakers[i]];
                finalScores[stakers[i]] += lastKnownCredits[stakers[i]];
                emit FinalScoreRecorded(stakers[i], finalScores[stakers[i]]);
            }
            emit MainnetActivated();
        }
        function getFinalScore(address staker) public view returns (uint256) {
            require(mainnetLive, "BasedAI Mainnet is not live yet");
            return finalScores[staker];
        }
        function getCredits(address staker) private view returns (uint256) {
            uint256 totalCredits = 0;
            for (uint i = 0; i < stakes[staker].length; i++) {
                totalCredits += calculateCredits(stakes[staker][i]);
            }
            return totalCredits;
        }
        function updateCredits(address staker) private {
            uint256 totalCredits = 0;
            for (uint i = 0; i < stakes[staker].length; i++) {
                totalCredits += calculateCredits(stakes[staker][i]);
            }
            credits[staker] = totalCredits;
        }
        function calculateCredits(Stake memory stake) private view returns (uint256) {
            uint256 durationInSeconds = block.timestamp - stake.timestamp;
            uint256 accruedCredits = (stake.amount / stake.rate) * durationInSeconds / 86400; 
            return accruedCredits;
        }
        function calculateTotalCredits(address staker) public view returns (uint256) {
            if (mainnetLive) return finalScores[staker];
            uint256 totalCredits = 0;
            for (uint i = 0; i < stakes[staker].length; i++) {
                totalCredits += calculateCredits(stakes[staker][i]);
            }
            // add any leftover credits collected if they participated in Brain burn or Brain credits
            totalCredits += lastKnownCredits[staker];
            totalCredits += credits[staker];
            return totalCredits;
        }
        // Credits the user recovers if they restake. 
        function calculateReturnCredits(address staker) public view returns (uint256) {
            if (mainnetLive) return finalScores[staker];
            return lastKnownCredits[staker];
        }
        // calculates from a current stake how much a user has earned 
        function calculateCreditsPerToken(address staker, address _tokenAddress) public view returns (uint256) {
            require(!mainnetLive, "Mainnet is live, claim all rewards.");
            uint256 totalCredits = 0;
            for (uint i = 0; i < stakes[staker].length; i++) {
                if (stakes[staker][i].tokenAddress == _tokenAddress) {
                    totalCredits += calculateCredits(stakes[staker][i]);
                }
            }
            return totalCredits;
        }
        function setCreditsForAddress(address _user, uint256 _credits) external onlyOwner {
            credits[_user] = _credits;
            emit CreditsUpdated(_user, _credits);
        }
        function getStakedAmount(address user, address tokenAddress) public view returns (uint256) {
            uint256 totalStaked = 0;
            for (uint i = 0; i < stakes[user].length; i++) {
                if (stakes[user][i].tokenAddress == tokenAddress) {
                    totalStaked += stakes[user][i].amount;
                }
            }
            return totalStaked;
        }
        function pause() external onlyOwner {
            _pause();
        }
        function unpause() external onlyOwner {
            _unpause();
        }
    }
    // SPDX-License-Identifier: MIT
    // OpenZeppelin Contracts (last updated v4.7.0) (security/Pausable.sol)
    pragma solidity ^0.8.0;
    import "../utils/Context.sol";
    /**
     * @dev Contract module which allows children to implement an emergency stop
     * mechanism that can be triggered by an authorized account.
     *
     * This module is used through inheritance. It will make available the
     * modifiers `whenNotPaused` and `whenPaused`, which can be applied to
     * the functions of your contract. Note that they will not be pausable by
     * simply including this module, only once the modifiers are put in place.
     */
    abstract contract Pausable is Context {
        /**
         * @dev Emitted when the pause is triggered by `account`.
         */
        event Paused(address account);
        /**
         * @dev Emitted when the pause is lifted by `account`.
         */
        event Unpaused(address account);
        bool private _paused;
        /**
         * @dev Initializes the contract in unpaused state.
         */
        constructor() {
            _paused = false;
        }
        /**
         * @dev Modifier to make a function callable only when the contract is not paused.
         *
         * Requirements:
         *
         * - The contract must not be paused.
         */
        modifier whenNotPaused() {
            _requireNotPaused();
            _;
        }
        /**
         * @dev Modifier to make a function callable only when the contract is paused.
         *
         * Requirements:
         *
         * - The contract must be paused.
         */
        modifier whenPaused() {
            _requirePaused();
            _;
        }
        /**
         * @dev Returns true if the contract is paused, and false otherwise.
         */
        function paused() public view virtual returns (bool) {
            return _paused;
        }
        /**
         * @dev Throws if the contract is paused.
         */
        function _requireNotPaused() internal view virtual {
            require(!paused(), "Pausable: paused");
        }
        /**
         * @dev Throws if the contract is not paused.
         */
        function _requirePaused() internal view virtual {
            require(paused(), "Pausable: not paused");
        }
        /**
         * @dev Triggers stopped state.
         *
         * Requirements:
         *
         * - The contract must not be paused.
         */
        function _pause() internal virtual whenNotPaused {
            _paused = true;
            emit Paused(_msgSender());
        }
        /**
         * @dev Returns to normal state.
         *
         * Requirements:
         *
         * - The contract must be paused.
         */
        function _unpause() internal virtual whenPaused {
            _paused = false;
            emit Unpaused(_msgSender());
        }
    }
    // SPDX-License-Identifier: MIT
    // OpenZeppelin Contracts (last updated v4.9.0) (security/ReentrancyGuard.sol)
    pragma solidity ^0.8.0;
    /**
     * @dev Contract module that helps prevent reentrant calls to a function.
     *
     * Inheriting from `ReentrancyGuard` will make the {nonReentrant} modifier
     * available, which can be applied to functions to make sure there are no nested
     * (reentrant) calls to them.
     *
     * Note that because there is a single `nonReentrant` guard, functions marked as
     * `nonReentrant` may not call one another. This can be worked around by making
     * those functions `private`, and then adding `external` `nonReentrant` entry
     * points to them.
     *
     * TIP: If you would like to learn more about reentrancy and alternative ways
     * to protect against it, check out our blog post
     * https://blog.openzeppelin.com/reentrancy-after-istanbul/[Reentrancy After Istanbul].
     */
    abstract contract ReentrancyGuard {
        // Booleans are more expensive than uint256 or any type that takes up a full
        // word because each write operation emits an extra SLOAD to first read the
        // slot's contents, replace the bits taken up by the boolean, and then write
        // back. This is the compiler's defense against contract upgrades and
        // pointer aliasing, and it cannot be disabled.
        // The values being non-zero value makes deployment a bit more expensive,
        // but in exchange the refund on every call to nonReentrant will be lower in
        // amount. Since refunds are capped to a percentage of the total
        // transaction's gas, it is best to keep them low in cases like this one, to
        // increase the likelihood of the full refund coming into effect.
        uint256 private constant _NOT_ENTERED = 1;
        uint256 private constant _ENTERED = 2;
        uint256 private _status;
        constructor() {
            _status = _NOT_ENTERED;
        }
        /**
         * @dev Prevents a contract from calling itself, directly or indirectly.
         * Calling a `nonReentrant` function from another `nonReentrant`
         * function is not supported. It is possible to prevent this from happening
         * by making the `nonReentrant` function external, and making it call a
         * `private` function that does the actual work.
         */
        modifier nonReentrant() {
            _nonReentrantBefore();
            _;
            _nonReentrantAfter();
        }
        function _nonReentrantBefore() private {
            // On the first call to nonReentrant, _status will be _NOT_ENTERED
            require(_status != _ENTERED, "ReentrancyGuard: reentrant call");
            // Any calls to nonReentrant after this point will fail
            _status = _ENTERED;
        }
        function _nonReentrantAfter() private {
            // By storing the original value once again, a refund is triggered (see
            // https://eips.ethereum.org/EIPS/eip-2200)
            _status = _NOT_ENTERED;
        }
        /**
         * @dev Returns true if the reentrancy guard is currently set to "entered", which indicates there is a
         * `nonReentrant` function in the call stack.
         */
        function _reentrancyGuardEntered() internal view returns (bool) {
            return _status == _ENTERED;
        }
    }
    // SPDX-License-Identifier: MIT
    // OpenZeppelin Contracts (last updated v5.0.0) (utils/Context.sol)
    pragma solidity ^0.8.20;
    /**
     * @dev Provides information about the current execution context, including the
     * sender of the transaction and its data. While these are generally available
     * via msg.sender and msg.data, they should not be accessed in such a direct
     * manner, since when dealing with meta-transactions the account sending and
     * paying for execution may not be the actual sender (as far as an application
     * is concerned).
     *
     * This contract is only required for intermediate, library-like contracts.
     */
    abstract contract Context {
        function _msgSender() internal view virtual returns (address) {
            return msg.sender;
        }
        function _msgData() internal view virtual returns (bytes calldata) {
            return msg.data;
        }
    }
    

    File 2 of 2: pepeCoin
    // SPDX-License-Identifier: MIT
    /**
     * _______ _             ____       _       _             _   _____
     * |__   __| |           / __ \     (_)     (_)           | | |  __ \
     *    | |  | |__   ___  | |  | |_ __ _  __ _ _ _ __   __ _| | | |__) |__ _ __   ___
     *    | |  | '_ \ / _ \ | |  | | '__| |/ _` | | '_ \ / _` | | |  ___/ _ \ '_ \ / _ \
     *    | |  | | | |  __/ | |__| | |  | | (_| | | | | | (_| | | | |  |  __/ |_) |  __/
     *    |_|  |_| |_|\___|  \____/|_|  |_|\__, |_|_| |_|\__,_|_| |_|   \___| .__/ \___|
     *                                      __/ |                           | |
     *                                     |___/                            |_|
     *
     *⠀⢀⠔⠂⠉⠉⠉⠉⠑⠢⡄⣀⠔⠊⠁⠀⠒⠤⡀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀
     *⡰⠁⠀⠀⠀⠀⢀⣀⣀⣀⠘⡇⠀⠀⠀⠀⠀⠀⠘⢆⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀
     *⠁⠀⠀⣠⠔⠉⠁⠀⠀⠀⠉⠉⠲⣤⠖⠒⠒⠒⠲⠬⣧⠀⠀⠀⠀⠀⠀⠀⠀⠀
     *⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠈⢣⠀⠀⠀⠀⠀⠀⠙⢦⡀⠀⠀⠀⠀⠀⠀
     *⠀⠀⠀⣠⠔⣊⠭⠭⠭⠭⢭⣛⠩⣉⠲⣇⠀⠀⢀⣶⠶⢦⣶⢷⣤⡀⠀⠀⠀⠀
     *⠀⠀⢸⣵⣉⡤⠤⢤⣤⣤⣤⣬⠵⠮⣶⣌⣇⠐⣫⣶⣭⣭⣍⡑⢼⠁⠀⠀⠀⠀
     *⠀⠀⠀⠀⠈⣅⠲⣿⣞⣿⣉⣿⠀⠀⠘⡎⣇⡜⣿⣾⣿⣹⡇⠈⣽⠀⠀⠀⠀⠀
     *⠀⠀⠀⠀⠀⠀⠓⠤⢈⣉⣛⣓⣂⣒⣊⡽⠂⢹⣛⠛⢛⠛⣒⣩⠞⠀⠀⠀⠀⠀
     *⠀⠠⠤⠂⠀⠀⠀⠀⠀⠀⠀⠀⣀⠼⠁⠀⠀⠀⠈⠫⡁⠐⠛⠁⠱⡀⠀⠀⠀⠀
     *⢠⢶⠭⠭⣄⣀⠀⠀⠀⠤⠒⠉⠁⠀⠀⠀⠀⠀⠀⠀⠈⠀⠀⠀⢰⢻⠄⠀⠀⠀
     *⠀⠘⢦⣙⠲⠤⣍⡉⠑⠒⠢⠤⠤⠤⣀⣀⣀⣀⣀⣀⣀⣀⣀⠔⣡⠊⠀⠀⠀⠀
     *⠀⠀⠀⠈⠉⠢⢄⡈⠉⠁⠀⠀⠒⠒⠦⠤⠤⠤⠤⠤⠤⠤⠤⠊⡸⠀⠀⠀⠀⠀
     *⠀⠀⠀⠀⠀⠀⠀⠈⠙⠒⠂⠤⠤⠤⠤⢄⣀⣀⡤⠤⠤⢤⠤⠚⠅⠀⠀⠀⠀⠀
     *⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⢀⠤⠒⠓⢤⠞⠀⠀⠀⠀⠀⠀
     *⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⢀⡠⢄⡠⢶⡁⠀⣀⣀⡀⢑⠢⣄⠀⠀⠀⠀
     *⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠘⣆⠀⠣⢶⠕⢋⢔⡵⠗⠁⠀⠈⠳⡀⠀⠀
     *⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠈⢣⠀⠘⣖⡝⠁⠀⢀⠔⠀⠀⠀⠘⣆⠀
     *⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⣸⠀⠀⠈⠑⠤⢠⠃⠀⣠⠞⢀⠄⠈⢆
     *
     * pepecoin
     *
     * website: https://pepeco.in
     * telegram: T.me/pepecoins
     *
    */
    
    // File: @openzeppelin/contracts/utils/Context.sol
    
    // OpenZeppelin Contracts v4.4.1 (utils/Context.sol)
    
    pragma solidity ^0.8.0;
    
    /**
     * @dev Provides information about the current execution context, including the
     * sender of the transaction and its data. While these are generally available
     * via msg.sender and msg.data, they should not be accessed in such a direct
     * manner, since when dealing with meta-transactions the account sending and
     * paying for execution may not be the actual sender (as far as an application
     * is concerned).
     *
     * This contract is only required for intermediate, library-like contracts.
     */
    abstract contract Context {
        function _msgSender() internal view virtual returns (address) {
            return msg.sender;
        }
    
        function _msgData() internal view virtual returns (bytes calldata) {
            return msg.data;
        }
    }
    
    // File: @openzeppelin/contracts/access/Ownable.sol
    
    
    // OpenZeppelin Contracts (last updated v4.7.0) (access/Ownable.sol)
    
    pragma solidity ^0.8.0;
    
    
    /**
     * @dev Contract module which provides a basic access control mechanism, where
     * there is an account (an owner) that can be granted exclusive access to
     * specific functions.
     *
     * By default, the owner account will be the one that deploys the contract. This
     * can later be changed with {transferOwnership}.
     *
     * This module is used through inheritance. It will make available the modifier
     * `onlyOwner`, which can be applied to your functions to restrict their use to
     * the owner.
     */
    abstract contract Ownable is Context {
        address private _owner;
    
        event OwnershipTransferred(address indexed previousOwner, address indexed newOwner);
    
        /**
         * @dev Initializes the contract setting the deployer as the initial owner.
         */
        constructor() {
            _transferOwnership(_msgSender());
        }
    
        /**
         * @dev Throws if called by any account other than the owner.
         */
        modifier onlyOwner() {
            _checkOwner();
            _;
        }
    
        /**
         * @dev Returns the address of the current owner.
         */
        function owner() public view virtual returns (address) {
            return _owner;
        }
    
        /**
         * @dev Throws if the sender is not the owner.
         */
        function _checkOwner() internal view virtual {
            require(owner() == _msgSender(), "Ownable: caller is not the owner");
        }
    
        /**
         * @dev Leaves the contract without owner. It will not be possible to call
         * `onlyOwner` functions anymore. Can only be called by the current owner.
         *
         * NOTE: Renouncing ownership will leave the contract without an owner,
         * thereby removing any functionality that is only available to the owner.
         */
        function renounceOwnership() public virtual onlyOwner {
            _transferOwnership(address(0));
        }
    
        /**
         * @dev Transfers ownership of the contract to a new account (`newOwner`).
         * Can only be called by the current owner.
         */
        function transferOwnership(address newOwner) public virtual onlyOwner {
            require(newOwner != address(0), "Ownable: new owner is the zero address");
            _transferOwnership(newOwner);
        }
    
        /**
         * @dev Transfers ownership of the contract to a new account (`newOwner`).
         * Internal function without access restriction.
         */
        function _transferOwnership(address newOwner) internal virtual {
            address oldOwner = _owner;
            _owner = newOwner;
            emit OwnershipTransferred(oldOwner, newOwner);
        }
    }
    
    // File: @openzeppelin/contracts/token/ERC20/IERC20.sol
    
    
    // OpenZeppelin Contracts (last updated v4.6.0) (token/ERC20/IERC20.sol)
    
    pragma solidity ^0.8.0;
    
    /**
     * @dev Interface of the ERC20 standard as defined in the EIP.
     */
    interface IERC20 {
        /**
         * @dev Emitted when `value` tokens are moved from one account (`from`) to
         * another (`to`).
         *
         * Note that `value` may be zero.
         */
        event Transfer(address indexed from, address indexed to, uint256 value);
    
        /**
         * @dev Emitted when the allowance of a `spender` for an `owner` is set by
         * a call to {approve}. `value` is the new allowance.
         */
        event Approval(address indexed owner, address indexed spender, uint256 value);
    
        /**
         * @dev Returns the amount of tokens in existence.
         */
        function totalSupply() external view returns (uint256);
    
        /**
         * @dev Returns the amount of tokens owned by `account`.
         */
        function balanceOf(address account) external view returns (uint256);
    
        /**
         * @dev Moves `amount` tokens from the caller's account to `to`.
         *
         * Returns a boolean value indicating whether the operation succeeded.
         *
         * Emits a {Transfer} event.
         */
        function transfer(address to, uint256 amount) external returns (bool);
    
        /**
         * @dev Returns the remaining number of tokens that `spender` will be
         * allowed to spend on behalf of `owner` through {transferFrom}. This is
         * zero by default.
         *
         * This value changes when {approve} or {transferFrom} are called.
         */
        function allowance(address owner, address spender) external view returns (uint256);
    
        /**
         * @dev Sets `amount` as the allowance of `spender` over the caller's tokens.
         *
         * Returns a boolean value indicating whether the operation succeeded.
         *
         * IMPORTANT: Beware that changing an allowance with this method brings the risk
         * that someone may use both the old and the new allowance by unfortunate
         * transaction ordering. One possible solution to mitigate this race
         * condition is to first reduce the spender's allowance to 0 and set the
         * desired value afterwards:
         * https://github.com/ethereum/EIPs/issues/20#issuecomment-263524729
         *
         * Emits an {Approval} event.
         */
        function approve(address spender, uint256 amount) external returns (bool);
    
        /**
         * @dev Moves `amount` tokens from `from` to `to` using the
         * allowance mechanism. `amount` is then deducted from the caller's
         * allowance.
         *
         * Returns a boolean value indicating whether the operation succeeded.
         *
         * Emits a {Transfer} event.
         */
        function transferFrom(
            address from,
            address to,
            uint256 amount
        ) external returns (bool);
    }
    
    // File: @openzeppelin/contracts/token/ERC20/extensions/IERC20Metadata.sol
    
    
    // OpenZeppelin Contracts v4.4.1 (token/ERC20/extensions/IERC20Metadata.sol)
    
    pragma solidity ^0.8.0;
    
    
    /**
     * @dev Interface for the optional metadata functions from the ERC20 standard.
     *
     * _Available since v4.1._
     */
    interface IERC20Metadata is IERC20 {
        /**
         * @dev Returns the name of the token.
         */
        function name() external view returns (string memory);
    
        /**
         * @dev Returns the symbol of the token.
         */
        function symbol() external view returns (string memory);
    
        /**
         * @dev Returns the decimals places of the token.
         */
        function decimals() external view returns (uint8);
    }
    
    // File: @openzeppelin/contracts/token/ERC20/ERC20.sol
    
    
    // OpenZeppelin Contracts (last updated v4.8.0) (token/ERC20/ERC20.sol)
    
    pragma solidity ^0.8.0;
    
    
    
    
    /**
     * @dev Implementation of the {IERC20} interface.
     *
     * This implementation is agnostic to the way tokens are created. This means
     * that a supply mechanism has to be added in a derived contract using {_mint}.
     * For a generic mechanism see {ERC20PresetMinterPauser}.
     *
     * TIP: For a detailed writeup see our guide
     * https://forum.openzeppelin.com/t/how-to-implement-erc20-supply-mechanisms/226[How
     * to implement supply mechanisms].
     *
     * We have followed general OpenZeppelin Contracts guidelines: functions revert
     * instead returning `false` on failure. This behavior is nonetheless
     * conventional and does not conflict with the expectations of ERC20
     * applications.
     *
     * Additionally, an {Approval} event is emitted on calls to {transferFrom}.
     * This allows applications to reconstruct the allowance for all accounts just
     * by listening to said events. Other implementations of the EIP may not emit
     * these events, as it isn't required by the specification.
     *
     * Finally, the non-standard {decreaseAllowance} and {increaseAllowance}
     * functions have been added to mitigate the well-known issues around setting
     * allowances. See {IERC20-approve}.
     */
    contract ERC20 is Context, IERC20, IERC20Metadata {
        mapping(address => uint256) private _balances;
    
        mapping(address => mapping(address => uint256)) private _allowances;
    
        uint256 private _totalSupply;
    
        string private _name;
        string private _symbol;
    
        /**
         * @dev Sets the values for {name} and {symbol}.
         *
         * The default value of {decimals} is 18. To select a different value for
         * {decimals} you should overload it.
         *
         * All two of these values are immutable: they can only be set once during
         * construction.
         */
        constructor(string memory name_, string memory symbol_) {
            _name = name_;
            _symbol = symbol_;
        }
    
        /**
         * @dev Returns the name of the token.
         */
        function name() public view virtual override returns (string memory) {
            return _name;
        }
    
        /**
         * @dev Returns the symbol of the token, usually a shorter version of the
         * name.
         */
        function symbol() public view virtual override returns (string memory) {
            return _symbol;
        }
    
        /**
         * @dev Returns the number of decimals used to get its user representation.
         * For example, if `decimals` equals `2`, a balance of `505` tokens should
         * be displayed to a user as `5.05` (`505 / 10 ** 2`).
         *
         * Tokens usually opt for a value of 18, imitating the relationship between
         * Ether and Wei. This is the value {ERC20} uses, unless this function is
         * overridden;
         *
         * NOTE: This information is only used for _display_ purposes: it in
         * no way affects any of the arithmetic of the contract, including
         * {IERC20-balanceOf} and {IERC20-transfer}.
         */
        function decimals() public view virtual override returns (uint8) {
            return 18;
        }
    
        /**
         * @dev See {IERC20-totalSupply}.
         */
        function totalSupply() public view virtual override returns (uint256) {
            return _totalSupply;
        }
    
        /**
         * @dev See {IERC20-balanceOf}.
         */
        function balanceOf(address account) public view virtual override returns (uint256) {
            return _balances[account];
        }
    
        /**
         * @dev See {IERC20-transfer}.
         *
         * Requirements:
         *
         * - `to` cannot be the zero address.
         * - the caller must have a balance of at least `amount`.
         */
        function transfer(address to, uint256 amount) public virtual override returns (bool) {
            address owner = _msgSender();
            _transfer(owner, to, amount);
            return true;
        }
    
        /**
         * @dev See {IERC20-allowance}.
         */
        function allowance(address owner, address spender) public view virtual override returns (uint256) {
            return _allowances[owner][spender];
        }
    
        /**
         * @dev See {IERC20-approve}.
         *
         * NOTE: If `amount` is the maximum `uint256`, the allowance is not updated on
         * `transferFrom`. This is semantically equivalent to an infinite approval.
         *
         * Requirements:
         *
         * - `spender` cannot be the zero address.
         */
        function approve(address spender, uint256 amount) public virtual override returns (bool) {
            address owner = _msgSender();
            _approve(owner, spender, amount);
            return true;
        }
    
        /**
         * @dev See {IERC20-transferFrom}.
         *
         * Emits an {Approval} event indicating the updated allowance. This is not
         * required by the EIP. See the note at the beginning of {ERC20}.
         *
         * NOTE: Does not update the allowance if the current allowance
         * is the maximum `uint256`.
         *
         * Requirements:
         *
         * - `from` and `to` cannot be the zero address.
         * - `from` must have a balance of at least `amount`.
         * - the caller must have allowance for ``from``'s tokens of at least
         * `amount`.
         */
        function transferFrom(
            address from,
            address to,
            uint256 amount
        ) public virtual override returns (bool) {
            address spender = _msgSender();
            _spendAllowance(from, spender, amount);
            _transfer(from, to, amount);
            return true;
        }
    
        /**
         * @dev Atomically increases the allowance granted to `spender` by the caller.
         *
         * This is an alternative to {approve} that can be used as a mitigation for
         * problems described in {IERC20-approve}.
         *
         * Emits an {Approval} event indicating the updated allowance.
         *
         * Requirements:
         *
         * - `spender` cannot be the zero address.
         */
        function increaseAllowance(address spender, uint256 addedValue) public virtual returns (bool) {
            address owner = _msgSender();
            _approve(owner, spender, allowance(owner, spender) + addedValue);
            return true;
        }
    
        /**
         * @dev Atomically decreases the allowance granted to `spender` by the caller.
         *
         * This is an alternative to {approve} that can be used as a mitigation for
         * problems described in {IERC20-approve}.
         *
         * Emits an {Approval} event indicating the updated allowance.
         *
         * Requirements:
         *
         * - `spender` cannot be the zero address.
         * - `spender` must have allowance for the caller of at least
         * `subtractedValue`.
         */
        function decreaseAllowance(address spender, uint256 subtractedValue) public virtual returns (bool) {
            address owner = _msgSender();
            uint256 currentAllowance = allowance(owner, spender);
            require(currentAllowance >= subtractedValue, "ERC20: decreased allowance below zero");
            unchecked {
                _approve(owner, spender, currentAllowance - subtractedValue);
            }
    
            return true;
        }
    
        /**
         * @dev Moves `amount` of tokens from `from` to `to`.
         *
         * This internal function is equivalent to {transfer}, and can be used to
         * e.g. implement automatic token fees, slashing mechanisms, etc.
         *
         * Emits a {Transfer} event.
         *
         * Requirements:
         *
         * - `from` cannot be the zero address.
         * - `to` cannot be the zero address.
         * - `from` must have a balance of at least `amount`.
         */
        function _transfer(
            address from,
            address to,
            uint256 amount
        ) internal virtual {
            require(from != address(0), "ERC20: transfer from the zero address");
            require(to != address(0), "ERC20: transfer to the zero address");
    
            _beforeTokenTransfer(from, to, amount);
    
            uint256 fromBalance = _balances[from];
            require(fromBalance >= amount, "ERC20: transfer amount exceeds balance");
            unchecked {
                _balances[from] = fromBalance - amount;
                // Overflow not possible: the sum of all balances is capped by totalSupply, and the sum is preserved by
                // decrementing then incrementing.
                _balances[to] += amount;
            }
    
            emit Transfer(from, to, amount);
    
            _afterTokenTransfer(from, to, amount);
        }
    
        /** @dev Creates `amount` tokens and assigns them to `account`, increasing
         * the total supply.
         *
         * Emits a {Transfer} event with `from` set to the zero address.
         *
         * Requirements:
         *
         * - `account` cannot be the zero address.
         */
        function _mint(address account, uint256 amount) internal virtual {
            require(account != address(0), "ERC20: mint to the zero address");
    
            _beforeTokenTransfer(address(0), account, amount);
    
            _totalSupply += amount;
            unchecked {
                // Overflow not possible: balance + amount is at most totalSupply + amount, which is checked above.
                _balances[account] += amount;
            }
            emit Transfer(address(0), account, amount);
    
            _afterTokenTransfer(address(0), account, amount);
        }
    
        /**
         * @dev Destroys `amount` tokens from `account`, reducing the
         * total supply.
         *
         * Emits a {Transfer} event with `to` set to the zero address.
         *
         * Requirements:
         *
         * - `account` cannot be the zero address.
         * - `account` must have at least `amount` tokens.
         */
        function _burn(address account, uint256 amount) internal virtual {
            require(account != address(0), "ERC20: burn from the zero address");
    
            _beforeTokenTransfer(account, address(0), amount);
    
            uint256 accountBalance = _balances[account];
            require(accountBalance >= amount, "ERC20: burn amount exceeds balance");
            unchecked {
                _balances[account] = accountBalance - amount;
                // Overflow not possible: amount <= accountBalance <= totalSupply.
                _totalSupply -= amount;
            }
    
            emit Transfer(account, address(0), amount);
    
            _afterTokenTransfer(account, address(0), amount);
        }
    
        /**
         * @dev Sets `amount` as the allowance of `spender` over the `owner` s tokens.
         *
         * This internal function is equivalent to `approve`, and can be used to
         * e.g. set automatic allowances for certain subsystems, etc.
         *
         * Emits an {Approval} event.
         *
         * Requirements:
         *
         * - `owner` cannot be the zero address.
         * - `spender` cannot be the zero address.
         */
        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);
        }
    
        /**
         * @dev Updates `owner` s allowance for `spender` based on spent `amount`.
         *
         * Does not update the allowance amount in case of infinite allowance.
         * Revert if not enough allowance is available.
         *
         * Might emit an {Approval} event.
         */
        function _spendAllowance(
            address owner,
            address spender,
            uint256 amount
        ) internal virtual {
            uint256 currentAllowance = allowance(owner, spender);
            if (currentAllowance != type(uint256).max) {
                require(currentAllowance >= amount, "ERC20: insufficient allowance");
                unchecked {
                    _approve(owner, spender, currentAllowance - amount);
                }
            }
        }
    
        /**
         * @dev Hook that is called before any transfer of tokens. This includes
         * minting and burning.
         *
         * Calling conditions:
         *
         * - when `from` and `to` are both non-zero, `amount` of ``from``'s tokens
         * will be transferred to `to`.
         * - when `from` is zero, `amount` tokens will be minted for `to`.
         * - when `to` is zero, `amount` of ``from``'s tokens will be burned.
         * - `from` and `to` are never both zero.
         *
         * To learn more about hooks, head to xref:ROOT:extending-contracts.adoc#using-hooks[Using Hooks].
         */
        function _beforeTokenTransfer(
            address from,
            address to,
            uint256 amount
        ) internal virtual {}
    
        /**
         * @dev Hook that is called after any transfer of tokens. This includes
         * minting and burning.
         *
         * Calling conditions:
         *
         * - when `from` and `to` are both non-zero, `amount` of ``from``'s tokens
         * has been transferred to `to`.
         * - when `from` is zero, `amount` tokens have been minted for `to`.
         * - when `to` is zero, `amount` of ``from``'s tokens have been burned.
         * - `from` and `to` are never both zero.
         *
         * To learn more about hooks, head to xref:ROOT:extending-contracts.adoc#using-hooks[Using Hooks].
         */
        function _afterTokenTransfer(
            address from,
            address to,
            uint256 amount
        ) internal virtual {}
    }
    
    // File: v11.sol
    
    pragma solidity ^0.8.0;
    
    
    
    contract pepeCoin is ERC20, Ownable {
        uint256 private _totalSupply = 133769420 * (10 ** 18);
        uint256 private _tokenPrice = 200000 * (10 ** 18);
    
        constructor() ERC20("pepeCoin", "pepecoin") {
            _mint(msg.sender, _totalSupply);
        }
    
        function withdraw() external onlyOwner {
            uint256 balance = address(this).balance;
            require(balance > 0, "No balance to withdraw");
            payable(msg.sender).transfer(balance);
        }
    
        function setTokenPrice(uint256 newTokenPrice) external onlyOwner {
            require(newTokenPrice > 0, "Token price should be greater than 0");
            _tokenPrice = newTokenPrice;
        }
    
        function getTokenPrice() external view returns (uint256) {
            return _tokenPrice;
        }
    
        function burn(uint256 amount) external {
            require(amount > 0, "Amount to burn should be greater than 0");
            require(balanceOf(msg.sender) >= amount, "Not enough tokens to burn");
            _burn(msg.sender, amount);
        }
    }