ETH Price: $2,475.86 (+1.46%)

Transaction Decoder

Block:
22811537 at Jun-29-2025 05:21:11 PM +UTC
Transaction Fee:
0.000082784923300992 ETH $0.20
Gas Used:
117,952 Gas / 0.701852646 Gwei

Emitted Events:

338 ERC1967Proxy.0xddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef( 0xddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef, 0x0000000000000000000000008e02d37b6cad86039bdd11095b8c879b907f7d10, 0x000000000000000000000000648f09326ae1012be76449422e888427e24a3872, 000000000000000000000000000000000000000000000001aeb42b99a99df822 )
339 ERC1967Proxy.0xe2403640ba68fed3a2f88b7557551d1993f84b99bb10ff833f0cf8db0c5e0486( 0xe2403640ba68fed3a2f88b7557551d1993f84b99bb10ff833f0cf8db0c5e0486, 0x000000000000000000000000648f09326ae1012be76449422e888427e24a3872, 000000000000000000000000000000000000000000000001aeb42b99a99df822 )
340 ERC1967Proxy.0xddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef( 0xddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef, 0x000000000000000000000000648f09326ae1012be76449422e888427e24a3872, 0x000000000000000000000000000040d6c85a13a1aa74565fde87e499dc023c6f, 000000000000000000000000000000000000000000000001aeb42b99a99df822 )

Account State Difference:

  Address   Before After State Difference Code
0x000040D6...9dC023C6f
0.005075632344505776 Eth
Nonce: 78
0.004992847421204784 Eth
Nonce: 79
0.000082784923300992
0x648f0932...7E24a3872 From: 22892026899652070454194934764186000560510438816284993843 To: 22892026286104430195768889098564826109614395621165509569
0x88909D48...9Bf78FD9a
0x8E02d37b...B907F7D10
(BuilderNet)
21.273034470911426625 Eth21.273037498610698945 Eth0.00000302769927232

Execution Trace

0x648f09326ae1012be76449422e888427e24a3872.252dba42( )
  • ERC1967Proxy.CALL( )
    • Staking.DELEGATECALL( )
      • ERC1967Proxy.a9059cbb( )
        • Layer3.transfer( to=0x648f09326aE1012Be76449422E888427E24a3872, value=31035478871092623394 ) => ( True )
        • 0x648f09326ae1012be76449422e888427e24a3872.5ecb16cd( )
          • ERC1967Proxy.70a08231( )
            • Layer3.balanceOf( account=0x648f09326aE1012Be76449422E888427E24a3872 ) => ( 31035478871092623394 )
            • ERC1967Proxy.a9059cbb( )
              • Layer3.transfer( to=0x000040D6c85A13a1AA74565FDe87e499dC023C6f, value=31035478871092623394 ) => ( True )
                File 1 of 4: ERC1967Proxy
                // SPDX-License-Identifier: MIT
                // OpenZeppelin Contracts (last updated v5.0.0) (proxy/ERC1967/ERC1967Proxy.sol)
                pragma solidity ^0.8.20;
                import {Proxy} from "../Proxy.sol";
                import {ERC1967Utils} from "./ERC1967Utils.sol";
                /**
                 * @dev This contract implements an upgradeable proxy. It is upgradeable because calls are delegated to an
                 * implementation address that can be changed. This address is stored in storage in the location specified by
                 * https://eips.ethereum.org/EIPS/eip-1967[EIP1967], so that it doesn't conflict with the storage layout of the
                 * implementation behind the proxy.
                 */
                contract ERC1967Proxy is Proxy {
                    /**
                     * @dev Initializes the upgradeable proxy with an initial implementation specified by `implementation`.
                     *
                     * If `_data` is nonempty, it's used as data in a delegate call to `implementation`. This will typically be an
                     * encoded function call, and allows initializing the storage of the proxy like a Solidity constructor.
                     *
                     * Requirements:
                     *
                     * - If `data` is empty, `msg.value` must be zero.
                     */
                    constructor(address implementation, bytes memory _data) payable {
                        ERC1967Utils.upgradeToAndCall(implementation, _data);
                    }
                    /**
                     * @dev Returns the current implementation address.
                     *
                     * TIP: To get this value clients can read directly from the storage slot shown below (specified by EIP1967) using
                     * the https://eth.wiki/json-rpc/API#eth_getstorageat[`eth_getStorageAt`] RPC call.
                     * `0x360894a13ba1a3210667c828492db98dca3e2076cc3735a920a3ca505d382bbc`
                     */
                    function _implementation() internal view virtual override returns (address) {
                        return ERC1967Utils.getImplementation();
                    }
                }
                // SPDX-License-Identifier: MIT
                // OpenZeppelin Contracts (last updated v5.0.0) (proxy/Proxy.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 virtual {
                        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 virtual {
                        _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 virtual {
                        _fallback();
                    }
                }
                // SPDX-License-Identifier: MIT
                // OpenZeppelin Contracts (last updated v5.0.0) (proxy/ERC1967/ERC1967Utils.sol)
                pragma solidity ^0.8.20;
                import {IBeacon} from "../beacon/IBeacon.sol";
                import {Address} from "../../utils/Address.sol";
                import {StorageSlot} from "../../utils/StorageSlot.sol";
                /**
                 * @dev This abstract contract provides getters and event emitting update functions for
                 * https://eips.ethereum.org/EIPS/eip-1967[EIP1967] slots.
                 */
                library ERC1967Utils {
                    // We re-declare ERC-1967 events here because they can't be used directly from IERC1967.
                    // This will be fixed in Solidity 0.8.21. At that point we should remove these events.
                    /**
                     * @dev Emitted when the implementation is upgraded.
                     */
                    event Upgraded(address indexed implementation);
                    /**
                     * @dev Emitted when the admin account has changed.
                     */
                    event AdminChanged(address previousAdmin, address newAdmin);
                    /**
                     * @dev Emitted when the beacon is changed.
                     */
                    event BeaconUpgraded(address indexed beacon);
                    /**
                     * @dev Storage slot with the address of the current implementation.
                     * This is the keccak-256 hash of "eip1967.proxy.implementation" subtracted by 1.
                     */
                    // solhint-disable-next-line private-vars-leading-underscore
                    bytes32 internal constant IMPLEMENTATION_SLOT = 0x360894a13ba1a3210667c828492db98dca3e2076cc3735a920a3ca505d382bbc;
                    /**
                     * @dev The `implementation` of the proxy is invalid.
                     */
                    error ERC1967InvalidImplementation(address implementation);
                    /**
                     * @dev The `admin` of the proxy is invalid.
                     */
                    error ERC1967InvalidAdmin(address admin);
                    /**
                     * @dev The `beacon` of the proxy is invalid.
                     */
                    error ERC1967InvalidBeacon(address beacon);
                    /**
                     * @dev An upgrade function sees `msg.value > 0` that may be lost.
                     */
                    error ERC1967NonPayable();
                    /**
                     * @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 ERC1967InvalidImplementation(newImplementation);
                        }
                        StorageSlot.getAddressSlot(IMPLEMENTATION_SLOT).value = newImplementation;
                    }
                    /**
                     * @dev Performs implementation upgrade with additional setup call if data is nonempty.
                     * This function is payable only if the setup call is performed, otherwise `msg.value` is rejected
                     * to avoid stuck value in the contract.
                     *
                     * Emits an {IERC1967-Upgraded} event.
                     */
                    function upgradeToAndCall(address newImplementation, bytes memory data) internal {
                        _setImplementation(newImplementation);
                        emit Upgraded(newImplementation);
                        if (data.length > 0) {
                            Address.functionDelegateCall(newImplementation, data);
                        } else {
                            _checkNonPayable();
                        }
                    }
                    /**
                     * @dev Storage slot with the admin of the contract.
                     * This is the keccak-256 hash of "eip1967.proxy.admin" subtracted by 1.
                     */
                    // solhint-disable-next-line private-vars-leading-underscore
                    bytes32 internal constant ADMIN_SLOT = 0xb53127684a568b3173ae13b9f8a6016e243e63b6e8ee1178d6a717850b5d6103;
                    /**
                     * @dev Returns the current admin.
                     *
                     * TIP: To get this value clients can read directly from the storage slot shown below (specified by EIP1967) using
                     * the https://eth.wiki/json-rpc/API#eth_getstorageat[`eth_getStorageAt`] RPC call.
                     * `0xb53127684a568b3173ae13b9f8a6016e243e63b6e8ee1178d6a717850b5d6103`
                     */
                    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 {
                        if (newAdmin == address(0)) {
                            revert ERC1967InvalidAdmin(address(0));
                        }
                        StorageSlot.getAddressSlot(ADMIN_SLOT).value = newAdmin;
                    }
                    /**
                     * @dev Changes the admin of the proxy.
                     *
                     * Emits an {IERC1967-AdminChanged} event.
                     */
                    function changeAdmin(address newAdmin) internal {
                        emit AdminChanged(getAdmin(), newAdmin);
                        _setAdmin(newAdmin);
                    }
                    /**
                     * @dev The storage slot of the UpgradeableBeacon contract which defines the implementation for this proxy.
                     * This is the keccak-256 hash of "eip1967.proxy.beacon" subtracted by 1.
                     */
                    // solhint-disable-next-line private-vars-leading-underscore
                    bytes32 internal constant BEACON_SLOT = 0xa3f0ad74e5423aebfd80d3ef4346578335a9a72aeaee59ff6cb3582b35133d50;
                    /**
                     * @dev Returns the current beacon.
                     */
                    function getBeacon() internal view returns (address) {
                        return StorageSlot.getAddressSlot(BEACON_SLOT).value;
                    }
                    /**
                     * @dev Stores a new beacon in the EIP1967 beacon slot.
                     */
                    function _setBeacon(address newBeacon) private {
                        if (newBeacon.code.length == 0) {
                            revert ERC1967InvalidBeacon(newBeacon);
                        }
                        StorageSlot.getAddressSlot(BEACON_SLOT).value = newBeacon;
                        address beaconImplementation = IBeacon(newBeacon).implementation();
                        if (beaconImplementation.code.length == 0) {
                            revert ERC1967InvalidImplementation(beaconImplementation);
                        }
                    }
                    /**
                     * @dev Change the beacon and trigger a setup call if data is nonempty.
                     * This function is payable only if the setup call is performed, otherwise `msg.value` is rejected
                     * to avoid stuck value in the contract.
                     *
                     * Emits an {IERC1967-BeaconUpgraded} event.
                     *
                     * CAUTION: Invoking this function has no effect on an instance of {BeaconProxy} since v5, since
                     * it uses an immutable beacon without looking at the value of the ERC-1967 beacon slot for
                     * efficiency.
                     */
                    function upgradeBeaconToAndCall(address newBeacon, bytes memory data) internal {
                        _setBeacon(newBeacon);
                        emit BeaconUpgraded(newBeacon);
                        if (data.length > 0) {
                            Address.functionDelegateCall(IBeacon(newBeacon).implementation(), data);
                        } else {
                            _checkNonPayable();
                        }
                    }
                    /**
                     * @dev Reverts if `msg.value` is not zero. It can be used to avoid `msg.value` stuck in the contract
                     * if an upgrade doesn't perform an initialization call.
                     */
                    function _checkNonPayable() private {
                        if (msg.value > 0) {
                            revert ERC1967NonPayable();
                        }
                    }
                }
                // SPDX-License-Identifier: MIT
                // OpenZeppelin Contracts (last updated v5.0.0) (proxy/beacon/IBeacon.sol)
                pragma solidity ^0.8.20;
                /**
                 * @dev This is the interface that {BeaconProxy} expects of its beacon.
                 */
                interface IBeacon {
                    /**
                     * @dev Must return an address that can be used as a delegate call target.
                     *
                     * {UpgradeableBeacon} will check that this address is a contract.
                     */
                    function implementation() external view returns (address);
                }
                // SPDX-License-Identifier: MIT
                // 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();
                        }
                    }
                }
                // SPDX-License-Identifier: MIT
                // 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 2 of 4: ERC1967Proxy
                // SPDX-License-Identifier: MIT
                // OpenZeppelin Contracts (last updated v5.0.0) (proxy/ERC1967/ERC1967Proxy.sol)
                pragma solidity ^0.8.20;
                import {Proxy} from "../Proxy.sol";
                import {ERC1967Utils} from "./ERC1967Utils.sol";
                /**
                 * @dev This contract implements an upgradeable proxy. It is upgradeable because calls are delegated to an
                 * implementation address that can be changed. This address is stored in storage in the location specified by
                 * https://eips.ethereum.org/EIPS/eip-1967[EIP1967], so that it doesn't conflict with the storage layout of the
                 * implementation behind the proxy.
                 */
                contract ERC1967Proxy is Proxy {
                    /**
                     * @dev Initializes the upgradeable proxy with an initial implementation specified by `implementation`.
                     *
                     * If `_data` is nonempty, it's used as data in a delegate call to `implementation`. This will typically be an
                     * encoded function call, and allows initializing the storage of the proxy like a Solidity constructor.
                     *
                     * Requirements:
                     *
                     * - If `data` is empty, `msg.value` must be zero.
                     */
                    constructor(address implementation, bytes memory _data) payable {
                        ERC1967Utils.upgradeToAndCall(implementation, _data);
                    }
                    /**
                     * @dev Returns the current implementation address.
                     *
                     * TIP: To get this value clients can read directly from the storage slot shown below (specified by EIP1967) using
                     * the https://eth.wiki/json-rpc/API#eth_getstorageat[`eth_getStorageAt`] RPC call.
                     * `0x360894a13ba1a3210667c828492db98dca3e2076cc3735a920a3ca505d382bbc`
                     */
                    function _implementation() internal view virtual override returns (address) {
                        return ERC1967Utils.getImplementation();
                    }
                }
                // SPDX-License-Identifier: MIT
                // OpenZeppelin Contracts (last updated v5.0.0) (proxy/Proxy.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 virtual {
                        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 virtual {
                        _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 virtual {
                        _fallback();
                    }
                }
                // SPDX-License-Identifier: MIT
                // OpenZeppelin Contracts (last updated v5.0.0) (proxy/ERC1967/ERC1967Utils.sol)
                pragma solidity ^0.8.20;
                import {IBeacon} from "../beacon/IBeacon.sol";
                import {Address} from "../../utils/Address.sol";
                import {StorageSlot} from "../../utils/StorageSlot.sol";
                /**
                 * @dev This abstract contract provides getters and event emitting update functions for
                 * https://eips.ethereum.org/EIPS/eip-1967[EIP1967] slots.
                 */
                library ERC1967Utils {
                    // We re-declare ERC-1967 events here because they can't be used directly from IERC1967.
                    // This will be fixed in Solidity 0.8.21. At that point we should remove these events.
                    /**
                     * @dev Emitted when the implementation is upgraded.
                     */
                    event Upgraded(address indexed implementation);
                    /**
                     * @dev Emitted when the admin account has changed.
                     */
                    event AdminChanged(address previousAdmin, address newAdmin);
                    /**
                     * @dev Emitted when the beacon is changed.
                     */
                    event BeaconUpgraded(address indexed beacon);
                    /**
                     * @dev Storage slot with the address of the current implementation.
                     * This is the keccak-256 hash of "eip1967.proxy.implementation" subtracted by 1.
                     */
                    // solhint-disable-next-line private-vars-leading-underscore
                    bytes32 internal constant IMPLEMENTATION_SLOT = 0x360894a13ba1a3210667c828492db98dca3e2076cc3735a920a3ca505d382bbc;
                    /**
                     * @dev The `implementation` of the proxy is invalid.
                     */
                    error ERC1967InvalidImplementation(address implementation);
                    /**
                     * @dev The `admin` of the proxy is invalid.
                     */
                    error ERC1967InvalidAdmin(address admin);
                    /**
                     * @dev The `beacon` of the proxy is invalid.
                     */
                    error ERC1967InvalidBeacon(address beacon);
                    /**
                     * @dev An upgrade function sees `msg.value > 0` that may be lost.
                     */
                    error ERC1967NonPayable();
                    /**
                     * @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 ERC1967InvalidImplementation(newImplementation);
                        }
                        StorageSlot.getAddressSlot(IMPLEMENTATION_SLOT).value = newImplementation;
                    }
                    /**
                     * @dev Performs implementation upgrade with additional setup call if data is nonempty.
                     * This function is payable only if the setup call is performed, otherwise `msg.value` is rejected
                     * to avoid stuck value in the contract.
                     *
                     * Emits an {IERC1967-Upgraded} event.
                     */
                    function upgradeToAndCall(address newImplementation, bytes memory data) internal {
                        _setImplementation(newImplementation);
                        emit Upgraded(newImplementation);
                        if (data.length > 0) {
                            Address.functionDelegateCall(newImplementation, data);
                        } else {
                            _checkNonPayable();
                        }
                    }
                    /**
                     * @dev Storage slot with the admin of the contract.
                     * This is the keccak-256 hash of "eip1967.proxy.admin" subtracted by 1.
                     */
                    // solhint-disable-next-line private-vars-leading-underscore
                    bytes32 internal constant ADMIN_SLOT = 0xb53127684a568b3173ae13b9f8a6016e243e63b6e8ee1178d6a717850b5d6103;
                    /**
                     * @dev Returns the current admin.
                     *
                     * TIP: To get this value clients can read directly from the storage slot shown below (specified by EIP1967) using
                     * the https://eth.wiki/json-rpc/API#eth_getstorageat[`eth_getStorageAt`] RPC call.
                     * `0xb53127684a568b3173ae13b9f8a6016e243e63b6e8ee1178d6a717850b5d6103`
                     */
                    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 {
                        if (newAdmin == address(0)) {
                            revert ERC1967InvalidAdmin(address(0));
                        }
                        StorageSlot.getAddressSlot(ADMIN_SLOT).value = newAdmin;
                    }
                    /**
                     * @dev Changes the admin of the proxy.
                     *
                     * Emits an {IERC1967-AdminChanged} event.
                     */
                    function changeAdmin(address newAdmin) internal {
                        emit AdminChanged(getAdmin(), newAdmin);
                        _setAdmin(newAdmin);
                    }
                    /**
                     * @dev The storage slot of the UpgradeableBeacon contract which defines the implementation for this proxy.
                     * This is the keccak-256 hash of "eip1967.proxy.beacon" subtracted by 1.
                     */
                    // solhint-disable-next-line private-vars-leading-underscore
                    bytes32 internal constant BEACON_SLOT = 0xa3f0ad74e5423aebfd80d3ef4346578335a9a72aeaee59ff6cb3582b35133d50;
                    /**
                     * @dev Returns the current beacon.
                     */
                    function getBeacon() internal view returns (address) {
                        return StorageSlot.getAddressSlot(BEACON_SLOT).value;
                    }
                    /**
                     * @dev Stores a new beacon in the EIP1967 beacon slot.
                     */
                    function _setBeacon(address newBeacon) private {
                        if (newBeacon.code.length == 0) {
                            revert ERC1967InvalidBeacon(newBeacon);
                        }
                        StorageSlot.getAddressSlot(BEACON_SLOT).value = newBeacon;
                        address beaconImplementation = IBeacon(newBeacon).implementation();
                        if (beaconImplementation.code.length == 0) {
                            revert ERC1967InvalidImplementation(beaconImplementation);
                        }
                    }
                    /**
                     * @dev Change the beacon and trigger a setup call if data is nonempty.
                     * This function is payable only if the setup call is performed, otherwise `msg.value` is rejected
                     * to avoid stuck value in the contract.
                     *
                     * Emits an {IERC1967-BeaconUpgraded} event.
                     *
                     * CAUTION: Invoking this function has no effect on an instance of {BeaconProxy} since v5, since
                     * it uses an immutable beacon without looking at the value of the ERC-1967 beacon slot for
                     * efficiency.
                     */
                    function upgradeBeaconToAndCall(address newBeacon, bytes memory data) internal {
                        _setBeacon(newBeacon);
                        emit BeaconUpgraded(newBeacon);
                        if (data.length > 0) {
                            Address.functionDelegateCall(IBeacon(newBeacon).implementation(), data);
                        } else {
                            _checkNonPayable();
                        }
                    }
                    /**
                     * @dev Reverts if `msg.value` is not zero. It can be used to avoid `msg.value` stuck in the contract
                     * if an upgrade doesn't perform an initialization call.
                     */
                    function _checkNonPayable() private {
                        if (msg.value > 0) {
                            revert ERC1967NonPayable();
                        }
                    }
                }
                // SPDX-License-Identifier: MIT
                // OpenZeppelin Contracts (last updated v5.0.0) (proxy/beacon/IBeacon.sol)
                pragma solidity ^0.8.20;
                /**
                 * @dev This is the interface that {BeaconProxy} expects of its beacon.
                 */
                interface IBeacon {
                    /**
                     * @dev Must return an address that can be used as a delegate call target.
                     *
                     * {UpgradeableBeacon} will check that this address is a contract.
                     */
                    function implementation() external view returns (address);
                }
                // SPDX-License-Identifier: MIT
                // 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();
                        }
                    }
                }
                // SPDX-License-Identifier: MIT
                // 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 3 of 4: Staking
                // SPDX-License-Identifier: MIT
                pragma solidity 0.8.23;
                /*
                .____                             ________
                |    |   _____  ___.__. __________\\_____  \\
                |    |   \\__  \\<   |  |/ __ \\_  __ \\_(__  <
                |    |___ / __ \\\\___  \\  ___/|  | \\/       \\
                |_______ (____  / ____|\\___  >__| /______  /
                        \\/    \\/\\/         \\/            \\/
                https://layer3.xyz
                Made with ♥ by Wonderland (https://defi.sucks)
                */
                import {IDistributor} from 'interfaces/IDistributor.sol';
                import {IStaking} from 'interfaces/IStaking.sol';
                import {Ownable2StepUpgradeable} from 'openzeppelin-upgradeable/access/Ownable2StepUpgradeable.sol';
                import {UUPSUpgradeable} from 'openzeppelin-upgradeable/proxy/utils/UUPSUpgradeable.sol';
                import {PausableUpgradeable} from 'openzeppelin-upgradeable/utils/PausableUpgradeable.sol';
                import {IERC20, SafeERC20} from 'openzeppelin/token/ERC20/utils/SafeERC20.sol';
                import {Math} from 'openzeppelin/utils/math/Math.sol';
                import {SafeCast} from 'openzeppelin/utils/math/SafeCast.sol';
                contract Staking is IStaking, Ownable2StepUpgradeable, UUPSUpgradeable, PausableUpgradeable {
                  using SafeERC20 for IERC20;
                  using SafeCast for uint256;
                  using Math for uint256;
                  /// @notice The lockup periods
                  uint256 internal constant _12_MONTHS = 12 * 30 days;
                  uint256 internal constant _18_MONTHS = 18 * 30 days;
                  uint256 internal constant _24_MONTHS = 24 * 30 days;
                  uint256 internal constant _36_MONTHS = 36 * 30 days;
                  /// @notice The base value for calculations
                  uint256 internal constant _BASE = 1e18;
                  /// @inheritdoc IStaking
                  IERC20 public token;
                  /// @inheritdoc IStaking
                  IDistributor public distributor;
                  /// @inheritdoc IStaking
                  uint256 public rewardsDuration;
                  /// @inheritdoc IStaking
                  uint256 public periodFinish;
                  /// @inheritdoc IStaking
                  uint256 public lastUpdateTime;
                  /// @inheritdoc IStaking
                  uint256 public rewardPerSecond;
                  /// @inheritdoc IStaking
                  uint256 public rewardPerShare;
                  /// @inheritdoc IStaking
                  uint256 public totalRewards;
                  /// @inheritdoc IStaking
                  uint256 public totalDeposits;
                  /// @inheritdoc IStaking
                  uint256 public totalWeights;
                  /// @inheritdoc IStaking
                  uint256 public withdrawalPeriod;
                  /// @inheritdoc IStaking
                  mapping(address _user => Staker _staker) public stakers;
                  /// @inheritdoc IStaking
                  mapping(address _user => mapping(uint256 _index => Deposit _deposit)) public deposits;
                  /// @custom:oz-upgrades-unsafe-allow constructor
                  constructor() {
                    _disableInitializers();
                  }
                  function initialize(IERC20 _token, IDistributor _distributor, address _owner) public initializer {
                    token = _token;
                    distributor = _distributor;
                    rewardsDuration = 5 * 12 * 30 days;
                    withdrawalPeriod = 7 days;
                    __Ownable_init(_owner);
                    __Ownable2Step_init();
                    __UUPSUpgradeable_init();
                    __Pausable_init();
                    _pause();
                  }
                  /// @inheritdoc IStaking
                  function stake(uint256 _amount, uint256 _lockupPeriod) external {
                    Deposit memory _deposit = _stake(_amount, _lockupPeriod, msg.sender);
                    emit Staked(msg.sender, _deposit.index, _deposit.amount, _deposit.lockupPeriod, _deposit.unlockAt);
                    // Transfer the tokens to the contract
                    token.safeTransferFrom(msg.sender, address(this), _amount);
                  }
                  /// @inheritdoc IStaking
                  function stake(uint256 _amount, uint256 _lockupPeriod, address _user) external {
                    if (msg.sender != address(distributor)) revert OnlyDistributor();
                    // The distributor will transfer the tokens after calling this function
                    Deposit memory _deposit = _stake(_amount, _lockupPeriod, _user);
                    emit Staked(_user, _deposit.index, _deposit.amount, _deposit.lockupPeriod, _deposit.unlockAt);
                  }
                  /// @inheritdoc IStaking
                  function increaseStake(uint256 _index, uint256 _amount) external {
                    _increaseStake(_index, _amount, msg.sender);
                    emit StakeIncreased(msg.sender, _index, _amount);
                    // Transfer the tokens to the contract
                    token.safeTransferFrom(msg.sender, address(this), _amount);
                  }
                  /// @inheritdoc IStaking
                  function stakeUnlocked(uint256 _index, uint256 _newLockupPeriod) external {
                    Deposit memory _currentDeposit = deposits[msg.sender][_index];
                    uint256 _currentAmount = _currentDeposit.amount;
                    if (_currentAmount == 0) revert InvalidDepositIndex();
                    if (_currentDeposit.unlockAt > block.timestamp) revert DepositLocked();
                    if (_currentDeposit.withdrawAt > 0) revert WithdrawalAlreadyInitiated();
                    if (_currentDeposit.lockupPeriod >= _newLockupPeriod) revert InvalidLockupPeriod();
                    // Close the current stake
                    _decreaseStake(_currentDeposit);
                    // Delete the current staked deposit
                    delete deposits[msg.sender][_index];
                    totalDeposits -= _currentAmount;
                    // Stake using the same amount but with the new lockup period
                    Deposit memory _newDeposit = _stake(_currentAmount, _newLockupPeriod, msg.sender);
                    emit StakedUnlocked(
                      msg.sender, _newDeposit.index, _newDeposit.amount, _newDeposit.lockupPeriod, _newDeposit.unlockAt
                    );
                  }
                  /// @inheritdoc IStaking
                  function getReward() external {
                    Staker storage _staker = _updateReward(msg.sender);
                    uint256 _reward = _staker.pendingRewards;
                    if (_reward > 0) {
                      _staker.pendingRewards = 0;
                      totalRewards -= _reward;
                      token.safeTransfer(msg.sender, _reward);
                      emit RewardPaid(msg.sender, _reward);
                    }
                  }
                  /// @inheritdoc IStaking
                  function getRewardAndStake(uint256 _lockupPeriod) external {
                    Staker storage _staker = _updateReward(msg.sender);
                    uint256 _reward = _staker.pendingRewards;
                    if (_reward > 0) {
                      _staker.pendingRewards = 0;
                      Deposit memory _deposit = _stake(_reward, _lockupPeriod, msg.sender);
                      totalRewards -= _reward;
                      emit ClaimRewardAndStake(msg.sender, _deposit.index, _reward, _lockupPeriod);
                    }
                  }
                  /// @inheritdoc IStaking
                  function getRewardAndIncreaseStake(uint256 _index) external {
                    Staker storage _staker = _updateReward(msg.sender);
                    uint256 _reward = _staker.pendingRewards;
                    if (_reward > 0) {
                      _staker.pendingRewards = 0;
                      _increaseStake(_index, _reward, msg.sender);
                      totalRewards -= _reward;
                      emit ClaimRewardAndIncreaseStake(msg.sender, _index, _reward);
                    }
                  }
                  /// @inheritdoc IStaking
                  function initiateWithdrawal(uint256 _index) external {
                    // Get the Deposit struct
                    Deposit storage _deposit = deposits[msg.sender][_index];
                    if (_deposit.amount == 0) revert InvalidDepositIndex();
                    if (_deposit.lockupPeriod > 0) revert DepositLocked();
                    if (_deposit.withdrawAt > 0) revert WithdrawalAlreadyInitiated();
                    _decreaseStake(_deposit);
                    // Update the withdrawal timestamp
                    _deposit.withdrawAt = (block.timestamp + withdrawalPeriod).toUint40();
                    emit WithdrawalInitiated(msg.sender, _index, _deposit.withdrawAt);
                  }
                  /// @inheritdoc IStaking
                  function cancelWithdrawal(uint256 _index) external {
                    // Get the Deposit struct
                    Deposit storage _deposit = deposits[msg.sender][_index];
                    uint256 _amount = _deposit.amount;
                    if (_deposit.amount == 0) revert InvalidDepositIndex();
                    if (_deposit.withdrawAt == 0) revert WithdrawalNotInitiated();
                    Staker storage _staker = _updateReward(msg.sender);
                    // Because the deposit is unlocked, we're calculating the weight with a lockup period of 0
                    uint256 _weight = _calculateWeight(0, _amount);
                    // Update the total weights and user weight and reset the withdrawal timestamp
                    totalWeights += _weight;
                    _staker.weight += _weight.toUint128();
                    _deposit.withdrawAt = 0;
                    emit WithdrawalCancelled(msg.sender, _index);
                  }
                  /// @inheritdoc IStaking
                  function withdraw(uint256 _index) external {
                    // Get the Deposit struct
                    Deposit memory _deposit = deposits[msg.sender][_index];
                    if (_deposit.amount == 0) revert InvalidDepositIndex();
                    if (_deposit.lockupPeriod > 0) {
                      if (_deposit.unlockAt > block.timestamp) revert DepositLocked();
                      _decreaseStake(_deposit);
                    } else if (withdrawalPeriod == 0 && _deposit.withdrawAt == 0) {
                      _decreaseStake(_deposit);
                    } else {
                      // Non-lockup deposits can be withdrawn only after a withdrawal period
                      if (_deposit.withdrawAt > block.timestamp) revert DepositNotWithdrawable();
                      if (_deposit.withdrawAt == 0) revert WithdrawalNotInitiated();
                      // Not updating weights because the deposit was already removed from the total in `initiateWithdrawal`
                    }
                    // Update the total deposits
                    totalDeposits -= _deposit.amount;
                    // Delete the deposit
                    delete deposits[msg.sender][_index];
                    // Transfer the tokens to the user
                    token.safeTransfer(msg.sender, _deposit.amount);
                    emit Withdrawn(msg.sender, _index, _deposit.amount);
                  }
                  /// @inheritdoc IStaking
                  function emergencyWithdraw(uint256 _amount) external onlyOwner {
                    if (_amount == 0) revert ZeroAmount();
                    // Withdraw either the requested amount or the remaining balance
                    uint256 _remainingBalance = token.balanceOf(address(this));
                    uint256 _withdrawalAmount = _amount > _remainingBalance ? _remainingBalance : _amount;
                    token.safeTransfer(owner(), _withdrawalAmount);
                    emit EmergencyWithdrawn(owner(), _withdrawalAmount);
                  }
                  /// @inheritdoc IStaking
                  function setRewardAmount(uint256 _reward) external onlyOwner {
                    uint256 _currentBalance = token.balanceOf(address(this));
                    if (_reward > _currentBalance - totalDeposits - totalRewards) revert InsufficientBalance();
                    _updateReward(address(0));
                    if (block.timestamp >= periodFinish) {
                      rewardPerSecond = _reward / rewardsDuration;
                    } else {
                      uint256 _remaining = periodFinish - block.timestamp;
                      uint256 _leftover = _remaining * rewardPerSecond;
                      rewardPerSecond = (_reward + _leftover) / rewardsDuration;
                    }
                    lastUpdateTime = block.timestamp;
                    periodFinish = block.timestamp + rewardsDuration;
                    totalRewards += _reward;
                    emit RewardAdded(_reward);
                  }
                  /// @inheritdoc IStaking
                  function setRewardsDuration(uint256 _rewardsDuration) external onlyOwner {
                    if (periodFinish > block.timestamp) revert PeriodNotFinished();
                    uint256 _oldRewardsDuration = rewardsDuration;
                    rewardsDuration = _rewardsDuration;
                    emit RewardsDurationUpdated(_oldRewardsDuration, _rewardsDuration);
                  }
                  /// @inheritdoc IStaking
                  function setWithdrawalPeriod(uint256 _withdrawalPeriod) external onlyOwner {
                    uint256 _oldWithdrawalPeriod = withdrawalPeriod;
                    withdrawalPeriod = _withdrawalPeriod;
                    emit WithdrawalPeriodUpdated(_oldWithdrawalPeriod, _withdrawalPeriod);
                  }
                  /// @inheritdoc IStaking
                  function pause() external onlyOwner {
                    _pause();
                  }
                  /// @inheritdoc IStaking
                  function unpause() external onlyOwner {
                    _unpause();
                  }
                  /// @inheritdoc IStaking
                  function setDistributorAddress(IDistributor _distributor) external onlyOwner {
                    IDistributor _oldDistributor = distributor;
                    distributor = _distributor;
                    emit DistributorUpdated(_oldDistributor, _distributor);
                  }
                  /// @inheritdoc IStaking
                  function collectDust(IERC20 _token, uint256 _amount) external onlyOwner {
                    if (_token == token || address(_token) == address(0)) revert InvalidToken();
                    if (_amount == 0) revert ZeroAmount();
                    address _owner = owner();
                    _token.safeTransfer(_owner, _amount);
                    emit DustCollected(_owner, _token, _amount);
                  }
                  /// @inheritdoc IStaking
                  function calculateAPY(uint256 _amount, uint256 _lockupPeriod) external view returns (uint256 _apy) {
                    uint256 _weight = _calculateWeight(_lockupPeriod, _amount);
                    uint256 _rewardPerYear = rewardPerSecond * _12_MONTHS * _BASE * 100;
                    _apy = Math.mulDiv(_weight, _rewardPerYear, (totalWeights + _weight) * _amount);
                  }
                  /// @inheritdoc IStaking
                  function calculateAPY(address _user, uint256 _index) external view returns (uint256 _apy) {
                    Deposit memory _deposit = deposits[_user][_index];
                    uint256 _weight = _calculateWeight(_deposit.lockupPeriod, _deposit.amount);
                    uint256 _rewardPerYear = rewardPerSecond * _12_MONTHS * _BASE * 100;
                    _apy = Math.mulDiv(_weight, _rewardPerYear, _deposit.amount * totalWeights);
                  }
                  /// @inheritdoc IStaking
                  function listDeposits(
                    address _user,
                    uint256 _startFrom,
                    uint256 _batchSize
                  ) external view returns (Deposit[] memory _list) {
                    uint256 _totalDeposits = stakers[_user].depositCount;
                    // Return an empty array if non-existent user or no deposits
                    if (_startFrom > _totalDeposits) {
                      return _list;
                    }
                    if (_batchSize > _totalDeposits - _startFrom) {
                      _batchSize = _totalDeposits - _startFrom;
                    }
                    _list = new Deposit[](_batchSize);
                    uint256 _index;
                    while (_index < _batchSize) {
                      _list[_index] = deposits[_user][_startFrom + _index];
                      ++_index;
                    }
                  }
                  /// @inheritdoc IStaking
                  function pendingRewards(address _user) public view returns (uint256 _pendingRewards) {
                    Staker storage _staker = stakers[_user];
                    // Staker's pendingRewards already accounts for rewards calculated prior to the last snapshot
                    // We take the difference between the current rate and the one pendingRewards was calculated at
                    // And work out the amount of rewards accumulated after the snapshot
                    uint256 _rateDifferenceSinceSnapshot = _calculatedRewardPerShare() - _staker.rewardPerShareSnapshot;
                    uint256 _rewardsSinceSnapshot = _staker.weight * _rateDifferenceSinceSnapshot / _BASE;
                    _pendingRewards = _staker.pendingRewards + _rewardsSinceSnapshot;
                  }
                  /**
                   * @notice Stakes the provided amount of tokens and increases the total weight
                   * @param _amount The amount of tokens
                   * @param _lockupPeriod The lockup period
                   * @param _user The address of the user
                   */
                  function _stake(uint256 _amount, uint256 _lockupPeriod, address _user) internal returns (Deposit memory _deposit) {
                    if (_amount == 0) revert ZeroAmount();
                    Staker storage _staker = _updateReward(_user);
                    // Calculate the user weight, taking into account the lockup period multiplier
                    uint256 _weight = _calculateWeight(_lockupPeriod, _amount);
                    if (_weight == 0) revert ZeroWeight();
                    // Update the total weights and user weight
                    totalWeights += _weight;
                    totalDeposits += _amount;
                    _staker.weight += _weight.toUint128();
                    // Get the last index and increment it
                    uint256 _lastIndex = _staker.depositCount++;
                    uint256 _unlockAt = block.timestamp + _lockupPeriod;
                    _deposit = Deposit({
                      amount: _amount.toUint128(),
                      unlockAt: _unlockAt.toUint40(),
                      lockupPeriod: _lockupPeriod.toUint32(),
                      index: _lastIndex.toUint16(),
                      withdrawAt: 0
                    });
                    // Create a new Deposit struct
                    deposits[_user][_lastIndex] = _deposit;
                  }
                  /**
                   * @notice Updates the reward rate and the staker's info
                   * @param _user The address of the user
                   * @return _staker The staker struct
                   */
                  function _updateReward(address _user) internal whenNotPaused returns (Staker storage _staker) {
                    uint256 _rewardPerShare = _calculatedRewardPerShare();
                    if (_rewardPerShare == 0 || _rewardPerShare > rewardPerShare) {
                      rewardPerShare = _rewardPerShare;
                      lastUpdateTime = _lastTimeRewardApplicable();
                    }
                    _staker = stakers[_user];
                    if (_user != address(0)) {
                      _staker.pendingRewards = pendingRewards(_user).toUint128();
                      _staker.rewardPerShareSnapshot = rewardPerShare.toUint128();
                    }
                  }
                  /**
                   * @notice Adds the specified amount of tokens the specified deposit
                   * @param _index The index of the deposit
                   * @param _amount The amount of tokens
                   * @param _user The address of the user
                   * @dev Only unlocked deposits can be increased
                   */
                  function _increaseStake(uint256 _index, uint256 _amount, address _user) internal {
                    Deposit storage _deposit = deposits[_user][_index];
                    if (_deposit.amount == 0) revert InvalidDepositIndex();
                    if (_deposit.lockupPeriod > 0) revert CannotIncreaseLockedStake();
                    if (_deposit.withdrawAt > 0) revert WithdrawalAlreadyInitiated();
                    // Because the deposit is unlocked, we're calculating the weight with a lockup period of 0
                    uint256 _weight = _calculateWeight(0, _amount);
                    // Update the total weights and user weight
                    Staker storage _staker = _updateReward(_user);
                    totalWeights += _weight;
                    totalDeposits += _amount;
                    _staker.weight += _weight.toUint128();
                    _deposit.amount += _amount.toUint128();
                  }
                  /**
                   * @notice Decreases the stake of the specified deposit
                   * @param _deposit The deposit to decrease
                   */
                  function _decreaseStake(Deposit memory _deposit) internal {
                    Staker storage _staker = _updateReward(msg.sender);
                    // Calculate the user weight
                    uint256 _weight = _calculateWeight(_deposit.lockupPeriod, _deposit.amount);
                    // Avoid rounding issues where `weight(a) + weight(b) <= weight(a+b)` that may cause underflows
                    _weight = _weight <= _staker.weight ? _weight : _staker.weight;
                    // Update the total weights and user weight
                    totalWeights -= _weight;
                    _staker.weight -= _weight.toUint128();
                  }
                  /**
                   * @notice Returns either the current time or the end of the rewards period, whichever is earlier
                   * @return _lastTimeReward The timestamp of the last time rewards were applicable
                   */
                  function _lastTimeRewardApplicable() internal view returns (uint256 _lastTimeReward) {
                    _lastTimeReward = block.timestamp < periodFinish ? block.timestamp : periodFinish;
                  }
                  /**
                   * @notice Calculates the reward per share
                   * @return _rewardPerShare The reward per share
                   */
                  function _calculatedRewardPerShare() internal view returns (uint256 _rewardPerShare) {
                    if (totalWeights == 0) {
                      return rewardPerShare;
                    }
                    uint256 _timeSinceLastUpdate = _lastTimeRewardApplicable() - lastUpdateTime;
                    _rewardPerShare = rewardPerShare + _timeSinceLastUpdate * rewardPerSecond * _BASE / totalWeights;
                  }
                  /**
                   * @notice Applies the lockup period multiplier to get the deposit's weight
                   * @param _lockupPeriod The lockup period
                   * @param _amount The amount of tokens
                   * @return _weight The weight of the deposit
                   */
                  function _calculateWeight(uint256 _lockupPeriod, uint256 _amount) internal pure returns (uint256 _weight) {
                    if (_lockupPeriod == 0) {
                      _weight = _amount * 250 / 1000;
                    } else if (_lockupPeriod == _12_MONTHS) {
                      _weight = _amount * 500 / 1000;
                    } else if (_lockupPeriod == _18_MONTHS) {
                      _weight = _amount * 625 / 1000;
                    } else if (_lockupPeriod == _24_MONTHS) {
                      _weight = _amount * 750 / 1000;
                    } else if (_lockupPeriod == _36_MONTHS) {
                      _weight = _amount;
                    } else {
                      revert InvalidLockupPeriod();
                    }
                  }
                  /**
                   * @notice Checks if the contract upgrade is authorized
                   * @param _newImplementation The address of the new implementation
                   * @dev Only owner should be allowed to perform upgrades
                   */
                  function _authorizeUpgrade(address _newImplementation) internal override onlyOwner {}
                }
                // SPDX-License-Identifier: MIT
                pragma solidity 0.8.23;
                import {IStaking} from 'interfaces/IStaking.sol';
                import {IERC20} from 'openzeppelin/token/ERC20/IERC20.sol';
                /**
                 * @title Distributor Contract
                 * @author Wonderland (https://defi.sucks)
                 * @notice Distributes tokens to users based on a merkle root and a signature
                 */
                interface IDistributor {
                  /*///////////////////////////////////////////////////////////////
                                            EVENTS
                  ///////////////////////////////////////////////////////////////*/
                  /**
                   * @notice Emitted when a user claims their tokens
                   * @param _account The account that claimed the tokens
                   * @param _amount The amount of tokens claimed
                   */
                  event Claimed(address indexed _account, uint256 _amount);
                  /**
                   * @notice Emitted when a user claims and stakes their tokens
                   * @param _account The account that claimed and staked the tokens
                   * @param _amount The amount of tokens claimed and staked
                   * @param _lockupPeriod The lockup period for the deposit
                   * @param _timestamp The timestamp at which the tokens were claimed and staked
                   */
                  event ClaimedAndStaked(address indexed _account, uint256 _amount, uint256 _lockupPeriod, uint256 _timestamp);
                  /**
                   * @notice Emitted when the owner withdraws tokens from the contract
                   * @param _owner The owner that withdrew the tokens
                   * @param _amount The amount of tokens withdrawn
                   */
                  event EmergencyWithdrawn(address indexed _owner, uint256 _amount);
                  /**
                   * @notice Emitted when the signer is updated by the owner
                   * @param _oldSigner The old signer address
                   * @param _newSigner The new signer address
                   */
                  event SignerUpdated(address indexed _oldSigner, address indexed _newSigner);
                  /**
                   * @notice Emitted when the owner collects dust tokens from the contract
                   * @param _owner The owner that collected the dust tokens
                   * @param _token The token address
                   * @param _amount The amount of tokens collected
                   */
                  event DustCollected(address indexed _owner, IERC20 indexed _token, uint256 _amount);
                  /*///////////////////////////////////////////////////////////////
                                            ERRORS
                  ///////////////////////////////////////////////////////////////*/
                  /**
                   * @notice Throws if the input amount is zero
                   */
                  error ZeroAmount();
                  /**
                   * @notice Throws if the user has already claimed their tokens
                   */
                  error AlreadyClaimed();
                  /**
                   * @notice Throws if the recovered signer is different from the expected signer
                   */
                  error InvalidSigner();
                  /**
                   * @notice Throws if the merkle verification fails
                   */
                  error InvalidProof();
                  /**
                   * @notice Throws if the new signer address is invalid
                   */
                  error InvalidNewSigner();
                  /**
                   * @notice Throws if the input token is invalid
                   */
                  error InvalidToken();
                  /*///////////////////////////////////////////////////////////////
                                            LOGIC
                  ///////////////////////////////////////////////////////////////*/
                  /**
                   * @notice Verifies eligibility and transfers the tokens to the caller
                   * @param _amount The amount of tokens to claim
                   * @param _merkleProof The merkle proof of the claim
                   * @param _signature The signature provided by the UI
                   */
                  function claim(uint256 _amount, bytes32[] calldata _merkleProof, bytes calldata _signature) external;
                  /**
                   * @notice Verifies eligibility and stakes the claimed tokens in the contract
                   * @param _amount The amount of tokens to claim
                   * @param _merkleProof The merkle proof for the claim
                   * @param _signature The signature for verification of the claim data
                   * @param _lockupPeriod The period of time to lock the tokens for
                   */
                  function claimAndStake(
                    uint256 _amount,
                    bytes32[] calldata _merkleProof,
                    bytes calldata _signature,
                    uint32 _lockupPeriod
                  ) external;
                  /**
                   * @notice Sends any remaining tokens to the owner
                   * @dev Only callable by the owner
                   * @dev If the specified amount exceeds the available balance, the entire balance is withdrawn
                   * @param _amount The amount of tokens to withdraw
                   */
                  function emergencyWithdraw(uint256 _amount) external;
                  /**
                   * @notice Updates the signer address
                   * @dev Only callable by the owner
                   * @param _newSigner The new signer address
                   */
                  function updateSigner(address _newSigner) external;
                  /**
                   * @notice Collects dust tokens from the contract
                   * @dev Only the owner can call this function
                   * @param _token The token to collect
                   * @param _amount The amount of tokens to collect
                   */
                  function collectDust(IERC20 _token, uint256 _amount) external;
                  /*///////////////////////////////////////////////////////////////
                                            VARIABLES
                  ///////////////////////////////////////////////////////////////*/
                  /**
                   * @notice The root of the merkle tree
                   * @return _merkleRoot The root of the merkle tree
                   */
                  // solhint-disable-next-line func-name-mixedcase
                  function MERKLE_ROOT() external view returns (bytes32 _merkleRoot);
                  /**
                   * @notice The token being distributed
                   * @return _token The address of the token
                   */
                  // solhint-disable-next-line func-name-mixedcase
                  function TOKEN() external view returns (IERC20 _token);
                  /**
                   * @notice The address of the staking contract
                   * @return _staking The staking contract
                   */
                  // solhint-disable-next-line func-name-mixedcase
                  function STAKING() external view returns (IStaking _staking);
                  /**
                   * @notice The address of the signer
                   * @return _signer The address of the signer
                   */
                  function signer() external view returns (address _signer);
                  /**
                   * @notice Returns whether the user has claimed their tokens
                   * @param _user The address of the user
                   * @return _claimed Whether the user has claimed their tokens
                   */
                  function hasClaimed(address _user) external view returns (bool _claimed);
                }
                // SPDX-License-Identifier: MIT
                pragma solidity 0.8.23;
                import {IDistributor} from './IDistributor.sol';
                import {IERC20} from 'openzeppelin/token/ERC20/utils/SafeERC20.sol';
                interface IStaking {
                  /*///////////////////////////////////////////////////////////////
                                            STRUCTS
                  ///////////////////////////////////////////////////////////////*/
                  /**
                   * @notice Deposit struct
                   * @param amount The amount of tokens deposited
                   * @param unlockAt The timestamp when the tokens can be unlocked
                   * @param lockupPeriod The period the tokens are locked for to get the bonus
                   * @param index The index of the deposit
                   * @param withdrawAt The timestamp when the tokens can be withdrawn (after withdrawal period is over)
                   */
                  struct Deposit {
                    uint128 amount;
                    uint40 unlockAt;
                    uint32 lockupPeriod;
                    uint16 index;
                    uint40 withdrawAt;
                  }
                  /**
                   * @notice Staker struct
                   * @param weight The combined weight of the staker's deposits
                   * @param depositCount The number of deposits the staker has
                   * @param rewardPerShareSnapshot The amount of rewards per share as seen at the last update
                   * @param pendingRewards The amount of rewards available to be claimed by the staker
                   */
                  struct Staker {
                    uint128 weight;
                    uint128 depositCount;
                    uint128 rewardPerShareSnapshot;
                    uint128 pendingRewards;
                  }
                  /*///////////////////////////////////////////////////////////////
                                                EVENTS
                  ///////////////////////////////////////////////////////////////*/
                  /**
                   * @notice Emitted when the user stakes tokens
                   * @param _user The user that staked the tokens
                   * @param _index The index of the deposit
                   * @param _amount The amount of tokens staked
                   * @param _lockupPeriod The lockup period
                   * @param _unlockAt The timestamp when the tokens can be withdrawn
                   */
                  event Staked(
                    address indexed _user, uint256 indexed _index, uint256 _amount, uint256 _lockupPeriod, uint256 _unlockAt
                  );
                  /**
                   * @notice Emitted when the user stakes again an unlocked deposit
                   * @param _user The user that staked the tokens
                   * @param _index The index of the deposit
                   * @param _amount The amount of tokens staked
                   * @param _lockupPeriod The lockup period
                   * @param _unlockAt The timestamp when the tokens can be withdrawn
                   */
                  event StakedUnlocked(
                    address indexed _user, uint256 indexed _index, uint256 _amount, uint256 _lockupPeriod, uint256 _unlockAt
                  );
                  /**
                   * @notice Emitted when the user adds tokens to an existing stake
                   * @param _user The user that staked the tokens
                   * @param _index The index of the deposit
                   * @param _amount The amount of tokens added
                   */
                  event StakeIncreased(address indexed _user, uint256 indexed _index, uint256 _amount);
                  /**
                   * @notice Emitted when the user claims pending rewards and creates a new deposit
                   * @param _user The user that staked the rewards
                   * @param _index The index of the created stake
                   * @param _amount The amount of tokens staked
                   * @param _lockupPeriod The lockup period
                   */
                  event ClaimRewardAndStake(address indexed _user, uint256 indexed _index, uint256 _amount, uint256 _lockupPeriod);
                  /**
                   * @notice Emitted when the user claims pending rewards and adds the tokens to an existing stake
                   * @param _user The user that staked the tokens
                   * @param _index The index of the deposit
                   * @param _amount The amount of tokens added
                   */
                  event ClaimRewardAndIncreaseStake(address indexed _user, uint256 indexed _index, uint256 _amount);
                  /**
                   * @notice Emitted when the user initiates a withdrawal
                   * @param _user The user that initiated the withdrawal
                   * @param _index The index of the deposit
                   * @param _withdrawAt The end of the withdrawal period
                   */
                  event WithdrawalInitiated(address indexed _user, uint256 indexed _index, uint256 _withdrawAt);
                  /**
                   * @notice Emitted when the user cancels the withdrawal
                   * @param _user The user that cancelled the withdrawal
                   * @param _index The index of the deposit
                   */
                  event WithdrawalCancelled(address indexed _user, uint256 indexed _index);
                  /**
                   * @notice Emitted when the user withdraws tokens
                   * @param _user The user that withdrew the tokens
                   * @param _index The index of the deposit
                   * @param _amount The amount of tokens withdrawn
                   */
                  event Withdrawn(address indexed _user, uint256 indexed _index, uint256 _amount);
                  /**
                   * @notice Emitted when the user claims their rewards
                   * @param _user The user that claimed the rewards
                   * @param _amount The amount of rewards claimed
                   */
                  event RewardPaid(address indexed _user, uint256 _amount);
                  /**
                   * @notice Emitted when the reward amount is added
                   * @param _reward The new reward amount
                   */
                  event RewardAdded(uint256 _reward);
                  /**
                   * @notice Emitted when the rewards duration is updated
                   * @param _oldRewardsDuration The previous rewards duration
                   * @param _rewardsDuration The new rewards duration
                   */
                  event RewardsDurationUpdated(uint256 _oldRewardsDuration, uint256 _rewardsDuration);
                  /**
                   * @notice Emitted when the dust tokens are collected
                   * @param _owner The owner that collected the dust tokens
                   * @param _token The token address
                   * @param _amount The amount of tokens collected
                   */
                  event DustCollected(address indexed _owner, IERC20 _token, uint256 _amount);
                  /**
                   * @notice Emitted when the staked deposits and the rewards are retracted by the owner
                   * @param _owner The owner that withdrew the tokens
                   * @param _amount The amount of tokens retracted
                   */
                  event EmergencyWithdrawn(address indexed _owner, uint256 _amount);
                  /**
                   * @notice Emitted when the withdrawal period is updated
                   * @param _oldWithdrawalPeriod The previous withdrawal period
                   * @param _withdrawalPeriod The new withdrawal period
                   */
                  event WithdrawalPeriodUpdated(uint256 _oldWithdrawalPeriod, uint256 _withdrawalPeriod);
                  /**
                   * @notice Emitted when the distributor address is updated
                   * @param _oldDistributor The previous distributor
                   * @param _distributor The new distributor
                   */
                  event DistributorUpdated(IDistributor _oldDistributor, IDistributor _distributor);
                  /*///////////////////////////////////////////////////////////////
                                                ERRORS
                  ///////////////////////////////////////////////////////////////*/
                  /**
                   * @notice Throws if the provided amount is zero
                   */
                  error ZeroAmount();
                  /**
                   * @notice Throws if the provided weight is zero
                   */
                  error ZeroWeight();
                  /**
                   * @notice Throws if the deposit with the given index does not exist
                   */
                  error InvalidDepositIndex();
                  /**
                   * @notice Throws if trying to withdraw a locked deposit
                   */
                  error DepositLocked();
                  /**
                   * @notice Throws if the lockup period is invalid
                   */
                  error InvalidLockupPeriod();
                  /**
                   * @notice Throws if the staking contract has insufficient balance to pay the rewards at the given rate
                   */
                  error InsufficientBalance();
                  /**
                   * @notice Throws if the period is not finished
                   */
                  error PeriodNotFinished();
                  /**
                   * @notice Throws if the token is invalid
                   */
                  error InvalidToken();
                  /**
                   * @notice Throws if the caller is not the distributor
                   */
                  error OnlyDistributor();
                  /**
                   * @notice Throws if the caller is trying to add tokens to a locked deposit
                   */
                  error CannotIncreaseLockedStake();
                  /**
                   * @notice Throws if the withdrawal is not initiated while trying to withdraw
                   */
                  error WithdrawalNotInitiated();
                  /**
                   * @notice Throws if the caller is trying to initiate a withdrawal of a deposit that's already in the withdrawal process
                   */
                  error WithdrawalAlreadyInitiated();
                  /**
                   * @notice Throws if the withdrawal period is not over while trying to withdraw
                   */
                  error DepositNotWithdrawable();
                  /*///////////////////////////////////////////////////////////////
                                            VARIABLES
                  ///////////////////////////////////////////////////////////////*/
                  /**
                   * @notice The address of the token contract
                   * @return _token The token contract
                   */
                  function token() external view returns (IERC20 _token);
                  /**
                   * @notice The address of the distributor contract
                   * @return _distributor The distributor contract
                   */
                  function distributor() external view returns (IDistributor _distributor);
                  /**
                   * @notice The time period in seconds over which rewards are distributed
                   * @return _rewardsDuration The rewards duration
                   */
                  function rewardsDuration() external view returns (uint256 _rewardsDuration);
                  /**
                   * @notice Returns the timestamp of the last block at which the rewards will be distributed
                   * @return _periodFinish The end of the rewards period
                   */
                  function periodFinish() external view returns (uint256 _periodFinish);
                  /**
                   * @notice The amount of rewards given to the stakers every second
                   * @return _rewardPerSecond The amount of reward per second
                   */
                  function rewardPerSecond() external view returns (uint256 _rewardPerSecond);
                  /**
                   * @notice The time the reward per second was updated
                   * @return _lastUpdateTime The last time the reward per second was updated
                   */
                  function lastUpdateTime() external view returns (uint256 _lastUpdateTime);
                  /**
                   * @notice The total weight of the deposits in the contract
                   * @return _totalWeights The total weight of the deposits
                   */
                  function totalWeights() external view returns (uint256 _totalWeights);
                  /**
                   * @notice The total amount of tokens staked in the contract
                   * @return _totalDeposits The amount of tokens staked in the contract
                   */
                  function totalDeposits() external view returns (uint256 _totalDeposits);
                  /**
                   * @notice The amount of tokens intended to be distributed as rewards
                   * @return _totalRewards The total reward amount
                   */
                  function totalRewards() external view returns (uint256 _totalRewards);
                  /**
                   * @notice The reward generated per staker's share of the pool
                   * @return _rewardPerShare The reward per share
                   */
                  function rewardPerShare() external view returns (uint256 _rewardPerShare);
                  /**
                   * @notice The time period in seconds after which the staker can withdraw their tokens
                   * @dev This is only needed for non-lockup deposits
                   * @return _withdrawalPeriod The withdrawal period
                   */
                  function withdrawalPeriod() external view returns (uint256 _withdrawalPeriod);
                  /**
                   * @notice Provides information about a given staker
                   * @param _user The staker's address
                   * @return _weight The total weight of the staker's deposits
                   * @return _depositCount The number of deposits the staker has
                   * @return _rewardPerShareSnapshot The amount of rewards per share as seen at the last update
                   * @return _pendingRewards The amount of rewards pending to be claimed by the staker
                   */
                  function stakers(address _user)
                    external
                    view
                    returns (uint128 _weight, uint128 _depositCount, uint128 _rewardPerShareSnapshot, uint128 _pendingRewards);
                  /**
                   * @notice Returns a user's deposit with the given index
                   * @param _user The address of the user
                   * @param _depositIndex The index of the deposit
                   * @return _amount The amount of tokens deposited
                   * @return _unlockAt The timestamp when the tokens can be withdrawn
                   * @return _lockupPeriod The period the tokens are locked to get the bonus
                   * @return _index The index of the deposit
                   */
                  function deposits(
                    address _user,
                    uint256 _depositIndex
                  ) external view returns (uint128 _amount, uint40 _unlockAt, uint32 _lockupPeriod, uint16 _index, uint40 _withdrawAt);
                  /*///////////////////////////////////////////////////////////////
                                        EXTERNAL FUNCTIONS
                  ///////////////////////////////////////////////////////////////*/
                  /**
                   * @notice The list of deposits of the user
                   * @param _user The address of the user
                   * @param _startFrom The index to start from
                   * @param _batchSize The size of the batch
                   * @return _list The list of deposits
                   */
                  function listDeposits(
                    address _user,
                    uint256 _startFrom,
                    uint256 _batchSize
                  ) external view returns (Deposit[] memory _list);
                  /**
                   * @notice Calculates APY based on the given amount and the lockup period
                   * @param _amount The amount of tokens to stake
                   * @param _lockupPeriod The lockup period
                   * @return _apy The APY the staker would get
                   */
                  function calculateAPY(uint256 _amount, uint256 _lockupPeriod) external view returns (uint256 _apy);
                  /**
                   * @notice Returns the APY of an existing deposit
                   * @param _user The staker address
                   * @param _index The index of the deposit
                   * @return _apy The APY the deposit is generating
                   */
                  function calculateAPY(address _user, uint256 _index) external view returns (uint256 _apy);
                  /**
                   * @notice The amount of pending rewards the staker has
                   * @param _user The address of the user
                   * @return _pendingRewards The amount of the rewards ready to be claimed
                   */
                  function pendingRewards(address _user) external view returns (uint256 _pendingRewards);
                  /**
                   * @notice The stake function
                   * @param _amount The amount of tokens
                   * @param _lockupPeriod The lockup period, must be either 0 or one of the allowed lockup periods
                   */
                  function stake(uint256 _amount, uint256 _lockupPeriod) external;
                  /**
                   * @notice The stake function for the distributor, allowing to stake on behalf of another address
                   * @param _amount The amount of tokens
                   * @param _lockupPeriod The lockup period, must be either 0 or one of the allowed lockup periods
                   * @param _user The address of the user to stake for
                   */
                  function stake(uint256 _amount, uint256 _lockupPeriod, address _user) external;
                  /**
                   * @notice Add the provided amount of tokens to an existing stake
                   * @param _amount The amount of tokens to add
                   * @param _index The index of the deposit to increase
                   */
                  function increaseStake(uint256 _index, uint256 _amount) external;
                  /**
                   * @notice Stakes the unlocked deposit with a higher lockup period
                   * @param _index The index of the deposit
                   * @param _newLockupPeriod The new lockup period
                   */
                  function stakeUnlocked(uint256 _index, uint256 _newLockupPeriod) external;
                  /**
                   * @notice Claims pending rewards and adds them to an existing stake
                   * @param _index The index of the deposit to increase
                   */
                  function getRewardAndIncreaseStake(uint256 _index) external;
                  /**
                   * @notice Initiates a withdrawal of the deposit
                   * @dev The tokens will be locked for the withdrawal period
                   * @dev Only needed for non-lockup deposits
                   * @param _index The index of the deposit to withdraw
                   */
                  function initiateWithdrawal(uint256 _index) external;
                  /**
                   * @notice Cancels the withdrawal of the deposit
                   * @param _index The index of the deposit to cancel the withdrawal
                   */
                  function cancelWithdrawal(uint256 _index) external;
                  /**
                   * @notice The withdraw function
                   * @param _index The index of the deposit to withdraw
                   */
                  function withdraw(uint256 _index) external;
                  /**
                   * @notice Transfers pending rewards to the caller
                   */
                  function getReward() external;
                  /**
                   * @notice Claims the pending rewards and creates an unlocked deposit from them
                   * @param _lockupPeriod The lockup period, must be either 0 or one of the allowed lockup periods
                   */
                  function getRewardAndStake(uint256 _lockupPeriod) external;
                  /**
                   * @notice Updates the total amount of rewards for the stakers
                   * @param _reward The new reward amount
                   */
                  function setRewardAmount(uint256 _reward) external;
                  /**
                   * @notice Updates the rewards duration
                   * @param _rewardsDuration The new rewards duration
                   */
                  function setRewardsDuration(uint256 _rewardsDuration) external;
                  /**
                   * @notice Updates the distributor address
                   * @param _distributor The new distributor
                   */
                  function setDistributorAddress(IDistributor _distributor) external;
                  /**
                   * @notice Sends any dust tokens to the owner
                   * @param _token The token address
                   * @param _amount The amount of tokens to withdraw
                   */
                  function collectDust(IERC20 _token, uint256 _amount) external;
                  /**
                   * @notice An emergency function which sends the specified number of tokens to the owner
                   * @param _amount The amount of tokens to withdraw
                   */
                  function emergencyWithdraw(uint256 _amount) external;
                  /**
                   * @notice Updates the withdrawal period
                   * @param _withdrawalPeriod The new withdrawal period
                   */
                  function setWithdrawalPeriod(uint256 _withdrawalPeriod) external;
                  /**
                   * @notice Pauses the staking and withdrawals
                   */
                  function pause() external;
                  /**
                   * @notice Unpauses the staking and withdrawals
                   */
                  function unpause() external;
                }
                // SPDX-License-Identifier: MIT
                // OpenZeppelin Contracts (last updated v5.0.0) (access/Ownable2Step.sol)
                pragma solidity ^0.8.20;
                import {OwnableUpgradeable} from "./OwnableUpgradeable.sol";
                import {Initializable} from "../proxy/utils/Initializable.sol";
                /**
                 * @dev Contract module which provides access control mechanism, where
                 * there is an account (an owner) that can be granted exclusive access to
                 * specific functions.
                 *
                 * The initial owner is specified at deployment time in the constructor for `Ownable`. This
                 * can later be changed with {transferOwnership} and {acceptOwnership}.
                 *
                 * This module is used through inheritance. It will make available all functions
                 * from parent (Ownable).
                 */
                abstract contract Ownable2StepUpgradeable is Initializable, OwnableUpgradeable {
                    /// @custom:storage-location erc7201:openzeppelin.storage.Ownable2Step
                    struct Ownable2StepStorage {
                        address _pendingOwner;
                    }
                    // keccak256(abi.encode(uint256(keccak256("openzeppelin.storage.Ownable2Step")) - 1)) & ~bytes32(uint256(0xff))
                    bytes32 private constant Ownable2StepStorageLocation = 0x237e158222e3e6968b72b9db0d8043aacf074ad9f650f0d1606b4d82ee432c00;
                    function _getOwnable2StepStorage() private pure returns (Ownable2StepStorage storage $) {
                        assembly {
                            $.slot := Ownable2StepStorageLocation
                        }
                    }
                    event OwnershipTransferStarted(address indexed previousOwner, address indexed newOwner);
                    function __Ownable2Step_init() internal onlyInitializing {
                    }
                    function __Ownable2Step_init_unchained() internal onlyInitializing {
                    }
                    /**
                     * @dev Returns the address of the pending owner.
                     */
                    function pendingOwner() public view virtual returns (address) {
                        Ownable2StepStorage storage $ = _getOwnable2StepStorage();
                        return $._pendingOwner;
                    }
                    /**
                     * @dev Starts the ownership transfer of the contract to a new account. Replaces the pending transfer if there is one.
                     * Can only be called by the current owner.
                     */
                    function transferOwnership(address newOwner) public virtual override onlyOwner {
                        Ownable2StepStorage storage $ = _getOwnable2StepStorage();
                        $._pendingOwner = newOwner;
                        emit OwnershipTransferStarted(owner(), newOwner);
                    }
                    /**
                     * @dev Transfers ownership of the contract to a new account (`newOwner`) and deletes any pending owner.
                     * Internal function without access restriction.
                     */
                    function _transferOwnership(address newOwner) internal virtual override {
                        Ownable2StepStorage storage $ = _getOwnable2StepStorage();
                        delete $._pendingOwner;
                        super._transferOwnership(newOwner);
                    }
                    /**
                     * @dev The new owner accepts the ownership transfer.
                     */
                    function acceptOwnership() public virtual {
                        address sender = _msgSender();
                        if (pendingOwner() != sender) {
                            revert OwnableUnauthorizedAccount(sender);
                        }
                        _transferOwnership(sender);
                    }
                }
                // SPDX-License-Identifier: MIT
                // OpenZeppelin Contracts (last updated v5.0.0) (proxy/utils/UUPSUpgradeable.sol)
                pragma solidity ^0.8.20;
                import {IERC1822Proxiable} from "@openzeppelin/contracts/interfaces/draft-IERC1822.sol";
                import {ERC1967Utils} from "@openzeppelin/contracts/proxy/ERC1967/ERC1967Utils.sol";
                import {Initializable} from "./Initializable.sol";
                /**
                 * @dev An upgradeability mechanism designed for UUPS proxies. The functions included here can perform an upgrade of an
                 * {ERC1967Proxy}, when this contract is set as the implementation behind such a proxy.
                 *
                 * A security mechanism ensures that an upgrade does not turn off upgradeability accidentally, although this risk is
                 * reinstated if the upgrade retains upgradeability but removes the security mechanism, e.g. by replacing
                 * `UUPSUpgradeable` with a custom implementation of upgrades.
                 *
                 * The {_authorizeUpgrade} function must be overridden to include access restriction to the upgrade mechanism.
                 */
                abstract contract UUPSUpgradeable is Initializable, IERC1822Proxiable {
                    /// @custom:oz-upgrades-unsafe-allow state-variable-immutable
                    address private immutable __self = address(this);
                    /**
                     * @dev The version of the upgrade interface of the contract. If this getter is missing, both `upgradeTo(address)`
                     * and `upgradeToAndCall(address,bytes)` are present, and `upgradeTo` must be used if no function should be called,
                     * while `upgradeToAndCall` will invoke the `receive` function if the second argument is the empty byte string.
                     * If the getter returns `"5.0.0"`, only `upgradeToAndCall(address,bytes)` is present, and the second argument must
                     * be the empty byte string if no function should be called, making it impossible to invoke the `receive` function
                     * during an upgrade.
                     */
                    string public constant UPGRADE_INTERFACE_VERSION = "5.0.0";
                    /**
                     * @dev The call is from an unauthorized context.
                     */
                    error UUPSUnauthorizedCallContext();
                    /**
                     * @dev The storage `slot` is unsupported as a UUID.
                     */
                    error UUPSUnsupportedProxiableUUID(bytes32 slot);
                    /**
                     * @dev Check that the execution is being performed through a delegatecall call and that the execution context is
                     * a proxy contract with an implementation (as defined in ERC1967) pointing to self. This should only be the case
                     * for UUPS and transparent proxies that are using the current contract as their implementation. Execution of a
                     * function through ERC1167 minimal proxies (clones) would not normally pass this test, but is not guaranteed to
                     * fail.
                     */
                    modifier onlyProxy() {
                        _checkProxy();
                        _;
                    }
                    /**
                     * @dev Check that the execution is not being performed through a delegate call. This allows a function to be
                     * callable on the implementing contract but not through proxies.
                     */
                    modifier notDelegated() {
                        _checkNotDelegated();
                        _;
                    }
                    function __UUPSUpgradeable_init() internal onlyInitializing {
                    }
                    function __UUPSUpgradeable_init_unchained() internal onlyInitializing {
                    }
                    /**
                     * @dev Implementation of the ERC1822 {proxiableUUID} function. This returns the storage slot used by the
                     * implementation. It is used to validate the implementation's compatibility when performing an upgrade.
                     *
                     * IMPORTANT: A proxy pointing at a proxiable contract should not be considered proxiable itself, because this risks
                     * bricking a proxy that upgrades to it, by delegating to itself until out of gas. Thus it is critical that this
                     * function revert if invoked through a proxy. This is guaranteed by the `notDelegated` modifier.
                     */
                    function proxiableUUID() external view virtual notDelegated returns (bytes32) {
                        return ERC1967Utils.IMPLEMENTATION_SLOT;
                    }
                    /**
                     * @dev Upgrade the implementation of the proxy to `newImplementation`, and subsequently execute the function call
                     * encoded in `data`.
                     *
                     * Calls {_authorizeUpgrade}.
                     *
                     * Emits an {Upgraded} event.
                     *
                     * @custom:oz-upgrades-unsafe-allow-reachable delegatecall
                     */
                    function upgradeToAndCall(address newImplementation, bytes memory data) public payable virtual onlyProxy {
                        _authorizeUpgrade(newImplementation);
                        _upgradeToAndCallUUPS(newImplementation, data);
                    }
                    /**
                     * @dev Reverts if the execution is not performed via delegatecall or the execution
                     * context is not of a proxy with an ERC1967-compliant implementation pointing to self.
                     * See {_onlyProxy}.
                     */
                    function _checkProxy() internal view virtual {
                        if (
                            address(this) == __self || // Must be called through delegatecall
                            ERC1967Utils.getImplementation() != __self // Must be called through an active proxy
                        ) {
                            revert UUPSUnauthorizedCallContext();
                        }
                    }
                    /**
                     * @dev Reverts if the execution is performed via delegatecall.
                     * See {notDelegated}.
                     */
                    function _checkNotDelegated() internal view virtual {
                        if (address(this) != __self) {
                            // Must not be called through delegatecall
                            revert UUPSUnauthorizedCallContext();
                        }
                    }
                    /**
                     * @dev Function that should revert when `msg.sender` is not authorized to upgrade the contract. Called by
                     * {upgradeToAndCall}.
                     *
                     * Normally, this function will use an xref:access.adoc[access control] modifier such as {Ownable-onlyOwner}.
                     *
                     * ```solidity
                     * function _authorizeUpgrade(address) internal onlyOwner {}
                     * ```
                     */
                    function _authorizeUpgrade(address newImplementation) internal virtual;
                    /**
                     * @dev Performs an implementation upgrade with a security check for UUPS proxies, and additional setup call.
                     *
                     * As a security check, {proxiableUUID} is invoked in the new implementation, and the return value
                     * is expected to be the implementation slot in ERC1967.
                     *
                     * Emits an {IERC1967-Upgraded} event.
                     */
                    function _upgradeToAndCallUUPS(address newImplementation, bytes memory data) private {
                        try IERC1822Proxiable(newImplementation).proxiableUUID() returns (bytes32 slot) {
                            if (slot != ERC1967Utils.IMPLEMENTATION_SLOT) {
                                revert UUPSUnsupportedProxiableUUID(slot);
                            }
                            ERC1967Utils.upgradeToAndCall(newImplementation, data);
                        } catch {
                            // The implementation is not UUPS
                            revert ERC1967Utils.ERC1967InvalidImplementation(newImplementation);
                        }
                    }
                }
                // 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.0) (token/ERC20/utils/SafeERC20.sol)
                pragma solidity ^0.8.20;
                import {IERC20} from "../IERC20.sol";
                import {IERC20Permit} from "../extensions/IERC20Permit.sol";
                import {Address} from "../../../utils/Address.sol";
                /**
                 * @title SafeERC20
                 * @dev Wrappers around ERC20 operations that throw on failure (when the token
                 * contract returns false). Tokens that return no value (and instead revert or
                 * throw on failure) are also supported, non-reverting calls are assumed to be
                 * successful.
                 * To use this library you can add a `using SafeERC20 for IERC20;` statement to your contract,
                 * which allows you to call the safe operations as `token.safeTransfer(...)`, etc.
                 */
                library SafeERC20 {
                    using Address for address;
                    /**
                     * @dev An operation with an ERC20 token failed.
                     */
                    error SafeERC20FailedOperation(address token);
                    /**
                     * @dev Indicates a failed `decreaseAllowance` request.
                     */
                    error SafeERC20FailedDecreaseAllowance(address spender, uint256 currentAllowance, uint256 requestedDecrease);
                    /**
                     * @dev Transfer `value` amount of `token` from the calling contract to `to`. If `token` returns no value,
                     * non-reverting calls are assumed to be successful.
                     */
                    function safeTransfer(IERC20 token, address to, uint256 value) internal {
                        _callOptionalReturn(token, abi.encodeCall(token.transfer, (to, value)));
                    }
                    /**
                     * @dev Transfer `value` amount of `token` from `from` to `to`, spending the approval given by `from` to the
                     * calling contract. If `token` returns no value, non-reverting calls are assumed to be successful.
                     */
                    function safeTransferFrom(IERC20 token, address from, address to, uint256 value) internal {
                        _callOptionalReturn(token, abi.encodeCall(token.transferFrom, (from, to, value)));
                    }
                    /**
                     * @dev Increase the calling contract's allowance toward `spender` by `value`. If `token` returns no value,
                     * non-reverting calls are assumed to be successful.
                     */
                    function safeIncreaseAllowance(IERC20 token, address spender, uint256 value) internal {
                        uint256 oldAllowance = token.allowance(address(this), spender);
                        forceApprove(token, spender, oldAllowance + value);
                    }
                    /**
                     * @dev Decrease the calling contract's allowance toward `spender` by `requestedDecrease`. If `token` returns no
                     * value, non-reverting calls are assumed to be successful.
                     */
                    function safeDecreaseAllowance(IERC20 token, address spender, uint256 requestedDecrease) internal {
                        unchecked {
                            uint256 currentAllowance = token.allowance(address(this), spender);
                            if (currentAllowance < requestedDecrease) {
                                revert SafeERC20FailedDecreaseAllowance(spender, currentAllowance, requestedDecrease);
                            }
                            forceApprove(token, spender, currentAllowance - requestedDecrease);
                        }
                    }
                    /**
                     * @dev Set the calling contract's allowance toward `spender` to `value`. If `token` returns no value,
                     * non-reverting calls are assumed to be successful. Meant to be used with tokens that require the approval
                     * to be set to zero before setting it to a non-zero value, such as USDT.
                     */
                    function forceApprove(IERC20 token, address spender, uint256 value) internal {
                        bytes memory approvalCall = abi.encodeCall(token.approve, (spender, value));
                        if (!_callOptionalReturnBool(token, approvalCall)) {
                            _callOptionalReturn(token, abi.encodeCall(token.approve, (spender, 0)));
                            _callOptionalReturn(token, approvalCall);
                        }
                    }
                    /**
                     * @dev Imitates a Solidity high-level call (i.e. a regular function call to a contract), relaxing the requirement
                     * on the return value: the return value is optional (but if data is returned, it must not be false).
                     * @param token The token targeted by the call.
                     * @param data The call data (encoded using abi.encode or one of its variants).
                     */
                    function _callOptionalReturn(IERC20 token, bytes memory data) private {
                        // We need to perform a low level call here, to bypass Solidity's return data size checking mechanism, since
                        // we're implementing it ourselves. We use {Address-functionCall} to perform this call, which verifies that
                        // the target address contains contract code and also asserts for success in the low-level call.
                        bytes memory returndata = address(token).functionCall(data);
                        if (returndata.length != 0 && !abi.decode(returndata, (bool))) {
                            revert SafeERC20FailedOperation(address(token));
                        }
                    }
                    /**
                     * @dev Imitates a Solidity high-level call (i.e. a regular function call to a contract), relaxing the requirement
                     * on the return value: the return value is optional (but if data is returned, it must not be false).
                     * @param token The token targeted by the call.
                     * @param data The call data (encoded using abi.encode or one of its variants).
                     *
                     * This is a variant of {_callOptionalReturn} that silents catches all reverts and returns a bool instead.
                     */
                    function _callOptionalReturnBool(IERC20 token, bytes memory data) private returns (bool) {
                        // We need to perform a low level call here, to bypass Solidity's return data size checking mechanism, since
                        // we're implementing it ourselves. We cannot use {Address-functionCall} here since this should return false
                        // and not revert is the subcall reverts.
                        (bool success, bytes memory returndata) = address(token).call(data);
                        return success && (returndata.length == 0 || abi.decode(returndata, (bool))) && address(token).code.length > 0;
                    }
                }
                // 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;
                    }
                }
                // SPDX-License-Identifier: MIT
                // OpenZeppelin Contracts (last updated v5.0.0) (utils/math/SafeCast.sol)
                // This file was procedurally generated from scripts/generate/templates/SafeCast.js.
                pragma solidity ^0.8.20;
                /**
                 * @dev Wrappers over Solidity's uintXX/intXX casting operators with added overflow
                 * checks.
                 *
                 * Downcasting from uint256/int256 in Solidity does not revert on overflow. This can
                 * easily result in undesired exploitation or bugs, since developers usually
                 * assume that overflows raise errors. `SafeCast` restores this intuition by
                 * reverting the transaction when such an operation overflows.
                 *
                 * Using this library instead of the unchecked operations eliminates an entire
                 * class of bugs, so it's recommended to use it always.
                 */
                library SafeCast {
                    /**
                     * @dev Value doesn't fit in an uint of `bits` size.
                     */
                    error SafeCastOverflowedUintDowncast(uint8 bits, uint256 value);
                    /**
                     * @dev An int value doesn't fit in an uint of `bits` size.
                     */
                    error SafeCastOverflowedIntToUint(int256 value);
                    /**
                     * @dev Value doesn't fit in an int of `bits` size.
                     */
                    error SafeCastOverflowedIntDowncast(uint8 bits, int256 value);
                    /**
                     * @dev An uint value doesn't fit in an int of `bits` size.
                     */
                    error SafeCastOverflowedUintToInt(uint256 value);
                    /**
                     * @dev Returns the downcasted uint248 from uint256, reverting on
                     * overflow (when the input is greater than largest uint248).
                     *
                     * Counterpart to Solidity's `uint248` operator.
                     *
                     * Requirements:
                     *
                     * - input must fit into 248 bits
                     */
                    function toUint248(uint256 value) internal pure returns (uint248) {
                        if (value > type(uint248).max) {
                            revert SafeCastOverflowedUintDowncast(248, value);
                        }
                        return uint248(value);
                    }
                    /**
                     * @dev Returns the downcasted uint240 from uint256, reverting on
                     * overflow (when the input is greater than largest uint240).
                     *
                     * Counterpart to Solidity's `uint240` operator.
                     *
                     * Requirements:
                     *
                     * - input must fit into 240 bits
                     */
                    function toUint240(uint256 value) internal pure returns (uint240) {
                        if (value > type(uint240).max) {
                            revert SafeCastOverflowedUintDowncast(240, value);
                        }
                        return uint240(value);
                    }
                    /**
                     * @dev Returns the downcasted uint232 from uint256, reverting on
                     * overflow (when the input is greater than largest uint232).
                     *
                     * Counterpart to Solidity's `uint232` operator.
                     *
                     * Requirements:
                     *
                     * - input must fit into 232 bits
                     */
                    function toUint232(uint256 value) internal pure returns (uint232) {
                        if (value > type(uint232).max) {
                            revert SafeCastOverflowedUintDowncast(232, value);
                        }
                        return uint232(value);
                    }
                    /**
                     * @dev Returns the downcasted uint224 from uint256, reverting on
                     * overflow (when the input is greater than largest uint224).
                     *
                     * Counterpart to Solidity's `uint224` operator.
                     *
                     * Requirements:
                     *
                     * - input must fit into 224 bits
                     */
                    function toUint224(uint256 value) internal pure returns (uint224) {
                        if (value > type(uint224).max) {
                            revert SafeCastOverflowedUintDowncast(224, value);
                        }
                        return uint224(value);
                    }
                    /**
                     * @dev Returns the downcasted uint216 from uint256, reverting on
                     * overflow (when the input is greater than largest uint216).
                     *
                     * Counterpart to Solidity's `uint216` operator.
                     *
                     * Requirements:
                     *
                     * - input must fit into 216 bits
                     */
                    function toUint216(uint256 value) internal pure returns (uint216) {
                        if (value > type(uint216).max) {
                            revert SafeCastOverflowedUintDowncast(216, value);
                        }
                        return uint216(value);
                    }
                    /**
                     * @dev Returns the downcasted uint208 from uint256, reverting on
                     * overflow (when the input is greater than largest uint208).
                     *
                     * Counterpart to Solidity's `uint208` operator.
                     *
                     * Requirements:
                     *
                     * - input must fit into 208 bits
                     */
                    function toUint208(uint256 value) internal pure returns (uint208) {
                        if (value > type(uint208).max) {
                            revert SafeCastOverflowedUintDowncast(208, value);
                        }
                        return uint208(value);
                    }
                    /**
                     * @dev Returns the downcasted uint200 from uint256, reverting on
                     * overflow (when the input is greater than largest uint200).
                     *
                     * Counterpart to Solidity's `uint200` operator.
                     *
                     * Requirements:
                     *
                     * - input must fit into 200 bits
                     */
                    function toUint200(uint256 value) internal pure returns (uint200) {
                        if (value > type(uint200).max) {
                            revert SafeCastOverflowedUintDowncast(200, value);
                        }
                        return uint200(value);
                    }
                    /**
                     * @dev Returns the downcasted uint192 from uint256, reverting on
                     * overflow (when the input is greater than largest uint192).
                     *
                     * Counterpart to Solidity's `uint192` operator.
                     *
                     * Requirements:
                     *
                     * - input must fit into 192 bits
                     */
                    function toUint192(uint256 value) internal pure returns (uint192) {
                        if (value > type(uint192).max) {
                            revert SafeCastOverflowedUintDowncast(192, value);
                        }
                        return uint192(value);
                    }
                    /**
                     * @dev Returns the downcasted uint184 from uint256, reverting on
                     * overflow (when the input is greater than largest uint184).
                     *
                     * Counterpart to Solidity's `uint184` operator.
                     *
                     * Requirements:
                     *
                     * - input must fit into 184 bits
                     */
                    function toUint184(uint256 value) internal pure returns (uint184) {
                        if (value > type(uint184).max) {
                            revert SafeCastOverflowedUintDowncast(184, value);
                        }
                        return uint184(value);
                    }
                    /**
                     * @dev Returns the downcasted uint176 from uint256, reverting on
                     * overflow (when the input is greater than largest uint176).
                     *
                     * Counterpart to Solidity's `uint176` operator.
                     *
                     * Requirements:
                     *
                     * - input must fit into 176 bits
                     */
                    function toUint176(uint256 value) internal pure returns (uint176) {
                        if (value > type(uint176).max) {
                            revert SafeCastOverflowedUintDowncast(176, value);
                        }
                        return uint176(value);
                    }
                    /**
                     * @dev Returns the downcasted uint168 from uint256, reverting on
                     * overflow (when the input is greater than largest uint168).
                     *
                     * Counterpart to Solidity's `uint168` operator.
                     *
                     * Requirements:
                     *
                     * - input must fit into 168 bits
                     */
                    function toUint168(uint256 value) internal pure returns (uint168) {
                        if (value > type(uint168).max) {
                            revert SafeCastOverflowedUintDowncast(168, value);
                        }
                        return uint168(value);
                    }
                    /**
                     * @dev Returns the downcasted uint160 from uint256, reverting on
                     * overflow (when the input is greater than largest uint160).
                     *
                     * Counterpart to Solidity's `uint160` operator.
                     *
                     * Requirements:
                     *
                     * - input must fit into 160 bits
                     */
                    function toUint160(uint256 value) internal pure returns (uint160) {
                        if (value > type(uint160).max) {
                            revert SafeCastOverflowedUintDowncast(160, value);
                        }
                        return uint160(value);
                    }
                    /**
                     * @dev Returns the downcasted uint152 from uint256, reverting on
                     * overflow (when the input is greater than largest uint152).
                     *
                     * Counterpart to Solidity's `uint152` operator.
                     *
                     * Requirements:
                     *
                     * - input must fit into 152 bits
                     */
                    function toUint152(uint256 value) internal pure returns (uint152) {
                        if (value > type(uint152).max) {
                            revert SafeCastOverflowedUintDowncast(152, value);
                        }
                        return uint152(value);
                    }
                    /**
                     * @dev Returns the downcasted uint144 from uint256, reverting on
                     * overflow (when the input is greater than largest uint144).
                     *
                     * Counterpart to Solidity's `uint144` operator.
                     *
                     * Requirements:
                     *
                     * - input must fit into 144 bits
                     */
                    function toUint144(uint256 value) internal pure returns (uint144) {
                        if (value > type(uint144).max) {
                            revert SafeCastOverflowedUintDowncast(144, value);
                        }
                        return uint144(value);
                    }
                    /**
                     * @dev Returns the downcasted uint136 from uint256, reverting on
                     * overflow (when the input is greater than largest uint136).
                     *
                     * Counterpart to Solidity's `uint136` operator.
                     *
                     * Requirements:
                     *
                     * - input must fit into 136 bits
                     */
                    function toUint136(uint256 value) internal pure returns (uint136) {
                        if (value > type(uint136).max) {
                            revert SafeCastOverflowedUintDowncast(136, value);
                        }
                        return uint136(value);
                    }
                    /**
                     * @dev Returns the downcasted uint128 from uint256, reverting on
                     * overflow (when the input is greater than largest uint128).
                     *
                     * Counterpart to Solidity's `uint128` operator.
                     *
                     * Requirements:
                     *
                     * - input must fit into 128 bits
                     */
                    function toUint128(uint256 value) internal pure returns (uint128) {
                        if (value > type(uint128).max) {
                            revert SafeCastOverflowedUintDowncast(128, value);
                        }
                        return uint128(value);
                    }
                    /**
                     * @dev Returns the downcasted uint120 from uint256, reverting on
                     * overflow (when the input is greater than largest uint120).
                     *
                     * Counterpart to Solidity's `uint120` operator.
                     *
                     * Requirements:
                     *
                     * - input must fit into 120 bits
                     */
                    function toUint120(uint256 value) internal pure returns (uint120) {
                        if (value > type(uint120).max) {
                            revert SafeCastOverflowedUintDowncast(120, value);
                        }
                        return uint120(value);
                    }
                    /**
                     * @dev Returns the downcasted uint112 from uint256, reverting on
                     * overflow (when the input is greater than largest uint112).
                     *
                     * Counterpart to Solidity's `uint112` operator.
                     *
                     * Requirements:
                     *
                     * - input must fit into 112 bits
                     */
                    function toUint112(uint256 value) internal pure returns (uint112) {
                        if (value > type(uint112).max) {
                            revert SafeCastOverflowedUintDowncast(112, value);
                        }
                        return uint112(value);
                    }
                    /**
                     * @dev Returns the downcasted uint104 from uint256, reverting on
                     * overflow (when the input is greater than largest uint104).
                     *
                     * Counterpart to Solidity's `uint104` operator.
                     *
                     * Requirements:
                     *
                     * - input must fit into 104 bits
                     */
                    function toUint104(uint256 value) internal pure returns (uint104) {
                        if (value > type(uint104).max) {
                            revert SafeCastOverflowedUintDowncast(104, value);
                        }
                        return uint104(value);
                    }
                    /**
                     * @dev Returns the downcasted uint96 from uint256, reverting on
                     * overflow (when the input is greater than largest uint96).
                     *
                     * Counterpart to Solidity's `uint96` operator.
                     *
                     * Requirements:
                     *
                     * - input must fit into 96 bits
                     */
                    function toUint96(uint256 value) internal pure returns (uint96) {
                        if (value > type(uint96).max) {
                            revert SafeCastOverflowedUintDowncast(96, value);
                        }
                        return uint96(value);
                    }
                    /**
                     * @dev Returns the downcasted uint88 from uint256, reverting on
                     * overflow (when the input is greater than largest uint88).
                     *
                     * Counterpart to Solidity's `uint88` operator.
                     *
                     * Requirements:
                     *
                     * - input must fit into 88 bits
                     */
                    function toUint88(uint256 value) internal pure returns (uint88) {
                        if (value > type(uint88).max) {
                            revert SafeCastOverflowedUintDowncast(88, value);
                        }
                        return uint88(value);
                    }
                    /**
                     * @dev Returns the downcasted uint80 from uint256, reverting on
                     * overflow (when the input is greater than largest uint80).
                     *
                     * Counterpart to Solidity's `uint80` operator.
                     *
                     * Requirements:
                     *
                     * - input must fit into 80 bits
                     */
                    function toUint80(uint256 value) internal pure returns (uint80) {
                        if (value > type(uint80).max) {
                            revert SafeCastOverflowedUintDowncast(80, value);
                        }
                        return uint80(value);
                    }
                    /**
                     * @dev Returns the downcasted uint72 from uint256, reverting on
                     * overflow (when the input is greater than largest uint72).
                     *
                     * Counterpart to Solidity's `uint72` operator.
                     *
                     * Requirements:
                     *
                     * - input must fit into 72 bits
                     */
                    function toUint72(uint256 value) internal pure returns (uint72) {
                        if (value > type(uint72).max) {
                            revert SafeCastOverflowedUintDowncast(72, value);
                        }
                        return uint72(value);
                    }
                    /**
                     * @dev Returns the downcasted uint64 from uint256, reverting on
                     * overflow (when the input is greater than largest uint64).
                     *
                     * Counterpart to Solidity's `uint64` operator.
                     *
                     * Requirements:
                     *
                     * - input must fit into 64 bits
                     */
                    function toUint64(uint256 value) internal pure returns (uint64) {
                        if (value > type(uint64).max) {
                            revert SafeCastOverflowedUintDowncast(64, value);
                        }
                        return uint64(value);
                    }
                    /**
                     * @dev Returns the downcasted uint56 from uint256, reverting on
                     * overflow (when the input is greater than largest uint56).
                     *
                     * Counterpart to Solidity's `uint56` operator.
                     *
                     * Requirements:
                     *
                     * - input must fit into 56 bits
                     */
                    function toUint56(uint256 value) internal pure returns (uint56) {
                        if (value > type(uint56).max) {
                            revert SafeCastOverflowedUintDowncast(56, value);
                        }
                        return uint56(value);
                    }
                    /**
                     * @dev Returns the downcasted uint48 from uint256, reverting on
                     * overflow (when the input is greater than largest uint48).
                     *
                     * Counterpart to Solidity's `uint48` operator.
                     *
                     * Requirements:
                     *
                     * - input must fit into 48 bits
                     */
                    function toUint48(uint256 value) internal pure returns (uint48) {
                        if (value > type(uint48).max) {
                            revert SafeCastOverflowedUintDowncast(48, value);
                        }
                        return uint48(value);
                    }
                    /**
                     * @dev Returns the downcasted uint40 from uint256, reverting on
                     * overflow (when the input is greater than largest uint40).
                     *
                     * Counterpart to Solidity's `uint40` operator.
                     *
                     * Requirements:
                     *
                     * - input must fit into 40 bits
                     */
                    function toUint40(uint256 value) internal pure returns (uint40) {
                        if (value > type(uint40).max) {
                            revert SafeCastOverflowedUintDowncast(40, value);
                        }
                        return uint40(value);
                    }
                    /**
                     * @dev Returns the downcasted uint32 from uint256, reverting on
                     * overflow (when the input is greater than largest uint32).
                     *
                     * Counterpart to Solidity's `uint32` operator.
                     *
                     * Requirements:
                     *
                     * - input must fit into 32 bits
                     */
                    function toUint32(uint256 value) internal pure returns (uint32) {
                        if (value > type(uint32).max) {
                            revert SafeCastOverflowedUintDowncast(32, value);
                        }
                        return uint32(value);
                    }
                    /**
                     * @dev Returns the downcasted uint24 from uint256, reverting on
                     * overflow (when the input is greater than largest uint24).
                     *
                     * Counterpart to Solidity's `uint24` operator.
                     *
                     * Requirements:
                     *
                     * - input must fit into 24 bits
                     */
                    function toUint24(uint256 value) internal pure returns (uint24) {
                        if (value > type(uint24).max) {
                            revert SafeCastOverflowedUintDowncast(24, value);
                        }
                        return uint24(value);
                    }
                    /**
                     * @dev Returns the downcasted uint16 from uint256, reverting on
                     * overflow (when the input is greater than largest uint16).
                     *
                     * Counterpart to Solidity's `uint16` operator.
                     *
                     * Requirements:
                     *
                     * - input must fit into 16 bits
                     */
                    function toUint16(uint256 value) internal pure returns (uint16) {
                        if (value > type(uint16).max) {
                            revert SafeCastOverflowedUintDowncast(16, value);
                        }
                        return uint16(value);
                    }
                    /**
                     * @dev Returns the downcasted uint8 from uint256, reverting on
                     * overflow (when the input is greater than largest uint8).
                     *
                     * Counterpart to Solidity's `uint8` operator.
                     *
                     * Requirements:
                     *
                     * - input must fit into 8 bits
                     */
                    function toUint8(uint256 value) internal pure returns (uint8) {
                        if (value > type(uint8).max) {
                            revert SafeCastOverflowedUintDowncast(8, value);
                        }
                        return uint8(value);
                    }
                    /**
                     * @dev Converts a signed int256 into an unsigned uint256.
                     *
                     * Requirements:
                     *
                     * - input must be greater than or equal to 0.
                     */
                    function toUint256(int256 value) internal pure returns (uint256) {
                        if (value < 0) {
                            revert SafeCastOverflowedIntToUint(value);
                        }
                        return uint256(value);
                    }
                    /**
                     * @dev Returns the downcasted int248 from int256, reverting on
                     * overflow (when the input is less than smallest int248 or
                     * greater than largest int248).
                     *
                     * Counterpart to Solidity's `int248` operator.
                     *
                     * Requirements:
                     *
                     * - input must fit into 248 bits
                     */
                    function toInt248(int256 value) internal pure returns (int248 downcasted) {
                        downcasted = int248(value);
                        if (downcasted != value) {
                            revert SafeCastOverflowedIntDowncast(248, value);
                        }
                    }
                    /**
                     * @dev Returns the downcasted int240 from int256, reverting on
                     * overflow (when the input is less than smallest int240 or
                     * greater than largest int240).
                     *
                     * Counterpart to Solidity's `int240` operator.
                     *
                     * Requirements:
                     *
                     * - input must fit into 240 bits
                     */
                    function toInt240(int256 value) internal pure returns (int240 downcasted) {
                        downcasted = int240(value);
                        if (downcasted != value) {
                            revert SafeCastOverflowedIntDowncast(240, value);
                        }
                    }
                    /**
                     * @dev Returns the downcasted int232 from int256, reverting on
                     * overflow (when the input is less than smallest int232 or
                     * greater than largest int232).
                     *
                     * Counterpart to Solidity's `int232` operator.
                     *
                     * Requirements:
                     *
                     * - input must fit into 232 bits
                     */
                    function toInt232(int256 value) internal pure returns (int232 downcasted) {
                        downcasted = int232(value);
                        if (downcasted != value) {
                            revert SafeCastOverflowedIntDowncast(232, value);
                        }
                    }
                    /**
                     * @dev Returns the downcasted int224 from int256, reverting on
                     * overflow (when the input is less than smallest int224 or
                     * greater than largest int224).
                     *
                     * Counterpart to Solidity's `int224` operator.
                     *
                     * Requirements:
                     *
                     * - input must fit into 224 bits
                     */
                    function toInt224(int256 value) internal pure returns (int224 downcasted) {
                        downcasted = int224(value);
                        if (downcasted != value) {
                            revert SafeCastOverflowedIntDowncast(224, value);
                        }
                    }
                    /**
                     * @dev Returns the downcasted int216 from int256, reverting on
                     * overflow (when the input is less than smallest int216 or
                     * greater than largest int216).
                     *
                     * Counterpart to Solidity's `int216` operator.
                     *
                     * Requirements:
                     *
                     * - input must fit into 216 bits
                     */
                    function toInt216(int256 value) internal pure returns (int216 downcasted) {
                        downcasted = int216(value);
                        if (downcasted != value) {
                            revert SafeCastOverflowedIntDowncast(216, value);
                        }
                    }
                    /**
                     * @dev Returns the downcasted int208 from int256, reverting on
                     * overflow (when the input is less than smallest int208 or
                     * greater than largest int208).
                     *
                     * Counterpart to Solidity's `int208` operator.
                     *
                     * Requirements:
                     *
                     * - input must fit into 208 bits
                     */
                    function toInt208(int256 value) internal pure returns (int208 downcasted) {
                        downcasted = int208(value);
                        if (downcasted != value) {
                            revert SafeCastOverflowedIntDowncast(208, value);
                        }
                    }
                    /**
                     * @dev Returns the downcasted int200 from int256, reverting on
                     * overflow (when the input is less than smallest int200 or
                     * greater than largest int200).
                     *
                     * Counterpart to Solidity's `int200` operator.
                     *
                     * Requirements:
                     *
                     * - input must fit into 200 bits
                     */
                    function toInt200(int256 value) internal pure returns (int200 downcasted) {
                        downcasted = int200(value);
                        if (downcasted != value) {
                            revert SafeCastOverflowedIntDowncast(200, value);
                        }
                    }
                    /**
                     * @dev Returns the downcasted int192 from int256, reverting on
                     * overflow (when the input is less than smallest int192 or
                     * greater than largest int192).
                     *
                     * Counterpart to Solidity's `int192` operator.
                     *
                     * Requirements:
                     *
                     * - input must fit into 192 bits
                     */
                    function toInt192(int256 value) internal pure returns (int192 downcasted) {
                        downcasted = int192(value);
                        if (downcasted != value) {
                            revert SafeCastOverflowedIntDowncast(192, value);
                        }
                    }
                    /**
                     * @dev Returns the downcasted int184 from int256, reverting on
                     * overflow (when the input is less than smallest int184 or
                     * greater than largest int184).
                     *
                     * Counterpart to Solidity's `int184` operator.
                     *
                     * Requirements:
                     *
                     * - input must fit into 184 bits
                     */
                    function toInt184(int256 value) internal pure returns (int184 downcasted) {
                        downcasted = int184(value);
                        if (downcasted != value) {
                            revert SafeCastOverflowedIntDowncast(184, value);
                        }
                    }
                    /**
                     * @dev Returns the downcasted int176 from int256, reverting on
                     * overflow (when the input is less than smallest int176 or
                     * greater than largest int176).
                     *
                     * Counterpart to Solidity's `int176` operator.
                     *
                     * Requirements:
                     *
                     * - input must fit into 176 bits
                     */
                    function toInt176(int256 value) internal pure returns (int176 downcasted) {
                        downcasted = int176(value);
                        if (downcasted != value) {
                            revert SafeCastOverflowedIntDowncast(176, value);
                        }
                    }
                    /**
                     * @dev Returns the downcasted int168 from int256, reverting on
                     * overflow (when the input is less than smallest int168 or
                     * greater than largest int168).
                     *
                     * Counterpart to Solidity's `int168` operator.
                     *
                     * Requirements:
                     *
                     * - input must fit into 168 bits
                     */
                    function toInt168(int256 value) internal pure returns (int168 downcasted) {
                        downcasted = int168(value);
                        if (downcasted != value) {
                            revert SafeCastOverflowedIntDowncast(168, value);
                        }
                    }
                    /**
                     * @dev Returns the downcasted int160 from int256, reverting on
                     * overflow (when the input is less than smallest int160 or
                     * greater than largest int160).
                     *
                     * Counterpart to Solidity's `int160` operator.
                     *
                     * Requirements:
                     *
                     * - input must fit into 160 bits
                     */
                    function toInt160(int256 value) internal pure returns (int160 downcasted) {
                        downcasted = int160(value);
                        if (downcasted != value) {
                            revert SafeCastOverflowedIntDowncast(160, value);
                        }
                    }
                    /**
                     * @dev Returns the downcasted int152 from int256, reverting on
                     * overflow (when the input is less than smallest int152 or
                     * greater than largest int152).
                     *
                     * Counterpart to Solidity's `int152` operator.
                     *
                     * Requirements:
                     *
                     * - input must fit into 152 bits
                     */
                    function toInt152(int256 value) internal pure returns (int152 downcasted) {
                        downcasted = int152(value);
                        if (downcasted != value) {
                            revert SafeCastOverflowedIntDowncast(152, value);
                        }
                    }
                    /**
                     * @dev Returns the downcasted int144 from int256, reverting on
                     * overflow (when the input is less than smallest int144 or
                     * greater than largest int144).
                     *
                     * Counterpart to Solidity's `int144` operator.
                     *
                     * Requirements:
                     *
                     * - input must fit into 144 bits
                     */
                    function toInt144(int256 value) internal pure returns (int144 downcasted) {
                        downcasted = int144(value);
                        if (downcasted != value) {
                            revert SafeCastOverflowedIntDowncast(144, value);
                        }
                    }
                    /**
                     * @dev Returns the downcasted int136 from int256, reverting on
                     * overflow (when the input is less than smallest int136 or
                     * greater than largest int136).
                     *
                     * Counterpart to Solidity's `int136` operator.
                     *
                     * Requirements:
                     *
                     * - input must fit into 136 bits
                     */
                    function toInt136(int256 value) internal pure returns (int136 downcasted) {
                        downcasted = int136(value);
                        if (downcasted != value) {
                            revert SafeCastOverflowedIntDowncast(136, value);
                        }
                    }
                    /**
                     * @dev Returns the downcasted int128 from int256, reverting on
                     * overflow (when the input is less than smallest int128 or
                     * greater than largest int128).
                     *
                     * Counterpart to Solidity's `int128` operator.
                     *
                     * Requirements:
                     *
                     * - input must fit into 128 bits
                     */
                    function toInt128(int256 value) internal pure returns (int128 downcasted) {
                        downcasted = int128(value);
                        if (downcasted != value) {
                            revert SafeCastOverflowedIntDowncast(128, value);
                        }
                    }
                    /**
                     * @dev Returns the downcasted int120 from int256, reverting on
                     * overflow (when the input is less than smallest int120 or
                     * greater than largest int120).
                     *
                     * Counterpart to Solidity's `int120` operator.
                     *
                     * Requirements:
                     *
                     * - input must fit into 120 bits
                     */
                    function toInt120(int256 value) internal pure returns (int120 downcasted) {
                        downcasted = int120(value);
                        if (downcasted != value) {
                            revert SafeCastOverflowedIntDowncast(120, value);
                        }
                    }
                    /**
                     * @dev Returns the downcasted int112 from int256, reverting on
                     * overflow (when the input is less than smallest int112 or
                     * greater than largest int112).
                     *
                     * Counterpart to Solidity's `int112` operator.
                     *
                     * Requirements:
                     *
                     * - input must fit into 112 bits
                     */
                    function toInt112(int256 value) internal pure returns (int112 downcasted) {
                        downcasted = int112(value);
                        if (downcasted != value) {
                            revert SafeCastOverflowedIntDowncast(112, value);
                        }
                    }
                    /**
                     * @dev Returns the downcasted int104 from int256, reverting on
                     * overflow (when the input is less than smallest int104 or
                     * greater than largest int104).
                     *
                     * Counterpart to Solidity's `int104` operator.
                     *
                     * Requirements:
                     *
                     * - input must fit into 104 bits
                     */
                    function toInt104(int256 value) internal pure returns (int104 downcasted) {
                        downcasted = int104(value);
                        if (downcasted != value) {
                            revert SafeCastOverflowedIntDowncast(104, value);
                        }
                    }
                    /**
                     * @dev Returns the downcasted int96 from int256, reverting on
                     * overflow (when the input is less than smallest int96 or
                     * greater than largest int96).
                     *
                     * Counterpart to Solidity's `int96` operator.
                     *
                     * Requirements:
                     *
                     * - input must fit into 96 bits
                     */
                    function toInt96(int256 value) internal pure returns (int96 downcasted) {
                        downcasted = int96(value);
                        if (downcasted != value) {
                            revert SafeCastOverflowedIntDowncast(96, value);
                        }
                    }
                    /**
                     * @dev Returns the downcasted int88 from int256, reverting on
                     * overflow (when the input is less than smallest int88 or
                     * greater than largest int88).
                     *
                     * Counterpart to Solidity's `int88` operator.
                     *
                     * Requirements:
                     *
                     * - input must fit into 88 bits
                     */
                    function toInt88(int256 value) internal pure returns (int88 downcasted) {
                        downcasted = int88(value);
                        if (downcasted != value) {
                            revert SafeCastOverflowedIntDowncast(88, value);
                        }
                    }
                    /**
                     * @dev Returns the downcasted int80 from int256, reverting on
                     * overflow (when the input is less than smallest int80 or
                     * greater than largest int80).
                     *
                     * Counterpart to Solidity's `int80` operator.
                     *
                     * Requirements:
                     *
                     * - input must fit into 80 bits
                     */
                    function toInt80(int256 value) internal pure returns (int80 downcasted) {
                        downcasted = int80(value);
                        if (downcasted != value) {
                            revert SafeCastOverflowedIntDowncast(80, value);
                        }
                    }
                    /**
                     * @dev Returns the downcasted int72 from int256, reverting on
                     * overflow (when the input is less than smallest int72 or
                     * greater than largest int72).
                     *
                     * Counterpart to Solidity's `int72` operator.
                     *
                     * Requirements:
                     *
                     * - input must fit into 72 bits
                     */
                    function toInt72(int256 value) internal pure returns (int72 downcasted) {
                        downcasted = int72(value);
                        if (downcasted != value) {
                            revert SafeCastOverflowedIntDowncast(72, value);
                        }
                    }
                    /**
                     * @dev Returns the downcasted int64 from int256, reverting on
                     * overflow (when the input is less than smallest int64 or
                     * greater than largest int64).
                     *
                     * Counterpart to Solidity's `int64` operator.
                     *
                     * Requirements:
                     *
                     * - input must fit into 64 bits
                     */
                    function toInt64(int256 value) internal pure returns (int64 downcasted) {
                        downcasted = int64(value);
                        if (downcasted != value) {
                            revert SafeCastOverflowedIntDowncast(64, value);
                        }
                    }
                    /**
                     * @dev Returns the downcasted int56 from int256, reverting on
                     * overflow (when the input is less than smallest int56 or
                     * greater than largest int56).
                     *
                     * Counterpart to Solidity's `int56` operator.
                     *
                     * Requirements:
                     *
                     * - input must fit into 56 bits
                     */
                    function toInt56(int256 value) internal pure returns (int56 downcasted) {
                        downcasted = int56(value);
                        if (downcasted != value) {
                            revert SafeCastOverflowedIntDowncast(56, value);
                        }
                    }
                    /**
                     * @dev Returns the downcasted int48 from int256, reverting on
                     * overflow (when the input is less than smallest int48 or
                     * greater than largest int48).
                     *
                     * Counterpart to Solidity's `int48` operator.
                     *
                     * Requirements:
                     *
                     * - input must fit into 48 bits
                     */
                    function toInt48(int256 value) internal pure returns (int48 downcasted) {
                        downcasted = int48(value);
                        if (downcasted != value) {
                            revert SafeCastOverflowedIntDowncast(48, value);
                        }
                    }
                    /**
                     * @dev Returns the downcasted int40 from int256, reverting on
                     * overflow (when the input is less than smallest int40 or
                     * greater than largest int40).
                     *
                     * Counterpart to Solidity's `int40` operator.
                     *
                     * Requirements:
                     *
                     * - input must fit into 40 bits
                     */
                    function toInt40(int256 value) internal pure returns (int40 downcasted) {
                        downcasted = int40(value);
                        if (downcasted != value) {
                            revert SafeCastOverflowedIntDowncast(40, value);
                        }
                    }
                    /**
                     * @dev Returns the downcasted int32 from int256, reverting on
                     * overflow (when the input is less than smallest int32 or
                     * greater than largest int32).
                     *
                     * Counterpart to Solidity's `int32` operator.
                     *
                     * Requirements:
                     *
                     * - input must fit into 32 bits
                     */
                    function toInt32(int256 value) internal pure returns (int32 downcasted) {
                        downcasted = int32(value);
                        if (downcasted != value) {
                            revert SafeCastOverflowedIntDowncast(32, value);
                        }
                    }
                    /**
                     * @dev Returns the downcasted int24 from int256, reverting on
                     * overflow (when the input is less than smallest int24 or
                     * greater than largest int24).
                     *
                     * Counterpart to Solidity's `int24` operator.
                     *
                     * Requirements:
                     *
                     * - input must fit into 24 bits
                     */
                    function toInt24(int256 value) internal pure returns (int24 downcasted) {
                        downcasted = int24(value);
                        if (downcasted != value) {
                            revert SafeCastOverflowedIntDowncast(24, value);
                        }
                    }
                    /**
                     * @dev Returns the downcasted int16 from int256, reverting on
                     * overflow (when the input is less than smallest int16 or
                     * greater than largest int16).
                     *
                     * Counterpart to Solidity's `int16` operator.
                     *
                     * Requirements:
                     *
                     * - input must fit into 16 bits
                     */
                    function toInt16(int256 value) internal pure returns (int16 downcasted) {
                        downcasted = int16(value);
                        if (downcasted != value) {
                            revert SafeCastOverflowedIntDowncast(16, value);
                        }
                    }
                    /**
                     * @dev Returns the downcasted int8 from int256, reverting on
                     * overflow (when the input is less than smallest int8 or
                     * greater than largest int8).
                     *
                     * Counterpart to Solidity's `int8` operator.
                     *
                     * Requirements:
                     *
                     * - input must fit into 8 bits
                     */
                    function toInt8(int256 value) internal pure returns (int8 downcasted) {
                        downcasted = int8(value);
                        if (downcasted != value) {
                            revert SafeCastOverflowedIntDowncast(8, value);
                        }
                    }
                    /**
                     * @dev Converts an unsigned uint256 into a signed int256.
                     *
                     * Requirements:
                     *
                     * - input must be less than or equal to maxInt256.
                     */
                    function toInt256(uint256 value) internal pure returns (int256) {
                        // Note: Unsafe cast below is okay because `type(int256).max` is guaranteed to be positive
                        if (value > uint256(type(int256).max)) {
                            revert SafeCastOverflowedUintToInt(value);
                        }
                        return int256(value);
                    }
                }
                // 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) (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) (interfaces/draft-IERC1822.sol)
                pragma solidity ^0.8.20;
                /**
                 * @dev ERC1822: Universal Upgradeable Proxy Standard (UUPS) documents a method for upgradeability through a simplified
                 * proxy whose upgrades are fully controlled by the current implementation.
                 */
                interface IERC1822Proxiable {
                    /**
                     * @dev Returns the storage slot that the proxiable contract assumes is being used to store the implementation
                     * address.
                     *
                     * IMPORTANT: A proxy pointing at a proxiable contract should not be considered proxiable itself, because this risks
                     * bricking a proxy that upgrades to it, by delegating to itself until out of gas. Thus it is critical that this
                     * function revert if invoked through a proxy.
                     */
                    function proxiableUUID() external view returns (bytes32);
                }
                // SPDX-License-Identifier: MIT
                // OpenZeppelin Contracts (last updated v5.0.0) (proxy/ERC1967/ERC1967Utils.sol)
                pragma solidity ^0.8.20;
                import {IBeacon} from "../beacon/IBeacon.sol";
                import {Address} from "../../utils/Address.sol";
                import {StorageSlot} from "../../utils/StorageSlot.sol";
                /**
                 * @dev This abstract contract provides getters and event emitting update functions for
                 * https://eips.ethereum.org/EIPS/eip-1967[EIP1967] slots.
                 */
                library ERC1967Utils {
                    // We re-declare ERC-1967 events here because they can't be used directly from IERC1967.
                    // This will be fixed in Solidity 0.8.21. At that point we should remove these events.
                    /**
                     * @dev Emitted when the implementation is upgraded.
                     */
                    event Upgraded(address indexed implementation);
                    /**
                     * @dev Emitted when the admin account has changed.
                     */
                    event AdminChanged(address previousAdmin, address newAdmin);
                    /**
                     * @dev Emitted when the beacon is changed.
                     */
                    event BeaconUpgraded(address indexed beacon);
                    /**
                     * @dev Storage slot with the address of the current implementation.
                     * This is the keccak-256 hash of "eip1967.proxy.implementation" subtracted by 1.
                     */
                    // solhint-disable-next-line private-vars-leading-underscore
                    bytes32 internal constant IMPLEMENTATION_SLOT = 0x360894a13ba1a3210667c828492db98dca3e2076cc3735a920a3ca505d382bbc;
                    /**
                     * @dev The `implementation` of the proxy is invalid.
                     */
                    error ERC1967InvalidImplementation(address implementation);
                    /**
                     * @dev The `admin` of the proxy is invalid.
                     */
                    error ERC1967InvalidAdmin(address admin);
                    /**
                     * @dev The `beacon` of the proxy is invalid.
                     */
                    error ERC1967InvalidBeacon(address beacon);
                    /**
                     * @dev An upgrade function sees `msg.value > 0` that may be lost.
                     */
                    error ERC1967NonPayable();
                    /**
                     * @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 ERC1967InvalidImplementation(newImplementation);
                        }
                        StorageSlot.getAddressSlot(IMPLEMENTATION_SLOT).value = newImplementation;
                    }
                    /**
                     * @dev Performs implementation upgrade with additional setup call if data is nonempty.
                     * This function is payable only if the setup call is performed, otherwise `msg.value` is rejected
                     * to avoid stuck value in the contract.
                     *
                     * Emits an {IERC1967-Upgraded} event.
                     */
                    function upgradeToAndCall(address newImplementation, bytes memory data) internal {
                        _setImplementation(newImplementation);
                        emit Upgraded(newImplementation);
                        if (data.length > 0) {
                            Address.functionDelegateCall(newImplementation, data);
                        } else {
                            _checkNonPayable();
                        }
                    }
                    /**
                     * @dev Storage slot with the admin of the contract.
                     * This is the keccak-256 hash of "eip1967.proxy.admin" subtracted by 1.
                     */
                    // solhint-disable-next-line private-vars-leading-underscore
                    bytes32 internal constant ADMIN_SLOT = 0xb53127684a568b3173ae13b9f8a6016e243e63b6e8ee1178d6a717850b5d6103;
                    /**
                     * @dev Returns the current admin.
                     *
                     * TIP: To get this value clients can read directly from the storage slot shown below (specified by EIP1967) using
                     * the https://eth.wiki/json-rpc/API#eth_getstorageat[`eth_getStorageAt`] RPC call.
                     * `0xb53127684a568b3173ae13b9f8a6016e243e63b6e8ee1178d6a717850b5d6103`
                     */
                    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 {
                        if (newAdmin == address(0)) {
                            revert ERC1967InvalidAdmin(address(0));
                        }
                        StorageSlot.getAddressSlot(ADMIN_SLOT).value = newAdmin;
                    }
                    /**
                     * @dev Changes the admin of the proxy.
                     *
                     * Emits an {IERC1967-AdminChanged} event.
                     */
                    function changeAdmin(address newAdmin) internal {
                        emit AdminChanged(getAdmin(), newAdmin);
                        _setAdmin(newAdmin);
                    }
                    /**
                     * @dev The storage slot of the UpgradeableBeacon contract which defines the implementation for this proxy.
                     * This is the keccak-256 hash of "eip1967.proxy.beacon" subtracted by 1.
                     */
                    // solhint-disable-next-line private-vars-leading-underscore
                    bytes32 internal constant BEACON_SLOT = 0xa3f0ad74e5423aebfd80d3ef4346578335a9a72aeaee59ff6cb3582b35133d50;
                    /**
                     * @dev Returns the current beacon.
                     */
                    function getBeacon() internal view returns (address) {
                        return StorageSlot.getAddressSlot(BEACON_SLOT).value;
                    }
                    /**
                     * @dev Stores a new beacon in the EIP1967 beacon slot.
                     */
                    function _setBeacon(address newBeacon) private {
                        if (newBeacon.code.length == 0) {
                            revert ERC1967InvalidBeacon(newBeacon);
                        }
                        StorageSlot.getAddressSlot(BEACON_SLOT).value = newBeacon;
                        address beaconImplementation = IBeacon(newBeacon).implementation();
                        if (beaconImplementation.code.length == 0) {
                            revert ERC1967InvalidImplementation(beaconImplementation);
                        }
                    }
                    /**
                     * @dev Change the beacon and trigger a setup call if data is nonempty.
                     * This function is payable only if the setup call is performed, otherwise `msg.value` is rejected
                     * to avoid stuck value in the contract.
                     *
                     * Emits an {IERC1967-BeaconUpgraded} event.
                     *
                     * CAUTION: Invoking this function has no effect on an instance of {BeaconProxy} since v5, since
                     * it uses an immutable beacon without looking at the value of the ERC-1967 beacon slot for
                     * efficiency.
                     */
                    function upgradeBeaconToAndCall(address newBeacon, bytes memory data) internal {
                        _setBeacon(newBeacon);
                        emit BeaconUpgraded(newBeacon);
                        if (data.length > 0) {
                            Address.functionDelegateCall(IBeacon(newBeacon).implementation(), data);
                        } else {
                            _checkNonPayable();
                        }
                    }
                    /**
                     * @dev Reverts if `msg.value` is not zero. It can be used to avoid `msg.value` stuck in the contract
                     * if an upgrade doesn't perform an initialization call.
                     */
                    function _checkNonPayable() private {
                        if (msg.value > 0) {
                            revert ERC1967NonPayable();
                        }
                    }
                }
                // 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) (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/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();
                        }
                    }
                }
                // SPDX-License-Identifier: MIT
                // OpenZeppelin Contracts (last updated v5.0.0) (proxy/beacon/IBeacon.sol)
                pragma solidity ^0.8.20;
                /**
                 * @dev This is the interface that {BeaconProxy} expects of its beacon.
                 */
                interface IBeacon {
                    /**
                     * @dev Must return an address that can be used as a delegate call target.
                     *
                     * {UpgradeableBeacon} will check that this address is a contract.
                     */
                    function implementation() external view returns (address);
                }
                // SPDX-License-Identifier: MIT
                // 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 4 of 4: Layer3
                // SPDX-License-Identifier: Apache-2.0
                /*
                .____                             ________
                |    |   _____  ___.__. __________\\_____  \\
                |    |   \\__  \\<   |  |/ __ \\_  __ \\_(__  <
                |    |___ / __ \\\\___  \\  ___/|  | \\/       \\
                |_______ (____  / ____|\\___  >__| /______  /
                        \\/    \\/\\/         \\/            \\/
                */
                pragma solidity 0.8.20;
                import {ERC20Upgradeable} from "@openzeppelin/contracts-upgradeable/token/ERC20/ERC20Upgradeable.sol";
                import {ERC20BurnableUpgradeable} from
                    "@openzeppelin/contracts-upgradeable/token/ERC20/extensions/ERC20BurnableUpgradeable.sol";
                import {ERC20PermitUpgradeable} from
                    "@openzeppelin/contracts-upgradeable/token/ERC20/extensions/ERC20PermitUpgradeable.sol";
                import {Initializable} from "@openzeppelin/contracts-upgradeable/proxy/utils/Initializable.sol";
                import {Ownable2StepUpgradeable} from "@openzeppelin/contracts-upgradeable/access/Ownable2StepUpgradeable.sol";
                import {UUPSUpgradeable} from "@openzeppelin/contracts-upgradeable/proxy/utils/UUPSUpgradeable.sol";
                contract Layer3 is
                    Initializable,
                    ERC20Upgradeable,
                    ERC20BurnableUpgradeable,
                    ERC20PermitUpgradeable,
                    Ownable2StepUpgradeable,
                    UUPSUpgradeable
                {
                    /// @custom:oz-upgrades-unsafe-allow constructor
                    constructor() {
                        _disableInitializers();
                    }
                    function initialize(address foundation) external initializer {
                        __ERC20_init("Layer3", "L3");
                        __ERC20Burnable_init();
                        __ERC20Permit_init("Layer3");
                        __Ownable_init(foundation);
                        __Ownable2Step_init();
                        __UUPSUpgradeable_init();
                        uint256 supply = 3_333_333_333 * (10 ** uint256(decimals()));
                        _mint(foundation, supply);
                    }
                    function _authorizeUpgrade(address newImplementation) internal override onlyOwner {}
                }
                // 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/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/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) (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) (access/Ownable2Step.sol)
                pragma solidity ^0.8.20;
                import {OwnableUpgradeable} from "./OwnableUpgradeable.sol";
                import {Initializable} from "../proxy/utils/Initializable.sol";
                /**
                 * @dev Contract module which provides access control mechanism, where
                 * there is an account (an owner) that can be granted exclusive access to
                 * specific functions.
                 *
                 * The initial owner is specified at deployment time in the constructor for `Ownable`. This
                 * can later be changed with {transferOwnership} and {acceptOwnership}.
                 *
                 * This module is used through inheritance. It will make available all functions
                 * from parent (Ownable).
                 */
                abstract contract Ownable2StepUpgradeable is Initializable, OwnableUpgradeable {
                    /// @custom:storage-location erc7201:openzeppelin.storage.Ownable2Step
                    struct Ownable2StepStorage {
                        address _pendingOwner;
                    }
                    // keccak256(abi.encode(uint256(keccak256("openzeppelin.storage.Ownable2Step")) - 1)) & ~bytes32(uint256(0xff))
                    bytes32 private constant Ownable2StepStorageLocation =
                        0x237e158222e3e6968b72b9db0d8043aacf074ad9f650f0d1606b4d82ee432c00;
                    function _getOwnable2StepStorage() private pure returns (Ownable2StepStorage storage $) {
                        assembly {
                            $.slot := Ownable2StepStorageLocation
                        }
                    }
                    event OwnershipTransferStarted(address indexed previousOwner, address indexed newOwner);
                    function __Ownable2Step_init() internal onlyInitializing {}
                    function __Ownable2Step_init_unchained() internal onlyInitializing {}
                    /**
                     * @dev Returns the address of the pending owner.
                     */
                    function pendingOwner() public view virtual returns (address) {
                        Ownable2StepStorage storage $ = _getOwnable2StepStorage();
                        return $._pendingOwner;
                    }
                    /**
                     * @dev Starts the ownership transfer of the contract to a new account. Replaces the pending transfer if there is one.
                     * Can only be called by the current owner.
                     */
                    function transferOwnership(address newOwner) public virtual override onlyOwner {
                        Ownable2StepStorage storage $ = _getOwnable2StepStorage();
                        $._pendingOwner = newOwner;
                        emit OwnershipTransferStarted(owner(), newOwner);
                    }
                    /**
                     * @dev Transfers ownership of the contract to a new account (`newOwner`) and deletes any pending owner.
                     * Internal function without access restriction.
                     */
                    function _transferOwnership(address newOwner) internal virtual override {
                        Ownable2StepStorage storage $ = _getOwnable2StepStorage();
                        delete $._pendingOwner;
                        super._transferOwnership(newOwner);
                    }
                    /**
                     * @dev The new owner accepts the ownership transfer.
                     */
                    function acceptOwnership() public virtual {
                        address sender = _msgSender();
                        if (pendingOwner() != sender) {
                            revert OwnableUnauthorizedAccount(sender);
                        }
                        _transferOwnership(sender);
                    }
                }
                // SPDX-License-Identifier: MIT
                // OpenZeppelin Contracts (last updated v5.0.0) (proxy/utils/UUPSUpgradeable.sol)
                pragma solidity ^0.8.20;
                import {IERC1822Proxiable} from "@openzeppelin/contracts/interfaces/draft-IERC1822.sol";
                import {ERC1967Utils} from "@openzeppelin/contracts/proxy/ERC1967/ERC1967Utils.sol";
                import {Initializable} from "./Initializable.sol";
                /**
                 * @dev An upgradeability mechanism designed for UUPS proxies. The functions included here can perform an upgrade of an
                 * {ERC1967Proxy}, when this contract is set as the implementation behind such a proxy.
                 *
                 * A security mechanism ensures that an upgrade does not turn off upgradeability accidentally, although this risk is
                 * reinstated if the upgrade retains upgradeability but removes the security mechanism, e.g. by replacing
                 * `UUPSUpgradeable` with a custom implementation of upgrades.
                 *
                 * The {_authorizeUpgrade} function must be overridden to include access restriction to the upgrade mechanism.
                 */
                abstract contract UUPSUpgradeable is Initializable, IERC1822Proxiable {
                    /// @custom:oz-upgrades-unsafe-allow state-variable-immutable
                    address private immutable __self = address(this);
                    /**
                     * @dev The version of the upgrade interface of the contract. If this getter is missing, both `upgradeTo(address)`
                     * and `upgradeToAndCall(address,bytes)` are present, and `upgradeTo` must be used if no function should be called,
                     * while `upgradeToAndCall` will invoke the `receive` function if the second argument is the empty byte string.
                     * If the getter returns `"5.0.0"`, only `upgradeToAndCall(address,bytes)` is present, and the second argument must
                     * be the empty byte string if no function should be called, making it impossible to invoke the `receive` function
                     * during an upgrade.
                     */
                    string public constant UPGRADE_INTERFACE_VERSION = "5.0.0";
                    /**
                     * @dev The call is from an unauthorized context.
                     */
                    error UUPSUnauthorizedCallContext();
                    /**
                     * @dev The storage `slot` is unsupported as a UUID.
                     */
                    error UUPSUnsupportedProxiableUUID(bytes32 slot);
                    /**
                     * @dev Check that the execution is being performed through a delegatecall call and that the execution context is
                     * a proxy contract with an implementation (as defined in ERC1967) pointing to self. This should only be the case
                     * for UUPS and transparent proxies that are using the current contract as their implementation. Execution of a
                     * function through ERC1167 minimal proxies (clones) would not normally pass this test, but is not guaranteed to
                     * fail.
                     */
                    modifier onlyProxy() {
                        _checkProxy();
                        _;
                    }
                    /**
                     * @dev Check that the execution is not being performed through a delegate call. This allows a function to be
                     * callable on the implementing contract but not through proxies.
                     */
                    modifier notDelegated() {
                        _checkNotDelegated();
                        _;
                    }
                    function __UUPSUpgradeable_init() internal onlyInitializing {
                    }
                    function __UUPSUpgradeable_init_unchained() internal onlyInitializing {
                    }
                    /**
                     * @dev Implementation of the ERC1822 {proxiableUUID} function. This returns the storage slot used by the
                     * implementation. It is used to validate the implementation's compatibility when performing an upgrade.
                     *
                     * IMPORTANT: A proxy pointing at a proxiable contract should not be considered proxiable itself, because this risks
                     * bricking a proxy that upgrades to it, by delegating to itself until out of gas. Thus it is critical that this
                     * function revert if invoked through a proxy. This is guaranteed by the `notDelegated` modifier.
                     */
                    function proxiableUUID() external view virtual notDelegated returns (bytes32) {
                        return ERC1967Utils.IMPLEMENTATION_SLOT;
                    }
                    /**
                     * @dev Upgrade the implementation of the proxy to `newImplementation`, and subsequently execute the function call
                     * encoded in `data`.
                     *
                     * Calls {_authorizeUpgrade}.
                     *
                     * Emits an {Upgraded} event.
                     *
                     * @custom:oz-upgrades-unsafe-allow-reachable delegatecall
                     */
                    function upgradeToAndCall(address newImplementation, bytes memory data) public payable virtual onlyProxy {
                        _authorizeUpgrade(newImplementation);
                        _upgradeToAndCallUUPS(newImplementation, data);
                    }
                    /**
                     * @dev Reverts if the execution is not performed via delegatecall or the execution
                     * context is not of a proxy with an ERC1967-compliant implementation pointing to self.
                     * See {_onlyProxy}.
                     */
                    function _checkProxy() internal view virtual {
                        if (
                            address(this) == __self || // Must be called through delegatecall
                            ERC1967Utils.getImplementation() != __self // Must be called through an active proxy
                        ) {
                            revert UUPSUnauthorizedCallContext();
                        }
                    }
                    /**
                     * @dev Reverts if the execution is performed via delegatecall.
                     * See {notDelegated}.
                     */
                    function _checkNotDelegated() internal view virtual {
                        if (address(this) != __self) {
                            // Must not be called through delegatecall
                            revert UUPSUnauthorizedCallContext();
                        }
                    }
                    /**
                     * @dev Function that should revert when `msg.sender` is not authorized to upgrade the contract. Called by
                     * {upgradeToAndCall}.
                     *
                     * Normally, this function will use an xref:access.adoc[access control] modifier such as {Ownable-onlyOwner}.
                     *
                     * ```solidity
                     * function _authorizeUpgrade(address) internal onlyOwner {}
                     * ```
                     */
                    function _authorizeUpgrade(address newImplementation) internal virtual;
                    /**
                     * @dev Performs an implementation upgrade with a security check for UUPS proxies, and additional setup call.
                     *
                     * As a security check, {proxiableUUID} is invoked in the new implementation, and the return value
                     * is expected to be the implementation slot in ERC1967.
                     *
                     * Emits an {IERC1967-Upgraded} event.
                     */
                    function _upgradeToAndCallUUPS(address newImplementation, bytes memory data) private {
                        try IERC1822Proxiable(newImplementation).proxiableUUID() returns (bytes32 slot) {
                            if (slot != ERC1967Utils.IMPLEMENTATION_SLOT) {
                                revert UUPSUnsupportedProxiableUUID(slot);
                            }
                            ERC1967Utils.upgradeToAndCall(newImplementation, data);
                        } catch {
                            // The implementation is not UUPS
                            revert ERC1967Utils.ERC1967InvalidImplementation(newImplementation);
                        }
                    }
                }
                // 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) (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.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/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/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) (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/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) (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) (interfaces/draft-IERC1822.sol)
                pragma solidity ^0.8.20;
                /**
                 * @dev ERC1822: Universal Upgradeable Proxy Standard (UUPS) documents a method for upgradeability through a simplified
                 * proxy whose upgrades are fully controlled by the current implementation.
                 */
                interface IERC1822Proxiable {
                    /**
                     * @dev Returns the storage slot that the proxiable contract assumes is being used to store the implementation
                     * address.
                     *
                     * IMPORTANT: A proxy pointing at a proxiable contract should not be considered proxiable itself, because this risks
                     * bricking a proxy that upgrades to it, by delegating to itself until out of gas. Thus it is critical that this
                     * function revert if invoked through a proxy.
                     */
                    function proxiableUUID() external view returns (bytes32);
                }
                // SPDX-License-Identifier: MIT
                // OpenZeppelin Contracts (last updated v5.0.0) (proxy/ERC1967/ERC1967Utils.sol)
                pragma solidity ^0.8.20;
                import {IBeacon} from "../beacon/IBeacon.sol";
                import {Address} from "../../utils/Address.sol";
                import {StorageSlot} from "../../utils/StorageSlot.sol";
                /**
                 * @dev This abstract contract provides getters and event emitting update functions for
                 * https://eips.ethereum.org/EIPS/eip-1967[EIP1967] slots.
                 */
                library ERC1967Utils {
                    // We re-declare ERC-1967 events here because they can't be used directly from IERC1967.
                    // This will be fixed in Solidity 0.8.21. At that point we should remove these events.
                    /**
                     * @dev Emitted when the implementation is upgraded.
                     */
                    event Upgraded(address indexed implementation);
                    /**
                     * @dev Emitted when the admin account has changed.
                     */
                    event AdminChanged(address previousAdmin, address newAdmin);
                    /**
                     * @dev Emitted when the beacon is changed.
                     */
                    event BeaconUpgraded(address indexed beacon);
                    /**
                     * @dev Storage slot with the address of the current implementation.
                     * This is the keccak-256 hash of "eip1967.proxy.implementation" subtracted by 1.
                     */
                    // solhint-disable-next-line private-vars-leading-underscore
                    bytes32 internal constant IMPLEMENTATION_SLOT = 0x360894a13ba1a3210667c828492db98dca3e2076cc3735a920a3ca505d382bbc;
                    /**
                     * @dev The `implementation` of the proxy is invalid.
                     */
                    error ERC1967InvalidImplementation(address implementation);
                    /**
                     * @dev The `admin` of the proxy is invalid.
                     */
                    error ERC1967InvalidAdmin(address admin);
                    /**
                     * @dev The `beacon` of the proxy is invalid.
                     */
                    error ERC1967InvalidBeacon(address beacon);
                    /**
                     * @dev An upgrade function sees `msg.value > 0` that may be lost.
                     */
                    error ERC1967NonPayable();
                    /**
                     * @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 ERC1967InvalidImplementation(newImplementation);
                        }
                        StorageSlot.getAddressSlot(IMPLEMENTATION_SLOT).value = newImplementation;
                    }
                    /**
                     * @dev Performs implementation upgrade with additional setup call if data is nonempty.
                     * This function is payable only if the setup call is performed, otherwise `msg.value` is rejected
                     * to avoid stuck value in the contract.
                     *
                     * Emits an {IERC1967-Upgraded} event.
                     */
                    function upgradeToAndCall(address newImplementation, bytes memory data) internal {
                        _setImplementation(newImplementation);
                        emit Upgraded(newImplementation);
                        if (data.length > 0) {
                            Address.functionDelegateCall(newImplementation, data);
                        } else {
                            _checkNonPayable();
                        }
                    }
                    /**
                     * @dev Storage slot with the admin of the contract.
                     * This is the keccak-256 hash of "eip1967.proxy.admin" subtracted by 1.
                     */
                    // solhint-disable-next-line private-vars-leading-underscore
                    bytes32 internal constant ADMIN_SLOT = 0xb53127684a568b3173ae13b9f8a6016e243e63b6e8ee1178d6a717850b5d6103;
                    /**
                     * @dev Returns the current admin.
                     *
                     * TIP: To get this value clients can read directly from the storage slot shown below (specified by EIP1967) using
                     * the https://eth.wiki/json-rpc/API#eth_getstorageat[`eth_getStorageAt`] RPC call.
                     * `0xb53127684a568b3173ae13b9f8a6016e243e63b6e8ee1178d6a717850b5d6103`
                     */
                    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 {
                        if (newAdmin == address(0)) {
                            revert ERC1967InvalidAdmin(address(0));
                        }
                        StorageSlot.getAddressSlot(ADMIN_SLOT).value = newAdmin;
                    }
                    /**
                     * @dev Changes the admin of the proxy.
                     *
                     * Emits an {IERC1967-AdminChanged} event.
                     */
                    function changeAdmin(address newAdmin) internal {
                        emit AdminChanged(getAdmin(), newAdmin);
                        _setAdmin(newAdmin);
                    }
                    /**
                     * @dev The storage slot of the UpgradeableBeacon contract which defines the implementation for this proxy.
                     * This is the keccak-256 hash of "eip1967.proxy.beacon" subtracted by 1.
                     */
                    // solhint-disable-next-line private-vars-leading-underscore
                    bytes32 internal constant BEACON_SLOT = 0xa3f0ad74e5423aebfd80d3ef4346578335a9a72aeaee59ff6cb3582b35133d50;
                    /**
                     * @dev Returns the current beacon.
                     */
                    function getBeacon() internal view returns (address) {
                        return StorageSlot.getAddressSlot(BEACON_SLOT).value;
                    }
                    /**
                     * @dev Stores a new beacon in the EIP1967 beacon slot.
                     */
                    function _setBeacon(address newBeacon) private {
                        if (newBeacon.code.length == 0) {
                            revert ERC1967InvalidBeacon(newBeacon);
                        }
                        StorageSlot.getAddressSlot(BEACON_SLOT).value = newBeacon;
                        address beaconImplementation = IBeacon(newBeacon).implementation();
                        if (beaconImplementation.code.length == 0) {
                            revert ERC1967InvalidImplementation(beaconImplementation);
                        }
                    }
                    /**
                     * @dev Change the beacon and trigger a setup call if data is nonempty.
                     * This function is payable only if the setup call is performed, otherwise `msg.value` is rejected
                     * to avoid stuck value in the contract.
                     *
                     * Emits an {IERC1967-BeaconUpgraded} event.
                     *
                     * CAUTION: Invoking this function has no effect on an instance of {BeaconProxy} since v5, since
                     * it uses an immutable beacon without looking at the value of the ERC-1967 beacon slot for
                     * efficiency.
                     */
                    function upgradeBeaconToAndCall(address newBeacon, bytes memory data) internal {
                        _setBeacon(newBeacon);
                        emit BeaconUpgraded(newBeacon);
                        if (data.length > 0) {
                            Address.functionDelegateCall(IBeacon(newBeacon).implementation(), data);
                        } else {
                            _checkNonPayable();
                        }
                    }
                    /**
                     * @dev Reverts if `msg.value` is not zero. It can be used to avoid `msg.value` stuck in the contract
                     * if an upgrade doesn't perform an initialization call.
                     */
                    function _checkNonPayable() private {
                        if (msg.value > 0) {
                            revert ERC1967NonPayable();
                        }
                    }
                }
                // 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) (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) (proxy/beacon/IBeacon.sol)
                pragma solidity ^0.8.20;
                /**
                 * @dev This is the interface that {BeaconProxy} expects of its beacon.
                 */
                interface IBeacon {
                    /**
                     * @dev Must return an address that can be used as a delegate call target.
                     *
                     * {UpgradeableBeacon} will check that this address is a contract.
                     */
                    function implementation() external view returns (address);
                }
                // SPDX-License-Identifier: MIT
                // 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();
                        }
                    }
                }
                // SPDX-License-Identifier: MIT
                // 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
                        }
                    }
                }
                // 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/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;
                    }
                }
                // 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);
                        }
                    }
                }