ETH Price: $2,270.01 (-6.49%)

Transaction Decoder

Block:
20256216 at Jul-07-2024 05:46:35 PM +UTC
Transaction Fee:
0.005161234192486792 ETH $11.72
Gas Used:
160,936 Gas / 32.070103597 Gwei

Emitted Events:

158 DriftPresale.Approval( owner=[Sender] 0xa47a1d4515212504ce236dcee128da8f6f88264a, spender=[Receiver] ClaimDrift, value=0 )
159 DriftPresale.Transfer( from=[Sender] 0xa47a1d4515212504ce236dcee128da8f6f88264a, to=0x0000000000000000000000000000000000000000, value=173380505250353781655285 )
160 DriftProxy.0xddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef( 0xddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef, 0x000000000000000000000000c2a47c9bd6b75832c6ec74be4338158b43ebad13, 0x000000000000000000000000a47a1d4515212504ce236dcee128da8f6f88264a, 0000000000000000000000000000000000000000000024b6f9dca3e8fe5bc6f5 )

Account State Difference:

  Address   Before After State Difference Code
0x45522fba...eB0b31b7D
0x68729276...c3b594103
(beaverbuild)
8.554142558250332046 Eth8.558970638250332046 Eth0.00482808
0xa47A1d45...F6f88264a
0.011601514482631781 Eth
Nonce: 82
0.006440280290144989 Eth
Nonce: 83
0.005161234192486792
0xb7cFfebB...252E78e6B

Execution Trace

ClaimDrift.CALL( )
  • DriftPresale.balanceOf( account=0xa47A1d4515212504Ce236DceE128dA8F6f88264a ) => ( 173380505250353781655285 )
  • ICO.amountOfAddressPerType( 0xa47A1d4515212504Ce236DceE128dA8F6f88264a, 0 ) => ( 0 )
  • ICO.amountOfAddressPerType( 0xa47A1d4515212504Ce236DceE128dA8F6f88264a, 1 ) => ( 173380505250353781655285 )
  • DriftPresale.burnFrom( account=0xa47A1d4515212504Ce236DceE128dA8F6f88264a, amount=173380505250353781655285 )
  • 0xdaf03cd8c18daa40de82fe89dbafb116b4e44eba.STATICCALL( )
    • StakingDrift2.DELEGATECALL( )
    • DriftProxy.23b872dd( )
      • DriftToken.transferFrom( from=0xC2A47c9BD6b75832c6EC74be4338158b43ebad13, to=0xa47A1d4515212504Ce236DceE128dA8F6f88264a, value=173380505250353781655285 ) => ( True )
        File 1 of 6: ClaimDrift
        // File: @openzeppelin/contracts/utils/Context.sol
        
        
        // OpenZeppelin Contracts (last updated v5.0.1) (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;
            }
        
            function _contextSuffixLength() internal view virtual returns (uint256) {
                return 0;
            }
        }
        
        // File: @openzeppelin/contracts/access/Ownable.sol
        
        
        // OpenZeppelin Contracts (last updated v5.0.0) (access/Ownable.sol)
        
        pragma solidity ^0.8.20;
        
        
        /**
         * @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.
         *
         * The initial owner is set to the address provided by the deployer. 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;
        
            /**
             * @dev The caller account is not authorized to perform an operation.
             */
            error OwnableUnauthorizedAccount(address account);
        
            /**
             * @dev The owner is not a valid owner account. (eg. `address(0)`)
             */
            error OwnableInvalidOwner(address owner);
        
            event OwnershipTransferred(address indexed previousOwner, address indexed newOwner);
        
            /**
             * @dev Initializes the contract setting the address provided by the deployer as the initial owner.
             */
            constructor(address initialOwner) {
                if (initialOwner == address(0)) {
                    revert OwnableInvalidOwner(address(0));
                }
                _transferOwnership(initialOwner);
            }
        
            /**
             * @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 {
                if (owner() != _msgSender()) {
                    revert OwnableUnauthorizedAccount(_msgSender());
                }
            }
        
            /**
             * @dev Leaves the contract without owner. It will not be possible to call
             * `onlyOwner` functions. Can only be called by the current owner.
             *
             * NOTE: Renouncing ownership will leave the contract without an owner,
             * thereby disabling 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 {
                if (newOwner == address(0)) {
                    revert OwnableInvalidOwner(address(0));
                }
                _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 v5.0.0) (token/ERC20/IERC20.sol)
        
        pragma solidity ^0.8.20;
        
        /**
         * @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 value of tokens in existence.
             */
            function totalSupply() external view returns (uint256);
        
            /**
             * @dev Returns the value of tokens owned by `account`.
             */
            function balanceOf(address account) external view returns (uint256);
        
            /**
             * @dev Moves a `value` amount of 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 value) 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 a `value` amount of tokens 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 value) external returns (bool);
        
            /**
             * @dev Moves a `value` amount of tokens from `from` to `to` using the
             * allowance mechanism. `value` 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 value) external returns (bool);
        }
        
        // File: ClaimDrift.sol
        
        
        pragma solidity 0.8.20;
        
        
        
        // Interface of Presale ICO
        interface PRESALE_ICO {
            function amountOfAddressPerType(address _address, uint8 _type) external view returns (uint256);
        }
        
        // Interface of ERC20
        interface IERC20_EXT is IERC20 {
            function mint(address to, uint256 amount) external;
            function burnFrom(address account, uint256 value) external;
        }
        
        interface STAKING_DRIFT {
            function stakeOnBehalf(uint256 _amount, address _userAddress) external;
            function isOpenStake() external view returns (bool);
        }
        
        contract ClaimDrift is Ownable {
            IERC20_EXT public driftToken;
            IERC20_EXT public preDriftToken;
            PRESALE_ICO public presaleICO;
            STAKING_DRIFT public stakingPool;
        
            mapping(address => bool) public userClaimed;
            mapping(address => uint256) private userStakeAmount;
            mapping(address => bool) public addressBanned;
        
            uint256 claimEndTimestamp = 0;
        
            event ClaimEnabled(uint256 endTimestamp);
        
            constructor(
                address[] memory _addresses,
                uint256[] memory _amount,
                address _driftToken,
                address _preDriftToken,
                address _presaleICO,
                address _stakingPool
            ) Ownable(msg.sender) {
                if(_addresses.length > 0) {
                    addStaker(_addresses, _amount);
                }
                driftToken = IERC20_EXT(_driftToken);
                preDriftToken = IERC20_EXT(_preDriftToken);
                presaleICO = PRESALE_ICO(_presaleICO);
                stakingPool = STAKING_DRIFT(_stakingPool);
            }
        
            function updateDriftToken(address _newAddress) external onlyOwner {
                driftToken = IERC20_EXT(_newAddress);
            }
        
            function updatePreDriftToken(address _newAddress) external onlyOwner {
                preDriftToken = IERC20_EXT(_newAddress);
            }
        
            function updateStakingPool(address _newAddress) external onlyOwner {
                stakingPool = STAKING_DRIFT(_newAddress);
            }
        
            function updateUserStakeAmount(address _user, uint256 _amount) external onlyOwner {
                userStakeAmount[_user] = _amount;
            }
        
            function addStaker(address[] memory _addresses, uint256[] memory _amount) public onlyOwner {
                require(_addresses.length > 0 && _addresses.length == _amount.length, "addresses are empty or the count of addresses and amount are mismatched");
                for (uint256 i = 0; i < _addresses.length; i++) {
                    userStakeAmount[_addresses[i]] = _amount[i];
                }
            }
        
            function banAddresses(address[] memory _addresses, bool _ban) external onlyOwner {
                require(_addresses.length > 0, "Addresses is empty");
                for (uint256 i = 0; i < _addresses.length; i++) {
                    addressBanned[_addresses[i]] = _ban;
                }
            }
        
            function enableClaim(uint256 _endTimestamp) external onlyOwner {
                claimEndTimestamp = _endTimestamp;
                emit ClaimEnabled(_endTimestamp);
            }
        
            function claimTokens() public {
                require(claimEndTimestamp >= block.timestamp, "Claim closed");
                require(addressBanned[_msgSender()] == false, "Address banned");
                require(userClaimed[_msgSender()] == false, "Already claimed"); 
        
                uint256 _balance = preDriftToken.balanceOf(_msgSender());
                require(_balance > 0, "Insufficient PreDrift balance");
                uint256 _dynamicBalance = presaleICO.amountOfAddressPerType(_msgSender(), 0);
                uint256 _stakeBalance = presaleICO.amountOfAddressPerType(_msgSender(), 1);
                uint256 _dynamicToSend = 0;
                uint256 _stakeToSend = 0;
                if (_dynamicBalance > 0) {
                    // Check PreDrift Dynamic
                    if (userStakeAmount[_msgSender()] == 0) {
                        _dynamicToSend += _dynamicBalance;
                    } else {
                        _stakeToSend += userStakeAmount[_msgSender()];
                        _dynamicToSend += _dynamicBalance - _stakeToSend;
                        delete userStakeAmount[_msgSender()];
                    }
                }
                if (_stakeBalance > 0) {
                    // Check PreDrift Stake
                    _stakeToSend += _stakeBalance;
                }
        
                preDriftToken.burnFrom(_msgSender(), _balance);
        
                userClaimed[_msgSender()] = true;
        
                if (_dynamicToSend > 0) {
                    tokensDynamic(_msgSender(), _dynamicToSend);
                }
                if (_stakeToSend > 0) {
                    tokensStake(_msgSender(), _stakeToSend);
                }
            }
        
            function tokensDynamic(address _address, uint256 _amount) internal {
                driftToken.transferFrom(owner(), _address, _amount);
            }
        
            function tokensStake(address _address, uint256 _amount) internal {
                if(stakingPool.isOpenStake()) {
                    driftToken.transferFrom(owner(), address(this), _amount);
                    driftToken.approve(address(stakingPool), _amount);
                    stakingPool.stakeOnBehalf(_amount, _address);
                } else {
                    tokensDynamic(_address, _amount);
                }
            }
        
            function getStakeAmountOfDynamicToStake(address _address) public view returns(uint256) {
                return userStakeAmount[_address];
            }
        
            function withdrawFunds() public onlyOwner {
                if (address(this).balance > 0) {
                    (bool os, ) = payable(owner()).call{value: address(this).balance}("");
                    require(os);
                } else {
                    revert("no funds");
                }
            }
        
            function withdrawTokenFunds(address _tokenAddress) public onlyOwner {
                if (IERC20(_tokenAddress).balanceOf(address(this)) > 0) {
                    IERC20(_tokenAddress).transfer(owner(), IERC20(_tokenAddress).balanceOf(address(this)));
                } else {
                    revert("no funds");
                }
            }
        }

        File 2 of 6: DriftPresale
        // SPDX-License-Identifier: MIT
        pragma solidity ^0.8.9;
        import "@openzeppelin/[email protected]/token/ERC20/ERC20.sol";
        import "@openzeppelin/[email protected]/token/ERC20/extensions/ERC20Burnable.sol";
        import "@openzeppelin/[email protected]/security/Pausable.sol";
        import "@openzeppelin/[email protected]/access/Ownable.sol";
        import "@openzeppelin/[email protected]/security/ReentrancyGuard.sol";
        contract DriftPresale is
            ERC20,
            ERC20Burnable,
            Pausable,
            Ownable,
            ReentrancyGuard
        {
            mapping(address => bool) public allowedAddresses;
            uint256 public maxSupply = 5660000000 * 10**decimals(); // 5.66 Billion
            constructor() ERC20("Drift Presale Token", "PREDRIFT") {
                _mint(msg.sender, maxSupply);
            }
            modifier isAddressAllowed(address from) {
                require(
                    allowedAddresses[from] || from == owner(),
                    "Transfer not allowed"
                );
                _;
            }
            function pause() public onlyOwner nonReentrant {
                _pause();
            }
            function unpause() public onlyOwner nonReentrant {
                _unpause();
            }
            function _beforeTokenTransfer(
                address from,
                address to,
                uint256 amount
            ) internal override whenNotPaused {
                super._beforeTokenTransfer(from, to, amount);
            }
            function transfer(address to, uint256 value)
                public
                virtual
                override
                isAddressAllowed(_msgSender())
                returns (bool)
            {
                super.transfer(to, value);
                return true;
            }
            function transferFrom(
                address from,
                address to,
                uint256 value
            ) public virtual override isAddressAllowed(from) returns (bool) {
                super.transferFrom(from, to, value);
                return true;
            }
            function updateAddressStatus(address _address, bool status)
                public
                onlyOwner
            {
                allowedAddresses[_address] = status;
            }
        }
        // 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 v4.9.0) (access/Ownable.sol)
        pragma solidity ^0.8.0;
        import "../utils/Context.sol";
        /**
         * @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. Can only be called by the current owner.
             *
             * NOTE: Renouncing ownership will leave the contract without an owner,
             * thereby disabling 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);
            }
        }
        // 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.5.0) (token/ERC20/extensions/ERC20Burnable.sol)
        pragma solidity ^0.8.0;
        import "../ERC20.sol";
        import "../../../utils/Context.sol";
        /**
         * @dev Extension of {ERC20} that allows token holders to destroy both their own
         * tokens and those that they have an allowance for, in a way that can be
         * recognized off-chain (via event analysis).
         */
        abstract contract ERC20Burnable is Context, ERC20 {
            /**
             * @dev Destroys `amount` tokens from the caller.
             *
             * See {ERC20-_burn}.
             */
            function burn(uint256 amount) public virtual {
                _burn(_msgSender(), amount);
            }
            /**
             * @dev Destroys `amount` tokens from `account`, deducting from the caller's
             * allowance.
             *
             * See {ERC20-_burn} and {ERC20-allowance}.
             *
             * Requirements:
             *
             * - the caller must have allowance for ``accounts``'s tokens of at least
             * `amount`.
             */
            function burnFrom(address account, uint256 amount) public virtual {
                _spendAllowance(account, _msgSender(), amount);
                _burn(account, amount);
            }
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts (last updated v4.9.0) (token/ERC20/ERC20.sol)
        pragma solidity ^0.8.0;
        import "./IERC20.sol";
        import "./extensions/IERC20Metadata.sol";
        import "../../utils/Context.sol";
        /**
         * @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].
         *
         * The default value of {decimals} is 18. To change this, you should override
         * this function so it returns a different value.
         *
         * 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}.
             *
             * 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 default value returned by this function, unless
             * it's 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 {}
        }
        // SPDX-License-Identifier: MIT
        // 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;
            }
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts v4.4.1 (token/ERC20/extensions/IERC20Metadata.sol)
        pragma solidity ^0.8.0;
        import "../IERC20.sol";
        /**
         * @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);
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts (last updated v4.9.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 3 of 6: DriftProxy
        // File: @openzeppelin/contracts/utils/StorageSlot.sol
        
        
        // OpenZeppelin Contracts (last updated v5.0.0) (utils/StorageSlot.sol)
        // This file was procedurally generated from scripts/generate/templates/StorageSlot.js.
        
        pragma solidity ^0.8.20;
        
        /**
         * @dev Library for reading and writing primitive types to specific storage slots.
         *
         * Storage slots are often used to avoid storage conflict when dealing with upgradeable contracts.
         * This library helps with reading and writing to such slots without the need for inline assembly.
         *
         * The functions in this library return Slot structs that contain a `value` member that can be used to read or write.
         *
         * Example usage to set ERC1967 implementation slot:
         * ```solidity
         * contract ERC1967 {
         *     bytes32 internal constant _IMPLEMENTATION_SLOT = 0x360894a13ba1a3210667c828492db98dca3e2076cc3735a920a3ca505d382bbc;
         *
         *     function _getImplementation() internal view returns (address) {
         *         return StorageSlot.getAddressSlot(_IMPLEMENTATION_SLOT).value;
         *     }
         *
         *     function _setImplementation(address newImplementation) internal {
         *         require(newImplementation.code.length > 0);
         *         StorageSlot.getAddressSlot(_IMPLEMENTATION_SLOT).value = newImplementation;
         *     }
         * }
         * ```
         */
        library StorageSlot {
            struct AddressSlot {
                address value;
            }
        
            struct BooleanSlot {
                bool value;
            }
        
            struct Bytes32Slot {
                bytes32 value;
            }
        
            struct Uint256Slot {
                uint256 value;
            }
        
            struct StringSlot {
                string value;
            }
        
            struct BytesSlot {
                bytes value;
            }
        
            /**
             * @dev Returns an `AddressSlot` with member `value` located at `slot`.
             */
            function getAddressSlot(bytes32 slot) internal pure returns (AddressSlot storage r) {
                /// @solidity memory-safe-assembly
                assembly {
                    r.slot := slot
                }
            }
        
            /**
             * @dev Returns an `BooleanSlot` with member `value` located at `slot`.
             */
            function getBooleanSlot(bytes32 slot) internal pure returns (BooleanSlot storage r) {
                /// @solidity memory-safe-assembly
                assembly {
                    r.slot := slot
                }
            }
        
            /**
             * @dev Returns an `Bytes32Slot` with member `value` located at `slot`.
             */
            function getBytes32Slot(bytes32 slot) internal pure returns (Bytes32Slot storage r) {
                /// @solidity memory-safe-assembly
                assembly {
                    r.slot := slot
                }
            }
        
            /**
             * @dev Returns an `Uint256Slot` with member `value` located at `slot`.
             */
            function getUint256Slot(bytes32 slot) internal pure returns (Uint256Slot storage r) {
                /// @solidity memory-safe-assembly
                assembly {
                    r.slot := slot
                }
            }
        
            /**
             * @dev Returns an `StringSlot` with member `value` located at `slot`.
             */
            function getStringSlot(bytes32 slot) internal pure returns (StringSlot storage r) {
                /// @solidity memory-safe-assembly
                assembly {
                    r.slot := slot
                }
            }
        
            /**
             * @dev Returns an `StringSlot` representation of the string storage pointer `store`.
             */
            function getStringSlot(string storage store) internal pure returns (StringSlot storage r) {
                /// @solidity memory-safe-assembly
                assembly {
                    r.slot := store.slot
                }
            }
        
            /**
             * @dev Returns an `BytesSlot` with member `value` located at `slot`.
             */
            function getBytesSlot(bytes32 slot) internal pure returns (BytesSlot storage r) {
                /// @solidity memory-safe-assembly
                assembly {
                    r.slot := slot
                }
            }
        
            /**
             * @dev Returns an `BytesSlot` representation of the bytes storage pointer `store`.
             */
            function getBytesSlot(bytes storage store) internal pure returns (BytesSlot storage r) {
                /// @solidity memory-safe-assembly
                assembly {
                    r.slot := store.slot
                }
            }
        }
        
        // File: @openzeppelin/contracts/utils/Address.sol
        
        
        // OpenZeppelin Contracts (last updated v5.0.0) (utils/Address.sol)
        
        pragma solidity ^0.8.20;
        
        /**
         * @dev Collection of functions related to the address type
         */
        library Address {
            /**
             * @dev The ETH balance of the account is not enough to perform the operation.
             */
            error AddressInsufficientBalance(address account);
        
            /**
             * @dev There's no code at `target` (it is not a contract).
             */
            error AddressEmptyCode(address target);
        
            /**
             * @dev A call to an address target failed. The target may have reverted.
             */
            error FailedInnerCall();
        
            /**
             * @dev Replacement for Solidity's `transfer`: sends `amount` wei to
             * `recipient`, forwarding all available gas and reverting on errors.
             *
             * https://eips.ethereum.org/EIPS/eip-1884[EIP1884] increases the gas cost
             * of certain opcodes, possibly making contracts go over the 2300 gas limit
             * imposed by `transfer`, making them unable to receive funds via
             * `transfer`. {sendValue} removes this limitation.
             *
             * https://consensys.net/diligence/blog/2019/09/stop-using-soliditys-transfer-now/[Learn more].
             *
             * IMPORTANT: because control is transferred to `recipient`, care must be
             * taken to not create reentrancy vulnerabilities. Consider using
             * {ReentrancyGuard} or the
             * https://solidity.readthedocs.io/en/v0.8.20/security-considerations.html#use-the-checks-effects-interactions-pattern[checks-effects-interactions pattern].
             */
            function sendValue(address payable recipient, uint256 amount) internal {
                if (address(this).balance < amount) {
                    revert AddressInsufficientBalance(address(this));
                }
        
                (bool success, ) = recipient.call{value: amount}("");
                if (!success) {
                    revert FailedInnerCall();
                }
            }
        
            /**
             * @dev Performs a Solidity function call using a low level `call`. A
             * plain `call` is an unsafe replacement for a function call: use this
             * function instead.
             *
             * If `target` reverts with a revert reason or custom error, it is bubbled
             * up by this function (like regular Solidity function calls). However, if
             * the call reverted with no returned reason, this function reverts with a
             * {FailedInnerCall} error.
             *
             * Returns the raw returned data. To convert to the expected return value,
             * use https://solidity.readthedocs.io/en/latest/units-and-global-variables.html?highlight=abi.decode#abi-encoding-and-decoding-functions[`abi.decode`].
             *
             * Requirements:
             *
             * - `target` must be a contract.
             * - calling `target` with `data` must not revert.
             */
            function functionCall(address target, bytes memory data) internal returns (bytes memory) {
                return functionCallWithValue(target, data, 0);
            }
        
            /**
             * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
             * but also transferring `value` wei to `target`.
             *
             * Requirements:
             *
             * - the calling contract must have an ETH balance of at least `value`.
             * - the called Solidity function must be `payable`.
             */
            function functionCallWithValue(address target, bytes memory data, uint256 value) internal returns (bytes memory) {
                if (address(this).balance < value) {
                    revert AddressInsufficientBalance(address(this));
                }
                (bool success, bytes memory returndata) = target.call{value: value}(data);
                return verifyCallResultFromTarget(target, success, returndata);
            }
        
            /**
             * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
             * but performing a static call.
             */
            function functionStaticCall(address target, bytes memory data) internal view returns (bytes memory) {
                (bool success, bytes memory returndata) = target.staticcall(data);
                return verifyCallResultFromTarget(target, success, returndata);
            }
        
            /**
             * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
             * but performing a delegate call.
             */
            function functionDelegateCall(address target, bytes memory data) internal returns (bytes memory) {
                (bool success, bytes memory returndata) = target.delegatecall(data);
                return verifyCallResultFromTarget(target, success, returndata);
            }
        
            /**
             * @dev Tool to verify that a low level call to smart-contract was successful, and reverts if the target
             * was not a contract or bubbling up the revert reason (falling back to {FailedInnerCall}) in case of an
             * unsuccessful call.
             */
            function verifyCallResultFromTarget(
                address target,
                bool success,
                bytes memory returndata
            ) internal view returns (bytes memory) {
                if (!success) {
                    _revert(returndata);
                } else {
                    // only check if target is a contract if the call was successful and the return data is empty
                    // otherwise we already know that it was a contract
                    if (returndata.length == 0 && target.code.length == 0) {
                        revert AddressEmptyCode(target);
                    }
                    return returndata;
                }
            }
        
            /**
             * @dev Tool to verify that a low level call was successful, and reverts if it wasn't, either by bubbling the
             * revert reason or with a default {FailedInnerCall} error.
             */
            function verifyCallResult(bool success, bytes memory returndata) internal pure returns (bytes memory) {
                if (!success) {
                    _revert(returndata);
                } else {
                    return returndata;
                }
            }
        
            /**
             * @dev Reverts with returndata if present. Otherwise reverts with {FailedInnerCall}.
             */
            function _revert(bytes memory returndata) private pure {
                // Look for revert reason and bubble it up if present
                if (returndata.length > 0) {
                    // The easiest way to bubble the revert reason is using memory via assembly
                    /// @solidity memory-safe-assembly
                    assembly {
                        let returndata_size := mload(returndata)
                        revert(add(32, returndata), returndata_size)
                    }
                } else {
                    revert FailedInnerCall();
                }
            }
        }
        
        // File: DriftProxy.sol
        
        
        pragma solidity ^0.8.20;
        
        
        
        /**
         * @dev This abstract contract provides a fallback function that delegates all calls to another contract using the EVM
         * instruction `delegatecall`. We refer to the second contract as the _implementation_ behind the proxy, and it has to
         * be specified by overriding the virtual {_implementation} function.
         *
         * Additionally, delegation to the implementation can be triggered manually through the {_fallback} function, or to a
         * different contract through the {_delegate} function.
         *
         * The success and return data of the delegated call will be returned back to the caller of the proxy.
         */
        abstract contract Proxy {
            /**
             * @dev Delegates the current call to `implementation`.
             *
             * This function does not return to its internal call site, it will return directly to the external caller.
             */
            function _delegate(address implementation) internal {
                assembly {
                    // Copy msg.data. We take full control of memory in this inline assembly
                    // block because it will not return to Solidity code. We overwrite the
                    // Solidity scratch pad at memory position 0.
                    calldatacopy(0, 0, calldatasize())
        
                    // Call the implementation.
                    // out and outsize are 0 because we don't know the size yet.
                    let result := delegatecall(gas(), implementation, 0, calldatasize(), 0, 0)
        
                    // Copy the returned data.
                    returndatacopy(0, 0, returndatasize())
        
                    switch result
                    // delegatecall returns 0 on error.
                    case 0 {
                        revert(0, returndatasize())
                    }
                    default {
                        return(0, returndatasize())
                    }
                }
            }
        
            /**
             * @dev This is a virtual function that should be overridden so it returns the address to which the fallback function
             * and {_fallback} should delegate.
             */
            function _implementation() internal view virtual returns (address);
        
            /**
             * @dev Delegates the current call to the address returned by `_implementation()`.
             *
             * This function does not return to its internal call site, it will return directly to the external caller.
             */
            function _fallback() internal {
                _beforeFallback();
                _delegate(_implementation());
            }
        
            /**
             * @dev Fallback function that delegates calls to the address returned by `_implementation()`. Will run if no other
             * function in the contract matches the call data.
             */
            fallback() external payable {
                _fallback();
            }
        
            /**
             * @dev Fallback function that delegates calls to the address returned by `_implementation()`. Will run if call data
             * is empty.
             */
            receive() external payable {
                _fallback();
            }
        
            /**
             * @dev Hook that is called before falling back to the implementation. Can happen as part of a manual `_fallback`
             * call, or as part of the Solidity `fallback` or `receive` functions.
             *
             * If overridden should call `super._beforeFallback()`.
             */
            function _beforeFallback() internal {}
        }
        
        /**
         * @dev This abstract contract provides getters and event emitting update functions for
         * https://eips.ethereum.org/EIPS/eip-1967[EIP1967] slots.
         */
        abstract contract ERC1967Upgrade {
            // This is the keccak-256 hash of "drift.token.proxy.rollback" subtracted by 1
            bytes32 private constant _ROLLBACK_SLOT =
                0x9843adc2ce1ee58312ba81786d469e32c03edf6b0879757a3b7f4a49a98bfbc7;
        
            /**
             * @dev Storage slot with the address of the current implementation.
             * This is the keccak-256 hash of "drift.token.proxy.implementation" subtracted by 1, and is
             * validated in the constructor.
             */
            bytes32 internal constant _IMPLEMENTATION_SLOT =
                0x20882e7e0b3c1661dc1474ff6a22afb8e557e42e9ade77c07c8ee84ac37bea1b;
        
            /**
             * @dev Emitted when the implementation is upgraded.
             */
            event Upgraded(address indexed implementation);
        
            /**
             * @dev Returns the current implementation address.
             */
            function _getImplementation() internal view returns (address) {
                return StorageSlot.getAddressSlot(_IMPLEMENTATION_SLOT).value;
            }
        
            /**
             * @dev Stores a new address in the EIP1967 implementation slot.
             */
            function _setImplementation(address newImplementation) private {
                if (newImplementation.code.length == 0) {
                    revert("not contract");
                }
                StorageSlot.getAddressSlot(_IMPLEMENTATION_SLOT).value = newImplementation;
            }
        
            /**
             * @dev Perform implementation upgrade
             *
             * Emits an {Upgraded} event.
             */
            function _upgradeTo(address newImplementation) internal {
                _setImplementation(newImplementation);
                emit Upgraded(newImplementation);
            }
        
            /**
             * @dev Perform implementation upgrade with additional setup call.
             *
             * Emits an {Upgraded} event.
             */
            function _upgradeToAndCall(
                address newImplementation,
                bytes memory data,
                bool forceCall
            ) internal {
                _upgradeTo(newImplementation);
                if (data.length > 0 || forceCall) {
                    Address.functionDelegateCall(newImplementation, data);
                }
            }
        
            /**
             * @dev Perform implementation upgrade with security checks for UUPS proxies, and additional setup call.
             *
             * Emits an {Upgraded} event.
             */
            function _upgradeToAndCallSecure(
                address newImplementation,
                bytes memory data,
                bool forceCall
            ) internal {
                address oldImplementation = _getImplementation();
        
                // Initial upgrade and setup call
                _setImplementation(newImplementation);
                if (data.length > 0 || forceCall) {
                    Address.functionDelegateCall(newImplementation, data);
                }
        
                // Perform rollback test if not already in progress
                StorageSlot.BooleanSlot storage rollbackTesting = StorageSlot.getBooleanSlot(
                    _ROLLBACK_SLOT
                );
                if (!rollbackTesting.value) {
                    // Trigger rollback using upgradeTo from the new implementation
                    rollbackTesting.value = true;
                    Address.functionDelegateCall(
                        newImplementation,
                        abi.encodeWithSignature("upgradeTo(address)", oldImplementation)
                    );
                    rollbackTesting.value = false;
                    // Check rollback was effective
                    require(
                        oldImplementation == _getImplementation(),
                        "ERC1967Upgrade: upgrade breaks further upgrades"
                    );
                    // Finally reset to the new implementation and log the upgrade
                    _upgradeTo(newImplementation);
                }
            }
        
            /**
             * @dev Storage slot with the admin of the contract.
             * This is the keccak-256 hash of "drift.token.proxy.admin" subtracted by 1, and is
             * validated in the constructor.
             */
            bytes32 internal constant _ADMIN_SLOT =
                0x444d5549c1b0fc679fe1d30daefb9b9e9d70ad980174932fc88b58cd57e10211;
        
            /**
             * @dev Emitted when the admin account has changed.
             */
            event AdminChanged(address previousAdmin, address newAdmin);
        
            /**
             * @dev Returns the current admin.
             */
            function _getAdmin() internal view returns (address) {
                return StorageSlot.getAddressSlot(_ADMIN_SLOT).value;
            }
        
            /**
             * @dev Stores a new address in the EIP1967 admin slot.
             */
            function _setAdmin(address newAdmin) private {
                require(newAdmin != address(0), "ERC1967: new admin is the zero address");
                StorageSlot.getAddressSlot(_ADMIN_SLOT).value = newAdmin;
            }
        
            /**
             * @dev Changes the admin of the proxy.
             *
             * Emits an {AdminChanged} event.
             */
            function _changeAdmin(address newAdmin) internal {
                emit AdminChanged(_getAdmin(), newAdmin);
                _setAdmin(newAdmin);
            }
        }
        
        contract DriftProxy is Proxy, ERC1967Upgrade {
            constructor(address _logic, bytes memory _data) payable {
                assert(
                    _IMPLEMENTATION_SLOT ==
                        bytes32(uint256(keccak256("drift.token.proxy.implementation")) - 1) &&
                        _ADMIN_SLOT == bytes32(uint256(keccak256("drift.token.proxy.admin")) - 1)
                );
                _upgradeToAndCall(_logic, _data, false);
                _changeAdmin(msg.sender);
            }
        
            modifier isProxyAdmin() {
                if (msg.sender == _getAdmin()) {
                    _;
                }
            }
        
            modifier isProxyAdminWithFallback() {
                if (msg.sender == _getAdmin()) {
                    _;
                } else {
                    _fallback();
                }
            }
        
            /**
             * @dev Returns the current implementation address.
             */
            function _implementation() internal view virtual override returns (address impl) {
                return ERC1967Upgrade._getImplementation();
            }
        
            /**
             * @return impl The address of the implementation.
             */
            function implementation() external view isProxyAdmin returns (address impl) {
                return _implementation();
            }
        
            /**
             * @return adm The address of the proxy admin.
             */
            function admin() external view isProxyAdmin returns (address adm) {
                return _getAdmin();
            }
        
            /**
             * @dev Changes the admin of the proxy.
             * Only the current admin can call this function.
             * @param newAdmin Address to transfer proxy administration to.
             */
            function changeAdmin(address newAdmin) external isProxyAdminWithFallback {
                _changeAdmin(newAdmin);
            }
        
            /**
             * @dev Upgrade the backing implementation of the proxy.
             * Only the admin can call this function.
             * @param newImplementation Address of the new implementation.
             */
            function upgradeTo(address newImplementation) external isProxyAdminWithFallback {
                _upgradeTo(newImplementation);
            }
        
            /**
             * @dev Upgrade the backing implementation of the proxy and call a function
             * on the new implementation.
             * This is useful to initialize the proxied contract.
             * @param newImplementation Address of the new implementation.
             * @param data Data to send as msg.data in the low level call.
             * @param forceCall Boolean check to force call.
             * It should include the signature and the parameters of the function to be
             * called, as described in
             * https://solidity.readthedocs.io/en/develop/abi-spec.html#function-selector-and-argument-encoding.
             */
            function upgradeToAndCall(
                address newImplementation,
                bytes memory data,
                bool forceCall
            ) external payable isProxyAdminWithFallback {
                _upgradeToAndCall(newImplementation, data, forceCall);
            }
        
            /**
             * @dev Upgrade the backing implementation of the proxy and call a function
             * on the new implementation.
             * This is useful to initialize the proxied contract.
             * @param newImplementation Address of the new implementation.
             * @param data Data to send as msg.data in the low level call.
             * @param forceCall Boolean check to force call.
             * It should include the signature and the parameters of the function to be
             * called, as described in
             * https://solidity.readthedocs.io/en/develop/abi-spec.html#function-selector-and-argument-encoding.
             */
            function upgradeToAndCallSecure(
                address newImplementation,
                bytes memory data,
                bool forceCall
            ) external payable isProxyAdminWithFallback {
                _upgradeToAndCallSecure(newImplementation, data, forceCall);
            }
        }

        File 4 of 6: ICO
        // SPDX-License-Identifier: MIT
        pragma solidity ^0.8.9;
        import "./DriftPresale.sol";
        import "@openzeppelin/[email protected]/access/Ownable.sol";
        import "@chainlink/contracts/src/v0.8/interfaces/AggregatorV3Interface.sol";
        contract ICO is Ownable {
            DriftPresale public tokenAddress;
             AggregatorV3Interface internal priceFeed;
            enum Tier {
                none,
                gold,
                silver,
                bronze
            }
            enum userType {
                dynamic,
                stake
            }
            struct AmbassadorInfo {
                address ambassadorAddress;
                string ambassadorCode;
                Tier userTier;
                uint256 earning;
                uint256 referrals;
                uint256 amountRaised;
            }
            
            address[] public ambassadorsList; //Ambassador Array
            uint256 public totalReceivedFunds;
            uint256 public icoStartTime;
            uint256 public limitPerUser;
            
            uint256 public icoStage;
            uint256 public eligibleSilverAmbassador = 0.5 ether;
            uint256 public eligibleBronzeAmbassador = 0.25 ether;
            uint256 public noOfAmbassadors;
            uint256 public AmbassadorPayout;
            uint256 public stakingLimit;
            // uint256 public minimumBuyLimit = ;
            bool public isSalePaused = false;
            // Mapping (Stage => Price)
            mapping(uint256 => uint256) public prices;
            // Mapping (Stage => ETH target)
            mapping(uint256 => uint256) public icoTarget;
            mapping(uint256 => uint256) public tokensTransferred;
            mapping(uint256 => uint256) public receivedFunds;
            mapping(userType => uint256) public noOfTokens;
            // Decimals 2
            mapping(Tier => uint32[]) public percent;
            mapping(userType => address[]) private users;
            mapping(address => mapping(userType => bool)) public isAddressExist;
            mapping(address => mapping(userType => uint256))
                public amountOfAddressPerType;
            // Is code is enabled/valid
            mapping(string => bool) public isEnableCode;
            // Generated promo code against address
            mapping(address => string) public codeOfAddress;
            // Generated promo code
            mapping(string => address) public promoCode;
            // How much money the address invest
            mapping(address => AmbassadorInfo) internal ambassadorInfo;
            // Mapping Stage => Tier => Amount
            mapping(uint256 => mapping(Tier => uint256)) public totalTierRaised;
            mapping(Tier => mapping(string => bool)) public ambassadorCode;
            // How much money the address invest
            mapping(address => uint256) public investAmount;
            mapping(address => bool) public isAmbassadorEligible;
            mapping(address => uint256) public ambassadorPercentExt;
            modifier isCodeValid(string memory _code) {
                require(
                    isEnableCode[_code] || compareStringsbyBytes(_code, ""),
                    "Code is invalid"
                );
                _;
            }
            modifier isUsingOwnCode(string memory _code) {
                require(promoCode[_code] != msg.sender, "Can't use your own code");
                _;
            }
            // Is crowdsale Open
            function changeSaleStatus(bool _status) public onlyOwner  {
               isSalePaused=_status;
            }
            // bool public isSalePaused;
            constructor(DriftPresale _tokenAddress) {
                tokenAddress = _tokenAddress;
            priceFeed = AggregatorV3Interface(0x5f4eC3Df9cbd43714FE2740f5E3616155c5b8419);
                percent[Tier.gold] = [1000, 500];
                percent[Tier.silver] = [1000, 500];
                percent[Tier.bronze] = [750, 500];
                stakingLimit = 2000000000 * 10**tokenAddress.decimals();
                limitPerUser = 150000000 * 10**tokenAddress.decimals();
            }
            // Is crowdsale Open
            function isOpen() public view returns (bool) {
                return icoTarget[icoStage] > tokensTransferred[icoStage] && block.timestamp >= icoStartTime;
            }
            function updateStakingLimit(uint256 _newLimit) public onlyOwner {
                stakingLimit = _newLimit;
            }
            // Percentage with multiplied by 100, eg. 10% => 10*100 = 1000
            function updateAmbassadorPercent(address _address, uint256 _percent) public onlyOwner {
                ambassadorPercentExt[_address] = _percent;
            }
            // Create new crowdsale stage
            function createStage(
                uint256 _price,
                uint256 _startTime,
                uint256 _target
            ) public onlyOwner {
                require(!isOpen(), "Previous ICO is open");
                require(_price > 0, "Price is zero");
                require(_startTime > block.timestamp, "Invalid Time");
                icoStartTime = _startTime;        
                icoStage += 1;
                prices[icoStage] = _price;
                icoTarget[icoStage] = _target;
            }
            function calculateTokens(uint256 _amount) internal view returns (uint256) {
                uint256 tokens = ((_amount * 10**18) / getPriceRate(prices[icoStage]));
                return tokens;
            }
        //Conversion Rate
                function getPriceRate(uint _amount) public view returns (uint) {
                (, int price,,,) = priceFeed.latestRoundData();
                uint adjust_price = uint(price) * 1e10;
                uint usd = _amount ;
                uint rate = (usd * 1e18) / adjust_price;
                return rate;
            }
            // Buy tokens
            function buyTokens(string memory _code, bool _isStaker)
                public
                payable
                isCodeValid(_code)
                isUsingOwnCode(_code)
            {
                require(!isSalePaused,"Sale is Paused");
                require(msg.value >= getPriceRate(50000000000000000000), "Minimum Buy Limit is USD $50.00");
                require(tokenAddress.balanceOf(_msgSender()) <= limitPerUser, "Sorry! Looks like this wallet has reached Max. Purchase Limit per address. As per our Presale Rules, a single wallet cannot purchase more than 1.5% of total supply.");
                require(isOpen(), "Current stage of Presale has been filled completely. Please stay tuned for the next lap.");
                
                
                uint256 tokens = calculateTokens(msg.value);
                if (!compareStringsbyBytes(_code, "")) {
                    Tier tier;
                    if (ambassadorCode[Tier.gold][_code]) {
                        tier = Tier.gold;
                    } else if (ambassadorCode[Tier.silver][_code]) {
                        tier = Tier.silver;
                    } else if (ambassadorCode[Tier.bronze][_code]) {
                        tier = Tier.bronze;
                    }
                    uint256 extraToken = (tokens * percent[tier][1]) / 10**4;
                    tokens += extraToken;
                    
                    AmbassadorInfo storage _ambassadorInfoCode = ambassadorInfo[
                        promoCode[_code]
                    ];
                    uint256 ambassadorPer = ambassadorPercentExt[_ambassadorInfoCode.ambassadorAddress] != 0 ? ambassadorPercentExt[_ambassadorInfoCode.ambassadorAddress] : percent[tier][0];
                    uint256 ambassadorAmount = (msg.value * ambassadorPer) / 10**4;
                    totalTierRaised[icoStage][tier] += msg.value;
                    _ambassadorInfoCode.amountRaised += msg.value;
                    _ambassadorInfoCode.referrals += 1;
                    _ambassadorInfoCode.earning += ambassadorAmount;
                    AmbassadorPayout += ambassadorAmount;
                    (bool sentToAmbassador, ) = payable(promoCode[_code]).call{
                        value: ambassadorAmount
                    }("");
                    require(sentToAmbassador, "Sent to ambassador failed");
                }
                bool success = tokenAddress.transferFrom(
                    tokenAddress.owner(),
                    _msgSender(),
                    tokens
                );
                require(success, "Transfer failed");
                tokensTransferred[icoStage]+=tokens;
                userType _type = _isStaker ? userType.stake : userType.dynamic;
                require(
                    (_type == userType.stake &&
                        noOfTokens[userType.stake] < stakingLimit) ||
                        _type == userType.dynamic,
                    "Staking Pool Full. A max. 20% of entire supply can be staked during Presale stage. "
                );
                if (!isAddressExist[_msgSender()][_type]) {
                    isAddressExist[_msgSender()][_type] = true;
                    users[_type].push(_msgSender());
                }
                amountOfAddressPerType[_msgSender()][_type] += tokens;
                noOfTokens[_type] += tokens;
                receivedFunds[icoStage] += msg.value;
                totalReceivedFunds += msg.value;
                investAmount[_msgSender()] += msg.value;
                
                Tier _eligible = Tier.none;
                if (investAmount[_msgSender()] >= eligibleSilverAmbassador) {
                    _eligible = Tier.silver;
                } else if (investAmount[_msgSender()] >= eligibleBronzeAmbassador) {
                    _eligible = Tier.bronze;
                }
                AmbassadorInfo storage _ambassadorInfo = ambassadorInfo[_msgSender()];
                if (!isAmbassadorEligible[_msgSender()] && _eligible != Tier.none) {
                    
                    _ambassadorInfo.userTier = _eligible;
                    _ambassadorInfo.ambassadorAddress = _msgSender();
                    isAmbassadorEligible[_msgSender()] = true;
                    noOfAmbassadors++;
                } else if (_ambassadorInfo.userTier == Tier.bronze) {
                    string memory _codeOfAddress = codeOfAddress[_msgSender()];
                    delete ambassadorCode[Tier.bronze][_codeOfAddress];
                    _ambassadorInfo.userTier = _eligible;
                    ambassadorCode[_eligible][_codeOfAddress] = true;
                }
            }
            function getAmbassadorInfo(address _address)
                public
                view
                returns (
                    Tier _tier,
                    string memory _promocode,
                    address _ambassador,
                    uint256 _earnings,
                    uint256 _referrals,
                    uint256 _raised
                )
            {
                AmbassadorInfo memory _ambassadorInfo = ambassadorInfo[_address];
                return (
                    _ambassadorInfo.userTier,
                    _ambassadorInfo.ambassadorCode,
                    _ambassadorInfo.ambassadorAddress,
                    _ambassadorInfo.earning,
                    _ambassadorInfo.referrals,
                    _ambassadorInfo.amountRaised
                );
            }
            function changeStatusCode(string memory _code, bool status)
                public
                onlyOwner
            {
                isEnableCode[_code] = status;
            }
            function createAmbassadorCode(
                address _address,
                Tier _tier,
                string memory _code
            ) public onlyOwner {
                require(isNotAlreadyAmbassador(_address), "Already Ambasador");
                ambassadorsList.push(_address);
                require(!isEnableCode[_code], "Code already exist");
                isAmbassadorEligible[_address] = true;
                isEnableCode[_code] = true;
                codeOfAddress[_address] = _code;
                promoCode[_code] = _address;
                ambassadorCode[_tier][_code] = true;
                AmbassadorInfo storage _ambassadorInfo = ambassadorInfo[_address];
                _ambassadorInfo.ambassadorAddress = _address;
                _ambassadorInfo.ambassadorCode = _code;
                _ambassadorInfo.userTier = _tier;
                noOfAmbassadors++;
            }
            function createCode(string memory _code) public {
                require(isAmbassadorEligible[_msgSender()], "Sorry! You are not an ambassador yet. Please visit www.drifttoken.io during Presale and purchase a minimum of 0.25 ETH to become a Bronze Tier Ambassador. ");
                require(isNotAlreadyAmbassador(_msgSender()), "You have already assigned a Promo Code to this Ambassador Wallet. Please visit www.influ3nce.me/ambassador and connect this wallet to view your promo code. ");
                require(!compareStringsbyBytes(_code, ""), "Oops! Seems like you forgot to type-in your Promo Code.");
                require(!isEnableCode[_code], "Unfortunately, this Promo Code has already been assigned to an Ambassador. Please choose another Promo Code. ");
                ambassadorsList.push(_msgSender());
                isEnableCode[_code] = true;
                codeOfAddress[_msgSender()] = _code;
                promoCode[_code] = _msgSender();
                AmbassadorInfo storage _ambassadorInfo = ambassadorInfo[_msgSender()];
                ambassadorCode[_ambassadorInfo.userTier][_code] = true;
                _ambassadorInfo.ambassadorCode = _code;
            }
            function upgradeAmbassador(address _address, Tier _tier) public onlyOwner {
                AmbassadorInfo storage _ambassadorInfo = ambassadorInfo[_address];
                delete ambassadorCode[_ambassadorInfo.userTier][
                    codeOfAddress[_address]
                ];
                ambassadorCode[_tier][codeOfAddress[_address]] = true;
                _ambassadorInfo.userTier = _tier;
          
            }
            function isNotAlreadyAmbassador(address user) public view returns (bool) {
                return compareStringsbyBytes(codeOfAddress[user], "");
            }
            function getUsers(userType _type) public view returns (address[] memory) {
                return users[_type];
            }
            function getAmbassadorList() public view returns (address[] memory) {
                return ambassadorsList;
            }
            function withdrawFunds() public onlyOwner {
                (bool os, ) = payable(owner()).call{value: address(this).balance}("");
                require(os);
            }
            function compareStringsbyBytes(string memory s1, string memory s2)
                public
                pure
                returns (bool)
            {
                return
                    keccak256(abi.encodePacked(s1)) == keccak256(abi.encodePacked(s2));
            }
        }
        // SPDX-License-Identifier: MIT
        pragma solidity ^0.8.0;
        interface AggregatorV3Interface {
          function decimals() external view returns (uint8);
          function description() external view returns (string memory);
          function version() external view returns (uint256);
          function getRoundData(
            uint80 _roundId
          ) external view returns (uint80 roundId, int256 answer, uint256 startedAt, uint256 updatedAt, uint80 answeredInRound);
          function latestRoundData()
            external
            view
            returns (uint80 roundId, int256 answer, uint256 startedAt, uint256 updatedAt, uint80 answeredInRound);
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts (last updated v4.9.0) (access/Ownable.sol)
        pragma solidity ^0.8.0;
        import "../utils/Context.sol";
        /**
         * @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. Can only be called by the current owner.
             *
             * NOTE: Renouncing ownership will leave the contract without an owner,
             * thereby disabling 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);
            }
        }
        // SPDX-License-Identifier: MIT
        pragma solidity ^0.8.9;
        import "@openzeppelin/[email protected]/token/ERC20/ERC20.sol";
        import "@openzeppelin/[email protected]/token/ERC20/extensions/ERC20Burnable.sol";
        import "@openzeppelin/[email protected]/security/Pausable.sol";
        import "@openzeppelin/[email protected]/access/Ownable.sol";
        import "@openzeppelin/[email protected]/security/ReentrancyGuard.sol";
        contract DriftPresale is
            ERC20,
            ERC20Burnable,
            Pausable,
            Ownable,
            ReentrancyGuard
        {
            mapping(address => bool) public allowedAddresses;
            uint256 public maxSupply = 5660000000 * 10**decimals(); // 5.66 Billion
            constructor() ERC20("Drift Presale Token", "PREDRIFT") {
                _mint(msg.sender, maxSupply);
            }
            modifier isAddressAllowed(address from) {
                require(
                    allowedAddresses[from] || from == owner(),
                    "Transfer not allowed"
                );
                _;
            }
            function pause() public onlyOwner nonReentrant {
                _pause();
            }
            function unpause() public onlyOwner nonReentrant {
                _unpause();
            }
            function _beforeTokenTransfer(
                address from,
                address to,
                uint256 amount
            ) internal override whenNotPaused {
                super._beforeTokenTransfer(from, to, amount);
            }
            function transfer(address to, uint256 value)
                public
                virtual
                override
                isAddressAllowed(_msgSender())
                returns (bool)
            {
                super.transfer(to, value);
                return true;
            }
            function transferFrom(
                address from,
                address to,
                uint256 value
            ) public virtual override isAddressAllowed(from) returns (bool) {
                super.transferFrom(from, to, value);
                return true;
            }
            function updateAddressStatus(address _address, bool status)
                public
                onlyOwner
            {
                allowedAddresses[_address] = status;
            }
        }
        // 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 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.5.0) (token/ERC20/extensions/ERC20Burnable.sol)
        pragma solidity ^0.8.0;
        import "../ERC20.sol";
        import "../../../utils/Context.sol";
        /**
         * @dev Extension of {ERC20} that allows token holders to destroy both their own
         * tokens and those that they have an allowance for, in a way that can be
         * recognized off-chain (via event analysis).
         */
        abstract contract ERC20Burnable is Context, ERC20 {
            /**
             * @dev Destroys `amount` tokens from the caller.
             *
             * See {ERC20-_burn}.
             */
            function burn(uint256 amount) public virtual {
                _burn(_msgSender(), amount);
            }
            /**
             * @dev Destroys `amount` tokens from `account`, deducting from the caller's
             * allowance.
             *
             * See {ERC20-_burn} and {ERC20-allowance}.
             *
             * Requirements:
             *
             * - the caller must have allowance for ``accounts``'s tokens of at least
             * `amount`.
             */
            function burnFrom(address account, uint256 amount) public virtual {
                _spendAllowance(account, _msgSender(), amount);
                _burn(account, amount);
            }
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts (last updated v4.9.0) (token/ERC20/ERC20.sol)
        pragma solidity ^0.8.0;
        import "./IERC20.sol";
        import "./extensions/IERC20Metadata.sol";
        import "../../utils/Context.sol";
        /**
         * @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].
         *
         * The default value of {decimals} is 18. To change this, you should override
         * this function so it returns a different value.
         *
         * 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}.
             *
             * 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 default value returned by this function, unless
             * it's 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 {}
        }
        // SPDX-License-Identifier: MIT
        // 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;
            }
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts v4.4.1 (token/ERC20/extensions/IERC20Metadata.sol)
        pragma solidity ^0.8.0;
        import "../IERC20.sol";
        /**
         * @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);
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts (last updated v4.9.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 5 of 6: StakingDrift2
        // File: @openzeppelin/contracts-upgradeable/proxy/utils/Initializable.sol
        
        
        // OpenZeppelin Contracts (last updated v5.0.0) (proxy/utils/Initializable.sol)
        
        pragma solidity ^0.8.20;
        
        /**
         * @dev This is a base contract to aid in writing upgradeable contracts, or any kind of contract that will be deployed
         * behind a proxy. Since proxied contracts do not make use of a constructor, it's common to move constructor logic to an
         * external initializer function, usually called `initialize`. It then becomes necessary to protect this initializer
         * function so it can only be called once. The {initializer} modifier provided by this contract will have this effect.
         *
         * The initialization functions use a version number. Once a version number is used, it is consumed and cannot be
         * reused. This mechanism prevents re-execution of each "step" but allows the creation of new initialization steps in
         * case an upgrade adds a module that needs to be initialized.
         *
         * For example:
         *
         * [.hljs-theme-light.nopadding]
         * ```solidity
         * contract MyToken is ERC20Upgradeable {
         *     function initialize() initializer public {
         *         __ERC20_init("MyToken", "MTK");
         *     }
         * }
         *
         * contract MyTokenV2 is MyToken, ERC20PermitUpgradeable {
         *     function initializeV2() reinitializer(2) public {
         *         __ERC20Permit_init("MyToken");
         *     }
         * }
         * ```
         *
         * TIP: To avoid leaving the proxy in an uninitialized state, the initializer function should be called as early as
         * possible by providing the encoded function call as the `_data` argument to {ERC1967Proxy-constructor}.
         *
         * CAUTION: When used with inheritance, manual care must be taken to not invoke a parent initializer twice, or to ensure
         * that all initializers are idempotent. This is not verified automatically as constructors are by Solidity.
         *
         * [CAUTION]
         * ====
         * Avoid leaving a contract uninitialized.
         *
         * An uninitialized contract can be taken over by an attacker. This applies to both a proxy and its implementation
         * contract, which may impact the proxy. To prevent the implementation contract from being used, you should invoke
         * the {_disableInitializers} function in the constructor to automatically lock it when it is deployed:
         *
         * [.hljs-theme-light.nopadding]
         * ```
         * /// @custom:oz-upgrades-unsafe-allow constructor
         * constructor() {
         *     _disableInitializers();
         * }
         * ```
         * ====
         */
        abstract contract Initializable {
            /**
             * @dev Storage of the initializable contract.
             *
             * It's implemented on a custom ERC-7201 namespace to reduce the risk of storage collisions
             * when using with upgradeable contracts.
             *
             * @custom:storage-location erc7201:openzeppelin.storage.Initializable
             */
            struct InitializableStorage {
                /**
                 * @dev Indicates that the contract has been initialized.
                 */
                uint64 _initialized;
                /**
                 * @dev Indicates that the contract is in the process of being initialized.
                 */
                bool _initializing;
            }
        
            // keccak256(abi.encode(uint256(keccak256("openzeppelin.storage.Initializable")) - 1)) & ~bytes32(uint256(0xff))
            bytes32 private constant INITIALIZABLE_STORAGE = 0xf0c57e16840df040f15088dc2f81fe391c3923bec73e23a9662efc9c229c6a00;
        
            /**
             * @dev The contract is already initialized.
             */
            error InvalidInitialization();
        
            /**
             * @dev The contract is not initializing.
             */
            error NotInitializing();
        
            /**
             * @dev Triggered when the contract has been initialized or reinitialized.
             */
            event Initialized(uint64 version);
        
            /**
             * @dev A modifier that defines a protected initializer function that can be invoked at most once. In its scope,
             * `onlyInitializing` functions can be used to initialize parent contracts.
             *
             * Similar to `reinitializer(1)`, except that in the context of a constructor an `initializer` may be invoked any
             * number of times. This behavior in the constructor can be useful during testing and is not expected to be used in
             * production.
             *
             * Emits an {Initialized} event.
             */
            modifier initializer() {
                // solhint-disable-next-line var-name-mixedcase
                InitializableStorage storage $ = _getInitializableStorage();
        
                // Cache values to avoid duplicated sloads
                bool isTopLevelCall = !$._initializing;
                uint64 initialized = $._initialized;
        
                // Allowed calls:
                // - initialSetup: the contract is not in the initializing state and no previous version was
                //                 initialized
                // - construction: the contract is initialized at version 1 (no reininitialization) and the
                //                 current contract is just being deployed
                bool initialSetup = initialized == 0 && isTopLevelCall;
                bool construction = initialized == 1 && address(this).code.length == 0;
        
                if (!initialSetup && !construction) {
                    revert InvalidInitialization();
                }
                $._initialized = 1;
                if (isTopLevelCall) {
                    $._initializing = true;
                }
                _;
                if (isTopLevelCall) {
                    $._initializing = false;
                    emit Initialized(1);
                }
            }
        
            /**
             * @dev A modifier that defines a protected reinitializer function that can be invoked at most once, and only if the
             * contract hasn't been initialized to a greater version before. In its scope, `onlyInitializing` functions can be
             * used to initialize parent contracts.
             *
             * A reinitializer may be used after the original initialization step. This is essential to configure modules that
             * are added through upgrades and that require initialization.
             *
             * When `version` is 1, this modifier is similar to `initializer`, except that functions marked with `reinitializer`
             * cannot be nested. If one is invoked in the context of another, execution will revert.
             *
             * Note that versions can jump in increments greater than 1; this implies that if multiple reinitializers coexist in
             * a contract, executing them in the right order is up to the developer or operator.
             *
             * WARNING: Setting the version to 2**64 - 1 will prevent any future reinitialization.
             *
             * Emits an {Initialized} event.
             */
            modifier reinitializer(uint64 version) {
                // solhint-disable-next-line var-name-mixedcase
                InitializableStorage storage $ = _getInitializableStorage();
        
                if ($._initializing || $._initialized >= version) {
                    revert InvalidInitialization();
                }
                $._initialized = version;
                $._initializing = true;
                _;
                $._initializing = false;
                emit Initialized(version);
            }
        
            /**
             * @dev Modifier to protect an initialization function so that it can only be invoked by functions with the
             * {initializer} and {reinitializer} modifiers, directly or indirectly.
             */
            modifier onlyInitializing() {
                _checkInitializing();
                _;
            }
        
            /**
             * @dev Reverts if the contract is not in an initializing state. See {onlyInitializing}.
             */
            function _checkInitializing() internal view virtual {
                if (!_isInitializing()) {
                    revert NotInitializing();
                }
            }
        
            /**
             * @dev Locks the contract, preventing any future reinitialization. This cannot be part of an initializer call.
             * Calling this in the constructor of a contract will prevent that contract from being initialized or reinitialized
             * to any version. It is recommended to use this to lock implementation contracts that are designed to be called
             * through proxies.
             *
             * Emits an {Initialized} event the first time it is successfully executed.
             */
            function _disableInitializers() internal virtual {
                // solhint-disable-next-line var-name-mixedcase
                InitializableStorage storage $ = _getInitializableStorage();
        
                if ($._initializing) {
                    revert InvalidInitialization();
                }
                if ($._initialized != type(uint64).max) {
                    $._initialized = type(uint64).max;
                    emit Initialized(type(uint64).max);
                }
            }
        
            /**
             * @dev Returns the highest version that has been initialized. See {reinitializer}.
             */
            function _getInitializedVersion() internal view returns (uint64) {
                return _getInitializableStorage()._initialized;
            }
        
            /**
             * @dev Returns `true` if the contract is currently initializing. See {onlyInitializing}.
             */
            function _isInitializing() internal view returns (bool) {
                return _getInitializableStorage()._initializing;
            }
        
            /**
             * @dev Returns a pointer to the storage namespace.
             */
            // solhint-disable-next-line var-name-mixedcase
            function _getInitializableStorage() private pure returns (InitializableStorage storage $) {
                assembly {
                    $.slot := INITIALIZABLE_STORAGE
                }
            }
        }
        
        // File: @openzeppelin/contracts-upgradeable/security/ReentrancyGuardUpgradeable.sol
        
        
        // 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 ReentrancyGuardUpgradeable is Initializable {
            // 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;
        
            function __ReentrancyGuard_init() internal onlyInitializing {
                __ReentrancyGuard_init_unchained();
            }
        
            function __ReentrancyGuard_init_unchained() internal onlyInitializing {
                _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;
            }
        
            /**
             * @dev This empty reserved space is put in place to allow future versions to add new
             * variables without shifting down storage in the inheritance chain.
             * See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps
             */
            uint256[49] private __gap;
        }
        
        // File: @openzeppelin/contracts-upgradeable/utils/ContextUpgradeable.sol
        
        
        // OpenZeppelin Contracts (last updated v5.0.1) (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 ContextUpgradeable is Initializable {
            function __Context_init() internal onlyInitializing {
            }
        
            function __Context_init_unchained() internal onlyInitializing {
            }
            function _msgSender() internal view virtual returns (address) {
                return msg.sender;
            }
        
            function _msgData() internal view virtual returns (bytes calldata) {
                return msg.data;
            }
        
            function _contextSuffixLength() internal view virtual returns (uint256) {
                return 0;
            }
        }
        
        // File: @openzeppelin/contracts-upgradeable/access/OwnableUpgradeable.sol
        
        
        // OpenZeppelin Contracts (last updated v5.0.0) (access/Ownable.sol)
        
        pragma solidity ^0.8.20;
        
        
        
        /**
         * @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.
         *
         * The initial owner is set to the address provided by the deployer. 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 OwnableUpgradeable is Initializable, ContextUpgradeable {
            /// @custom:storage-location erc7201:openzeppelin.storage.Ownable
            struct OwnableStorage {
                address _owner;
            }
        
            // keccak256(abi.encode(uint256(keccak256("openzeppelin.storage.Ownable")) - 1)) & ~bytes32(uint256(0xff))
            bytes32 private constant OwnableStorageLocation = 0x9016d09d72d40fdae2fd8ceac6b6234c7706214fd39c1cd1e609a0528c199300;
        
            function _getOwnableStorage() private pure returns (OwnableStorage storage $) {
                assembly {
                    $.slot := OwnableStorageLocation
                }
            }
        
            /**
             * @dev The caller account is not authorized to perform an operation.
             */
            error OwnableUnauthorizedAccount(address account);
        
            /**
             * @dev The owner is not a valid owner account. (eg. `address(0)`)
             */
            error OwnableInvalidOwner(address owner);
        
            event OwnershipTransferred(address indexed previousOwner, address indexed newOwner);
        
            /**
             * @dev Initializes the contract setting the address provided by the deployer as the initial owner.
             */
            function __Ownable_init(address initialOwner) internal onlyInitializing {
                __Ownable_init_unchained(initialOwner);
            }
        
            function __Ownable_init_unchained(address initialOwner) internal onlyInitializing {
                if (initialOwner == address(0)) {
                    revert OwnableInvalidOwner(address(0));
                }
                _transferOwnership(initialOwner);
            }
        
            /**
             * @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) {
                OwnableStorage storage $ = _getOwnableStorage();
                return $._owner;
            }
        
            /**
             * @dev Throws if the sender is not the owner.
             */
            function _checkOwner() internal view virtual {
                if (owner() != _msgSender()) {
                    revert OwnableUnauthorizedAccount(_msgSender());
                }
            }
        
            /**
             * @dev Leaves the contract without owner. It will not be possible to call
             * `onlyOwner` functions. Can only be called by the current owner.
             *
             * NOTE: Renouncing ownership will leave the contract without an owner,
             * thereby disabling 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 {
                if (newOwner == address(0)) {
                    revert OwnableInvalidOwner(address(0));
                }
                _transferOwnership(newOwner);
            }
        
            /**
             * @dev Transfers ownership of the contract to a new account (`newOwner`).
             * Internal function without access restriction.
             */
            function _transferOwnership(address newOwner) internal virtual {
                OwnableStorage storage $ = _getOwnableStorage();
                address oldOwner = $._owner;
                $._owner = newOwner;
                emit OwnershipTransferred(oldOwner, newOwner);
            }
        }
        
        // File: @openzeppelin/contracts/utils/structs/EnumerableSet.sol
        
        
        // OpenZeppelin Contracts (last updated v5.0.0) (utils/structs/EnumerableSet.sol)
        // This file was procedurally generated from scripts/generate/templates/EnumerableSet.js.
        
        pragma solidity ^0.8.20;
        
        /**
         * @dev Library for managing
         * https://en.wikipedia.org/wiki/Set_(abstract_data_type)[sets] of primitive
         * types.
         *
         * Sets have the following properties:
         *
         * - Elements are added, removed, and checked for existence in constant time
         * (O(1)).
         * - Elements are enumerated in O(n). No guarantees are made on the ordering.
         *
         * ```solidity
         * contract Example {
         *     // Add the library methods
         *     using EnumerableSet for EnumerableSet.AddressSet;
         *
         *     // Declare a set state variable
         *     EnumerableSet.AddressSet private mySet;
         * }
         * ```
         *
         * As of v3.3.0, sets of type `bytes32` (`Bytes32Set`), `address` (`AddressSet`)
         * and `uint256` (`UintSet`) are supported.
         *
         * [WARNING]
         * ====
         * Trying to delete such a structure from storage will likely result in data corruption, rendering the structure
         * unusable.
         * See https://github.com/ethereum/solidity/pull/11843[ethereum/solidity#11843] for more info.
         *
         * In order to clean an EnumerableSet, you can either remove all elements one by one or create a fresh instance using an
         * array of EnumerableSet.
         * ====
         */
        library EnumerableSet {
            // To implement this library for multiple types with as little code
            // repetition as possible, we write it in terms of a generic Set type with
            // bytes32 values.
            // The Set implementation uses private functions, and user-facing
            // implementations (such as AddressSet) are just wrappers around the
            // underlying Set.
            // This means that we can only create new EnumerableSets for types that fit
            // in bytes32.
        
            struct Set {
                // Storage of set values
                bytes32[] _values;
                // Position is the index of the value in the `values` array plus 1.
                // Position 0 is used to mean a value is not in the set.
                mapping(bytes32 value => uint256) _positions;
            }
        
            /**
             * @dev Add a value to a set. O(1).
             *
             * Returns true if the value was added to the set, that is if it was not
             * already present.
             */
            function _add(Set storage set, bytes32 value) private returns (bool) {
                if (!_contains(set, value)) {
                    set._values.push(value);
                    // The value is stored at length-1, but we add 1 to all indexes
                    // and use 0 as a sentinel value
                    set._positions[value] = set._values.length;
                    return true;
                } else {
                    return false;
                }
            }
        
            /**
             * @dev Removes a value from a set. O(1).
             *
             * Returns true if the value was removed from the set, that is if it was
             * present.
             */
            function _remove(Set storage set, bytes32 value) private returns (bool) {
                // We cache the value's position to prevent multiple reads from the same storage slot
                uint256 position = set._positions[value];
        
                if (position != 0) {
                    // Equivalent to contains(set, value)
                    // To delete an element from the _values array in O(1), we swap the element to delete with the last one in
                    // the array, and then remove the last element (sometimes called as 'swap and pop').
                    // This modifies the order of the array, as noted in {at}.
        
                    uint256 valueIndex = position - 1;
                    uint256 lastIndex = set._values.length - 1;
        
                    if (valueIndex != lastIndex) {
                        bytes32 lastValue = set._values[lastIndex];
        
                        // Move the lastValue to the index where the value to delete is
                        set._values[valueIndex] = lastValue;
                        // Update the tracked position of the lastValue (that was just moved)
                        set._positions[lastValue] = position;
                    }
        
                    // Delete the slot where the moved value was stored
                    set._values.pop();
        
                    // Delete the tracked position for the deleted slot
                    delete set._positions[value];
        
                    return true;
                } else {
                    return false;
                }
            }
        
            /**
             * @dev Returns true if the value is in the set. O(1).
             */
            function _contains(Set storage set, bytes32 value) private view returns (bool) {
                return set._positions[value] != 0;
            }
        
            /**
             * @dev Returns the number of values on the set. O(1).
             */
            function _length(Set storage set) private view returns (uint256) {
                return set._values.length;
            }
        
            /**
             * @dev Returns the value stored at position `index` in the set. O(1).
             *
             * Note that there are no guarantees on the ordering of values inside the
             * array, and it may change when more values are added or removed.
             *
             * Requirements:
             *
             * - `index` must be strictly less than {length}.
             */
            function _at(Set storage set, uint256 index) private view returns (bytes32) {
                return set._values[index];
            }
        
            /**
             * @dev Return the entire set in an array
             *
             * WARNING: This operation will copy the entire storage to memory, which can be quite expensive. This is designed
             * to mostly be used by view accessors that are queried without any gas fees. Developers should keep in mind that
             * this function has an unbounded cost, and using it as part of a state-changing function may render the function
             * uncallable if the set grows to a point where copying to memory consumes too much gas to fit in a block.
             */
            function _values(Set storage set) private view returns (bytes32[] memory) {
                return set._values;
            }
        
            // Bytes32Set
        
            struct Bytes32Set {
                Set _inner;
            }
        
            /**
             * @dev Add a value to a set. O(1).
             *
             * Returns true if the value was added to the set, that is if it was not
             * already present.
             */
            function add(Bytes32Set storage set, bytes32 value) internal returns (bool) {
                return _add(set._inner, value);
            }
        
            /**
             * @dev Removes a value from a set. O(1).
             *
             * Returns true if the value was removed from the set, that is if it was
             * present.
             */
            function remove(Bytes32Set storage set, bytes32 value) internal returns (bool) {
                return _remove(set._inner, value);
            }
        
            /**
             * @dev Returns true if the value is in the set. O(1).
             */
            function contains(Bytes32Set storage set, bytes32 value) internal view returns (bool) {
                return _contains(set._inner, value);
            }
        
            /**
             * @dev Returns the number of values in the set. O(1).
             */
            function length(Bytes32Set storage set) internal view returns (uint256) {
                return _length(set._inner);
            }
        
            /**
             * @dev Returns the value stored at position `index` in the set. O(1).
             *
             * Note that there are no guarantees on the ordering of values inside the
             * array, and it may change when more values are added or removed.
             *
             * Requirements:
             *
             * - `index` must be strictly less than {length}.
             */
            function at(Bytes32Set storage set, uint256 index) internal view returns (bytes32) {
                return _at(set._inner, index);
            }
        
            /**
             * @dev Return the entire set in an array
             *
             * WARNING: This operation will copy the entire storage to memory, which can be quite expensive. This is designed
             * to mostly be used by view accessors that are queried without any gas fees. Developers should keep in mind that
             * this function has an unbounded cost, and using it as part of a state-changing function may render the function
             * uncallable if the set grows to a point where copying to memory consumes too much gas to fit in a block.
             */
            function values(Bytes32Set storage set) internal view returns (bytes32[] memory) {
                bytes32[] memory store = _values(set._inner);
                bytes32[] memory result;
        
                /// @solidity memory-safe-assembly
                assembly {
                    result := store
                }
        
                return result;
            }
        
            // AddressSet
        
            struct AddressSet {
                Set _inner;
            }
        
            /**
             * @dev Add a value to a set. O(1).
             *
             * Returns true if the value was added to the set, that is if it was not
             * already present.
             */
            function add(AddressSet storage set, address value) internal returns (bool) {
                return _add(set._inner, bytes32(uint256(uint160(value))));
            }
        
            /**
             * @dev Removes a value from a set. O(1).
             *
             * Returns true if the value was removed from the set, that is if it was
             * present.
             */
            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);
            }
        
            /**
             * @dev Returns the value stored at position `index` in the set. O(1).
             *
             * Note that there are no guarantees on the ordering of values inside the
             * array, and it may change when more values are added or removed.
             *
             * Requirements:
             *
             * - `index` must be strictly less than {length}.
             */
            function at(AddressSet storage set, uint256 index) internal view returns (address) {
                return address(uint160(uint256(_at(set._inner, index))));
            }
        
            /**
             * @dev Return the entire set in an array
             *
             * WARNING: This operation will copy the entire storage to memory, which can be quite expensive. This is designed
             * to mostly be used by view accessors that are queried without any gas fees. Developers should keep in mind that
             * this function has an unbounded cost, and using it as part of a state-changing function may render the function
             * uncallable if the set grows to a point where copying to memory consumes too much gas to fit in a block.
             */
            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;
            }
        
            // UintSet
        
            struct UintSet {
                Set _inner;
            }
        
            /**
             * @dev Add a value to a set. O(1).
             *
             * Returns true if the value was added to the set, that is if it was not
             * already present.
             */
            function add(UintSet storage set, uint256 value) internal returns (bool) {
                return _add(set._inner, bytes32(value));
            }
        
            /**
             * @dev Removes a value from a set. O(1).
             *
             * Returns true if the value was removed from the set, that is if it was
             * present.
             */
            function remove(UintSet storage set, uint256 value) internal returns (bool) {
                return _remove(set._inner, bytes32(value));
            }
        
            /**
             * @dev Returns true if the value is in the set. O(1).
             */
            function contains(UintSet storage set, uint256 value) internal view returns (bool) {
                return _contains(set._inner, bytes32(value));
            }
        
            /**
             * @dev Returns the number of values in the set. O(1).
             */
            function length(UintSet storage set) internal view returns (uint256) {
                return _length(set._inner);
            }
        
            /**
             * @dev Returns the value stored at position `index` in the set. O(1).
             *
             * Note that there are no guarantees on the ordering of values inside the
             * array, and it may change when more values are added or removed.
             *
             * Requirements:
             *
             * - `index` must be strictly less than {length}.
             */
            function at(UintSet storage set, uint256 index) internal view returns (uint256) {
                return uint256(_at(set._inner, index));
            }
        
            /**
             * @dev Return the entire set in an array
             *
             * WARNING: This operation will copy the entire storage to memory, which can be quite expensive. This is designed
             * to mostly be used by view accessors that are queried without any gas fees. Developers should keep in mind that
             * this function has an unbounded cost, and using it as part of a state-changing function may render the function
             * uncallable if the set grows to a point where copying to memory consumes too much gas to fit in a block.
             */
            function values(UintSet storage set) internal view returns (uint256[] memory) {
                bytes32[] memory store = _values(set._inner);
                uint256[] memory result;
        
                /// @solidity memory-safe-assembly
                assembly {
                    result := store
                }
        
                return result;
            }
        }
        
        // File: utils/AllowableAddressUpgradeable.sol
        
        
        pragma solidity ^0.8.4;
        
        
        
        contract AllowableAddress is OwnableUpgradeable {
        
            using EnumerableSet for EnumerableSet.AddressSet;
        
            // @dev the allowed minter addresses
            EnumerableSet.AddressSet internal s_minters;
            // @dev the allowed burner addresses
            EnumerableSet.AddressSet internal s_burners;
        
            /******* Variables *******/
            // blacklist address => bool
            mapping(address => bool) internal blacklistAddress;
        
            /******* Modifiers *******/
            modifier blacklistCheck(address _from, address _to) {
                require(!blacklistAddress[_from] && !blacklistAddress[_to], "Address is blacklisted");
                _;
            }
        
            modifier onlyMinter() {
                if (!isMinter(msg.sender)) revert SenderNotMinter(msg.sender);
                _;
            }
        
            modifier onlyBurner() {
                if (!isBurner(msg.sender)) revert SenderNotBurner(msg.sender);
                _;
            }
        
            /******* Events *******/
            error SenderNotMinter(address sender);
            error SenderNotBurner(address sender);
            error MaxSupplyExceeded(uint256 supplyAfterMint);
        
            event MintAccessGranted(address indexed minter);
            event BurnAccessGranted(address indexed burner);
            event MintAccessRevoked(address indexed minter);
            event BurnAccessRevoked(address indexed burner);
            event BlacklistUpdated(bool status);
        
            /******* Constructor *******/
            function __AllowableAddress_init(address _ownerAddress) internal {
                __Ownable_init(_ownerAddress);
            }
        
            /*
             * @notice Manage blacklisting to restrict addresses
             * @param addresses     Array of addresses
             * @param status        If blacklist passing true, false otherwise.
             */
            function manageBlacklist(address[] calldata addresses, bool status)
                external
                virtual
                onlyOwner
            {
                for (uint256 i; i < addresses.length; ++i) {
                    blacklistAddress[addresses[i]] = status;
                }
                emit BlacklistUpdated(status);
            }
        
            /*
             * @notice grants both mint and burn roles to `burnAndMinter`.
             * @notice Accessable to only owner.
             * @param burnAndMinter      Any address for mint and burn access
             */
            function grantMintAndBurnRoles(address burnAndMinter) external onlyOwner {
                grantMintRole(burnAndMinter);
                grantBurnRole(burnAndMinter);
            }
        
            /*
             * @notice Grants mint role to the given address.
             * @notice Accessable to only owner.
             * @param minter    Any address for mint access
             */
            function grantMintRole(address minter) public onlyOwner {
                if (s_minters.add(minter)) {
                    emit MintAccessGranted(minter);
                }
            }
        
            /*
             * @notice Grants burn role to the given address.
             * @notice Accessable to only owner.
             * @param burner    Any address for burn access
             */
            function grantBurnRole(address burner) public onlyOwner {
                if (s_burners.add(burner)) {
                    emit BurnAccessGranted(burner);
                }
            }
        
            /*
             * @notice Revokes mint role for the given address.
             * @notice Accessable to only owner.
             * @param minter    Any address for remove mint access
             */
            function revokeMintRole(address minter) public onlyOwner {
                if (s_minters.remove(minter)) {
                    emit MintAccessRevoked(minter);
                }
            }
        
            /*
             * @notice Revokes burn role from the given address.
             * @notice Accessable to only owner
             * @param burner    Any address for remove burn access
             */
            function revokeBurnRole(address burner) public onlyOwner {
                if (s_burners.remove(burner)) {
                    emit BurnAccessRevoked(burner);
                }
            }
        
            /*
             * @param _address      Any address
             * @return {the address is blacklisted or not}.
             */
            function isBlacklisted(address _address) public view returns (bool) {
                return blacklistAddress[_address];
            }
        
            /*
             * @return {all permissioned minters}
             */
            function getMinters() public view returns (address[] memory) {
                return s_minters.values();
            }
        
            /*
             * @return {all permissioned burners}
             */
            function getBurners() public view returns (address[] memory) {
                return s_burners.values();
            }
        
            /*
             * @notice Checks whether a given address is a minter for this token.
             * @return {true if the address is allowed to mint}.
             */
            function isMinter(address minter) public view returns (bool) {
                return s_minters.contains(minter);
            }
        
            /*
             * @notice Checks whether a given address is a burner for this token.
             * @return {true if the address is allowed to burn}.
             */
            function isBurner(address burner) public view returns (bool) {
                return s_burners.contains(burner);
            }
        }
        
        // File: @openzeppelin/contracts/utils/math/SafeMath.sol
        
        
        // OpenZeppelin Contracts (last updated v4.9.0) (utils/math/SafeMath.sol)
        
        pragma solidity ^0.8.0;
        
        // CAUTION
        // This version of SafeMath should only be used with Solidity 0.8 or later,
        // because it relies on the compiler's built in overflow checks.
        
        /**
         * @dev Wrappers over Solidity's arithmetic operations.
         *
         * NOTE: `SafeMath` is generally not needed starting with Solidity 0.8, since the compiler
         * now has built in overflow checking.
         */
        library SafeMath {
            /**
             * @dev Returns the addition of two unsigned integers, with an overflow flag.
             *
             * _Available since v3.4._
             */
            function tryAdd(uint256 a, uint256 b) internal pure returns (bool, uint256) {
                unchecked {
                    uint256 c = a + b;
                    if (c < a) return (false, 0);
                    return (true, c);
                }
            }
        
            /**
             * @dev Returns the subtraction of two unsigned integers, with an overflow flag.
             *
             * _Available since v3.4._
             */
            function trySub(uint256 a, uint256 b) internal pure returns (bool, uint256) {
                unchecked {
                    if (b > a) return (false, 0);
                    return (true, a - b);
                }
            }
        
            /**
             * @dev Returns the multiplication of two unsigned integers, with an overflow flag.
             *
             * _Available since v3.4._
             */
            function tryMul(uint256 a, uint256 b) internal pure returns (bool, uint256) {
                unchecked {
                    // Gas optimization: this is cheaper than requiring 'a' not being zero, but the
                    // benefit is lost if 'b' is also tested.
                    // See: https://github.com/OpenZeppelin/openzeppelin-contracts/pull/522
                    if (a == 0) return (true, 0);
                    uint256 c = a * b;
                    if (c / a != b) return (false, 0);
                    return (true, c);
                }
            }
        
            /**
             * @dev Returns the division of two unsigned integers, with a division by zero flag.
             *
             * _Available since v3.4._
             */
            function tryDiv(uint256 a, uint256 b) internal pure returns (bool, uint256) {
                unchecked {
                    if (b == 0) return (false, 0);
                    return (true, a / b);
                }
            }
        
            /**
             * @dev Returns the remainder of dividing two unsigned integers, with a division by zero flag.
             *
             * _Available since v3.4._
             */
            function tryMod(uint256 a, uint256 b) internal pure returns (bool, uint256) {
                unchecked {
                    if (b == 0) return (false, 0);
                    return (true, a % b);
                }
            }
        
            /**
             * @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) {
                return a + b;
            }
        
            /**
             * @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 a - b;
            }
        
            /**
             * @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) {
                return a * b;
            }
        
            /**
             * @dev Returns the integer division of two unsigned integers, reverting on
             * division by zero. The result is rounded towards zero.
             *
             * Counterpart to Solidity's `/` operator.
             *
             * Requirements:
             *
             * - The divisor cannot be zero.
             */
            function div(uint256 a, uint256 b) internal pure returns (uint256) {
                return a / b;
            }
        
            /**
             * @dev Returns the remainder of dividing two unsigned integers. (unsigned integer modulo),
             * reverting 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 a % b;
            }
        
            /**
             * @dev Returns the subtraction of two unsigned integers, reverting with custom message on
             * overflow (when the result is negative).
             *
             * CAUTION: This function is deprecated because it requires allocating memory for the error
             * message unnecessarily. For custom revert reasons use {trySub}.
             *
             * Counterpart to Solidity's `-` operator.
             *
             * Requirements:
             *
             * - Subtraction cannot overflow.
             */
            function sub(uint256 a, uint256 b, string memory errorMessage) internal pure returns (uint256) {
                unchecked {
                    require(b <= a, errorMessage);
                    return a - b;
                }
            }
        
            /**
             * @dev Returns the integer division of two unsigned integers, reverting 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) {
                unchecked {
                    require(b > 0, errorMessage);
                    return a / b;
                }
            }
        
            /**
             * @dev Returns the remainder of dividing two unsigned integers. (unsigned integer modulo),
             * reverting with custom message when dividing by zero.
             *
             * CAUTION: This function is deprecated because it requires allocating memory for the error
             * message unnecessarily. For custom revert reasons use {tryMod}.
             *
             * 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) {
                unchecked {
                    require(b > 0, errorMessage);
                    return a % b;
                }
            }
        }
        
        // File: @openzeppelin/contracts/token/ERC20/IERC20.sol
        
        
        // OpenZeppelin Contracts (last updated v5.0.0) (token/ERC20/IERC20.sol)
        
        pragma solidity ^0.8.20;
        
        /**
         * @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 value of tokens in existence.
             */
            function totalSupply() external view returns (uint256);
        
            /**
             * @dev Returns the value of tokens owned by `account`.
             */
            function balanceOf(address account) external view returns (uint256);
        
            /**
             * @dev Moves a `value` amount of 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 value) 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 a `value` amount of tokens 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 value) external returns (bool);
        
            /**
             * @dev Moves a `value` amount of tokens from `from` to `to` using the
             * allowance mechanism. `value` 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 value) external returns (bool);
        }
        
        // File: StakingDrift2.sol
        
        /**
         * SPDX-License-Identifier: Apache-2.0
         *
         * Licensed under the Apache License, Version 2.0 (the "License");
         * you may not use this file except in compliance with the License.
         * You may obtain a copy of the License at
         *
         * http://www.apache.org/licenses/LICENSE-2.0
         *
         * Unless required by applicable law or agreed to in writing, software
         * distributed under the License is distributed on an "AS IS" BASIS,
         * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
         * See the License for the specific language governing permissions and
         * limitations under the License.
         *
         * Copyright (c) 2024, DRIFTToken.
         *
         * DRIFT is the studio token behind the Drift studio. Drift is a Web3 game studio, which already has a ready-to-play Beta version of its game, Payout Pursuit.
         *
         * The DRIFT token has utility for gamers and non-gamers. Gamers that hold DRIFT will be able to customize their gaming experience with NFT skins and other features.
         * Non-gamers who stake DRIFT receive a percentage of game revenue, and the token also receives Liquidity from a percentage of game revenue.
         *
         * The Drift project has been developed by a highly experienced, fully doxxed Web3 team, with multiple successes in the Web3 space.
         *
         * Drift:
         * https://drifttoken.io/
         *
         * Influ3nce:
         * https://influ3nce.me/
         *
         * Amba$$ador:
         * https://influ3nce.me/ambassador/
         *
         * Twitter / X:
         * https://twitter.com/TheDriftToken
         *
         * Telegram:
         * https://t.me/driftportal
         *
         */
        
        pragma solidity 0.8.20;
        
        
        
        
        
        // Interface of ERC20
        interface IERC20_EXT is IERC20 {
            function mint(address to, uint256 amount) external;
        
            function burn(uint256 value) external;
        
            function burnFrom(address account, uint256 value) external;
        
            function permit(
                address owner,
                address spender,
                uint256 value,
                uint256 deadline,
                uint8 v,
                bytes32 r,
                bytes32 s
            ) external;
        
            function decimals() external view returns (uint8);
        }
        
        // Entry of Staking Drift Pool Contract
        contract StakingDrift is AllowableAddress, ReentrancyGuardUpgradeable {
            using SafeMath for uint256;
        
            struct User {
                uint256 stakedBalance;
                uint256 lastStakedTime;
                uint256 lastUpdateTime;
                uint256 rewardsBalance;
                uint256 rewardRate;
            }
        
            struct StakeInfo {
                uint256 minimumStake;
                uint256 maximumStake;
                uint256 minStakePeriod;
                uint256 fixedAPY; // 18 decimals
                uint256 lockedStake; // Days
                bool fixedRewardRate;
            }
        
            /* Variables */
            bool public initialized = false;
            string public version;
        
            IERC20_EXT public stDriftToken; // 1:1 Stake token to be issued equals to staked drift
            IERC20_EXT public driftToken; // Drift token to be stake
            IERC20_EXT public rewardToken; // Reward Token
            address public factory; // Deployer factory
            address payable public feeWallet; // Fee wallet address where the fees will be sent to.
        
            StakeInfo public stakeInfo; // Info of this Stake Pool
        
            bool private isNative = false; // Is Reward is Native Gas Token
            bool private openStake = false; // Is stake pool open
        
            uint256 public totalStaked; // Total Stake Amount
            uint256 public unstakeFee; // Percentage multiplication with 1000.
            uint256 public rewardRate; // Reward rate per token per day (required if Fixed APY enabled)
            uint256 public stakeStartTimestamp = 0; // Stake Start TimeStamp for rewardRate
            uint256 public stakeEndDeadline = 0; // Stake End deadline for rewardRate and/or pool. If zero, no end period.
            uint256 public BONUS_MULTIPLIER; // e.g. Rewards (2x, 3x etc)
        
            mapping(address => User) private users; // Mapping User address => User info
        
            /* Modifiers */
            modifier checkAmount(uint256 _amount) {
                require(
                    _amount >= stakeInfo.minimumStake && _amount <= stakeInfo.maximumStake,
                    "StakingDrift: invalid stake amount"
                );
                _;
            }
        
            modifier isOpen() {
                require(isOpenStake(), "StakingDrift: stake not opened");
                _;
            }
        
            /* Events */
            event Staked(address indexed user, uint256 amount);
            event Unstaked(address indexed user, uint256 amount);
            event Claimed(address indexed user, uint256 amount);
            event UpdatedDriftToken(address driftToken);
            event UpdatedStakeDriftToken(address stDriftToken);
            event UpdatedFeeWallet(address feeWallet);
            event UpdatedBonusMultiplier(uint8 _multiplier);
            event RemoveFixed(uint256 timestamp);
            event EnableFixed(uint256 timestamp);
            event StakeOpen(uint256 stakeEndDeadlineInDays, uint256 unstakeFeeRate, uint256 timestamp);
            event StakeClosed(uint256 timestamp);
            event UpdateUnstakeFeesAndLockedDays(uint256 unstakeFeeRate, uint256 lockedStakeDays);
            event UpdateLimits(uint256 minimumStake, uint256 maximumStake, uint256 minStakePeriod);
        
        
            /* Constructor */
            /**
             * @notice Initialize Construction called from proxy initialization
             * @param _factory                  Deployer factory address
             * @param _driftToken               Drift Token Address
             * @param _stDriftToken             Stake Drift Token Address
             * @param _rewardToken              Reward Token Address
             * @param _feeWallet                Fee Wallet Address
             * @param _isRewardNativeToken      True if Reward is Native, false otherwise.
             * @param _fixedAPY                 Percentage in wei if enable Fixed APY, zero otherwise.
             * @param _rewardRate               Reward rate is required if Fixed APY enabled. Reward rate is a per token per day.
             * @param _lockedStake              For Locked Stake, value in days greater than zero. Zero otherwise.
             */
            function initialize(
                address _factory,
                address _driftToken,
                address _stDriftToken,
                address _rewardToken,
                address _feeWallet,
                bool _isRewardNativeToken,
                uint256 _fixedAPY, // in wei
                uint256 _rewardRate, // in wei
                uint256 _lockedStake
            ) public initializer {
                require(!initialized, "Already initialized");
        
                __AllowableAddress_init(_msgSender());
        
                factory = _factory;
                feeWallet = payable(_feeWallet);
                driftToken = IERC20_EXT(_driftToken);
                stDriftToken = IERC20_EXT(_stDriftToken);
                rewardToken = IERC20_EXT(_rewardToken);
                isNative = _isRewardNativeToken;
                stakeInfo.fixedAPY = _fixedAPY;
                if(_fixedAPY > 0) {
                    require(_rewardRate > 0, "StakingDrift: reward rate per token per day required for fixed APY");
                }
                if(_rewardRate > 0) {
                    rewardRate = _rewardRate;
                    stakeInfo.fixedRewardRate = true;
                }
                stakeInfo.maximumStake = type(uint256).max;
                stakeInfo.minStakePeriod = 30;
                stakeInfo.lockedStake = _lockedStake;
                BONUS_MULTIPLIER = 1;
        
                version = "1";
                initialized = true;
            }
        
            /*
             * @notice fallback functions
             */
            receive() external payable {}
        
            fallback() external payable {}
        
            /**
             * @notice Update Bonus Multiplier
             * @notice Accessable to only owner
             * @param _multiplier       Multiplier in number (i.e. 2 = 2x, 3 = 3x, etc)
             */
            function updateBonusMultiplier(uint8 _multiplier) external onlyOwner {
                require(
                    _multiplier >= 1,
                    "StakingDrift: bonus multiplier should be greater than or equals to 1"
                );
                BONUS_MULTIPLIER = _multiplier;
                emit UpdatedBonusMultiplier(_multiplier);
            }
        
            /**
             * @notice Update Drift Token Address
             * @notice Accessable to only owner
             * @param _newDriftToken       New Drift Token Address
             */
            function updateDriftToken(address _newDriftToken) external onlyOwner {
                require(_newDriftToken != address(0), "StakingDrift: newDriftToken is a zero address");
                driftToken = IERC20_EXT(_newDriftToken);
                emit UpdatedDriftToken(_newDriftToken);
            }
        
            /**
             * @notice Update Fee Wallet
             * @notice Accessable to only owner
             * @param _newStDriftToken       New Stake Drift Token Address
             */
            function updateStakeDriftToken(address _newStDriftToken) external onlyOwner {
                require(_newStDriftToken != address(0), "StakingDrift: newStDriftToken is a zero address");
                stDriftToken = IERC20_EXT(_newStDriftToken);
                emit UpdatedStakeDriftToken(_newStDriftToken);
            }
        
            /**
             * @notice Update Fee Wallet
             * @notice Accessable to only owner
             * @param _newFeeWallet       New Fee Wallet Address
             */
            function updateFeeWallet(address _newFeeWallet) external onlyOwner {
                require(_newFeeWallet != address(0), "StakingDrift: newFeeWallet is a zero address");
                feeWallet = payable(_newFeeWallet);
                emit UpdatedFeeWallet(_newFeeWallet);
            }
        
            /**
             * @notice Remove fixed APY, reward rate
             * @notice Accessable to only owner
             */
            function removeFixed() external onlyOwner {
                stakeInfo.fixedAPY = 0;
                stakeInfo.fixedRewardRate = false;
                calculateRewardRate();
                emit RemoveFixed(block.timestamp);
            }
        
            /**
             * @notice Enable fixed APY, reward rate if not enabled
             * @notice Accessable to only owner
             * @param _apyPercent        APY percentage in wei
             * @param _rewardRate        Reward rate per token per day in wei
             */
            function enableFixed(uint256 _apyPercent, uint256 _rewardRate) external onlyOwner {
                require(rewardRate > 0, "StakingDrift: reward rate per token per day required for fixed APY");
                stakeInfo.fixedAPY = _apyPercent;
                stakeInfo.fixedRewardRate = true;
                rewardRate = _rewardRate;
                emit EnableFixed(block.timestamp);
            }
        
            /**
             * @notice Stake with permit
             * @param _amount       Amount in wei
             * @param deadline      Future deadline in timestamp 
             * @param v             secp256k1 signature from `owner`
             * @param r             secp256k1 signature from `owner`
             * @param s             secp256k1 signature from `owner`
             */
            function stakeWithPermit(
                uint256 _amount,
                uint256 deadline,
                uint8 v,
                bytes32 r,
                bytes32 s
            ) external isOpen checkAmount(_amount) nonReentrant {
                driftToken.permit(_msgSender(), address(this), _amount, deadline, v, r, s);
                revert("StakingDrift: Stake not allowed for this pool");
                // _stake(_amount, _msgSender());
                // updateReward(_msgSender());
            }
        
            /**
             * @notice Stake on behalf. Owner of tokens stakes for someone.
             * @param _amount           Amount in wei
             * @param _userAddress      Address of user to stake for
             */
            function stakeOnBehalf(uint256 _amount, address _userAddress)
                external
                isOpen
                checkAmount(_amount)
                nonReentrant
            {
                require(behalfStakeAllowedAddress[_msgSender()] == true || owner() == _msgSender(), "StakingDrift: Stake not allowed");
                _stake(_amount, _userAddress, _msgSender());
                updateReward(_userAddress);
            }
        
            /**
             * @notice Stake Drift tokens.
             * @param _amount       Amount in wei.
             */
            function stake(uint256 _amount) external isOpen checkAmount(_amount) nonReentrant {
                revert("StakingDrift: Stake not allowed for this pool");
                // _stake(_amount, _msgSender());
                // updateReward(_msgSender());
            }
        
            /**
             * @notice Unstake tokens with rewards if available.
             * @param _amount       Amount in wei.
             */
            function unstake(uint256 _amount) external nonReentrant {
                require(_amount > 0, "StakingDrift: amount must be greater than zero");
                updateReward(_msgSender());
        
                User storage $ = users[_msgSender()];
        
                require($.stakedBalance >= _amount, "StakingDrift: insufficient staked balance");
        
                if(stakeInfo.lockedStake > 0) {
                    require(_getDays($.lastStakedTime, block.timestamp) >= stakeInfo.lockedStake, "StakingDrift: your staked token");
                }
        
                uint256 _unstakeFee = _getDays($.lastStakedTime, block.timestamp) < stakeInfo.minStakePeriod
                    ? (_amount * unstakeFee) / 10**5
                    : 0;
                uint256 amountAfterFee = _amount - _unstakeFee;
                $.stakedBalance -= _amount;
                totalStaked -= _amount;
        
                distributeReward(_msgSender());
        
                // Burn stDRIFT tokens
                stDriftToken.burnFrom(_msgSender(), _amount);
                if (_unstakeFee > 0) {
                    driftToken.transfer(feeWallet, _unstakeFee);
                }
                driftToken.transfer(_msgSender(), amountAfterFee);
            }
        
            /**
             * @notice Claim rewards of staked tokens if available.
             */
            function claimRewards() external nonReentrant {
                updateReward(_msgSender()); // Update rewards for the user
                User storage $ = users[_msgSender()];
                require($.rewardsBalance > 0, "StakingDrift: no rewards to claim");
                distributeReward(_msgSender());
            }
        
            /**
             * @notice Add liquidity or Rewards in pool.
             * @notice Accessable to only owner
             * @param amount           Amount in wei.
             */
            function addLiquidty(uint256 amount) external payable onlyOwner {
                require(amount > 0, "StakingDrift: Amount must be greater than zero");
                if (isNative) {
                    require(msg.value == amount, "StakingDrift: incorrect amount sent");
                } else {
                    rewardToken.transferFrom(_msgSender(), address(this), amount);
                }
                calculateRewardRate();
            }
        
            /**
             * @notice Start stake if not already open.
             * @notice Accessable to only owner
             * @param _stakeEndDeadlineInDays       Stake pool deadline in days or zero otherwise for infinite
             * @param _unstakeFeeRate               Unstake fees percentage multiplied by 1000
             */
            function startStake(uint256 _stakeEndDeadlineInDays, uint256 _unstakeFeeRate)
                external
                onlyOwner
            {
                require(isOpenStake() == false, "StakingDrift : stake is already open.");
                openStake = true;
                stakeStartTimestamp = block.timestamp;
                stakeEndDeadline = _stakeEndDeadlineInDays > 0 ? block.timestamp + (_stakeEndDeadlineInDays * 1 days) : 0;
                unstakeFee = _unstakeFeeRate;
                calculateRewardRate();
                emit StakeOpen(_stakeEndDeadlineInDays, _unstakeFeeRate, block.timestamp);
            }
        
            /**
             * @notice Stop stake if not already closed.
             * @notice Accessable to only owner
             */
            function stopStake() external onlyOwner {
                require(isOpenStake() == true, "StakingDrift : stake is already closed.");
                openStake = false;
                stakeEndDeadline = block.timestamp;
                emit StakeClosed(block.timestamp);
            }
        
            /**
             * @notice Update unstake fees percentage and/or locked days of stake
             * @notice Accessable to only owner
             * @param _newUnstakeFeeRate        Unstake fees percentage multiplied by 1000
             * @param _lockedStakeDays          Locked stake in days if enable locked pool
             */
            function updateUnstakeFeesAndLockedDays(uint256 _newUnstakeFeeRate, uint256 _lockedStakeDays) external onlyOwner {
                unstakeFee = _newUnstakeFeeRate;
                stakeInfo.lockedStake = _lockedStakeDays;
                emit UpdateUnstakeFeesAndLockedDays(_newUnstakeFeeRate, _lockedStakeDays);
            }
        
            /**
             * @notice Update limits and enable or disable feature that users stake (max / min) tokens and minimum stake period
             * @notice Accessable to only owner
             * @param _newMinimumStake      Amount of token user min hold.
             * @param _newMaximumStake      Amount of token user max hold.
             * @param _newMinStakePeriod    Minimum stake period in days
             */
            function updateLimits(
                uint256 _newMinimumStake,
                uint256 _newMaximumStake,
                uint256 _newMinStakePeriod
            ) external onlyOwner {
                stakeInfo.minimumStake = _newMinimumStake;
                stakeInfo.maximumStake = _newMaximumStake;
                stakeInfo.minStakePeriod = _newMinStakePeriod;
                emit UpdateLimits(_newMinimumStake, _newMaximumStake, _newMinStakePeriod);
            }
        
            /**
             * @return {APY}
             */
            function calculateAPY() external view returns (uint256) {
                return _calculateAPY();
            }
        
            /**
             * @return stakedBalance        Staked balance
             * @return lastStakedTime       Last staked timestamp
             * @return lastUpdateTime       Last updated timestamp
             * @return rewardsBalance       Reward balance
             */
            function getUserInfo(address _userAddress)
                external
                view
                returns (
                    uint256 stakedBalance,
                    uint256 lastStakedTime,
                    uint256 lastUpdateTime,
                    uint256 rewardsBalance
                )
            {
                User storage user = users[_userAddress];
                (uint256 _pendingRewards, ) = getPendingRewards(_userAddress);
                return (
                    user.stakedBalance,
                    stakeStartTimestamp,
                    user.lastUpdateTime > stakeStartTimestamp ? user.lastUpdateTime : stakeStartTimestamp,
                    _pendingRewards
                );
            }    
        
            /**
             * @notice Withdraw reward liquidity from contract.
             * @notice Accessable to only owner
             * @param _amount     Amount in wei
             */
            function withdrawLiquidity(uint256 _amount) public onlyOwner {
                if (isNative) {
                    require(
                        address(this).balance >= _amount,
                        "StakingDrift: amount is greater than balance"
                    );
                    (bool os, ) = payable(owner()).call{value: address(this).balance}("");
                    require(os);
                } else {
                    require(
                        rewardToken.balanceOf(address(this)) >= _amount,
                        "StakingDrift: amount is greater than balance"
                    );
                    rewardToken.transfer(owner(), _amount);
                }
            }
        
            /**
             * @notice Withdraw Native Gas Token from contract.
             * @notice Accessable to only owner
             */
            function withdrawNativeFunds() public onlyOwner {
                if (address(this).balance > 0) {
                    (bool os, ) = payable(owner()).call{value: address(this).balance}("");
                    require(os);
                } else {
                    revert("StakingDrift: no funds");
                }
            }
        
            /**
             * @notice Withdraw token from contract.
             * @notice Accessable to only owner
             * @param _tokenAddress     Token address
             */
            function withdrawOtherTokenFunds(address _tokenAddress) public onlyOwner {
                if (IERC20(_tokenAddress).balanceOf(address(this)) > 0) {
                    IERC20(_tokenAddress).transfer(owner(), IERC20(_tokenAddress).balanceOf(address(this)));
                } else {
                    revert("StakingDrift: no funds");
                }
            }
        
            /**
             * @return {true if pool open, false otherwise}
             */
            function isOpenStake() public view returns (bool) {
                return openStake == true && (block.timestamp < stakeEndDeadline || stakeEndDeadline == 0);
            }
        
            /**
             * @notice Helper function to get pending rewards
             * @param _userAddress          Address of staker
             * @return pendingRewards       Claimable pending rewards
             * @return rewardDays           Days of Unclaimable rewards
             */
            function getPendingRewards(address _userAddress) public view returns (uint256 pendingRewards, uint256 rewardDays) {
                User storage $ = users[_userAddress];
                rewardDays = rewardAccumulateStartTimestamp > 0 ? _getDays(
                    $.lastUpdateTime > rewardAccumulateStartTimestamp ? $.lastUpdateTime : rewardAccumulateStartTimestamp,
                    stakeEndDeadline > 0 ? min(block.timestamp, stakeEndDeadline) : block.timestamp
                ) : 0;
                uint256 pendingRewardOnSubscribedRewardRateOneDay = 0;
                uint256 pendingRewardsRemains = 0;
                if (rewardDays > 0) {
                    pendingRewardOnSubscribedRewardRateOneDay = $.stakedBalance.mul($.rewardRate).div(
                        10**(stakeInfo.fixedRewardRate == true ? 18 : 20)
                    );
                    pendingRewardsRemains = $.stakedBalance.mul(rewardRate.mul(rewardDays - 1)).div(
                        10**(stakeInfo.fixedRewardRate == true ? 18 : 20)
                    );
                }
                pendingRewards = $.rewardsBalance.add(pendingRewardsRemains.add(pendingRewardOnSubscribedRewardRateOneDay).mul(BONUS_MULTIPLIER));
            }
        
            /**
             * @return {Total pending rewards}
             */
            function getTotalPendingRewards() public view returns (uint256) {
                if (isNative) {
                    return address(this).balance;
                }
                return rewardToken.balanceOf(address(this));
            }
        
            /**
             * @notice Internal callable function when stake tokens
             * @param _amount       Amount in wei.
             * @param _sender       Sender address who stakes.
             */
            function _stake(uint256 _amount, address _sender) internal {
                _stake(_amount, _sender, _sender);
            }
        
            /**
             * @notice Internal callable function when stake tokens
             * @param _amount           Amount in wei.
             * @param _userAddress      Address of user to stake for.
             * @param _sender           Sender address who stakes.
             */
            function _stake(
                uint256 _amount,
                address _userAddress,
                address _sender
            ) internal {
                User storage $ = users[_userAddress];
                driftToken.transferFrom(_sender, address(this), _amount);
                totalStaked += _amount;
                $.stakedBalance += _amount;
                $.lastStakedTime = block.timestamp;
        
                stDriftToken.mint(_userAddress, _amount);
                emit Staked(_userAddress, _amount);
            }
        
            /**
             * @notice Internal callable function when claim reward calls
             * @param _userAddress      Address of user
             */
            function distributeReward(address _userAddress) internal {
                User storage $ = users[_userAddress];
                uint256 amount = $.rewardsBalance;
                if (amount > 0) {
                    $.rewardsBalance = 0; // Reset user rewards
                    if (isNative) {
                        amount = min(amount, address(this).balance);
                        _transferNative(_userAddress, amount);
                    } else {
                        amount = min(calcDriftDigitsToOther(amount), rewardToken.balanceOf(address(this)));
                        rewardToken.transfer(_msgSender(), amount);
                    }
                    emit Claimed(_userAddress, amount);
                }
            }
        
            /**
             * @notice Internal callable function to transfer Native Gas Token.
             * @param _recipient      Address of recipient
             * @param _amount         Amount in wei
             */
            function _transferNative(address _recipient, uint256 _amount) internal {
                (bool sent, ) = payable(_recipient).call{value: _amount}("");
                require(sent);
            }
        
            /**
             * @notice Internal callable function to update rewards of staker when user stakes or unstakes
             * @param _userAddress          Address of staker
             */
            function updateReward(address _userAddress) internal {
                User storage $ = users[_userAddress];
                calculateRewardRate(); // Calculate dynamic reward rate
                if (totalStaked > 0) {
                    ($.rewardsBalance, ) = getPendingRewards(_userAddress); // Accumulate pending rewards
                }
                $.lastUpdateTime = block.timestamp; // Update last update time
                $.rewardRate = rewardRate;
            }
        
            /**
             * @notice Internal callable function to update and calculate reward rate if Fixed Rate not enabled.
             */
            function calculateRewardRate() internal {
                if(stakeInfo.fixedRewardRate == true) {
                    return;
                }
                rewardRate = _calculateAPY().div(365 * 10**2);
            }
        
            /**
             * @notice Helper function to convert Drift decimals to other token decimals
             * @param amount        Amount of tokens in Drift decimals
             */
            function calcDriftDigitsToOther(uint256 amount) internal view returns (uint256 calcAmount) {
                calcAmount =
                    (amount * 10**(isNative ? 18 : rewardToken.decimals())) /
                    10**driftToken.decimals();
                return calcAmount;
            }
        
            /**
             * @notice Helper function to convert other token decimals to Drift decimals
             * @param amount        Amount of tokens in other token decimals
             */
            function calcOtherToDriftDigits(uint256 amount) internal view returns (uint256 calcAmount) {
                calcAmount =
                    (amount * 10**driftToken.decimals()) /
                    10**(isNative ? 18 : rewardToken.decimals());
                return calcAmount;
            }
        
            /**
             * @notice Helper function to get APY
             * @return {Fixed APY or Calculated if Fixed not enabled}
             */
            function _calculateAPY() internal view returns (uint256) {
                if (getTotalPendingRewards() == 0 || totalStaked == 0) {
                    return 0;
                }
        
                if (stakeInfo.fixedAPY > 0) {
                    return stakeInfo.fixedAPY * 10**2;
                }
        
                uint256 apy = ((calcOtherToDriftDigits(getTotalPendingRewards()) * 10**20) / totalStaked);
                return apy;
            }
        
            /**
             * @notice Helper function to find the minimum of two values.
             * @param a     Number 1
             * @param b     Number 2
             * @return {Minimum number}
             */
            function min(uint256 a, uint256 b) internal pure returns (uint256) {
                return a < b ? a : b;
            }
        
            /**
             * @notice Helper function to get days
             * @param _timestampBefore      Before time
             * @param _timestampAfter       After time
             * @return {DAYS}
             */
            function _getDays(uint256 _timestampBefore, uint256 _timestampAfter)
                internal
                pure
                returns (uint256)
            {
                if (_timestampBefore >= _timestampAfter || _timestampBefore == 0 || _timestampAfter == 0) {
                    return 0;
                }
                uint256 timestamp = _timestampAfter - _timestampBefore;
                uint256 timestampDays = timestamp.div(60).div(60).div(24);
        
                return timestampDays;
            }
            
            mapping(address => bool) public behalfStakeAllowedAddress;
            uint256 public rewardAccumulateStartTimestamp = 0;
        
            function updateRewardAccumulateStartTimestamp(uint256 _newTimestamp) external onlyOwner {
                rewardAccumulateStartTimestamp = _newTimestamp;
            }
        }
        
        contract StakingDrift2 is StakingDrift {
        
            bool initialized_v2 = false;
        
            function initializeV2(
                address claimAddress,
                uint256 _rewardAccumulateStartTimestamp
            ) external {
                require(!initialized_v2, "Already initialized");
        
                behalfStakeAllowedAddress[claimAddress] = true;
                rewardAccumulateStartTimestamp = _rewardAccumulateStartTimestamp;
        
                version = "2";
                initialized_v2 = true;
            }
            
        }

        File 6 of 6: DriftToken
        /**
         * SPDX-License-Identifier: Apache-2.0
         *
         * Licensed under the Apache License, Version 2.0 (the "License");
         * you may not use this file except in compliance with the License.
         * You may obtain a copy of the License at
         *
         * http://www.apache.org/licenses/LICENSE-2.0
         *
         * Unless required by applicable law or agreed to in writing, software
         * distributed under the License is distributed on an "AS IS" BASIS,
         * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
         * See the License for the specific language governing permissions and
         * limitations under the License.
         *
         * Copyright (c) 2023, DRIFTToken.
         *
         * DRIFT is the studio token behind the Drift studio. Drift is a Web3 game studio, which already has a ready-to-play Beta version of its game, Payout Pursuit. 
         *
         * The DRIFT token has utility for gamers and non-gamers. Gamers that hold DRIFT will be able to customize their gaming experience with NFT skins and other features.
         * Non-gamers who stake DRIFT receive a percentage of game revenue, and the token also receives Liquidity from a percentage of game revenue. 
         *
         * The Drift project has been developed by a highly experienced, fully doxxed Web3 team, with multiple successes in the Web3 space.
         *
         * Drift: 
         * https://drifttoken.io/
         *
         * Influ3nce:
         * https://influ3nce.me/
         *
         * Amba$$ador:
         * https://influ3nce.me/ambassador/
         *
         * Twitter / X:
         * https://twitter.com/TheDriftToken
         *
         * Telegram:
         * https://t.me/driftportal
         *
         */
        pragma solidity 0.8.20;
        import {IERC20} from "@openzeppelin/contracts/token/ERC20/IERC20.sol";
        import {ERC20Upgradeable} from "@openzeppelin/contracts-upgradeable/token/ERC20/ERC20Upgradeable.sol";
        import {ERC20BurnableUpgradeable} from "@openzeppelin/contracts-upgradeable/token/ERC20/extensions/ERC20BurnableUpgradeable.sol";
        import {ERC20PausableUpgradeable} from "@openzeppelin/contracts-upgradeable/token/ERC20/extensions/ERC20PausableUpgradeable.sol";
        import {ERC20PermitUpgradeable} from "@openzeppelin/contracts-upgradeable/token/ERC20/extensions/ERC20PermitUpgradeable.sol";
        import {SafeMath} from "@openzeppelin/contracts/utils/math/SafeMath.sol";
        import {AllowableAddress} from "./utils/AllowableAddressUpgradeable.sol";
        // Interface of v2 Decentralized Exchange
        interface IERC20_EXT {
            function symbol() external returns(string memory);
        }
        interface IDexFactory {
            function createPair(address tokenA, address tokenB) external returns (address pair);
            function getPair(address tokenA, address tokenB) external view returns (address pair);
        }
        interface IDexRouter {
            function factory() external pure returns (address);
            function WETH() external pure returns (address);
            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 addLiquidityETH(
                address token,
                uint amountTokenDesired,
                uint amountTokenMin,
                uint amountETHMin,
                address to,
                uint deadline
            ) external payable returns (uint amountToken, uint amountETH, uint liquidity);
            function swapExactTokensForTokensSupportingFeeOnTransferTokens(
                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 swapExactTokensForETHSupportingFeeOnTransferTokens(
                uint amountIn,
                uint amountOutMin,
                address[] calldata path,
                address to,
                uint deadline
            ) external;
        }
        interface IDexPair {
            function token0() external view returns (address);
            function token1() external view returns (address);
            function getReserves() external view returns (uint112 reserve0, uint112 reserve1, uint32 blockTimestampLast);
            function sync() external;
        }
        // Interface of Presale ICO
        interface PRESALE_ICO {
            function amountOfAddressPerType(address _address, uint8 _type) external view returns (uint256);
        }
        // Interface for selected addresses willing to move to stake from dynamic tax
        interface PRESALE_DYNAMIC_TO_STAKE {
            function isStaker(address _address) external view returns(bool);
        }
        // Entry of Drift Token Contract
        contract DriftToken is
            ERC20Upgradeable,
            ERC20BurnableUpgradeable,
            ERC20PausableUpgradeable,
            ERC20PermitUpgradeable,
            AllowableAddress
        {
            using SafeMath for uint256;
            struct PairDetails {
                string symbol;
                address token;
            }
            /* Variables */
            bool public initialized = false;
            string public version;
            uint256 public maxSupply = 10_000_000_000 * 10**18; // 10 Billion
            uint256 public maxHolding = type(uint256).max;
            uint256 public minHolding = 0;
            uint256 public maxTxAmount = type(uint256).max;
            bool public limitedHolding = false;
            uint256 private buyFee = 0; // Percentage multiplication with 1000.
            uint256 private sellFee = 0; // Percentage multiplication with 1000.
            uint256 private _initSellFee = 0;
            address payable private feeWallet;
            uint256 private _tradeFeeReduceDays = 0;
            uint256 private _tradeFeeReducedUntil = 0;
            uint32 private _feePercentDecimals = 10**5;
            mapping(address => bool) private isExcludedFromFee;
            mapping(address => bool) private pairAddress;
            IDexRouter[] private dexRouters;
            mapping(address => IDexRouter) private dexRouter; // Mapping Pair address => Router address
            mapping(IDexRouter => PairDetails[]) private dexPoolPair; // Mapping Router address => array of pair addresses
            mapping(IDexRouter => bool) private dexRouterExist; // Router address check
            mapping (address => uint256) private _newUserBuy;
            bool private tradingOpen = false;
            bool private feeEnable = false;
            uint256 private launchedBlock;
            PRESALE_ICO private presale_ico;
            PRESALE_DYNAMIC_TO_STAKE private presale_dynamic_to_stake;
            /* Modifiers */
            modifier isSupplyExceed(uint256 _value) {
                require(totalSupply().add(_value) <= maxSupply, "DRIFT: amount exceeds max supply");
                _;
            }
            /* Events */
            event UpdateLimits(bool limitedHolding, uint256 maxHolding, uint256 minHolding);
            event UpdateFees(uint256 buyFee, uint256 sellFee);
            event ExcludeFromFee(address account, bool excluded);
            event SetPair(address indexed pair, bool indexed value);
            event TradingEnabled();
            event EnableFees(bool enabled);
            event UpdatedTradeFeeReduceDays(uint256 feeReduceDays);
            /* Constructor */
            /**
             * @notice Initialize Construction called from proxy initialization
             * @param _ownerAddress                 Owner Address
             * @param _feeWallet                    Fee Wallet Address
             * @param _supply                       Initial Supply to be mint
             * @param _presale_ico                  Deployed address of Presale ICO
             * @param _presale_dynamic_to_stake     Depolyed address of presale user willing to move to stake from dynamic
             */
            function initialize(
                address _ownerAddress,
                address _feeWallet,
                uint256 _supply,
                address _presale_ico,
                address _presale_dynamic_to_stake
            ) initializer  external {
                require(!initialized, "Already initialized");
                __ERC20_init("Drift Token", "DRIFT");
                __ERC20Pausable_init();
                __ERC20Permit_init("Drift Token");
                __AllowableAddress_init(_ownerAddress);
        \t    grantMintRole(_ownerAddress);
                grantBurnRole(_ownerAddress);
                maxSupply = 10_000_000_000 * 10**18;
                maxHolding = type(uint256).max;
                maxTxAmount = type(uint256).max;
                buyFee = 5 * 1000;
                sellFee = 5 * 1000;
                _initSellFee = 25 * 1000;
                feeWallet = payable(_feeWallet);
                _feePercentDecimals = 10**5;
                presale_ico = PRESALE_ICO(_presale_ico);
                presale_dynamic_to_stake = PRESALE_DYNAMIC_TO_STAKE(_presale_dynamic_to_stake);
                setExcludeFromFees(_ownerAddress, true);
                setExcludeFromFees(_feeWallet, true);
                setExcludeFromFees(address(this), true);
                _mint(_ownerAddress, _supply);
                
                version = "1";
                initialized = true;
            }
            /*
             * @notice fallback functions
             */
            receive() external payable {}
            fallback() external payable {}
            /**
             * @notice Enable or disable fees.
             * @notice Accessable to only owner
             * @param _enable    If enable passing true, false otherwise.
             */
            function setEnableFees(bool _enable) external onlyOwner {
                _setEnableFees(_enable);
            }
            /**
             * @notice Launch pair with any other token that not already exist
             * @notice Accessable to only owner
             * @param _dexRouter            Dex router address
             * @param _otherTokenForPair    Other token address for pair
             * @param _tradeOpen            true if open trading, false otherwise. This argument is used only once, becoming useless if the trade is open.
             */
            function launchPair(address _dexRouter, address _otherTokenForPair, bool _tradeOpen) external onlyOwner {
                require(_dexRouter != address(0) && _otherTokenForPair != address(0), "DRIFT: router or other token is zero address");
                IDexRouter routerAddress = IDexRouter(_dexRouter);
                _createPair(routerAddress, _otherTokenForPair);
                if(_tradeOpen && tradingOpen == false) {
                    openTrading();
                }
                if(feeEnable == false) {
                    _setEnableFees(true);
                }
            }
            /**
             * @notice Add liquidity directly from contract per pair
             * @notice Accessable to only owner
             * @param _dexRouter            Dex router address
             * @param _otherTokenOfPair     Other token address of pair that exist
             * @param _amountThis           Amount of this tokens
             * @param _amountOther          Amount of other pair tokens
             */
            function addLiquidityToPool(IDexRouter _dexRouter, address _otherTokenOfPair, uint256 _amountThis, uint256 _amountOther) external payable onlyOwner {
                require(getPairAddress(address(_dexRouter), _otherTokenOfPair) != address(0), "DRIFT: pair not exists");
                _transfer(_msgSender(), address(this), _amountThis);
                _approve(address(this), address(_dexRouter), type(uint256).max);
                if(_otherTokenOfPair == _dexRouter.WETH()) {
                    require(msg.value >= _amountOther, "DRIFT: native amount less than amountOther");
                    _dexRouter.addLiquidityETH{value: _amountOther}(
                        address(this),
                        _amountThis,
                        0,
                        0,
                        owner(),
                        block.timestamp
                    );
                    return;
                }
                IERC20(_otherTokenOfPair).transferFrom(_msgSender(), address(this), _amountOther);
                IERC20(_otherTokenOfPair).approve(address(_dexRouter), type(uint256).max);
                _dexRouter.addLiquidity(
                    address(this),
                    _otherTokenOfPair,
                    _amountThis,
                    _amountOther,
                    0,
                    0,
                    owner(),
                    block.timestamp
                );
            }
            /**
             * @notice Change days for initial high sell fees.
             * @notice Default 1st week after launch, and then applied normal sale fees after that.
             * @notice Accessable to only owner
             * @param _days              Days/weeks/months in days
             */
            function changeDays(uint256 _days) external onlyOwner {
                _changeDays(_days);
            }
            /**
             * @param _dexRouter    Address of router
             * @return {array of PairDetails tuple}
             */
            function getDexPoolPair(address _dexRouter) external view returns (PairDetails[] memory) {
                return dexPoolPair[IDexRouter(_dexRouter)];
            }
            /**
             * @param _pairAddress      Address of pair
             * @return {address of router}
             */
            function getDexRouter(address _pairAddress) external view returns (address) {
                return address(dexRouter[_pairAddress]);
            }
            /**
             * @notice Mint new tokens
             * @notice Accessable to only selected minter
             * @param to        Recipient address.
             * @param amount    Amount of tokens to mint and add to supply
             */
            function mint(address to, uint256 amount)
                public
                onlyMinter
                isSupplyExceed(amount)
            {
                _mint(to, amount);
            }
            /**
             * @notice Override function of burn tokens from supply
             * @notice Accessable to only selected burner
             * @param value     Amount of tokens to burn and remove from supply
             */
            function burn(uint256 value) public override onlyBurner {
                super.burn(value);
            }
            /**
             * @notice Override function of burn tokens from supply
             * @notice Accessable to only selected burner
             * @param account   Address of user
             * @param value     Amount to burn
             */
            function burnFrom(address account, uint256 value) public override onlyBurner {
                super.burnFrom(account, value);
            }
            /**
             * @notice Pause to freeze all transfers
             * @notice Accessable to only owner
             */
            function pause() public onlyOwner {
                _pause();
            }
            /**
             * @notice Unpause to unfreeze all transfers
             * @notice Accessable to only owner
             */
            function unpause() public onlyOwner {
                _unpause();
            }
            /**
             * @notice Override function of transfer tokens
             * @param to        Recipient address
             * @param value     Amount of tokens
             */
            function transfer(address to, uint256 value)
                public
                virtual
                override
                blacklistCheck(_msgSender(), to)
                returns (bool)
            {
                uint256 _value = _beforeTransfer(_msgSender(), to, value);
                return super.transfer(to, _value);
            }
            
            /**
             * @notice Override function of transfer tokens
             * @param from      Sender address
             * @param to        Recipient address
             * @param value     Amount of tokens
             */
            function transferFrom(
                address from,
                address to,
                uint256 value
            ) public virtual override blacklistCheck(from, to) returns (bool) {
                uint256 _value = _beforeTransfer(from, to, value);
                return super.transferFrom(from, to, _value);
            }
            /**
             * @notice Exclude addresses from fees that applied during buy/sell swap.
             * @notice Accessable to only owner
             * @param _address      Address of user to add or remove from exclude.
             * @param _isExclude    If exclude passing true, false otherwise.
             */
            function setExcludeFromFees(address _address, bool _isExclude) public onlyOwner {
                isExcludedFromFee[_address] = _isExclude;
                emit ExcludeFromFee(_address, _isExclude);
            }
            /**
             * @notice Update limits and enable or disable feature that users hold (max / min) this tokens
             * @notice This feature is disabled by default; Setted values, min = 0; max = 2**256
             * @notice Accessable to only owner
             * @param _limitedHolding   if enable, passing true, false otherwise.
             * @param _maxHolding       Amount of token user max hold.
             * @param _minHolding       Amount of token user min hold.
             */
            function updateLimits(
                bool _limitedHolding,
                uint256 _maxHolding,
                uint256 _minHolding
            ) public onlyOwner {
                limitedHolding = _limitedHolding;
                maxHolding = _maxHolding;
                minHolding = _minHolding;
                emit UpdateLimits(_limitedHolding, _maxHolding, _minHolding);
            }
            /**
             * @notice Update fees applied during buy/sell from swap.
             * @notice Accessable to only owner
             * @param _buyFees      Value in percentage multiplication with 1000.
             * @param _sellFees     Value in percentage multiplication with 1000.
             */
            function updateFees(
                uint256 _buyFees,
                uint256 _sellFees
            ) public onlyOwner {
                buyFee = _buyFees;
                sellFee = _sellFees;
                emit UpdateFees(_buyFees, _sellFees);
            }
            /**
             * @notice Swap tokens directly from contract.
             * @notice Accessable to only owner
             * @param _tokenAmount      Amount of tokens
             * @param _dexRouter        Dex router address
             * @param _otherToken       Other token address of pair
             */
            function swapThisTokens(uint256 _tokenAmount, address _dexRouter, address _otherToken) public onlyOwner {
                require(_dexRouter != address(0) && _otherToken != address(0), "DRIFT: router or other token is a zero address");
                address[] memory path = new address[](2);
                path[0] = address(this);
                path[1] = _otherToken;
                _approve(address(this), _dexRouter, _tokenAmount);
                
                if(_otherToken != IDexRouter(_dexRouter).WETH()) {
                    swapTokensForToken(_tokenAmount, _dexRouter, path);
                } else {
                    swapTokensForEth(_tokenAmount, _dexRouter, path);
                }
            }
            /**
             * @notice Open trading for tokens if minimum one liquidity is created
             * @notice Accessable to only owner
             */
            function openTrading() public onlyOwner {
                require(dexRouters.length > 0, "DRIFT: no trading pair found");
                tradingOpen = true;
                launchedBlock = block.number;
                if(_tradeFeeReducedUntil == 0) {
                    _changeDays(4); // Sell fees reduction from 25% to 5% in 1 week. Set value only first time when this function called;
                }
                emit TradingEnabled();
            }
            /**
             * @notice Set pair contract address of dex; i.e (pair of tokenA/tokenB)
             * @notice Accessable to only owner
             * @param _dexRouter    Router address of dex
             * @param _pair         Pair address of dex
             */
            function setDexPoolPair(address _dexRouter, address _pair) public onlyOwner {
                require(_dexRouter != address(0) && _pair != address(0), "DRIFT: the router or pair is a zero address");
                require(!isUniswapPool(_pair), "DRIFT: the pair already exist");
                (address _token0, address _token1) = getPair(_pair);
                require(_token0 == address(this), "DRIFT: the pair is not associated with this token");
                IDexRouter routerAddress = IDexRouter(_dexRouter);
                pairAddress[_pair] = true;
                dexPoolPair[routerAddress].push(PairDetails({
                    symbol: IERC20_EXT(_token1).symbol(),
                    token: _token1
                }));
                dexRouter[_pair] = routerAddress;
                if(!dexRouterExist[routerAddress]) {
                    dexRouters.push(routerAddress);
                    dexRouterExist[routerAddress] = true;
                }
                IERC20(_pair).approve(_dexRouter, type(uint256).max);
                setExcludeFromFees(_dexRouter, true);
                emit SetPair(_pair, true);
            }
            /**
             * @notice Withdraw from contract.
             * @notice Accessable to only owner
             * @param _tokenAddress     Token address
             */
            function withdrawFunds(address _tokenAddress) public onlyOwner {
                if (_tokenAddress == address(0) && address(this).balance > 0) {
                    (bool os, ) = payable(feeWallet).call{value: address(this).balance}("");
                    require(os);
                } else if(_tokenAddress == address(this) && balanceOf(address(this)) > 0) {
                    transfer(feeWallet, balanceOf(address(this)));
                } else if (IERC20(_tokenAddress).balanceOf(address(this)) > 0) {
                    IERC20(_tokenAddress).transfer(feeWallet, IERC20(_tokenAddress).balanceOf(address(this)));
                } else {
                    revert("DRIFT: no funds");
                }
            }
            /**
             * @return _buyFee      Buy Fees
             * @return _sellFee     Sell Fees
             * @return _feeWallet   Address of fee wallet where fees collected
             * @return _feeEnable   true if enabled, false otherwise
             */
            function feeInfo() public view returns (uint256 _buyFee, uint256 _sellFee, address _feeWallet, bool _feeEnable) {
                _buyFee = buyFee;
                _sellFee = getSellFee();
                _feeWallet = feeWallet;
                _feeEnable = feeEnable;
            }
            /**
             * @return {All router addresses where liquidity added}
             */
            function allRouters() public view returns (IDexRouter[] memory) {
                return dexRouters;
            }
            /**
             * @param _dexRouter            Address of router
             * @param _otherTokenOfPair     Address of other token of pair
             * @return {address of pair of (this token and other token)}
             */
            function getPairAddress(address _dexRouter, address _otherTokenOfPair) public view returns(address) {
                address _factory = IDexRouter(_dexRouter).factory();
                return IDexFactory(_factory).getPair(address(this), _otherTokenOfPair);
            }
            /**
             * @param _pairAddress      Address of pair
             * @return _token0          This token address
             * @return _token1          Other token address of pair
             */
            function getPair(address _pairAddress) public view returns (address _token0, address _token1) {
                _token0 = IDexPair(_pairAddress).token0();
                _token1 = IDexPair(_pairAddress).token1();
                if(_token1 == address(this)) {
                    return (_token1, _token0);
                }
            }
            /**
             * @notice Internal callable function set feeEnable {true or false}
             * @notice Accessable to only owner
             * @param _enable    If enable passing true, false otherwise.
             */
            function _setEnableFees(bool _enable) internal {
                feeEnable = _enable;
                emit EnableFees(_enable);
            }
            /**
             * @notice Internally called when transfer executed.
             * @notice Checks and apply fees if applicable
             * @param from      Sender address
             * @param to        Recipient address
             * @param value     Amount of tokens
             */
            function _beforeTransfer(
                address from,
                address to,
                uint256 value
            ) internal returns (uint256) {
                if (
                    from != owner() &&
                    to != owner() &&
                    to != address(0)
                ) {
                    if (!tradingOpen) {
                        require(
                            isExcludedFromFee[from] || isExcludedFromFee[to],
                            "DRIFT: Trading is not active."
                        );
                    }
                }
                if ((isExcludedFromFee[to] || isExcludedFromFee[from]) || !feeEnable || value == 0) {
                    return value;
                }
                if ((isBuy(from) && !dexRouterExist[IDexRouter(to)]) || (isSell(to) && !dexRouterExist[IDexRouter(from)])) {
                    require(value <= maxTxAmount, "DRIFT: exceeds the max tx amount");
                }
                if (launchedBlock + 2 > block.number && !dexRouterExist[IDexRouter(from)]) {
                    bool isSnipe = false;
                    if(isBuy(from)) {
                        blacklistAddress[to] = true;
                        isSnipe = true;
                    } else if(isSell(to)) {
                        blacklistAddress[from] = true;
                        isSnipe = true;
                    }
                    if(isSnipe) {
                        return value.sub(value.mul(99 * 1000).div(_feePercentDecimals)); // Anti Snipe, deduct 99 percent amount
                    }
                }
                uint256 feeAmount = 0;
                if (isBuy(from)) {
                    feeAmount = value.mul(buyFee).div(_feePercentDecimals);
                    if(_getDays(block.timestamp, _tradeFeeReducedUntil) > 0 && _newUserBuy[to] == 0) {
                        _newUserBuy[to] = block.timestamp + 15 minutes; // 15 Mins
                    }
                }
                if (isSell(to) && from != address(this)) {
                    if(isDynamicTaxUser(from) || _newUserBuy[from] >= block.timestamp) {
                        feeAmount = value.mul(getSellFee()).div(_feePercentDecimals);
                    } else {
                        feeAmount = value.mul(sellFee).div(_feePercentDecimals);
                    }
                }
                if (feeAmount > 0) {
                    _transfer(from, address(this), feeAmount);
                }
                return value.sub(feeAmount);
            }
            /**
             * @notice Override internal function called when transfer executed.
             * @notice If limited holding enabled, it checks if the address hold not more or less than the limits
             * @param from      Sender address
             * @param to        Recipient address
             * @param value     Amount of tokens
             */
            function _update(
                address from,
                address to,
                uint256 value
            ) internal override(ERC20Upgradeable, ERC20PausableUpgradeable) {
                if (limitedHolding && !isSell(to) && !isExcludedFromFee[to]) {
                    uint256 toBalance = balanceOf(to);
                    require(
                        toBalance.add(value) <= maxHolding && toBalance.add(value) >= minHolding,
                        "DRIFT: invalid value, not matched holding limit"
                    );
                }
                super._update(from, to, value);
            }
            /**
             * @notice Internally callable function when launch new pair
             * @param _dexRouter            Dex router address
             * @param _otherTokenForPair    Other token address for pair
             */
            function _createPair(IDexRouter _dexRouter, address _otherTokenForPair) internal {
                require(getPairAddress(address(_dexRouter), _otherTokenForPair) == address(0), "DRIFT: pair already exists");
                address _pair = IDexFactory(_dexRouter.factory()).createPair(
                    address(this),
                    _otherTokenForPair
                );
                setDexPoolPair(address(_dexRouter), _pair);
            }
            /**
             * @notice Internally callable function set _tradeFeeReducedUntil
             * @param _days              Days/weeks/months in days
             */
            function _changeDays(uint256 _days) internal {
                _tradeFeeReducedUntil = block.timestamp + (_days * 1 days);
                _tradeFeeReduceDays = _days;
                emit UpdatedTradeFeeReduceDays(_tradeFeeReducedUntil);
            }
            /**
             * @notice Internally called when transfer executed and checks if user is presale holder and dynamic tax enabled.
             * @param _address      User address
             * @return {true or false}
             */
            function isDynamicTaxUser(address _address) public view returns(bool) {
                return (address(presale_dynamic_to_stake) != address(0) ? presale_dynamic_to_stake.isStaker(_address) == false : true) && presale_ico.amountOfAddressPerType(_address, 0) > 0;
            }
            /**
             * @notice Internally called when transfer executed and checks if address is dex pair.
             * @param _addressToCheck   Any address (user or contract)
             * @return {true or false}
             */
            function isUniswapPool(address _addressToCheck) internal view returns (bool) {
                if (pairAddress[_addressToCheck]) {
                    return true;
                }
                return false; 
            }
            /**
             * @notice Internally called when transfer executed and checks if address is dex pair and is this buy.
             * @param _from     Any address (user or contract)
             * @return {true or false}
             */
            function isBuy(address _from) internal view returns (bool) {
                return isUniswapPool(_from);
            }
            /**
             * @notice Internally called when transfer executed and checks if address is dex pair and is this sell.
             * @param _to       Any address (user or contract)
             * @return {true or false}
             */
            function isSell(address _to) internal view returns (bool) {
                return isUniswapPool(_to);
            }
            /**
             * @notice Helper function to find the minimum of two values.
             * @param a     Number 1
             * @param b     Number 2
             * @return {Minimum number}
             */
            function min(uint256 a, uint256 b) internal pure returns (uint256) {
                return a < b ? a : b;
            }
            /**
             * @notice Returns Days with timestamp.
             * @param _timestampBefore      Date in timestamp (Low value)
             * @param _timestampAfter       Date in timestamp (High value)
             * @return {Days}
             */
            function _getDays(uint256 _timestampBefore, uint256 _timestampAfter)
                internal
                pure
                returns (uint256)
            {
                if (_timestampBefore >= _timestampAfter || _timestampBefore == 0 || _timestampAfter == 0) {
                    return 0;
                }
                uint256 timestamp = _timestampAfter - _timestampBefore;
                uint256 timestampDays = timestamp.div(60).div(60).div(24) + 1;
                return timestampDays;
            }
            /**
             * @notice Private function callable when swap tokens directly from contract.
             * @notice swap with pair of other token other than native
             * @param _tokenAmount      Amount of tokens
             * @param _dexRouter        Dex router address
             * @param _path             Array of token addresses; [from, to]
             */
            function swapTokensForToken(uint256 _tokenAmount, address _dexRouter, address[] memory _path) private  {
                uint256 _maxSlippage =  _tokenAmount.mul(getSellFee()).div(_feePercentDecimals);
                IDexRouter(_dexRouter).swapExactTokensForTokensSupportingFeeOnTransferTokens(
                    _tokenAmount,
                    _tokenAmount.sub(_maxSlippage), // Maximum slippage, 
                    _path,
                    feeWallet,
                    block.timestamp
                );
            }
            /**
             * @notice Private function callable when swap tokens directly from contract.
             * @notice swap with pair of native token
             * @param _tokenAmount      Amount of tokens
             * @param _dexRouter        Dex router address
             * @param _path             Array of token addresses; [from, to]
             */
            function swapTokensForEth(uint256 _tokenAmount, address _dexRouter, address[] memory _path) private  {
                uint256 _maxSlippage =  _tokenAmount.mul(getSellFee()).div(_feePercentDecimals);
                IDexRouter(_dexRouter).swapExactTokensForETHSupportingFeeOnTransferTokens(
                    _tokenAmount,
                    _tokenAmount.sub(_maxSlippage), // Maximum slippage, 
                    _path,
                    feeWallet,
                    block.timestamp
                );
            }
            /**
             * @notice Private function callable when transfer tokens and transfer is sell.
             * @return fees
             */
            function getSellFee() private view returns (uint256) {
                uint256 diff = (_initSellFee - sellFee);
                uint256 fees = diff.div(_tradeFeeReduceDays > 0 ? _tradeFeeReduceDays : 1).mul(_getDays(block.timestamp, _tradeFeeReducedUntil)).add(sellFee);
                return fees;
            }
            
            function getContractAddresses() external view returns (address, address) {
                return (address(presale_ico), address(presale_dynamic_to_stake));
            }
            function updateContractAddresses(address _ico, address _dynamictostake) external onlyOwner {
                presale_ico = PRESALE_ICO(_ico);
                presale_dynamic_to_stake = PRESALE_DYNAMIC_TO_STAKE(_dynamictostake);
            }
            function updateFeeWallet(address _newFeeWallet) external onlyOwner {
                feeWallet = payable(_newFeeWallet);
            }
        }
        // SPDX-License-Identifier: MIT
        pragma solidity ^0.8.4;
        import "@openzeppelin/contracts/utils/structs/EnumerableSet.sol";
        import "@openzeppelin/contracts-upgradeable/access/OwnableUpgradeable.sol";
        contract AllowableAddress is OwnableUpgradeable {
            using EnumerableSet for EnumerableSet.AddressSet;
            // @dev the allowed minter addresses
            EnumerableSet.AddressSet internal s_minters;
            // @dev the allowed burner addresses
            EnumerableSet.AddressSet internal s_burners;
            /******* Variables *******/
            // blacklist address => bool
            mapping(address => bool) internal blacklistAddress;
            /******* Modifiers *******/
            modifier blacklistCheck(address _from, address _to) {
                require(!blacklistAddress[_from] && !blacklistAddress[_to], "Address is blacklisted");
                _;
            }
            modifier onlyMinter() {
                if (!isMinter(msg.sender)) revert SenderNotMinter(msg.sender);
                _;
            }
            modifier onlyBurner() {
                if (!isBurner(msg.sender)) revert SenderNotBurner(msg.sender);
                _;
            }
            /******* Events *******/
            error SenderNotMinter(address sender);
            error SenderNotBurner(address sender);
            error MaxSupplyExceeded(uint256 supplyAfterMint);
            event MintAccessGranted(address indexed minter);
            event BurnAccessGranted(address indexed burner);
            event MintAccessRevoked(address indexed minter);
            event BurnAccessRevoked(address indexed burner);
            event BlacklistUpdated(bool status);
            /******* Constructor *******/
            function __AllowableAddress_init(address _ownerAddress) internal {
                __Ownable_init(_ownerAddress);
            }
            /*
             * @notice Manage blacklisting to restrict addresses
             * @param addresses     Array of addresses
             * @param status        If blacklist passing true, false otherwise.
             */
            function manageBlacklist(address[] calldata addresses, bool status)
                external
                virtual
                onlyOwner
            {
                for (uint256 i; i < addresses.length; ++i) {
                    blacklistAddress[addresses[i]] = status;
                }
                emit BlacklistUpdated(status);
            }
            /*
             * @notice grants both mint and burn roles to `burnAndMinter`.
             * @notice Accessable to only owner.
             * @param burnAndMinter      Any address for mint and burn access
             */
            function grantMintAndBurnRoles(address burnAndMinter) external onlyOwner {
                grantMintRole(burnAndMinter);
                grantBurnRole(burnAndMinter);
            }
            /*
             * @notice Grants mint role to the given address.
             * @notice Accessable to only owner.
             * @param minter    Any address for mint access
             */
            function grantMintRole(address minter) public onlyOwner {
                if (s_minters.add(minter)) {
                    emit MintAccessGranted(minter);
                }
            }
            /*
             * @notice Grants burn role to the given address.
             * @notice Accessable to only owner.
             * @param burner    Any address for burn access
             */
            function grantBurnRole(address burner) public onlyOwner {
                if (s_burners.add(burner)) {
                    emit BurnAccessGranted(burner);
                }
            }
            /*
             * @notice Revokes mint role for the given address.
             * @notice Accessable to only owner.
             * @param minter    Any address for remove mint access
             */
            function revokeMintRole(address minter) public onlyOwner {
                if (s_minters.remove(minter)) {
                    emit MintAccessRevoked(minter);
                }
            }
            /*
             * @notice Revokes burn role from the given address.
             * @notice Accessable to only owner
             * @param burner    Any address for remove burn access
             */
            function revokeBurnRole(address burner) public onlyOwner {
                if (s_burners.remove(burner)) {
                    emit BurnAccessRevoked(burner);
                }
            }
            /*
             * @param _address      Any address
             * @return {the address is blacklisted or not}.
             */
            function isBlacklisted(address _address) public view returns (bool) {
                return blacklistAddress[_address];
            }
            /*
             * @return {all permissioned minters}
             */
            function getMinters() public view returns (address[] memory) {
                return s_minters.values();
            }
            /*
             * @return {all permissioned burners}
             */
            function getBurners() public view returns (address[] memory) {
                return s_burners.values();
            }
            /*
             * @notice Checks whether a given address is a minter for this token.
             * @return {true if the address is allowed to mint}.
             */
            function isMinter(address minter) public view returns (bool) {
                return s_minters.contains(minter);
            }
            /*
             * @notice Checks whether a given address is a burner for this token.
             * @return {true if the address is allowed to burn}.
             */
            function isBurner(address burner) public view returns (bool) {
                return s_burners.contains(burner);
            }
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts (last updated v4.9.0) (utils/math/SafeMath.sol)
        pragma solidity ^0.8.0;
        // CAUTION
        // This version of SafeMath should only be used with Solidity 0.8 or later,
        // because it relies on the compiler's built in overflow checks.
        /**
         * @dev Wrappers over Solidity's arithmetic operations.
         *
         * NOTE: `SafeMath` is generally not needed starting with Solidity 0.8, since the compiler
         * now has built in overflow checking.
         */
        library SafeMath {
            /**
             * @dev Returns the addition of two unsigned integers, with an overflow flag.
             *
             * _Available since v3.4._
             */
            function tryAdd(uint256 a, uint256 b) internal pure returns (bool, uint256) {
                unchecked {
                    uint256 c = a + b;
                    if (c < a) return (false, 0);
                    return (true, c);
                }
            }
            /**
             * @dev Returns the subtraction of two unsigned integers, with an overflow flag.
             *
             * _Available since v3.4._
             */
            function trySub(uint256 a, uint256 b) internal pure returns (bool, uint256) {
                unchecked {
                    if (b > a) return (false, 0);
                    return (true, a - b);
                }
            }
            /**
             * @dev Returns the multiplication of two unsigned integers, with an overflow flag.
             *
             * _Available since v3.4._
             */
            function tryMul(uint256 a, uint256 b) internal pure returns (bool, uint256) {
                unchecked {
                    // Gas optimization: this is cheaper than requiring 'a' not being zero, but the
                    // benefit is lost if 'b' is also tested.
                    // See: https://github.com/OpenZeppelin/openzeppelin-contracts/pull/522
                    if (a == 0) return (true, 0);
                    uint256 c = a * b;
                    if (c / a != b) return (false, 0);
                    return (true, c);
                }
            }
            /**
             * @dev Returns the division of two unsigned integers, with a division by zero flag.
             *
             * _Available since v3.4._
             */
            function tryDiv(uint256 a, uint256 b) internal pure returns (bool, uint256) {
                unchecked {
                    if (b == 0) return (false, 0);
                    return (true, a / b);
                }
            }
            /**
             * @dev Returns the remainder of dividing two unsigned integers, with a division by zero flag.
             *
             * _Available since v3.4._
             */
            function tryMod(uint256 a, uint256 b) internal pure returns (bool, uint256) {
                unchecked {
                    if (b == 0) return (false, 0);
                    return (true, a % b);
                }
            }
            /**
             * @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) {
                return a + b;
            }
            /**
             * @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 a - b;
            }
            /**
             * @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) {
                return a * b;
            }
            /**
             * @dev Returns the integer division of two unsigned integers, reverting on
             * division by zero. The result is rounded towards zero.
             *
             * Counterpart to Solidity's `/` operator.
             *
             * Requirements:
             *
             * - The divisor cannot be zero.
             */
            function div(uint256 a, uint256 b) internal pure returns (uint256) {
                return a / b;
            }
            /**
             * @dev Returns the remainder of dividing two unsigned integers. (unsigned integer modulo),
             * reverting 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 a % b;
            }
            /**
             * @dev Returns the subtraction of two unsigned integers, reverting with custom message on
             * overflow (when the result is negative).
             *
             * CAUTION: This function is deprecated because it requires allocating memory for the error
             * message unnecessarily. For custom revert reasons use {trySub}.
             *
             * Counterpart to Solidity's `-` operator.
             *
             * Requirements:
             *
             * - Subtraction cannot overflow.
             */
            function sub(uint256 a, uint256 b, string memory errorMessage) internal pure returns (uint256) {
                unchecked {
                    require(b <= a, errorMessage);
                    return a - b;
                }
            }
            /**
             * @dev Returns the integer division of two unsigned integers, reverting 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) {
                unchecked {
                    require(b > 0, errorMessage);
                    return a / b;
                }
            }
            /**
             * @dev Returns the remainder of dividing two unsigned integers. (unsigned integer modulo),
             * reverting with custom message when dividing by zero.
             *
             * CAUTION: This function is deprecated because it requires allocating memory for the error
             * message unnecessarily. For custom revert reasons use {tryMod}.
             *
             * 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) {
                unchecked {
                    require(b > 0, errorMessage);
                    return a % b;
                }
            }
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts (last updated v5.0.0) (token/ERC20/extensions/ERC20Permit.sol)
        pragma solidity ^0.8.20;
        import {IERC20Permit} from "@openzeppelin/contracts/token/ERC20/extensions/IERC20Permit.sol";
        import {ERC20Upgradeable} from "../ERC20Upgradeable.sol";
        import {ECDSA} from "@openzeppelin/contracts/utils/cryptography/ECDSA.sol";
        import {EIP712Upgradeable} from "../../../utils/cryptography/EIP712Upgradeable.sol";
        import {NoncesUpgradeable} from "../../../utils/NoncesUpgradeable.sol";
        import {Initializable} from "../../../proxy/utils/Initializable.sol";
        /**
         * @dev Implementation of the ERC20 Permit extension allowing approvals to be made via signatures, as defined in
         * https://eips.ethereum.org/EIPS/eip-2612[EIP-2612].
         *
         * Adds the {permit} method, which can be used to change an account's ERC20 allowance (see {IERC20-allowance}) by
         * presenting a message signed by the account. By not relying on `{IERC20-approve}`, the token holder account doesn't
         * need to send a transaction, and thus is not required to hold Ether at all.
         */
        abstract contract ERC20PermitUpgradeable is Initializable, ERC20Upgradeable, IERC20Permit, EIP712Upgradeable, NoncesUpgradeable {
            bytes32 private constant PERMIT_TYPEHASH =
                keccak256("Permit(address owner,address spender,uint256 value,uint256 nonce,uint256 deadline)");
            /**
             * @dev Permit deadline has expired.
             */
            error ERC2612ExpiredSignature(uint256 deadline);
            /**
             * @dev Mismatched signature.
             */
            error ERC2612InvalidSigner(address signer, address owner);
            /**
             * @dev Initializes the {EIP712} domain separator using the `name` parameter, and setting `version` to `"1"`.
             *
             * It's a good idea to use the same `name` that is defined as the ERC20 token name.
             */
            function __ERC20Permit_init(string memory name) internal onlyInitializing {
                __EIP712_init_unchained(name, "1");
            }
            function __ERC20Permit_init_unchained(string memory) internal onlyInitializing {}
            /**
             * @inheritdoc IERC20Permit
             */
            function permit(
                address owner,
                address spender,
                uint256 value,
                uint256 deadline,
                uint8 v,
                bytes32 r,
                bytes32 s
            ) public virtual {
                if (block.timestamp > deadline) {
                    revert ERC2612ExpiredSignature(deadline);
                }
                bytes32 structHash = keccak256(abi.encode(PERMIT_TYPEHASH, owner, spender, value, _useNonce(owner), deadline));
                bytes32 hash = _hashTypedDataV4(structHash);
                address signer = ECDSA.recover(hash, v, r, s);
                if (signer != owner) {
                    revert ERC2612InvalidSigner(signer, owner);
                }
                _approve(owner, spender, value);
            }
            /**
             * @inheritdoc IERC20Permit
             */
            function nonces(address owner) public view virtual override(IERC20Permit, NoncesUpgradeable) returns (uint256) {
                return super.nonces(owner);
            }
            /**
             * @inheritdoc IERC20Permit
             */
            // solhint-disable-next-line func-name-mixedcase
            function DOMAIN_SEPARATOR() external view virtual returns (bytes32) {
                return _domainSeparatorV4();
            }
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts (last updated v5.0.0) (token/ERC20/extensions/ERC20Pausable.sol)
        pragma solidity ^0.8.20;
        import {ERC20Upgradeable} from "../ERC20Upgradeable.sol";
        import {PausableUpgradeable} from "../../../utils/PausableUpgradeable.sol";
        import {Initializable} from "../../../proxy/utils/Initializable.sol";
        /**
         * @dev ERC20 token with pausable token transfers, minting and burning.
         *
         * Useful for scenarios such as preventing trades until the end of an evaluation
         * period, or having an emergency switch for freezing all token transfers in the
         * event of a large bug.
         *
         * IMPORTANT: This contract does not include public pause and unpause functions. In
         * addition to inheriting this contract, you must define both functions, invoking the
         * {Pausable-_pause} and {Pausable-_unpause} internal functions, with appropriate
         * access control, e.g. using {AccessControl} or {Ownable}. Not doing so will
         * make the contract pause mechanism of the contract unreachable, and thus unusable.
         */
        abstract contract ERC20PausableUpgradeable is Initializable, ERC20Upgradeable, PausableUpgradeable {
            function __ERC20Pausable_init() internal onlyInitializing {
                __Pausable_init_unchained();
            }
            function __ERC20Pausable_init_unchained() internal onlyInitializing {
            }
            /**
             * @dev See {ERC20-_update}.
             *
             * Requirements:
             *
             * - the contract must not be paused.
             */
            function _update(address from, address to, uint256 value) internal virtual override whenNotPaused {
                super._update(from, to, value);
            }
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts (last updated v5.0.0) (token/ERC20/extensions/ERC20Burnable.sol)
        pragma solidity ^0.8.20;
        import {ERC20Upgradeable} from "../ERC20Upgradeable.sol";
        import {ContextUpgradeable} from "../../../utils/ContextUpgradeable.sol";
        import {Initializable} from "../../../proxy/utils/Initializable.sol";
        /**
         * @dev Extension of {ERC20} that allows token holders to destroy both their own
         * tokens and those that they have an allowance for, in a way that can be
         * recognized off-chain (via event analysis).
         */
        abstract contract ERC20BurnableUpgradeable is Initializable, ContextUpgradeable, ERC20Upgradeable {
            function __ERC20Burnable_init() internal onlyInitializing {
            }
            function __ERC20Burnable_init_unchained() internal onlyInitializing {
            }
            /**
             * @dev Destroys a `value` amount of tokens from the caller.
             *
             * See {ERC20-_burn}.
             */
            function burn(uint256 value) public virtual {
                _burn(_msgSender(), value);
            }
            /**
             * @dev Destroys a `value` amount of tokens from `account`, deducting from
             * the caller's allowance.
             *
             * See {ERC20-_burn} and {ERC20-allowance}.
             *
             * Requirements:
             *
             * - the caller must have allowance for ``accounts``'s tokens of at least
             * `value`.
             */
            function burnFrom(address account, uint256 value) public virtual {
                _spendAllowance(account, _msgSender(), value);
                _burn(account, value);
            }
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts (last updated v5.0.0) (token/ERC20/ERC20.sol)
        pragma solidity ^0.8.20;
        import {IERC20} from "@openzeppelin/contracts/token/ERC20/IERC20.sol";
        import {IERC20Metadata} from "@openzeppelin/contracts/token/ERC20/extensions/IERC20Metadata.sol";
        import {ContextUpgradeable} from "../../utils/ContextUpgradeable.sol";
        import {IERC20Errors} from "@openzeppelin/contracts/interfaces/draft-IERC6093.sol";
        import {Initializable} from "../../proxy/utils/Initializable.sol";
        /**
         * @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}.
         *
         * 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].
         *
         * The default value of {decimals} is 18. To change this, you should override
         * this function so it returns a different value.
         *
         * 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.
         */
        abstract contract ERC20Upgradeable is Initializable, ContextUpgradeable, IERC20, IERC20Metadata, IERC20Errors {
            /// @custom:storage-location erc7201:openzeppelin.storage.ERC20
            struct ERC20Storage {
                mapping(address account => uint256) _balances;
                mapping(address account => mapping(address spender => uint256)) _allowances;
                uint256 _totalSupply;
                string _name;
                string _symbol;
            }
            // keccak256(abi.encode(uint256(keccak256("openzeppelin.storage.ERC20")) - 1)) & ~bytes32(uint256(0xff))
            bytes32 private constant ERC20StorageLocation = 0x52c63247e1f47db19d5ce0460030c497f067ca4cebf71ba98eeadabe20bace00;
            function _getERC20Storage() private pure returns (ERC20Storage storage $) {
                assembly {
                    $.slot := ERC20StorageLocation
                }
            }
            /**
             * @dev Sets the values for {name} and {symbol}.
             *
             * All two of these values are immutable: they can only be set once during
             * construction.
             */
            function __ERC20_init(string memory name_, string memory symbol_) internal onlyInitializing {
                __ERC20_init_unchained(name_, symbol_);
            }
            function __ERC20_init_unchained(string memory name_, string memory symbol_) internal onlyInitializing {
                ERC20Storage storage $ = _getERC20Storage();
                $._name = name_;
                $._symbol = symbol_;
            }
            /**
             * @dev Returns the name of the token.
             */
            function name() public view virtual returns (string memory) {
                ERC20Storage storage $ = _getERC20Storage();
                return $._name;
            }
            /**
             * @dev Returns the symbol of the token, usually a shorter version of the
             * name.
             */
            function symbol() public view virtual returns (string memory) {
                ERC20Storage storage $ = _getERC20Storage();
                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 default value returned by this function, unless
             * it's 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 returns (uint8) {
                return 18;
            }
            /**
             * @dev See {IERC20-totalSupply}.
             */
            function totalSupply() public view virtual returns (uint256) {
                ERC20Storage storage $ = _getERC20Storage();
                return $._totalSupply;
            }
            /**
             * @dev See {IERC20-balanceOf}.
             */
            function balanceOf(address account) public view virtual returns (uint256) {
                ERC20Storage storage $ = _getERC20Storage();
                return $._balances[account];
            }
            /**
             * @dev See {IERC20-transfer}.
             *
             * Requirements:
             *
             * - `to` cannot be the zero address.
             * - the caller must have a balance of at least `value`.
             */
            function transfer(address to, uint256 value) public virtual returns (bool) {
                address owner = _msgSender();
                _transfer(owner, to, value);
                return true;
            }
            /**
             * @dev See {IERC20-allowance}.
             */
            function allowance(address owner, address spender) public view virtual returns (uint256) {
                ERC20Storage storage $ = _getERC20Storage();
                return $._allowances[owner][spender];
            }
            /**
             * @dev See {IERC20-approve}.
             *
             * NOTE: If `value` 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 value) public virtual returns (bool) {
                address owner = _msgSender();
                _approve(owner, spender, value);
                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 `value`.
             * - the caller must have allowance for ``from``'s tokens of at least
             * `value`.
             */
            function transferFrom(address from, address to, uint256 value) public virtual returns (bool) {
                address spender = _msgSender();
                _spendAllowance(from, spender, value);
                _transfer(from, to, value);
                return true;
            }
            /**
             * @dev Moves a `value` 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.
             *
             * NOTE: This function is not virtual, {_update} should be overridden instead.
             */
            function _transfer(address from, address to, uint256 value) internal {
                if (from == address(0)) {
                    revert ERC20InvalidSender(address(0));
                }
                if (to == address(0)) {
                    revert ERC20InvalidReceiver(address(0));
                }
                _update(from, to, value);
            }
            /**
             * @dev Transfers a `value` amount of tokens from `from` to `to`, or alternatively mints (or burns) if `from`
             * (or `to`) is the zero address. All customizations to transfers, mints, and burns should be done by overriding
             * this function.
             *
             * Emits a {Transfer} event.
             */
            function _update(address from, address to, uint256 value) internal virtual {
                ERC20Storage storage $ = _getERC20Storage();
                if (from == address(0)) {
                    // Overflow check required: The rest of the code assumes that totalSupply never overflows
                    $._totalSupply += value;
                } else {
                    uint256 fromBalance = $._balances[from];
                    if (fromBalance < value) {
                        revert ERC20InsufficientBalance(from, fromBalance, value);
                    }
                    unchecked {
                        // Overflow not possible: value <= fromBalance <= totalSupply.
                        $._balances[from] = fromBalance - value;
                    }
                }
                if (to == address(0)) {
                    unchecked {
                        // Overflow not possible: value <= totalSupply or value <= fromBalance <= totalSupply.
                        $._totalSupply -= value;
                    }
                } else {
                    unchecked {
                        // Overflow not possible: balance + value is at most totalSupply, which we know fits into a uint256.
                        $._balances[to] += value;
                    }
                }
                emit Transfer(from, to, value);
            }
            /**
             * @dev Creates a `value` amount of tokens and assigns them to `account`, by transferring it from address(0).
             * Relies on the `_update` mechanism
             *
             * Emits a {Transfer} event with `from` set to the zero address.
             *
             * NOTE: This function is not virtual, {_update} should be overridden instead.
             */
            function _mint(address account, uint256 value) internal {
                if (account == address(0)) {
                    revert ERC20InvalidReceiver(address(0));
                }
                _update(address(0), account, value);
            }
            /**
             * @dev Destroys a `value` amount of tokens from `account`, lowering the total supply.
             * Relies on the `_update` mechanism.
             *
             * Emits a {Transfer} event with `to` set to the zero address.
             *
             * NOTE: This function is not virtual, {_update} should be overridden instead
             */
            function _burn(address account, uint256 value) internal {
                if (account == address(0)) {
                    revert ERC20InvalidSender(address(0));
                }
                _update(account, address(0), value);
            }
            /**
             * @dev Sets `value` 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.
             *
             * Overrides to this logic should be done to the variant with an additional `bool emitEvent` argument.
             */
            function _approve(address owner, address spender, uint256 value) internal {
                _approve(owner, spender, value, true);
            }
            /**
             * @dev Variant of {_approve} with an optional flag to enable or disable the {Approval} event.
             *
             * By default (when calling {_approve}) the flag is set to true. On the other hand, approval changes made by
             * `_spendAllowance` during the `transferFrom` operation set the flag to false. This saves gas by not emitting any
             * `Approval` event during `transferFrom` operations.
             *
             * Anyone who wishes to continue emitting `Approval` events on the`transferFrom` operation can force the flag to
             * true using the following override:
             * ```
             * function _approve(address owner, address spender, uint256 value, bool) internal virtual override {
             *     super._approve(owner, spender, value, true);
             * }
             * ```
             *
             * Requirements are the same as {_approve}.
             */
            function _approve(address owner, address spender, uint256 value, bool emitEvent) internal virtual {
                ERC20Storage storage $ = _getERC20Storage();
                if (owner == address(0)) {
                    revert ERC20InvalidApprover(address(0));
                }
                if (spender == address(0)) {
                    revert ERC20InvalidSpender(address(0));
                }
                $._allowances[owner][spender] = value;
                if (emitEvent) {
                    emit Approval(owner, spender, value);
                }
            }
            /**
             * @dev Updates `owner` s allowance for `spender` based on spent `value`.
             *
             * Does not update the allowance value in case of infinite allowance.
             * Revert if not enough allowance is available.
             *
             * Does not emit an {Approval} event.
             */
            function _spendAllowance(address owner, address spender, uint256 value) internal virtual {
                uint256 currentAllowance = allowance(owner, spender);
                if (currentAllowance != type(uint256).max) {
                    if (currentAllowance < value) {
                        revert ERC20InsufficientAllowance(spender, currentAllowance, value);
                    }
                    unchecked {
                        _approve(owner, spender, currentAllowance - value, false);
                    }
                }
            }
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts (last updated v5.0.0) (token/ERC20/IERC20.sol)
        pragma solidity ^0.8.20;
        /**
         * @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 value of tokens in existence.
             */
            function totalSupply() external view returns (uint256);
            /**
             * @dev Returns the value of tokens owned by `account`.
             */
            function balanceOf(address account) external view returns (uint256);
            /**
             * @dev Moves a `value` amount of 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 value) 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 a `value` amount of tokens 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 value) external returns (bool);
            /**
             * @dev Moves a `value` amount of tokens from `from` to `to` using the
             * allowance mechanism. `value` 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 value) external returns (bool);
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts (last updated v5.0.0) (access/Ownable.sol)
        pragma solidity ^0.8.20;
        import {ContextUpgradeable} from "../utils/ContextUpgradeable.sol";
        import {Initializable} from "../proxy/utils/Initializable.sol";
        /**
         * @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.
         *
         * The initial owner is set to the address provided by the deployer. 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 OwnableUpgradeable is Initializable, ContextUpgradeable {
            /// @custom:storage-location erc7201:openzeppelin.storage.Ownable
            struct OwnableStorage {
                address _owner;
            }
            // keccak256(abi.encode(uint256(keccak256("openzeppelin.storage.Ownable")) - 1)) & ~bytes32(uint256(0xff))
            bytes32 private constant OwnableStorageLocation = 0x9016d09d72d40fdae2fd8ceac6b6234c7706214fd39c1cd1e609a0528c199300;
            function _getOwnableStorage() private pure returns (OwnableStorage storage $) {
                assembly {
                    $.slot := OwnableStorageLocation
                }
            }
            /**
             * @dev The caller account is not authorized to perform an operation.
             */
            error OwnableUnauthorizedAccount(address account);
            /**
             * @dev The owner is not a valid owner account. (eg. `address(0)`)
             */
            error OwnableInvalidOwner(address owner);
            event OwnershipTransferred(address indexed previousOwner, address indexed newOwner);
            /**
             * @dev Initializes the contract setting the address provided by the deployer as the initial owner.
             */
            function __Ownable_init(address initialOwner) internal onlyInitializing {
                __Ownable_init_unchained(initialOwner);
            }
            function __Ownable_init_unchained(address initialOwner) internal onlyInitializing {
                if (initialOwner == address(0)) {
                    revert OwnableInvalidOwner(address(0));
                }
                _transferOwnership(initialOwner);
            }
            /**
             * @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) {
                OwnableStorage storage $ = _getOwnableStorage();
                return $._owner;
            }
            /**
             * @dev Throws if the sender is not the owner.
             */
            function _checkOwner() internal view virtual {
                if (owner() != _msgSender()) {
                    revert OwnableUnauthorizedAccount(_msgSender());
                }
            }
            /**
             * @dev Leaves the contract without owner. It will not be possible to call
             * `onlyOwner` functions. Can only be called by the current owner.
             *
             * NOTE: Renouncing ownership will leave the contract without an owner,
             * thereby disabling 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 {
                if (newOwner == address(0)) {
                    revert OwnableInvalidOwner(address(0));
                }
                _transferOwnership(newOwner);
            }
            /**
             * @dev Transfers ownership of the contract to a new account (`newOwner`).
             * Internal function without access restriction.
             */
            function _transferOwnership(address newOwner) internal virtual {
                OwnableStorage storage $ = _getOwnableStorage();
                address oldOwner = $._owner;
                $._owner = newOwner;
                emit OwnershipTransferred(oldOwner, newOwner);
            }
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts (last updated v5.0.0) (utils/structs/EnumerableSet.sol)
        // This file was procedurally generated from scripts/generate/templates/EnumerableSet.js.
        pragma solidity ^0.8.20;
        /**
         * @dev Library for managing
         * https://en.wikipedia.org/wiki/Set_(abstract_data_type)[sets] of primitive
         * types.
         *
         * Sets have the following properties:
         *
         * - Elements are added, removed, and checked for existence in constant time
         * (O(1)).
         * - Elements are enumerated in O(n). No guarantees are made on the ordering.
         *
         * ```solidity
         * contract Example {
         *     // Add the library methods
         *     using EnumerableSet for EnumerableSet.AddressSet;
         *
         *     // Declare a set state variable
         *     EnumerableSet.AddressSet private mySet;
         * }
         * ```
         *
         * As of v3.3.0, sets of type `bytes32` (`Bytes32Set`), `address` (`AddressSet`)
         * and `uint256` (`UintSet`) are supported.
         *
         * [WARNING]
         * ====
         * Trying to delete such a structure from storage will likely result in data corruption, rendering the structure
         * unusable.
         * See https://github.com/ethereum/solidity/pull/11843[ethereum/solidity#11843] for more info.
         *
         * In order to clean an EnumerableSet, you can either remove all elements one by one or create a fresh instance using an
         * array of EnumerableSet.
         * ====
         */
        library EnumerableSet {
            // To implement this library for multiple types with as little code
            // repetition as possible, we write it in terms of a generic Set type with
            // bytes32 values.
            // The Set implementation uses private functions, and user-facing
            // implementations (such as AddressSet) are just wrappers around the
            // underlying Set.
            // This means that we can only create new EnumerableSets for types that fit
            // in bytes32.
            struct Set {
                // Storage of set values
                bytes32[] _values;
                // Position is the index of the value in the `values` array plus 1.
                // Position 0 is used to mean a value is not in the set.
                mapping(bytes32 value => uint256) _positions;
            }
            /**
             * @dev Add a value to a set. O(1).
             *
             * Returns true if the value was added to the set, that is if it was not
             * already present.
             */
            function _add(Set storage set, bytes32 value) private returns (bool) {
                if (!_contains(set, value)) {
                    set._values.push(value);
                    // The value is stored at length-1, but we add 1 to all indexes
                    // and use 0 as a sentinel value
                    set._positions[value] = set._values.length;
                    return true;
                } else {
                    return false;
                }
            }
            /**
             * @dev Removes a value from a set. O(1).
             *
             * Returns true if the value was removed from the set, that is if it was
             * present.
             */
            function _remove(Set storage set, bytes32 value) private returns (bool) {
                // We cache the value's position to prevent multiple reads from the same storage slot
                uint256 position = set._positions[value];
                if (position != 0) {
                    // Equivalent to contains(set, value)
                    // To delete an element from the _values array in O(1), we swap the element to delete with the last one in
                    // the array, and then remove the last element (sometimes called as 'swap and pop').
                    // This modifies the order of the array, as noted in {at}.
                    uint256 valueIndex = position - 1;
                    uint256 lastIndex = set._values.length - 1;
                    if (valueIndex != lastIndex) {
                        bytes32 lastValue = set._values[lastIndex];
                        // Move the lastValue to the index where the value to delete is
                        set._values[valueIndex] = lastValue;
                        // Update the tracked position of the lastValue (that was just moved)
                        set._positions[lastValue] = position;
                    }
                    // Delete the slot where the moved value was stored
                    set._values.pop();
                    // Delete the tracked position for the deleted slot
                    delete set._positions[value];
                    return true;
                } else {
                    return false;
                }
            }
            /**
             * @dev Returns true if the value is in the set. O(1).
             */
            function _contains(Set storage set, bytes32 value) private view returns (bool) {
                return set._positions[value] != 0;
            }
            /**
             * @dev Returns the number of values on the set. O(1).
             */
            function _length(Set storage set) private view returns (uint256) {
                return set._values.length;
            }
            /**
             * @dev Returns the value stored at position `index` in the set. O(1).
             *
             * Note that there are no guarantees on the ordering of values inside the
             * array, and it may change when more values are added or removed.
             *
             * Requirements:
             *
             * - `index` must be strictly less than {length}.
             */
            function _at(Set storage set, uint256 index) private view returns (bytes32) {
                return set._values[index];
            }
            /**
             * @dev Return the entire set in an array
             *
             * WARNING: This operation will copy the entire storage to memory, which can be quite expensive. This is designed
             * to mostly be used by view accessors that are queried without any gas fees. Developers should keep in mind that
             * this function has an unbounded cost, and using it as part of a state-changing function may render the function
             * uncallable if the set grows to a point where copying to memory consumes too much gas to fit in a block.
             */
            function _values(Set storage set) private view returns (bytes32[] memory) {
                return set._values;
            }
            // Bytes32Set
            struct Bytes32Set {
                Set _inner;
            }
            /**
             * @dev Add a value to a set. O(1).
             *
             * Returns true if the value was added to the set, that is if it was not
             * already present.
             */
            function add(Bytes32Set storage set, bytes32 value) internal returns (bool) {
                return _add(set._inner, value);
            }
            /**
             * @dev Removes a value from a set. O(1).
             *
             * Returns true if the value was removed from the set, that is if it was
             * present.
             */
            function remove(Bytes32Set storage set, bytes32 value) internal returns (bool) {
                return _remove(set._inner, value);
            }
            /**
             * @dev Returns true if the value is in the set. O(1).
             */
            function contains(Bytes32Set storage set, bytes32 value) internal view returns (bool) {
                return _contains(set._inner, value);
            }
            /**
             * @dev Returns the number of values in the set. O(1).
             */
            function length(Bytes32Set storage set) internal view returns (uint256) {
                return _length(set._inner);
            }
            /**
             * @dev Returns the value stored at position `index` in the set. O(1).
             *
             * Note that there are no guarantees on the ordering of values inside the
             * array, and it may change when more values are added or removed.
             *
             * Requirements:
             *
             * - `index` must be strictly less than {length}.
             */
            function at(Bytes32Set storage set, uint256 index) internal view returns (bytes32) {
                return _at(set._inner, index);
            }
            /**
             * @dev Return the entire set in an array
             *
             * WARNING: This operation will copy the entire storage to memory, which can be quite expensive. This is designed
             * to mostly be used by view accessors that are queried without any gas fees. Developers should keep in mind that
             * this function has an unbounded cost, and using it as part of a state-changing function may render the function
             * uncallable if the set grows to a point where copying to memory consumes too much gas to fit in a block.
             */
            function values(Bytes32Set storage set) internal view returns (bytes32[] memory) {
                bytes32[] memory store = _values(set._inner);
                bytes32[] memory result;
                /// @solidity memory-safe-assembly
                assembly {
                    result := store
                }
                return result;
            }
            // AddressSet
            struct AddressSet {
                Set _inner;
            }
            /**
             * @dev Add a value to a set. O(1).
             *
             * Returns true if the value was added to the set, that is if it was not
             * already present.
             */
            function add(AddressSet storage set, address value) internal returns (bool) {
                return _add(set._inner, bytes32(uint256(uint160(value))));
            }
            /**
             * @dev Removes a value from a set. O(1).
             *
             * Returns true if the value was removed from the set, that is if it was
             * present.
             */
            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);
            }
            /**
             * @dev Returns the value stored at position `index` in the set. O(1).
             *
             * Note that there are no guarantees on the ordering of values inside the
             * array, and it may change when more values are added or removed.
             *
             * Requirements:
             *
             * - `index` must be strictly less than {length}.
             */
            function at(AddressSet storage set, uint256 index) internal view returns (address) {
                return address(uint160(uint256(_at(set._inner, index))));
            }
            /**
             * @dev Return the entire set in an array
             *
             * WARNING: This operation will copy the entire storage to memory, which can be quite expensive. This is designed
             * to mostly be used by view accessors that are queried without any gas fees. Developers should keep in mind that
             * this function has an unbounded cost, and using it as part of a state-changing function may render the function
             * uncallable if the set grows to a point where copying to memory consumes too much gas to fit in a block.
             */
            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;
            }
            // UintSet
            struct UintSet {
                Set _inner;
            }
            /**
             * @dev Add a value to a set. O(1).
             *
             * Returns true if the value was added to the set, that is if it was not
             * already present.
             */
            function add(UintSet storage set, uint256 value) internal returns (bool) {
                return _add(set._inner, bytes32(value));
            }
            /**
             * @dev Removes a value from a set. O(1).
             *
             * Returns true if the value was removed from the set, that is if it was
             * present.
             */
            function remove(UintSet storage set, uint256 value) internal returns (bool) {
                return _remove(set._inner, bytes32(value));
            }
            /**
             * @dev Returns true if the value is in the set. O(1).
             */
            function contains(UintSet storage set, uint256 value) internal view returns (bool) {
                return _contains(set._inner, bytes32(value));
            }
            /**
             * @dev Returns the number of values in the set. O(1).
             */
            function length(UintSet storage set) internal view returns (uint256) {
                return _length(set._inner);
            }
            /**
             * @dev Returns the value stored at position `index` in the set. O(1).
             *
             * Note that there are no guarantees on the ordering of values inside the
             * array, and it may change when more values are added or removed.
             *
             * Requirements:
             *
             * - `index` must be strictly less than {length}.
             */
            function at(UintSet storage set, uint256 index) internal view returns (uint256) {
                return uint256(_at(set._inner, index));
            }
            /**
             * @dev Return the entire set in an array
             *
             * WARNING: This operation will copy the entire storage to memory, which can be quite expensive. This is designed
             * to mostly be used by view accessors that are queried without any gas fees. Developers should keep in mind that
             * this function has an unbounded cost, and using it as part of a state-changing function may render the function
             * uncallable if the set grows to a point where copying to memory consumes too much gas to fit in a block.
             */
            function values(UintSet storage set) internal view returns (uint256[] memory) {
                bytes32[] memory store = _values(set._inner);
                uint256[] memory result;
                /// @solidity memory-safe-assembly
                assembly {
                    result := store
                }
                return result;
            }
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts (last updated v5.0.0) (proxy/utils/Initializable.sol)
        pragma solidity ^0.8.20;
        /**
         * @dev This is a base contract to aid in writing upgradeable contracts, or any kind of contract that will be deployed
         * behind a proxy. Since proxied contracts do not make use of a constructor, it's common to move constructor logic to an
         * external initializer function, usually called `initialize`. It then becomes necessary to protect this initializer
         * function so it can only be called once. The {initializer} modifier provided by this contract will have this effect.
         *
         * The initialization functions use a version number. Once a version number is used, it is consumed and cannot be
         * reused. This mechanism prevents re-execution of each "step" but allows the creation of new initialization steps in
         * case an upgrade adds a module that needs to be initialized.
         *
         * For example:
         *
         * [.hljs-theme-light.nopadding]
         * ```solidity
         * contract MyToken is ERC20Upgradeable {
         *     function initialize() initializer public {
         *         __ERC20_init("MyToken", "MTK");
         *     }
         * }
         *
         * contract MyTokenV2 is MyToken, ERC20PermitUpgradeable {
         *     function initializeV2() reinitializer(2) public {
         *         __ERC20Permit_init("MyToken");
         *     }
         * }
         * ```
         *
         * TIP: To avoid leaving the proxy in an uninitialized state, the initializer function should be called as early as
         * possible by providing the encoded function call as the `_data` argument to {ERC1967Proxy-constructor}.
         *
         * CAUTION: When used with inheritance, manual care must be taken to not invoke a parent initializer twice, or to ensure
         * that all initializers are idempotent. This is not verified automatically as constructors are by Solidity.
         *
         * [CAUTION]
         * ====
         * Avoid leaving a contract uninitialized.
         *
         * An uninitialized contract can be taken over by an attacker. This applies to both a proxy and its implementation
         * contract, which may impact the proxy. To prevent the implementation contract from being used, you should invoke
         * the {_disableInitializers} function in the constructor to automatically lock it when it is deployed:
         *
         * [.hljs-theme-light.nopadding]
         * ```
         * /// @custom:oz-upgrades-unsafe-allow constructor
         * constructor() {
         *     _disableInitializers();
         * }
         * ```
         * ====
         */
        abstract contract Initializable {
            /**
             * @dev Storage of the initializable contract.
             *
             * It's implemented on a custom ERC-7201 namespace to reduce the risk of storage collisions
             * when using with upgradeable contracts.
             *
             * @custom:storage-location erc7201:openzeppelin.storage.Initializable
             */
            struct InitializableStorage {
                /**
                 * @dev Indicates that the contract has been initialized.
                 */
                uint64 _initialized;
                /**
                 * @dev Indicates that the contract is in the process of being initialized.
                 */
                bool _initializing;
            }
            // keccak256(abi.encode(uint256(keccak256("openzeppelin.storage.Initializable")) - 1)) & ~bytes32(uint256(0xff))
            bytes32 private constant INITIALIZABLE_STORAGE = 0xf0c57e16840df040f15088dc2f81fe391c3923bec73e23a9662efc9c229c6a00;
            /**
             * @dev The contract is already initialized.
             */
            error InvalidInitialization();
            /**
             * @dev The contract is not initializing.
             */
            error NotInitializing();
            /**
             * @dev Triggered when the contract has been initialized or reinitialized.
             */
            event Initialized(uint64 version);
            /**
             * @dev A modifier that defines a protected initializer function that can be invoked at most once. In its scope,
             * `onlyInitializing` functions can be used to initialize parent contracts.
             *
             * Similar to `reinitializer(1)`, except that in the context of a constructor an `initializer` may be invoked any
             * number of times. This behavior in the constructor can be useful during testing and is not expected to be used in
             * production.
             *
             * Emits an {Initialized} event.
             */
            modifier initializer() {
                // solhint-disable-next-line var-name-mixedcase
                InitializableStorage storage $ = _getInitializableStorage();
                // Cache values to avoid duplicated sloads
                bool isTopLevelCall = !$._initializing;
                uint64 initialized = $._initialized;
                // Allowed calls:
                // - initialSetup: the contract is not in the initializing state and no previous version was
                //                 initialized
                // - construction: the contract is initialized at version 1 (no reininitialization) and the
                //                 current contract is just being deployed
                bool initialSetup = initialized == 0 && isTopLevelCall;
                bool construction = initialized == 1 && address(this).code.length == 0;
                if (!initialSetup && !construction) {
                    revert InvalidInitialization();
                }
                $._initialized = 1;
                if (isTopLevelCall) {
                    $._initializing = true;
                }
                _;
                if (isTopLevelCall) {
                    $._initializing = false;
                    emit Initialized(1);
                }
            }
            /**
             * @dev A modifier that defines a protected reinitializer function that can be invoked at most once, and only if the
             * contract hasn't been initialized to a greater version before. In its scope, `onlyInitializing` functions can be
             * used to initialize parent contracts.
             *
             * A reinitializer may be used after the original initialization step. This is essential to configure modules that
             * are added through upgrades and that require initialization.
             *
             * When `version` is 1, this modifier is similar to `initializer`, except that functions marked with `reinitializer`
             * cannot be nested. If one is invoked in the context of another, execution will revert.
             *
             * Note that versions can jump in increments greater than 1; this implies that if multiple reinitializers coexist in
             * a contract, executing them in the right order is up to the developer or operator.
             *
             * WARNING: Setting the version to 2**64 - 1 will prevent any future reinitialization.
             *
             * Emits an {Initialized} event.
             */
            modifier reinitializer(uint64 version) {
                // solhint-disable-next-line var-name-mixedcase
                InitializableStorage storage $ = _getInitializableStorage();
                if ($._initializing || $._initialized >= version) {
                    revert InvalidInitialization();
                }
                $._initialized = version;
                $._initializing = true;
                _;
                $._initializing = false;
                emit Initialized(version);
            }
            /**
             * @dev Modifier to protect an initialization function so that it can only be invoked by functions with the
             * {initializer} and {reinitializer} modifiers, directly or indirectly.
             */
            modifier onlyInitializing() {
                _checkInitializing();
                _;
            }
            /**
             * @dev Reverts if the contract is not in an initializing state. See {onlyInitializing}.
             */
            function _checkInitializing() internal view virtual {
                if (!_isInitializing()) {
                    revert NotInitializing();
                }
            }
            /**
             * @dev Locks the contract, preventing any future reinitialization. This cannot be part of an initializer call.
             * Calling this in the constructor of a contract will prevent that contract from being initialized or reinitialized
             * to any version. It is recommended to use this to lock implementation contracts that are designed to be called
             * through proxies.
             *
             * Emits an {Initialized} event the first time it is successfully executed.
             */
            function _disableInitializers() internal virtual {
                // solhint-disable-next-line var-name-mixedcase
                InitializableStorage storage $ = _getInitializableStorage();
                if ($._initializing) {
                    revert InvalidInitialization();
                }
                if ($._initialized != type(uint64).max) {
                    $._initialized = type(uint64).max;
                    emit Initialized(type(uint64).max);
                }
            }
            /**
             * @dev Returns the highest version that has been initialized. See {reinitializer}.
             */
            function _getInitializedVersion() internal view returns (uint64) {
                return _getInitializableStorage()._initialized;
            }
            /**
             * @dev Returns `true` if the contract is currently initializing. See {onlyInitializing}.
             */
            function _isInitializing() internal view returns (bool) {
                return _getInitializableStorage()._initializing;
            }
            /**
             * @dev Returns a pointer to the storage namespace.
             */
            // solhint-disable-next-line var-name-mixedcase
            function _getInitializableStorage() private pure returns (InitializableStorage storage $) {
                assembly {
                    $.slot := INITIALIZABLE_STORAGE
                }
            }
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts (last updated v5.0.0) (utils/Nonces.sol)
        pragma solidity ^0.8.20;
        import {Initializable} from "../proxy/utils/Initializable.sol";
        /**
         * @dev Provides tracking nonces for addresses. Nonces will only increment.
         */
        abstract contract NoncesUpgradeable is Initializable {
            /**
             * @dev The nonce used for an `account` is not the expected current nonce.
             */
            error InvalidAccountNonce(address account, uint256 currentNonce);
            /// @custom:storage-location erc7201:openzeppelin.storage.Nonces
            struct NoncesStorage {
                mapping(address account => uint256) _nonces;
            }
            // keccak256(abi.encode(uint256(keccak256("openzeppelin.storage.Nonces")) - 1)) & ~bytes32(uint256(0xff))
            bytes32 private constant NoncesStorageLocation = 0x5ab42ced628888259c08ac98db1eb0cf702fc1501344311d8b100cd1bfe4bb00;
            function _getNoncesStorage() private pure returns (NoncesStorage storage $) {
                assembly {
                    $.slot := NoncesStorageLocation
                }
            }
            function __Nonces_init() internal onlyInitializing {
            }
            function __Nonces_init_unchained() internal onlyInitializing {
            }
            /**
             * @dev Returns the next unused nonce for an address.
             */
            function nonces(address owner) public view virtual returns (uint256) {
                NoncesStorage storage $ = _getNoncesStorage();
                return $._nonces[owner];
            }
            /**
             * @dev Consumes a nonce.
             *
             * Returns the current value and increments nonce.
             */
            function _useNonce(address owner) internal virtual returns (uint256) {
                NoncesStorage storage $ = _getNoncesStorage();
                // For each account, the nonce has an initial value of 0, can only be incremented by one, and cannot be
                // decremented or reset. This guarantees that the nonce never overflows.
                unchecked {
                    // It is important to do x++ and not ++x here.
                    return $._nonces[owner]++;
                }
            }
            /**
             * @dev Same as {_useNonce} but checking that `nonce` is the next valid for `owner`.
             */
            function _useCheckedNonce(address owner, uint256 nonce) internal virtual {
                uint256 current = _useNonce(owner);
                if (nonce != current) {
                    revert InvalidAccountNonce(owner, current);
                }
            }
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts (last updated v5.0.0) (utils/cryptography/EIP712.sol)
        pragma solidity ^0.8.20;
        import {MessageHashUtils} from "@openzeppelin/contracts/utils/cryptography/MessageHashUtils.sol";
        import {IERC5267} from "@openzeppelin/contracts/interfaces/IERC5267.sol";
        import {Initializable} from "../../proxy/utils/Initializable.sol";
        /**
         * @dev https://eips.ethereum.org/EIPS/eip-712[EIP 712] is a standard for hashing and signing of typed structured data.
         *
         * The encoding scheme specified in the EIP requires a domain separator and a hash of the typed structured data, whose
         * encoding is very generic and therefore its implementation in Solidity is not feasible, thus this contract
         * does not implement the encoding itself. Protocols need to implement the type-specific encoding they need in order to
         * produce the hash of their typed data using a combination of `abi.encode` and `keccak256`.
         *
         * This contract implements the EIP 712 domain separator ({_domainSeparatorV4}) that is used as part of the encoding
         * scheme, and the final step of the encoding to obtain the message digest that is then signed via ECDSA
         * ({_hashTypedDataV4}).
         *
         * The implementation of the domain separator was designed to be as efficient as possible while still properly updating
         * the chain id to protect against replay attacks on an eventual fork of the chain.
         *
         * NOTE: This contract implements the version of the encoding known as "v4", as implemented by the JSON RPC method
         * https://docs.metamask.io/guide/signing-data.html[`eth_signTypedDataV4` in MetaMask].
         *
         * NOTE: In the upgradeable version of this contract, the cached values will correspond to the address, and the domain
         * separator of the implementation contract. This will cause the {_domainSeparatorV4} function to always rebuild the
         * separator from the immutable values, which is cheaper than accessing a cached version in cold storage.
         */
        abstract contract EIP712Upgradeable is Initializable, IERC5267 {
            bytes32 private constant TYPE_HASH =
                keccak256("EIP712Domain(string name,string version,uint256 chainId,address verifyingContract)");
            /// @custom:storage-location erc7201:openzeppelin.storage.EIP712
            struct EIP712Storage {
                /// @custom:oz-renamed-from _HASHED_NAME
                bytes32 _hashedName;
                /// @custom:oz-renamed-from _HASHED_VERSION
                bytes32 _hashedVersion;
                string _name;
                string _version;
            }
            // keccak256(abi.encode(uint256(keccak256("openzeppelin.storage.EIP712")) - 1)) & ~bytes32(uint256(0xff))
            bytes32 private constant EIP712StorageLocation = 0xa16a46d94261c7517cc8ff89f61c0ce93598e3c849801011dee649a6a557d100;
            function _getEIP712Storage() private pure returns (EIP712Storage storage $) {
                assembly {
                    $.slot := EIP712StorageLocation
                }
            }
            /**
             * @dev Initializes the domain separator and parameter caches.
             *
             * The meaning of `name` and `version` is specified in
             * https://eips.ethereum.org/EIPS/eip-712#definition-of-domainseparator[EIP 712]:
             *
             * - `name`: the user readable name of the signing domain, i.e. the name of the DApp or the protocol.
             * - `version`: the current major version of the signing domain.
             *
             * NOTE: These parameters cannot be changed except through a xref:learn::upgrading-smart-contracts.adoc[smart
             * contract upgrade].
             */
            function __EIP712_init(string memory name, string memory version) internal onlyInitializing {
                __EIP712_init_unchained(name, version);
            }
            function __EIP712_init_unchained(string memory name, string memory version) internal onlyInitializing {
                EIP712Storage storage $ = _getEIP712Storage();
                $._name = name;
                $._version = version;
                // Reset prior values in storage if upgrading
                $._hashedName = 0;
                $._hashedVersion = 0;
            }
            /**
             * @dev Returns the domain separator for the current chain.
             */
            function _domainSeparatorV4() internal view returns (bytes32) {
                return _buildDomainSeparator();
            }
            function _buildDomainSeparator() private view returns (bytes32) {
                return keccak256(abi.encode(TYPE_HASH, _EIP712NameHash(), _EIP712VersionHash(), block.chainid, address(this)));
            }
            /**
             * @dev Given an already https://eips.ethereum.org/EIPS/eip-712#definition-of-hashstruct[hashed struct], this
             * function returns the hash of the fully encoded EIP712 message for this domain.
             *
             * This hash can be used together with {ECDSA-recover} to obtain the signer of a message. For example:
             *
             * ```solidity
             * bytes32 digest = _hashTypedDataV4(keccak256(abi.encode(
             *     keccak256("Mail(address to,string contents)"),
             *     mailTo,
             *     keccak256(bytes(mailContents))
             * )));
             * address signer = ECDSA.recover(digest, signature);
             * ```
             */
            function _hashTypedDataV4(bytes32 structHash) internal view virtual returns (bytes32) {
                return MessageHashUtils.toTypedDataHash(_domainSeparatorV4(), structHash);
            }
            /**
             * @dev See {IERC-5267}.
             */
            function eip712Domain()
                public
                view
                virtual
                returns (
                    bytes1 fields,
                    string memory name,
                    string memory version,
                    uint256 chainId,
                    address verifyingContract,
                    bytes32 salt,
                    uint256[] memory extensions
                )
            {
                EIP712Storage storage $ = _getEIP712Storage();
                // If the hashed name and version in storage are non-zero, the contract hasn't been properly initialized
                // and the EIP712 domain is not reliable, as it will be missing name and version.
                require($._hashedName == 0 && $._hashedVersion == 0, "EIP712: Uninitialized");
                return (
                    hex"0f", // 01111
                    _EIP712Name(),
                    _EIP712Version(),
                    block.chainid,
                    address(this),
                    bytes32(0),
                    new uint256[](0)
                );
            }
            /**
             * @dev The name parameter for the EIP712 domain.
             *
             * NOTE: This function reads from storage by default, but can be redefined to return a constant value if gas costs
             * are a concern.
             */
            function _EIP712Name() internal view virtual returns (string memory) {
                EIP712Storage storage $ = _getEIP712Storage();
                return $._name;
            }
            /**
             * @dev The version parameter for the EIP712 domain.
             *
             * NOTE: This function reads from storage by default, but can be redefined to return a constant value if gas costs
             * are a concern.
             */
            function _EIP712Version() internal view virtual returns (string memory) {
                EIP712Storage storage $ = _getEIP712Storage();
                return $._version;
            }
            /**
             * @dev The hash of the name parameter for the EIP712 domain.
             *
             * NOTE: In previous versions this function was virtual. In this version you should override `_EIP712Name` instead.
             */
            function _EIP712NameHash() internal view returns (bytes32) {
                EIP712Storage storage $ = _getEIP712Storage();
                string memory name = _EIP712Name();
                if (bytes(name).length > 0) {
                    return keccak256(bytes(name));
                } else {
                    // If the name is empty, the contract may have been upgraded without initializing the new storage.
                    // We return the name hash in storage if non-zero, otherwise we assume the name is empty by design.
                    bytes32 hashedName = $._hashedName;
                    if (hashedName != 0) {
                        return hashedName;
                    } else {
                        return keccak256("");
                    }
                }
            }
            /**
             * @dev The hash of the version parameter for the EIP712 domain.
             *
             * NOTE: In previous versions this function was virtual. In this version you should override `_EIP712Version` instead.
             */
            function _EIP712VersionHash() internal view returns (bytes32) {
                EIP712Storage storage $ = _getEIP712Storage();
                string memory version = _EIP712Version();
                if (bytes(version).length > 0) {
                    return keccak256(bytes(version));
                } else {
                    // If the version is empty, the contract may have been upgraded without initializing the new storage.
                    // We return the version hash in storage if non-zero, otherwise we assume the version is empty by design.
                    bytes32 hashedVersion = $._hashedVersion;
                    if (hashedVersion != 0) {
                        return hashedVersion;
                    } else {
                        return keccak256("");
                    }
                }
            }
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts (last updated v5.0.0) (utils/cryptography/ECDSA.sol)
        pragma solidity ^0.8.20;
        /**
         * @dev Elliptic Curve Digital Signature Algorithm (ECDSA) operations.
         *
         * These functions can be used to verify that a message was signed by the holder
         * of the private keys of a given address.
         */
        library ECDSA {
            enum RecoverError {
                NoError,
                InvalidSignature,
                InvalidSignatureLength,
                InvalidSignatureS
            }
            /**
             * @dev The signature derives the `address(0)`.
             */
            error ECDSAInvalidSignature();
            /**
             * @dev The signature has an invalid length.
             */
            error ECDSAInvalidSignatureLength(uint256 length);
            /**
             * @dev The signature has an S value that is in the upper half order.
             */
            error ECDSAInvalidSignatureS(bytes32 s);
            /**
             * @dev Returns the address that signed a hashed message (`hash`) with `signature` or an error. This will not
             * return address(0) without also returning an error description. Errors are documented using an enum (error type)
             * and a bytes32 providing additional information about the error.
             *
             * If no error is returned, then the address can be used for verification purposes.
             *
             * The `ecrecover` EVM precompile allows for malleable (non-unique) signatures:
             * this function rejects them by requiring the `s` value to be in the lower
             * half order, and the `v` value to be either 27 or 28.
             *
             * IMPORTANT: `hash` _must_ be the result of a hash operation for the
             * verification to be secure: it is possible to craft signatures that
             * recover to arbitrary addresses for non-hashed data. A safe way to ensure
             * this is by receiving a hash of the original message (which may otherwise
             * be too long), and then calling {MessageHashUtils-toEthSignedMessageHash} on it.
             *
             * Documentation for signature generation:
             * - with https://web3js.readthedocs.io/en/v1.3.4/web3-eth-accounts.html#sign[Web3.js]
             * - with https://docs.ethers.io/v5/api/signer/#Signer-signMessage[ethers]
             */
            function tryRecover(bytes32 hash, bytes memory signature) internal pure returns (address, RecoverError, bytes32) {
                if (signature.length == 65) {
                    bytes32 r;
                    bytes32 s;
                    uint8 v;
                    // ecrecover takes the signature parameters, and the only way to get them
                    // currently is to use assembly.
                    /// @solidity memory-safe-assembly
                    assembly {
                        r := mload(add(signature, 0x20))
                        s := mload(add(signature, 0x40))
                        v := byte(0, mload(add(signature, 0x60)))
                    }
                    return tryRecover(hash, v, r, s);
                } else {
                    return (address(0), RecoverError.InvalidSignatureLength, bytes32(signature.length));
                }
            }
            /**
             * @dev Returns the address that signed a hashed message (`hash`) with
             * `signature`. This address can then be used for verification purposes.
             *
             * The `ecrecover` EVM precompile allows for malleable (non-unique) signatures:
             * this function rejects them by requiring the `s` value to be in the lower
             * half order, and the `v` value to be either 27 or 28.
             *
             * IMPORTANT: `hash` _must_ be the result of a hash operation for the
             * verification to be secure: it is possible to craft signatures that
             * recover to arbitrary addresses for non-hashed data. A safe way to ensure
             * this is by receiving a hash of the original message (which may otherwise
             * be too long), and then calling {MessageHashUtils-toEthSignedMessageHash} on it.
             */
            function recover(bytes32 hash, bytes memory signature) internal pure returns (address) {
                (address recovered, RecoverError error, bytes32 errorArg) = tryRecover(hash, signature);
                _throwError(error, errorArg);
                return recovered;
            }
            /**
             * @dev Overload of {ECDSA-tryRecover} that receives the `r` and `vs` short-signature fields separately.
             *
             * See https://eips.ethereum.org/EIPS/eip-2098[EIP-2098 short signatures]
             */
            function tryRecover(bytes32 hash, bytes32 r, bytes32 vs) internal pure returns (address, RecoverError, bytes32) {
                unchecked {
                    bytes32 s = vs & bytes32(0x7fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff);
                    // We do not check for an overflow here since the shift operation results in 0 or 1.
                    uint8 v = uint8((uint256(vs) >> 255) + 27);
                    return tryRecover(hash, v, r, s);
                }
            }
            /**
             * @dev Overload of {ECDSA-recover} that receives the `r and `vs` short-signature fields separately.
             */
            function recover(bytes32 hash, bytes32 r, bytes32 vs) internal pure returns (address) {
                (address recovered, RecoverError error, bytes32 errorArg) = tryRecover(hash, r, vs);
                _throwError(error, errorArg);
                return recovered;
            }
            /**
             * @dev Overload of {ECDSA-tryRecover} that receives the `v`,
             * `r` and `s` signature fields separately.
             */
            function tryRecover(
                bytes32 hash,
                uint8 v,
                bytes32 r,
                bytes32 s
            ) internal pure returns (address, RecoverError, bytes32) {
                // EIP-2 still allows signature malleability for ecrecover(). Remove this possibility and make the signature
                // unique. Appendix F in the Ethereum Yellow paper (https://ethereum.github.io/yellowpaper/paper.pdf), defines
                // the valid range for s in (301): 0 < s < secp256k1n ÷ 2 + 1, and for v in (302): v ∈ {27, 28}. Most
                // signatures from current libraries generate a unique signature with an s-value in the lower half order.
                //
                // If your library generates malleable signatures, such as s-values in the upper range, calculate a new s-value
                // with 0xFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFEBAAEDCE6AF48A03BBFD25E8CD0364141 - s1 and flip v from 27 to 28 or
                // vice versa. If your library also generates signatures with 0/1 for v instead 27/28, add 27 to v to accept
                // these malleable signatures as well.
                if (uint256(s) > 0x7FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF5D576E7357A4501DDFE92F46681B20A0) {
                    return (address(0), RecoverError.InvalidSignatureS, s);
                }
                // If the signature is valid (and not malleable), return the signer address
                address signer = ecrecover(hash, v, r, s);
                if (signer == address(0)) {
                    return (address(0), RecoverError.InvalidSignature, bytes32(0));
                }
                return (signer, RecoverError.NoError, bytes32(0));
            }
            /**
             * @dev Overload of {ECDSA-recover} that receives the `v`,
             * `r` and `s` signature fields separately.
             */
            function recover(bytes32 hash, uint8 v, bytes32 r, bytes32 s) internal pure returns (address) {
                (address recovered, RecoverError error, bytes32 errorArg) = tryRecover(hash, v, r, s);
                _throwError(error, errorArg);
                return recovered;
            }
            /**
             * @dev Optionally reverts with the corresponding custom error according to the `error` argument provided.
             */
            function _throwError(RecoverError error, bytes32 errorArg) private pure {
                if (error == RecoverError.NoError) {
                    return; // no error: do nothing
                } else if (error == RecoverError.InvalidSignature) {
                    revert ECDSAInvalidSignature();
                } else if (error == RecoverError.InvalidSignatureLength) {
                    revert ECDSAInvalidSignatureLength(uint256(errorArg));
                } else if (error == RecoverError.InvalidSignatureS) {
                    revert ECDSAInvalidSignatureS(errorArg);
                }
            }
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts (last updated v5.0.0) (token/ERC20/extensions/IERC20Permit.sol)
        pragma solidity ^0.8.20;
        /**
         * @dev Interface of the ERC20 Permit extension allowing approvals to be made via signatures, as defined in
         * https://eips.ethereum.org/EIPS/eip-2612[EIP-2612].
         *
         * Adds the {permit} method, which can be used to change an account's ERC20 allowance (see {IERC20-allowance}) by
         * presenting a message signed by the account. By not relying on {IERC20-approve}, the token holder account doesn't
         * need to send a transaction, and thus is not required to hold Ether at all.
         *
         * ==== Security Considerations
         *
         * There are two important considerations concerning the use of `permit`. The first is that a valid permit signature
         * expresses an allowance, and it should not be assumed to convey additional meaning. In particular, it should not be
         * considered as an intention to spend the allowance in any specific way. The second is that because permits have
         * built-in replay protection and can be submitted by anyone, they can be frontrun. A protocol that uses permits should
         * take this into consideration and allow a `permit` call to fail. Combining these two aspects, a pattern that may be
         * generally recommended is:
         *
         * ```solidity
         * function doThingWithPermit(..., uint256 value, uint256 deadline, uint8 v, bytes32 r, bytes32 s) public {
         *     try token.permit(msg.sender, address(this), value, deadline, v, r, s) {} catch {}
         *     doThing(..., value);
         * }
         *
         * function doThing(..., uint256 value) public {
         *     token.safeTransferFrom(msg.sender, address(this), value);
         *     ...
         * }
         * ```
         *
         * Observe that: 1) `msg.sender` is used as the owner, leaving no ambiguity as to the signer intent, and 2) the use of
         * `try/catch` allows the permit to fail and makes the code tolerant to frontrunning. (See also
         * {SafeERC20-safeTransferFrom}).
         *
         * Additionally, note that smart contract wallets (such as Argent or Safe) are not able to produce permit signatures, so
         * contracts should have entry points that don't rely on permit.
         */
        interface IERC20Permit {
            /**
             * @dev Sets `value` as the allowance of `spender` over ``owner``'s tokens,
             * given ``owner``'s signed approval.
             *
             * IMPORTANT: The same issues {IERC20-approve} has related to transaction
             * ordering also apply here.
             *
             * Emits an {Approval} event.
             *
             * Requirements:
             *
             * - `spender` cannot be the zero address.
             * - `deadline` must be a timestamp in the future.
             * - `v`, `r` and `s` must be a valid `secp256k1` signature from `owner`
             * over the EIP712-formatted function arguments.
             * - the signature must use ``owner``'s current nonce (see {nonces}).
             *
             * For more information on the signature format, see the
             * https://eips.ethereum.org/EIPS/eip-2612#specification[relevant EIP
             * section].
             *
             * CAUTION: See Security Considerations above.
             */
            function permit(
                address owner,
                address spender,
                uint256 value,
                uint256 deadline,
                uint8 v,
                bytes32 r,
                bytes32 s
            ) external;
            /**
             * @dev Returns the current nonce for `owner`. This value must be
             * included whenever a signature is generated for {permit}.
             *
             * Every successful call to {permit} increases ``owner``'s nonce by one. This
             * prevents a signature from being used multiple times.
             */
            function nonces(address owner) external view returns (uint256);
            /**
             * @dev Returns the domain separator used in the encoding of the signature for {permit}, as defined by {EIP712}.
             */
            // solhint-disable-next-line func-name-mixedcase
            function DOMAIN_SEPARATOR() external view returns (bytes32);
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts (last updated v5.0.0) (utils/Pausable.sol)
        pragma solidity ^0.8.20;
        import {ContextUpgradeable} from "../utils/ContextUpgradeable.sol";
        import {Initializable} from "../proxy/utils/Initializable.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 PausableUpgradeable is Initializable, ContextUpgradeable {
            /// @custom:storage-location erc7201:openzeppelin.storage.Pausable
            struct PausableStorage {
                bool _paused;
            }
            // keccak256(abi.encode(uint256(keccak256("openzeppelin.storage.Pausable")) - 1)) & ~bytes32(uint256(0xff))
            bytes32 private constant PausableStorageLocation = 0xcd5ed15c6e187e77e9aee88184c21f4f2182ab5827cb3b7e07fbedcd63f03300;
            function _getPausableStorage() private pure returns (PausableStorage storage $) {
                assembly {
                    $.slot := PausableStorageLocation
                }
            }
            /**
             * @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);
            /**
             * @dev The operation failed because the contract is paused.
             */
            error EnforcedPause();
            /**
             * @dev The operation failed because the contract is not paused.
             */
            error ExpectedPause();
            /**
             * @dev Initializes the contract in unpaused state.
             */
            function __Pausable_init() internal onlyInitializing {
                __Pausable_init_unchained();
            }
            function __Pausable_init_unchained() internal onlyInitializing {
                PausableStorage storage $ = _getPausableStorage();
                $._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) {
                PausableStorage storage $ = _getPausableStorage();
                return $._paused;
            }
            /**
             * @dev Throws if the contract is paused.
             */
            function _requireNotPaused() internal view virtual {
                if (paused()) {
                    revert EnforcedPause();
                }
            }
            /**
             * @dev Throws if the contract is not paused.
             */
            function _requirePaused() internal view virtual {
                if (!paused()) {
                    revert ExpectedPause();
                }
            }
            /**
             * @dev Triggers stopped state.
             *
             * Requirements:
             *
             * - The contract must not be paused.
             */
            function _pause() internal virtual whenNotPaused {
                PausableStorage storage $ = _getPausableStorage();
                $._paused = true;
                emit Paused(_msgSender());
            }
            /**
             * @dev Returns to normal state.
             *
             * Requirements:
             *
             * - The contract must be paused.
             */
            function _unpause() internal virtual whenPaused {
                PausableStorage storage $ = _getPausableStorage();
                $._paused = false;
                emit Unpaused(_msgSender());
            }
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts (last updated v5.0.1) (utils/Context.sol)
        pragma solidity ^0.8.20;
        import {Initializable} from "../proxy/utils/Initializable.sol";
        /**
         * @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 ContextUpgradeable is Initializable {
            function __Context_init() internal onlyInitializing {
            }
            function __Context_init_unchained() internal onlyInitializing {
            }
            function _msgSender() internal view virtual returns (address) {
                return msg.sender;
            }
            function _msgData() internal view virtual returns (bytes calldata) {
                return msg.data;
            }
            function _contextSuffixLength() internal view virtual returns (uint256) {
                return 0;
            }
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts (last updated v5.0.0) (interfaces/draft-IERC6093.sol)
        pragma solidity ^0.8.20;
        /**
         * @dev Standard ERC20 Errors
         * Interface of the https://eips.ethereum.org/EIPS/eip-6093[ERC-6093] custom errors for ERC20 tokens.
         */
        interface IERC20Errors {
            /**
             * @dev Indicates an error related to the current `balance` of a `sender`. Used in transfers.
             * @param sender Address whose tokens are being transferred.
             * @param balance Current balance for the interacting account.
             * @param needed Minimum amount required to perform a transfer.
             */
            error ERC20InsufficientBalance(address sender, uint256 balance, uint256 needed);
            /**
             * @dev Indicates a failure with the token `sender`. Used in transfers.
             * @param sender Address whose tokens are being transferred.
             */
            error ERC20InvalidSender(address sender);
            /**
             * @dev Indicates a failure with the token `receiver`. Used in transfers.
             * @param receiver Address to which tokens are being transferred.
             */
            error ERC20InvalidReceiver(address receiver);
            /**
             * @dev Indicates a failure with the `spender`’s `allowance`. Used in transfers.
             * @param spender Address that may be allowed to operate on tokens without being their owner.
             * @param allowance Amount of tokens a `spender` is allowed to operate with.
             * @param needed Minimum amount required to perform a transfer.
             */
            error ERC20InsufficientAllowance(address spender, uint256 allowance, uint256 needed);
            /**
             * @dev Indicates a failure with the `approver` of a token to be approved. Used in approvals.
             * @param approver Address initiating an approval operation.
             */
            error ERC20InvalidApprover(address approver);
            /**
             * @dev Indicates a failure with the `spender` to be approved. Used in approvals.
             * @param spender Address that may be allowed to operate on tokens without being their owner.
             */
            error ERC20InvalidSpender(address spender);
        }
        /**
         * @dev Standard ERC721 Errors
         * Interface of the https://eips.ethereum.org/EIPS/eip-6093[ERC-6093] custom errors for ERC721 tokens.
         */
        interface IERC721Errors {
            /**
             * @dev Indicates that an address can't be an owner. For example, `address(0)` is a forbidden owner in EIP-20.
             * Used in balance queries.
             * @param owner Address of the current owner of a token.
             */
            error ERC721InvalidOwner(address owner);
            /**
             * @dev Indicates a `tokenId` whose `owner` is the zero address.
             * @param tokenId Identifier number of a token.
             */
            error ERC721NonexistentToken(uint256 tokenId);
            /**
             * @dev Indicates an error related to the ownership over a particular token. Used in transfers.
             * @param sender Address whose tokens are being transferred.
             * @param tokenId Identifier number of a token.
             * @param owner Address of the current owner of a token.
             */
            error ERC721IncorrectOwner(address sender, uint256 tokenId, address owner);
            /**
             * @dev Indicates a failure with the token `sender`. Used in transfers.
             * @param sender Address whose tokens are being transferred.
             */
            error ERC721InvalidSender(address sender);
            /**
             * @dev Indicates a failure with the token `receiver`. Used in transfers.
             * @param receiver Address to which tokens are being transferred.
             */
            error ERC721InvalidReceiver(address receiver);
            /**
             * @dev Indicates a failure with the `operator`’s approval. Used in transfers.
             * @param operator Address that may be allowed to operate on tokens without being their owner.
             * @param tokenId Identifier number of a token.
             */
            error ERC721InsufficientApproval(address operator, uint256 tokenId);
            /**
             * @dev Indicates a failure with the `approver` of a token to be approved. Used in approvals.
             * @param approver Address initiating an approval operation.
             */
            error ERC721InvalidApprover(address approver);
            /**
             * @dev Indicates a failure with the `operator` to be approved. Used in approvals.
             * @param operator Address that may be allowed to operate on tokens without being their owner.
             */
            error ERC721InvalidOperator(address operator);
        }
        /**
         * @dev Standard ERC1155 Errors
         * Interface of the https://eips.ethereum.org/EIPS/eip-6093[ERC-6093] custom errors for ERC1155 tokens.
         */
        interface IERC1155Errors {
            /**
             * @dev Indicates an error related to the current `balance` of a `sender`. Used in transfers.
             * @param sender Address whose tokens are being transferred.
             * @param balance Current balance for the interacting account.
             * @param needed Minimum amount required to perform a transfer.
             * @param tokenId Identifier number of a token.
             */
            error ERC1155InsufficientBalance(address sender, uint256 balance, uint256 needed, uint256 tokenId);
            /**
             * @dev Indicates a failure with the token `sender`. Used in transfers.
             * @param sender Address whose tokens are being transferred.
             */
            error ERC1155InvalidSender(address sender);
            /**
             * @dev Indicates a failure with the token `receiver`. Used in transfers.
             * @param receiver Address to which tokens are being transferred.
             */
            error ERC1155InvalidReceiver(address receiver);
            /**
             * @dev Indicates a failure with the `operator`’s approval. Used in transfers.
             * @param operator Address that may be allowed to operate on tokens without being their owner.
             * @param owner Address of the current owner of a token.
             */
            error ERC1155MissingApprovalForAll(address operator, address owner);
            /**
             * @dev Indicates a failure with the `approver` of a token to be approved. Used in approvals.
             * @param approver Address initiating an approval operation.
             */
            error ERC1155InvalidApprover(address approver);
            /**
             * @dev Indicates a failure with the `operator` to be approved. Used in approvals.
             * @param operator Address that may be allowed to operate on tokens without being their owner.
             */
            error ERC1155InvalidOperator(address operator);
            /**
             * @dev Indicates an array length mismatch between ids and values in a safeBatchTransferFrom operation.
             * Used in batch transfers.
             * @param idsLength Length of the array of token identifiers
             * @param valuesLength Length of the array of token amounts
             */
            error ERC1155InvalidArrayLength(uint256 idsLength, uint256 valuesLength);
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts (last updated v5.0.0) (token/ERC20/extensions/IERC20Metadata.sol)
        pragma solidity ^0.8.20;
        import {IERC20} from "../IERC20.sol";
        /**
         * @dev Interface for the optional metadata functions from the ERC20 standard.
         */
        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);
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts (last updated v5.0.0) (interfaces/IERC5267.sol)
        pragma solidity ^0.8.20;
        interface IERC5267 {
            /**
             * @dev MAY be emitted to signal that the domain could have changed.
             */
            event EIP712DomainChanged();
            /**
             * @dev returns the fields and values that describe the domain separator used by this contract for EIP-712
             * signature.
             */
            function eip712Domain()
                external
                view
                returns (
                    bytes1 fields,
                    string memory name,
                    string memory version,
                    uint256 chainId,
                    address verifyingContract,
                    bytes32 salt,
                    uint256[] memory extensions
                );
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts (last updated v5.0.0) (utils/cryptography/MessageHashUtils.sol)
        pragma solidity ^0.8.20;
        import {Strings} from "../Strings.sol";
        /**
         * @dev Signature message hash utilities for producing digests to be consumed by {ECDSA} recovery or signing.
         *
         * The library provides methods for generating a hash of a message that conforms to the
         * https://eips.ethereum.org/EIPS/eip-191[EIP 191] and https://eips.ethereum.org/EIPS/eip-712[EIP 712]
         * specifications.
         */
        library MessageHashUtils {
            /**
             * @dev Returns the keccak256 digest of an EIP-191 signed data with version
             * `0x45` (`personal_sign` messages).
             *
             * The digest is calculated by prefixing a bytes32 `messageHash` with
             * `"\\x19Ethereum Signed Message:\
        32"` and hashing the result. It corresponds with the
             * hash signed when using the https://eth.wiki/json-rpc/API#eth_sign[`eth_sign`] JSON-RPC method.
             *
             * NOTE: The `messageHash` parameter is intended to be the result of hashing a raw message with
             * keccak256, although any bytes32 value can be safely used because the final digest will
             * be re-hashed.
             *
             * See {ECDSA-recover}.
             */
            function toEthSignedMessageHash(bytes32 messageHash) internal pure returns (bytes32 digest) {
                /// @solidity memory-safe-assembly
                assembly {
                    mstore(0x00, "\\x19Ethereum Signed Message:\
        32") // 32 is the bytes-length of messageHash
                    mstore(0x1c, messageHash) // 0x1c (28) is the length of the prefix
                    digest := keccak256(0x00, 0x3c) // 0x3c is the length of the prefix (0x1c) + messageHash (0x20)
                }
            }
            /**
             * @dev Returns the keccak256 digest of an EIP-191 signed data with version
             * `0x45` (`personal_sign` messages).
             *
             * The digest is calculated by prefixing an arbitrary `message` with
             * `"\\x19Ethereum Signed Message:\
        " + len(message)` and hashing the result. It corresponds with the
             * hash signed when using the https://eth.wiki/json-rpc/API#eth_sign[`eth_sign`] JSON-RPC method.
             *
             * See {ECDSA-recover}.
             */
            function toEthSignedMessageHash(bytes memory message) internal pure returns (bytes32) {
                return
                    keccak256(bytes.concat("\\x19Ethereum Signed Message:\
        ", bytes(Strings.toString(message.length)), message));
            }
            /**
             * @dev Returns the keccak256 digest of an EIP-191 signed data with version
             * `0x00` (data with intended validator).
             *
             * The digest is calculated by prefixing an arbitrary `data` with `"\\x19\\x00"` and the intended
             * `validator` address. Then hashing the result.
             *
             * See {ECDSA-recover}.
             */
            function toDataWithIntendedValidatorHash(address validator, bytes memory data) internal pure returns (bytes32) {
                return keccak256(abi.encodePacked(hex"19_00", validator, data));
            }
            /**
             * @dev Returns the keccak256 digest of an EIP-712 typed data (EIP-191 version `0x01`).
             *
             * The digest is calculated from a `domainSeparator` and a `structHash`, by prefixing them with
             * `\\x19\\x01` and hashing the result. It corresponds to the hash signed by the
             * https://eips.ethereum.org/EIPS/eip-712[`eth_signTypedData`] JSON-RPC method as part of EIP-712.
             *
             * See {ECDSA-recover}.
             */
            function toTypedDataHash(bytes32 domainSeparator, bytes32 structHash) internal pure returns (bytes32 digest) {
                /// @solidity memory-safe-assembly
                assembly {
                    let ptr := mload(0x40)
                    mstore(ptr, hex"19_01")
                    mstore(add(ptr, 0x02), domainSeparator)
                    mstore(add(ptr, 0x22), structHash)
                    digest := keccak256(ptr, 0x42)
                }
            }
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts (last updated v5.0.0) (utils/Strings.sol)
        pragma solidity ^0.8.20;
        import {Math} from "./math/Math.sol";
        import {SignedMath} from "./math/SignedMath.sol";
        /**
         * @dev String operations.
         */
        library Strings {
            bytes16 private constant HEX_DIGITS = "0123456789abcdef";
            uint8 private constant ADDRESS_LENGTH = 20;
            /**
             * @dev The `value` string doesn't fit in the specified `length`.
             */
            error StringsInsufficientHexLength(uint256 value, uint256 length);
            /**
             * @dev Converts a `uint256` to its ASCII `string` decimal representation.
             */
            function toString(uint256 value) internal pure returns (string memory) {
                unchecked {
                    uint256 length = Math.log10(value) + 1;
                    string memory buffer = new string(length);
                    uint256 ptr;
                    /// @solidity memory-safe-assembly
                    assembly {
                        ptr := add(buffer, add(32, length))
                    }
                    while (true) {
                        ptr--;
                        /// @solidity memory-safe-assembly
                        assembly {
                            mstore8(ptr, byte(mod(value, 10), HEX_DIGITS))
                        }
                        value /= 10;
                        if (value == 0) break;
                    }
                    return buffer;
                }
            }
            /**
             * @dev Converts a `int256` to its ASCII `string` decimal representation.
             */
            function toStringSigned(int256 value) internal pure returns (string memory) {
                return string.concat(value < 0 ? "-" : "", toString(SignedMath.abs(value)));
            }
            /**
             * @dev Converts a `uint256` to its ASCII `string` hexadecimal representation.
             */
            function toHexString(uint256 value) internal pure returns (string memory) {
                unchecked {
                    return toHexString(value, Math.log256(value) + 1);
                }
            }
            /**
             * @dev Converts a `uint256` to its ASCII `string` hexadecimal representation with fixed length.
             */
            function toHexString(uint256 value, uint256 length) internal pure returns (string memory) {
                uint256 localValue = value;
                bytes memory buffer = new bytes(2 * length + 2);
                buffer[0] = "0";
                buffer[1] = "x";
                for (uint256 i = 2 * length + 1; i > 1; --i) {
                    buffer[i] = HEX_DIGITS[localValue & 0xf];
                    localValue >>= 4;
                }
                if (localValue != 0) {
                    revert StringsInsufficientHexLength(value, length);
                }
                return string(buffer);
            }
            /**
             * @dev Converts an `address` with fixed length of 20 bytes to its not checksummed ASCII `string` hexadecimal
             * representation.
             */
            function toHexString(address addr) internal pure returns (string memory) {
                return toHexString(uint256(uint160(addr)), ADDRESS_LENGTH);
            }
            /**
             * @dev Returns true if the two strings are equal.
             */
            function equal(string memory a, string memory b) internal pure returns (bool) {
                return bytes(a).length == bytes(b).length && keccak256(bytes(a)) == keccak256(bytes(b));
            }
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts (last updated v5.0.0) (utils/math/SignedMath.sol)
        pragma solidity ^0.8.20;
        /**
         * @dev Standard signed math utilities missing in the Solidity language.
         */
        library SignedMath {
            /**
             * @dev Returns the largest of two signed numbers.
             */
            function max(int256 a, int256 b) internal pure returns (int256) {
                return a > b ? a : b;
            }
            /**
             * @dev Returns the smallest of two signed numbers.
             */
            function min(int256 a, int256 b) internal pure returns (int256) {
                return a < b ? a : b;
            }
            /**
             * @dev Returns the average of two signed numbers without overflow.
             * The result is rounded towards zero.
             */
            function average(int256 a, int256 b) internal pure returns (int256) {
                // Formula from the book "Hacker's Delight"
                int256 x = (a & b) + ((a ^ b) >> 1);
                return x + (int256(uint256(x) >> 255) & (a ^ b));
            }
            /**
             * @dev Returns the absolute unsigned value of a signed value.
             */
            function abs(int256 n) internal pure returns (uint256) {
                unchecked {
                    // must be unchecked in order to support `n = type(int256).min`
                    return uint256(n >= 0 ? n : -n);
                }
            }
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts (last updated v5.0.0) (utils/math/Math.sol)
        pragma solidity ^0.8.20;
        /**
         * @dev Standard math utilities missing in the Solidity language.
         */
        library Math {
            /**
             * @dev Muldiv operation overflow.
             */
            error MathOverflowedMulDiv();
            enum Rounding {
                Floor, // Toward negative infinity
                Ceil, // Toward positive infinity
                Trunc, // Toward zero
                Expand // Away from zero
            }
            /**
             * @dev Returns the addition of two unsigned integers, with an overflow flag.
             */
            function tryAdd(uint256 a, uint256 b) internal pure returns (bool, uint256) {
                unchecked {
                    uint256 c = a + b;
                    if (c < a) return (false, 0);
                    return (true, c);
                }
            }
            /**
             * @dev Returns the subtraction of two unsigned integers, with an overflow flag.
             */
            function trySub(uint256 a, uint256 b) internal pure returns (bool, uint256) {
                unchecked {
                    if (b > a) return (false, 0);
                    return (true, a - b);
                }
            }
            /**
             * @dev Returns the multiplication of two unsigned integers, with an overflow flag.
             */
            function tryMul(uint256 a, uint256 b) internal pure returns (bool, uint256) {
                unchecked {
                    // Gas optimization: this is cheaper than requiring 'a' not being zero, but the
                    // benefit is lost if 'b' is also tested.
                    // See: https://github.com/OpenZeppelin/openzeppelin-contracts/pull/522
                    if (a == 0) return (true, 0);
                    uint256 c = a * b;
                    if (c / a != b) return (false, 0);
                    return (true, c);
                }
            }
            /**
             * @dev Returns the division of two unsigned integers, with a division by zero flag.
             */
            function tryDiv(uint256 a, uint256 b) internal pure returns (bool, uint256) {
                unchecked {
                    if (b == 0) return (false, 0);
                    return (true, a / b);
                }
            }
            /**
             * @dev Returns the remainder of dividing two unsigned integers, with a division by zero flag.
             */
            function tryMod(uint256 a, uint256 b) internal pure returns (bool, uint256) {
                unchecked {
                    if (b == 0) return (false, 0);
                    return (true, a % b);
                }
            }
            /**
             * @dev Returns the largest of two numbers.
             */
            function max(uint256 a, uint256 b) internal pure returns (uint256) {
                return a > b ? a : b;
            }
            /**
             * @dev Returns the smallest of two numbers.
             */
            function min(uint256 a, uint256 b) internal pure returns (uint256) {
                return a < b ? a : b;
            }
            /**
             * @dev Returns the average of two numbers. The result is rounded towards
             * zero.
             */
            function average(uint256 a, uint256 b) internal pure returns (uint256) {
                // (a + b) / 2 can overflow.
                return (a & b) + (a ^ b) / 2;
            }
            /**
             * @dev Returns the ceiling of the division of two numbers.
             *
             * This differs from standard division with `/` in that it rounds towards infinity instead
             * of rounding towards zero.
             */
            function ceilDiv(uint256 a, uint256 b) internal pure returns (uint256) {
                if (b == 0) {
                    // Guarantee the same behavior as in a regular Solidity division.
                    return a / b;
                }
                // (a + b - 1) / b can overflow on addition, so we distribute.
                return a == 0 ? 0 : (a - 1) / b + 1;
            }
            /**
             * @notice Calculates floor(x * y / denominator) with full precision. Throws if result overflows a uint256 or
             * denominator == 0.
             * @dev Original credit to Remco Bloemen under MIT license (https://xn--2-umb.com/21/muldiv) with further edits by
             * Uniswap Labs also under MIT license.
             */
            function mulDiv(uint256 x, uint256 y, uint256 denominator) internal pure returns (uint256 result) {
                unchecked {
                    // 512-bit multiply [prod1 prod0] = x * y. Compute the product mod 2^256 and mod 2^256 - 1, then use
                    // use the Chinese Remainder Theorem to reconstruct the 512 bit result. The result is stored in two 256
                    // variables such that product = prod1 * 2^256 + prod0.
                    uint256 prod0 = x * y; // Least significant 256 bits of the product
                    uint256 prod1; // Most significant 256 bits of the product
                    assembly {
                        let mm := mulmod(x, y, not(0))
                        prod1 := sub(sub(mm, prod0), lt(mm, prod0))
                    }
                    // Handle non-overflow cases, 256 by 256 division.
                    if (prod1 == 0) {
                        // Solidity will revert if denominator == 0, unlike the div opcode on its own.
                        // The surrounding unchecked block does not change this fact.
                        // See https://docs.soliditylang.org/en/latest/control-structures.html#checked-or-unchecked-arithmetic.
                        return prod0 / denominator;
                    }
                    // Make sure the result is less than 2^256. Also prevents denominator == 0.
                    if (denominator <= prod1) {
                        revert MathOverflowedMulDiv();
                    }
                    ///////////////////////////////////////////////
                    // 512 by 256 division.
                    ///////////////////////////////////////////////
                    // Make division exact by subtracting the remainder from [prod1 prod0].
                    uint256 remainder;
                    assembly {
                        // Compute remainder using mulmod.
                        remainder := mulmod(x, y, denominator)
                        // Subtract 256 bit number from 512 bit number.
                        prod1 := sub(prod1, gt(remainder, prod0))
                        prod0 := sub(prod0, remainder)
                    }
                    // Factor powers of two out of denominator and compute largest power of two divisor of denominator.
                    // Always >= 1. See https://cs.stackexchange.com/q/138556/92363.
                    uint256 twos = denominator & (0 - denominator);
                    assembly {
                        // Divide denominator by twos.
                        denominator := div(denominator, twos)
                        // Divide [prod1 prod0] by twos.
                        prod0 := div(prod0, twos)
                        // Flip twos such that it is 2^256 / twos. If twos is zero, then it becomes one.
                        twos := add(div(sub(0, twos), twos), 1)
                    }
                    // Shift in bits from prod1 into prod0.
                    prod0 |= prod1 * twos;
                    // Invert denominator mod 2^256. Now that denominator is an odd number, it has an inverse modulo 2^256 such
                    // that denominator * inv = 1 mod 2^256. Compute the inverse by starting with a seed that is correct for
                    // four bits. That is, denominator * inv = 1 mod 2^4.
                    uint256 inverse = (3 * denominator) ^ 2;
                    // Use the Newton-Raphson iteration to improve the precision. Thanks to Hensel's lifting lemma, this also
                    // works in modular arithmetic, doubling the correct bits in each step.
                    inverse *= 2 - denominator * inverse; // inverse mod 2^8
                    inverse *= 2 - denominator * inverse; // inverse mod 2^16
                    inverse *= 2 - denominator * inverse; // inverse mod 2^32
                    inverse *= 2 - denominator * inverse; // inverse mod 2^64
                    inverse *= 2 - denominator * inverse; // inverse mod 2^128
                    inverse *= 2 - denominator * inverse; // inverse mod 2^256
                    // Because the division is now exact we can divide by multiplying with the modular inverse of denominator.
                    // This will give us the correct result modulo 2^256. Since the preconditions guarantee that the outcome is
                    // less than 2^256, this is the final result. We don't need to compute the high bits of the result and prod1
                    // is no longer required.
                    result = prod0 * inverse;
                    return result;
                }
            }
            /**
             * @notice Calculates x * y / denominator with full precision, following the selected rounding direction.
             */
            function mulDiv(uint256 x, uint256 y, uint256 denominator, Rounding rounding) internal pure returns (uint256) {
                uint256 result = mulDiv(x, y, denominator);
                if (unsignedRoundsUp(rounding) && mulmod(x, y, denominator) > 0) {
                    result += 1;
                }
                return result;
            }
            /**
             * @dev Returns the square root of a number. If the number is not a perfect square, the value is rounded
             * towards zero.
             *
             * Inspired by Henry S. Warren, Jr.'s "Hacker's Delight" (Chapter 11).
             */
            function sqrt(uint256 a) internal pure returns (uint256) {
                if (a == 0) {
                    return 0;
                }
                // For our first guess, we get the biggest power of 2 which is smaller than the square root of the target.
                //
                // We know that the "msb" (most significant bit) of our target number `a` is a power of 2 such that we have
                // `msb(a) <= a < 2*msb(a)`. This value can be written `msb(a)=2**k` with `k=log2(a)`.
                //
                // This can be rewritten `2**log2(a) <= a < 2**(log2(a) + 1)`
                // → `sqrt(2**k) <= sqrt(a) < sqrt(2**(k+1))`
                // → `2**(k/2) <= sqrt(a) < 2**((k+1)/2) <= 2**(k/2 + 1)`
                //
                // Consequently, `2**(log2(a) / 2)` is a good first approximation of `sqrt(a)` with at least 1 correct bit.
                uint256 result = 1 << (log2(a) >> 1);
                // At this point `result` is an estimation with one bit of precision. We know the true value is a uint128,
                // since it is the square root of a uint256. Newton's method converges quadratically (precision doubles at
                // every iteration). We thus need at most 7 iteration to turn our partial result with one bit of precision
                // into the expected uint128 result.
                unchecked {
                    result = (result + a / result) >> 1;
                    result = (result + a / result) >> 1;
                    result = (result + a / result) >> 1;
                    result = (result + a / result) >> 1;
                    result = (result + a / result) >> 1;
                    result = (result + a / result) >> 1;
                    result = (result + a / result) >> 1;
                    return min(result, a / result);
                }
            }
            /**
             * @notice Calculates sqrt(a), following the selected rounding direction.
             */
            function sqrt(uint256 a, Rounding rounding) internal pure returns (uint256) {
                unchecked {
                    uint256 result = sqrt(a);
                    return result + (unsignedRoundsUp(rounding) && result * result < a ? 1 : 0);
                }
            }
            /**
             * @dev Return the log in base 2 of a positive value rounded towards zero.
             * Returns 0 if given 0.
             */
            function log2(uint256 value) internal pure returns (uint256) {
                uint256 result = 0;
                unchecked {
                    if (value >> 128 > 0) {
                        value >>= 128;
                        result += 128;
                    }
                    if (value >> 64 > 0) {
                        value >>= 64;
                        result += 64;
                    }
                    if (value >> 32 > 0) {
                        value >>= 32;
                        result += 32;
                    }
                    if (value >> 16 > 0) {
                        value >>= 16;
                        result += 16;
                    }
                    if (value >> 8 > 0) {
                        value >>= 8;
                        result += 8;
                    }
                    if (value >> 4 > 0) {
                        value >>= 4;
                        result += 4;
                    }
                    if (value >> 2 > 0) {
                        value >>= 2;
                        result += 2;
                    }
                    if (value >> 1 > 0) {
                        result += 1;
                    }
                }
                return result;
            }
            /**
             * @dev Return the log in base 2, following the selected rounding direction, of a positive value.
             * Returns 0 if given 0.
             */
            function log2(uint256 value, Rounding rounding) internal pure returns (uint256) {
                unchecked {
                    uint256 result = log2(value);
                    return result + (unsignedRoundsUp(rounding) && 1 << result < value ? 1 : 0);
                }
            }
            /**
             * @dev Return the log in base 10 of a positive value rounded towards zero.
             * Returns 0 if given 0.
             */
            function log10(uint256 value) internal pure returns (uint256) {
                uint256 result = 0;
                unchecked {
                    if (value >= 10 ** 64) {
                        value /= 10 ** 64;
                        result += 64;
                    }
                    if (value >= 10 ** 32) {
                        value /= 10 ** 32;
                        result += 32;
                    }
                    if (value >= 10 ** 16) {
                        value /= 10 ** 16;
                        result += 16;
                    }
                    if (value >= 10 ** 8) {
                        value /= 10 ** 8;
                        result += 8;
                    }
                    if (value >= 10 ** 4) {
                        value /= 10 ** 4;
                        result += 4;
                    }
                    if (value >= 10 ** 2) {
                        value /= 10 ** 2;
                        result += 2;
                    }
                    if (value >= 10 ** 1) {
                        result += 1;
                    }
                }
                return result;
            }
            /**
             * @dev Return the log in base 10, following the selected rounding direction, of a positive value.
             * Returns 0 if given 0.
             */
            function log10(uint256 value, Rounding rounding) internal pure returns (uint256) {
                unchecked {
                    uint256 result = log10(value);
                    return result + (unsignedRoundsUp(rounding) && 10 ** result < value ? 1 : 0);
                }
            }
            /**
             * @dev Return the log in base 256 of a positive value rounded towards zero.
             * Returns 0 if given 0.
             *
             * Adding one to the result gives the number of pairs of hex symbols needed to represent `value` as a hex string.
             */
            function log256(uint256 value) internal pure returns (uint256) {
                uint256 result = 0;
                unchecked {
                    if (value >> 128 > 0) {
                        value >>= 128;
                        result += 16;
                    }
                    if (value >> 64 > 0) {
                        value >>= 64;
                        result += 8;
                    }
                    if (value >> 32 > 0) {
                        value >>= 32;
                        result += 4;
                    }
                    if (value >> 16 > 0) {
                        value >>= 16;
                        result += 2;
                    }
                    if (value >> 8 > 0) {
                        result += 1;
                    }
                }
                return result;
            }
            /**
             * @dev Return the log in base 256, following the selected rounding direction, of a positive value.
             * Returns 0 if given 0.
             */
            function log256(uint256 value, Rounding rounding) internal pure returns (uint256) {
                unchecked {
                    uint256 result = log256(value);
                    return result + (unsignedRoundsUp(rounding) && 1 << (result << 3) < value ? 1 : 0);
                }
            }
            /**
             * @dev Returns whether a provided rounding mode is considered rounding up for unsigned integers.
             */
            function unsignedRoundsUp(Rounding rounding) internal pure returns (bool) {
                return uint8(rounding) % 2 == 1;
            }
        }