ETH Price: $2,692.49 (+6.24%)

Transaction Decoder

Block:
19890480 at May-17-2024 03:07:23 PM +UTC
Transaction Fee:
0.00240323864659592 ETH $6.47
Gas Used:
191,632 Gas / 12.540904685 Gwei

Emitted Events:

288 TransparentUpgradeableProxy.0xddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef( 0xddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef, 0x000000000000000000000000fae9b3bd6effed451fc82a5b612cbf8be420f9fc, 0x00000000000000000000000098083e22d12497c1516d3c49e7cc6cd2cd9dcba4, 00000000000000000000000000000000000000000000000013fb7bb9697b06fe )
289 TransparentUpgradeableProxy.0xe3d373fc0d45b583082ab11e27c1fc455bb6811ae97c1b7c62ab25c9c0559d21( 0xe3d373fc0d45b583082ab11e27c1fc455bb6811ae97c1b7c62ab25c9c0559d21, 0x000000000000000000000000fae9b3bd6effed451fc82a5b612cbf8be420f9fc, 0x000000000000000000000000f951e335afb289353dc249e82926178eac7ded78, 00000000000000000000000000000000000000000000000013fb7bb9697b06fe, 00000000000000000000000000000000000000000000000013fb7bb9697b06fe, 0000000000000000000000000000000000000000000000000000000000000004, 000000000000000000000000000000000000000000000000000000006659333f )

Account State Difference:

  Address   Before After State Difference Code
0x32bd822d...2C996D678
(Titan Builder)
6.984788546673831901 Eth6.984807709873831901 Eth0.0000191632
0x98083E22...2CD9dcBA4
0xFAE9b3bD...bE420f9fc
2.327805548741807175 Eth
Nonce: 70
2.325402310095211255 Eth
Nonce: 71
0.00240323864659592

Execution Trace

TransparentUpgradeableProxy.e320e0ed( )
  • EigenpieWithdrawManager.userQueuingForWithdraw( asset=0xf951E335afb289353dc249e82926178EaC7DEd78, mLRTamount=1439880542143842046 )
    • TransparentUpgradeableProxy.9be918e6( )
      • EigenpieConfig.isSupportedAsset( token=0xf951E335afb289353dc249e82926178EaC7DEd78 ) => ( isSupported=True )
      • TransparentUpgradeableProxy.e046dd2b( )
        • EigenpieConfig.mLRTReceiptByAsset( token=0xf951E335afb289353dc249e82926178EaC7DEd78 ) => ( mLRTReceipt=0x32bd822d615A3658A68b6fDD30c2fcb2C996D678 )
        • TransparentUpgradeableProxy.70a08231( )
          • MLRT.balanceOf( account=0xFAE9b3bD6efFed451Fc82A5B612cBf8bE420f9fc ) => ( 1439880542143842046 )
          • TransparentUpgradeableProxy.STATICCALL( )
            • EigenpieWithdrawManager.DELEGATECALL( )
            • TransparentUpgradeableProxy.STATICCALL( )
              • MLRT.DELEGATECALL( )
              • TransparentUpgradeableProxy.STATICCALL( )
                • EigenpieWithdrawManager.DELEGATECALL( )
                  • TransparentUpgradeableProxy.STATICCALL( )
                    • EigenpieWithdrawManager.DELEGATECALL( )
                    • TransparentUpgradeableProxy.23b872dd( )
                      • MLRT.transferFrom( from=0xFAE9b3bD6efFed451Fc82A5B612cBf8bE420f9fc, to=0x98083E22d12497c1516d3C49E7Cc6cD2CD9dcBA4, amount=1439880542143842046 ) => ( True )
                        File 1 of 6: TransparentUpgradeableProxy
                        // Sources flattened with hardhat v2.22.2 https://hardhat.org
                        
                        // SPDX-License-Identifier: MIT
                        
                        // File lib/openzeppelin-contracts/contracts/interfaces/draft-IERC1822.sol
                        
                        // Original license: SPDX_License_Identifier: MIT
                        // OpenZeppelin Contracts (last updated v4.5.0) (interfaces/draft-IERC1822.sol)
                        
                        pragma solidity ^0.8.0;
                        
                        /**
                         * @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);
                        }
                        
                        
                        // File lib/openzeppelin-contracts/contracts/interfaces/IERC1967.sol
                        
                        // Original license: SPDX_License_Identifier: MIT
                        // OpenZeppelin Contracts (last updated v4.9.0) (interfaces/IERC1967.sol)
                        
                        pragma solidity ^0.8.0;
                        
                        /**
                         * @dev ERC-1967: Proxy Storage Slots. This interface contains the events defined in the ERC.
                         *
                         * _Available since v4.8.3._
                         */
                        interface IERC1967 {
                            /**
                             * @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);
                        }
                        
                        
                        // File lib/openzeppelin-contracts/contracts/proxy/beacon/IBeacon.sol
                        
                        // Original license: SPDX_License_Identifier: MIT
                        // OpenZeppelin Contracts v4.4.1 (proxy/beacon/IBeacon.sol)
                        
                        pragma solidity ^0.8.0;
                        
                        /**
                         * @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.
                             *
                             * {BeaconProxy} will check that this address is a contract.
                             */
                            function implementation() external view returns (address);
                        }
                        
                        
                        // File lib/openzeppelin-contracts/contracts/utils/Address.sol
                        
                        // Original license: SPDX_License_Identifier: MIT
                        // OpenZeppelin Contracts (last updated v4.9.0) (utils/Address.sol)
                        
                        pragma solidity ^0.8.1;
                        
                        /**
                         * @dev Collection of functions related to the address type
                         */
                        library Address {
                            /**
                             * @dev Returns true if `account` is a contract.
                             *
                             * [IMPORTANT]
                             * ====
                             * It is unsafe to assume that an address for which this function returns
                             * false is an externally-owned account (EOA) and not a contract.
                             *
                             * Among others, `isContract` will return false for the following
                             * types of addresses:
                             *
                             *  - an externally-owned account
                             *  - a contract in construction
                             *  - an address where a contract will be created
                             *  - an address where a contract lived, but was destroyed
                             *
                             * Furthermore, `isContract` will also return true if the target contract within
                             * the same transaction is already scheduled for destruction by `SELFDESTRUCT`,
                             * which only has an effect at the end of a transaction.
                             * ====
                             *
                             * [IMPORTANT]
                             * ====
                             * You shouldn't rely on `isContract` to protect against flash loan attacks!
                             *
                             * Preventing calls from contracts is highly discouraged. It breaks composability, breaks support for smart wallets
                             * like Gnosis Safe, and does not provide security since it can be circumvented by calling from a contract
                             * constructor.
                             * ====
                             */
                            function isContract(address account) internal view returns (bool) {
                                // This method relies on extcodesize/address.code.length, which returns 0
                                // for contracts in construction, since the code is only stored at the end
                                // of the constructor execution.
                        
                                return account.code.length > 0;
                            }
                        
                            /**
                             * @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.0/security-considerations.html#use-the-checks-effects-interactions-pattern[checks-effects-interactions pattern].
                             */
                            function sendValue(address payable recipient, uint256 amount) internal {
                                require(address(this).balance >= amount, "Address: insufficient balance");
                        
                                (bool success, ) = recipient.call{value: amount}("");
                                require(success, "Address: unable to send value, recipient may have reverted");
                            }
                        
                            /**
                             * @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, it is bubbled up by this
                             * function (like regular Solidity function calls).
                             *
                             * 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.
                             *
                             * _Available since v3.1._
                             */
                            function functionCall(address target, bytes memory data) internal returns (bytes memory) {
                                return functionCallWithValue(target, data, 0, "Address: low-level call failed");
                            }
                        
                            /**
                             * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], but with
                             * `errorMessage` as a fallback revert reason when `target` reverts.
                             *
                             * _Available since v3.1._
                             */
                            function functionCall(
                                address target,
                                bytes memory data,
                                string memory errorMessage
                            ) internal returns (bytes memory) {
                                return functionCallWithValue(target, data, 0, errorMessage);
                            }
                        
                            /**
                             * @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`.
                             *
                             * _Available since v3.1._
                             */
                            function functionCallWithValue(address target, bytes memory data, uint256 value) internal returns (bytes memory) {
                                return functionCallWithValue(target, data, value, "Address: low-level call with value failed");
                            }
                        
                            /**
                             * @dev Same as {xref-Address-functionCallWithValue-address-bytes-uint256-}[`functionCallWithValue`], but
                             * with `errorMessage` as a fallback revert reason when `target` reverts.
                             *
                             * _Available since v3.1._
                             */
                            function functionCallWithValue(
                                address target,
                                bytes memory data,
                                uint256 value,
                                string memory errorMessage
                            ) internal returns (bytes memory) {
                                require(address(this).balance >= value, "Address: insufficient balance for call");
                                (bool success, bytes memory returndata) = target.call{value: value}(data);
                                return verifyCallResultFromTarget(target, success, returndata, errorMessage);
                            }
                        
                            /**
                             * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
                             * but performing a static call.
                             *
                             * _Available since v3.3._
                             */
                            function functionStaticCall(address target, bytes memory data) internal view returns (bytes memory) {
                                return functionStaticCall(target, data, "Address: low-level static call failed");
                            }
                        
                            /**
                             * @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`],
                             * but performing a static call.
                             *
                             * _Available since v3.3._
                             */
                            function functionStaticCall(
                                address target,
                                bytes memory data,
                                string memory errorMessage
                            ) internal view returns (bytes memory) {
                                (bool success, bytes memory returndata) = target.staticcall(data);
                                return verifyCallResultFromTarget(target, success, returndata, errorMessage);
                            }
                        
                            /**
                             * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
                             * but performing a delegate call.
                             *
                             * _Available since v3.4._
                             */
                            function functionDelegateCall(address target, bytes memory data) internal returns (bytes memory) {
                                return functionDelegateCall(target, data, "Address: low-level delegate call failed");
                            }
                        
                            /**
                             * @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`],
                             * but performing a delegate call.
                             *
                             * _Available since v3.4._
                             */
                            function functionDelegateCall(
                                address target,
                                bytes memory data,
                                string memory errorMessage
                            ) internal returns (bytes memory) {
                                (bool success, bytes memory returndata) = target.delegatecall(data);
                                return verifyCallResultFromTarget(target, success, returndata, errorMessage);
                            }
                        
                            /**
                             * @dev Tool to verify that a low level call to smart-contract was successful, and revert (either by bubbling
                             * the revert reason or using the provided one) in case of unsuccessful call or if target was not a contract.
                             *
                             * _Available since v4.8._
                             */
                            function verifyCallResultFromTarget(
                                address target,
                                bool success,
                                bytes memory returndata,
                                string memory errorMessage
                            ) internal view returns (bytes memory) {
                                if (success) {
                                    if (returndata.length == 0) {
                                        // only check isContract if the call was successful and the return data is empty
                                        // otherwise we already know that it was a contract
                                        require(isContract(target), "Address: call to non-contract");
                                    }
                                    return returndata;
                                } else {
                                    _revert(returndata, errorMessage);
                                }
                            }
                        
                            /**
                             * @dev Tool to verify that a low level call was successful, and revert if it wasn't, either by bubbling the
                             * revert reason or using the provided one.
                             *
                             * _Available since v4.3._
                             */
                            function verifyCallResult(
                                bool success,
                                bytes memory returndata,
                                string memory errorMessage
                            ) internal pure returns (bytes memory) {
                                if (success) {
                                    return returndata;
                                } else {
                                    _revert(returndata, errorMessage);
                                }
                            }
                        
                            function _revert(bytes memory returndata, string memory errorMessage) 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(errorMessage);
                                }
                            }
                        }
                        
                        
                        // File lib/openzeppelin-contracts/contracts/utils/StorageSlot.sol
                        
                        // Original license: SPDX_License_Identifier: MIT
                        // OpenZeppelin Contracts (last updated v4.9.0) (utils/StorageSlot.sol)
                        // This file was procedurally generated from scripts/generate/templates/StorageSlot.js.
                        
                        pragma solidity ^0.8.0;
                        
                        /**
                         * @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(Address.isContract(newImplementation), "ERC1967: new implementation is not a contract");
                         *         StorageSlot.getAddressSlot(_IMPLEMENTATION_SLOT).value = newImplementation;
                         *     }
                         * }
                         * ```
                         *
                         * _Available since v4.1 for `address`, `bool`, `bytes32`, `uint256`._
                         * _Available since v4.9 for `string`, `bytes`._
                         */
                        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 lib/openzeppelin-contracts/contracts/proxy/ERC1967/ERC1967Upgrade.sol
                        
                        // Original license: SPDX_License_Identifier: MIT
                        // OpenZeppelin Contracts (last updated v4.9.0) (proxy/ERC1967/ERC1967Upgrade.sol)
                        
                        pragma solidity ^0.8.2;
                        
                        
                        
                        
                        
                        /**
                         * @dev This abstract contract provides getters and event emitting update functions for
                         * https://eips.ethereum.org/EIPS/eip-1967[EIP1967] slots.
                         *
                         * _Available since v4.1._
                         */
                        abstract contract ERC1967Upgrade is IERC1967 {
                            // This is the keccak-256 hash of "eip1967.proxy.rollback" subtracted by 1
                            bytes32 private constant _ROLLBACK_SLOT = 0x4910fdfa16fed3260ed0e7147f7cc6da11a60208b5b9406d12a635614ffd9143;
                        
                            /**
                             * @dev Storage slot with the address of the current implementation.
                             * This is the keccak-256 hash of "eip1967.proxy.implementation" subtracted by 1, and is
                             * validated in the constructor.
                             */
                            bytes32 internal constant _IMPLEMENTATION_SLOT = 0x360894a13ba1a3210667c828492db98dca3e2076cc3735a920a3ca505d382bbc;
                        
                            /**
                             * @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 {
                                require(Address.isContract(newImplementation), "ERC1967: new implementation is not a contract");
                                StorageSlot.getAddressSlot(_IMPLEMENTATION_SLOT).value = newImplementation;
                            }
                        
                            /**
                             * @dev Perform implementation upgrade
                             *
                             * Emits an {Upgraded} event.
                             */
                            function _upgradeTo(address newImplementation) internal {
                                _setImplementation(newImplementation);
                                emit Upgraded(newImplementation);
                            }
                        
                            /**
                             * @dev Perform implementation upgrade with additional setup call.
                             *
                             * Emits an {Upgraded} event.
                             */
                            function _upgradeToAndCall(address newImplementation, bytes memory data, bool forceCall) internal {
                                _upgradeTo(newImplementation);
                                if (data.length > 0 || forceCall) {
                                    Address.functionDelegateCall(newImplementation, data);
                                }
                            }
                        
                            /**
                             * @dev Perform implementation upgrade with security checks for UUPS proxies, and additional setup call.
                             *
                             * Emits an {Upgraded} event.
                             */
                            function _upgradeToAndCallUUPS(address newImplementation, bytes memory data, bool forceCall) internal {
                                // Upgrades from old implementations will perform a rollback test. This test requires the new
                                // implementation to upgrade back to the old, non-ERC1822 compliant, implementation. Removing
                                // this special case will break upgrade paths from old UUPS implementation to new ones.
                                if (StorageSlot.getBooleanSlot(_ROLLBACK_SLOT).value) {
                                    _setImplementation(newImplementation);
                                } else {
                                    try IERC1822Proxiable(newImplementation).proxiableUUID() returns (bytes32 slot) {
                                        require(slot == _IMPLEMENTATION_SLOT, "ERC1967Upgrade: unsupported proxiableUUID");
                                    } catch {
                                        revert("ERC1967Upgrade: new implementation is not UUPS");
                                    }
                                    _upgradeToAndCall(newImplementation, data, forceCall);
                                }
                            }
                        
                            /**
                             * @dev Storage slot with the admin of the contract.
                             * This is the keccak-256 hash of "eip1967.proxy.admin" subtracted by 1, and is
                             * validated in the constructor.
                             */
                            bytes32 internal constant _ADMIN_SLOT = 0xb53127684a568b3173ae13b9f8a6016e243e63b6e8ee1178d6a717850b5d6103;
                        
                            /**
                             * @dev Returns the current admin.
                             */
                            function _getAdmin() internal view returns (address) {
                                return StorageSlot.getAddressSlot(_ADMIN_SLOT).value;
                            }
                        
                            /**
                             * @dev Stores a new address in the EIP1967 admin slot.
                             */
                            function _setAdmin(address newAdmin) private {
                                require(newAdmin != address(0), "ERC1967: new admin is the zero address");
                                StorageSlot.getAddressSlot(_ADMIN_SLOT).value = newAdmin;
                            }
                        
                            /**
                             * @dev Changes the admin of the proxy.
                             *
                             * Emits an {AdminChanged} event.
                             */
                            function _changeAdmin(address newAdmin) internal {
                                emit AdminChanged(_getAdmin(), newAdmin);
                                _setAdmin(newAdmin);
                            }
                        
                            /**
                             * @dev The storage slot of the UpgradeableBeacon contract which defines the implementation for this proxy.
                             * This is bytes32(uint256(keccak256('eip1967.proxy.beacon')) - 1)) and is validated in the constructor.
                             */
                            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 {
                                require(Address.isContract(newBeacon), "ERC1967: new beacon is not a contract");
                                require(
                                    Address.isContract(IBeacon(newBeacon).implementation()),
                                    "ERC1967: beacon implementation is not a contract"
                                );
                                StorageSlot.getAddressSlot(_BEACON_SLOT).value = newBeacon;
                            }
                        
                            /**
                             * @dev Perform beacon upgrade with additional setup call. Note: This upgrades the address of the beacon, it does
                             * not upgrade the implementation contained in the beacon (see {UpgradeableBeacon-_setImplementation} for that).
                             *
                             * Emits a {BeaconUpgraded} event.
                             */
                            function _upgradeBeaconToAndCall(address newBeacon, bytes memory data, bool forceCall) internal {
                                _setBeacon(newBeacon);
                                emit BeaconUpgraded(newBeacon);
                                if (data.length > 0 || forceCall) {
                                    Address.functionDelegateCall(IBeacon(newBeacon).implementation(), data);
                                }
                            }
                        }
                        
                        
                        // File lib/openzeppelin-contracts/contracts/proxy/Proxy.sol
                        
                        // Original license: SPDX_License_Identifier: MIT
                        // OpenZeppelin Contracts (last updated v4.6.0) (proxy/Proxy.sol)
                        
                        pragma solidity ^0.8.0;
                        
                        /**
                         * @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 {
                                _beforeFallback();
                                _delegate(_implementation());
                            }
                        
                            /**
                             * @dev Fallback function that delegates calls to the address returned by `_implementation()`. Will run if no other
                             * function in the contract matches the call data.
                             */
                            fallback() external payable virtual {
                                _fallback();
                            }
                        
                            /**
                             * @dev Fallback function that delegates calls to the address returned by `_implementation()`. Will run if call data
                             * is empty.
                             */
                            receive() external payable virtual {
                                _fallback();
                            }
                        
                            /**
                             * @dev Hook that is called before falling back to the implementation. Can happen as part of a manual `_fallback`
                             * call, or as part of the Solidity `fallback` or `receive` functions.
                             *
                             * If overridden should call `super._beforeFallback()`.
                             */
                            function _beforeFallback() internal virtual {}
                        }
                        
                        
                        // File lib/openzeppelin-contracts/contracts/proxy/ERC1967/ERC1967Proxy.sol
                        
                        // Original license: SPDX_License_Identifier: MIT
                        // OpenZeppelin Contracts (last updated v4.7.0) (proxy/ERC1967/ERC1967Proxy.sol)
                        
                        pragma solidity ^0.8.0;
                        
                        
                        /**
                         * @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, ERC1967Upgrade {
                            /**
                             * @dev Initializes the upgradeable proxy with an initial implementation specified by `_logic`.
                             *
                             * If `_data` is nonempty, it's used as data in a delegate call to `_logic`. This will typically be an encoded
                             * function call, and allows initializing the storage of the proxy like a Solidity constructor.
                             */
                            constructor(address _logic, bytes memory _data) payable {
                                _upgradeToAndCall(_logic, _data, false);
                            }
                        
                            /**
                             * @dev Returns the current implementation address.
                             */
                            function _implementation() internal view virtual override returns (address impl) {
                                return ERC1967Upgrade._getImplementation();
                            }
                        }
                        
                        
                        // File lib/openzeppelin-contracts/contracts/proxy/transparent/TransparentUpgradeableProxy.sol
                        
                        // Original license: SPDX_License_Identifier: MIT
                        // OpenZeppelin Contracts (last updated v4.9.0) (proxy/transparent/TransparentUpgradeableProxy.sol)
                        
                        pragma solidity ^0.8.0;
                        
                        /**
                         * @dev Interface for {TransparentUpgradeableProxy}. In order to implement transparency, {TransparentUpgradeableProxy}
                         * does not implement this interface directly, and some of its functions are implemented by an internal dispatch
                         * mechanism. The compiler is unaware that these functions are implemented by {TransparentUpgradeableProxy} and will not
                         * include them in the ABI so this interface must be used to interact with it.
                         */
                        interface ITransparentUpgradeableProxy is IERC1967 {
                            function admin() external view returns (address);
                        
                            function implementation() external view returns (address);
                        
                            function changeAdmin(address) external;
                        
                            function upgradeTo(address) external;
                        
                            function upgradeToAndCall(address, bytes memory) external payable;
                        }
                        
                        /**
                         * @dev This contract implements a proxy that is upgradeable by an admin.
                         *
                         * To avoid https://medium.com/nomic-labs-blog/malicious-backdoors-in-ethereum-proxies-62629adf3357[proxy selector
                         * clashing], which can potentially be used in an attack, this contract uses the
                         * https://blog.openzeppelin.com/the-transparent-proxy-pattern/[transparent proxy pattern]. This pattern implies two
                         * things that go hand in hand:
                         *
                         * 1. If any account other than the admin calls the proxy, the call will be forwarded to the implementation, even if
                         * that call matches one of the admin functions exposed by the proxy itself.
                         * 2. If the admin calls the proxy, it can access the admin functions, but its calls will never be forwarded to the
                         * implementation. If the admin tries to call a function on the implementation it will fail with an error that says
                         * "admin cannot fallback to proxy target".
                         *
                         * These properties mean that the admin account can only be used for admin actions like upgrading the proxy or changing
                         * the admin, so it's best if it's a dedicated account that is not used for anything else. This will avoid headaches due
                         * to sudden errors when trying to call a function from the proxy implementation.
                         *
                         * Our recommendation is for the dedicated account to be an instance of the {ProxyAdmin} contract. If set up this way,
                         * you should think of the `ProxyAdmin` instance as the real administrative interface of your proxy.
                         *
                         * NOTE: The real interface of this proxy is that defined in `ITransparentUpgradeableProxy`. This contract does not
                         * inherit from that interface, and instead the admin functions are implicitly implemented using a custom dispatch
                         * mechanism in `_fallback`. Consequently, the compiler will not produce an ABI for this contract. This is necessary to
                         * fully implement transparency without decoding reverts caused by selector clashes between the proxy and the
                         * implementation.
                         *
                         * WARNING: It is not recommended to extend this contract to add additional external functions. If you do so, the compiler
                         * will not check that there are no selector conflicts, due to the note above. A selector clash between any new function
                         * and the functions declared in {ITransparentUpgradeableProxy} will be resolved in favor of the new one. This could
                         * render the admin operations inaccessible, which could prevent upgradeability. Transparency may also be compromised.
                         */
                        contract TransparentUpgradeableProxy is ERC1967Proxy {
                            /**
                             * @dev Initializes an upgradeable proxy managed by `_admin`, backed by the implementation at `_logic`, and
                             * optionally initialized with `_data` as explained in {ERC1967Proxy-constructor}.
                             */
                            constructor(address _logic, address admin_, bytes memory _data) payable ERC1967Proxy(_logic, _data) {
                                _changeAdmin(admin_);
                            }
                        
                            /**
                             * @dev Modifier used internally that will delegate the call to the implementation unless the sender is the admin.
                             *
                             * CAUTION: This modifier is deprecated, as it could cause issues if the modified function has arguments, and the
                             * implementation provides a function with the same selector.
                             */
                            modifier ifAdmin() {
                                if (msg.sender == _getAdmin()) {
                                    _;
                                } else {
                                    _fallback();
                                }
                            }
                        
                            /**
                             * @dev If caller is the admin process the call internally, otherwise transparently fallback to the proxy behavior
                             */
                            function _fallback() internal virtual override {
                                if (msg.sender == _getAdmin()) {
                                    bytes memory ret;
                                    bytes4 selector = msg.sig;
                                    if (selector == ITransparentUpgradeableProxy.upgradeTo.selector) {
                                        ret = _dispatchUpgradeTo();
                                    } else if (selector == ITransparentUpgradeableProxy.upgradeToAndCall.selector) {
                                        ret = _dispatchUpgradeToAndCall();
                                    } else if (selector == ITransparentUpgradeableProxy.changeAdmin.selector) {
                                        ret = _dispatchChangeAdmin();
                                    } else if (selector == ITransparentUpgradeableProxy.admin.selector) {
                                        ret = _dispatchAdmin();
                                    } else if (selector == ITransparentUpgradeableProxy.implementation.selector) {
                                        ret = _dispatchImplementation();
                                    } else {
                                        revert("TransparentUpgradeableProxy: admin cannot fallback to proxy target");
                                    }
                                    assembly {
                                        return(add(ret, 0x20), mload(ret))
                                    }
                                } else {
                                    super._fallback();
                                }
                            }
                        
                            /**
                             * @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 _dispatchAdmin() private returns (bytes memory) {
                                _requireZeroValue();
                        
                                address admin = _getAdmin();
                                return abi.encode(admin);
                            }
                        
                            /**
                             * @dev Returns the current implementation.
                             *
                             * 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 _dispatchImplementation() private returns (bytes memory) {
                                _requireZeroValue();
                        
                                address implementation = _implementation();
                                return abi.encode(implementation);
                            }
                        
                            /**
                             * @dev Changes the admin of the proxy.
                             *
                             * Emits an {AdminChanged} event.
                             */
                            function _dispatchChangeAdmin() private returns (bytes memory) {
                                _requireZeroValue();
                        
                                address newAdmin = abi.decode(msg.data[4:], (address));
                                _changeAdmin(newAdmin);
                        
                                return "";
                            }
                        
                            /**
                             * @dev Upgrade the implementation of the proxy.
                             */
                            function _dispatchUpgradeTo() private returns (bytes memory) {
                                _requireZeroValue();
                        
                                address newImplementation = abi.decode(msg.data[4:], (address));
                                _upgradeToAndCall(newImplementation, bytes(""), false);
                        
                                return "";
                            }
                        
                            /**
                             * @dev Upgrade the implementation of the proxy, and then call a function from the new implementation as specified
                             * by `data`, which should be an encoded function call. This is useful to initialize new storage variables in the
                             * proxied contract.
                             */
                            function _dispatchUpgradeToAndCall() private returns (bytes memory) {
                                (address newImplementation, bytes memory data) = abi.decode(msg.data[4:], (address, bytes));
                                _upgradeToAndCall(newImplementation, data, true);
                        
                                return "";
                            }
                        
                            /**
                             * @dev Returns the current admin.
                             *
                             * CAUTION: This function is deprecated. Use {ERC1967Upgrade-_getAdmin} instead.
                             */
                            function _admin() internal view virtual returns (address) {
                                return _getAdmin();
                            }
                        
                            /**
                             * @dev To keep this contract fully transparent, all `ifAdmin` functions must be payable. This helper is here to
                             * emulate some proxy functions being non-payable while still allowing value to pass through.
                             */
                            function _requireZeroValue() private {
                                require(msg.value == 0);
                            }
                        }

                        File 2 of 6: TransparentUpgradeableProxy
                        // SPDX-License-Identifier: MIT
                        // OpenZeppelin Contracts (last updated v4.9.0) (proxy/transparent/TransparentUpgradeableProxy.sol)
                        pragma solidity ^0.8.0;
                        import "../ERC1967/ERC1967Proxy.sol";
                        /**
                         * @dev Interface for {TransparentUpgradeableProxy}. In order to implement transparency, {TransparentUpgradeableProxy}
                         * does not implement this interface directly, and some of its functions are implemented by an internal dispatch
                         * mechanism. The compiler is unaware that these functions are implemented by {TransparentUpgradeableProxy} and will not
                         * include them in the ABI so this interface must be used to interact with it.
                         */
                        interface ITransparentUpgradeableProxy is IERC1967 {
                            function admin() external view returns (address);
                            function implementation() external view returns (address);
                            function changeAdmin(address) external;
                            function upgradeTo(address) external;
                            function upgradeToAndCall(address, bytes memory) external payable;
                        }
                        /**
                         * @dev This contract implements a proxy that is upgradeable by an admin.
                         *
                         * To avoid https://medium.com/nomic-labs-blog/malicious-backdoors-in-ethereum-proxies-62629adf3357[proxy selector
                         * clashing], which can potentially be used in an attack, this contract uses the
                         * https://blog.openzeppelin.com/the-transparent-proxy-pattern/[transparent proxy pattern]. This pattern implies two
                         * things that go hand in hand:
                         *
                         * 1. If any account other than the admin calls the proxy, the call will be forwarded to the implementation, even if
                         * that call matches one of the admin functions exposed by the proxy itself.
                         * 2. If the admin calls the proxy, it can access the admin functions, but its calls will never be forwarded to the
                         * implementation. If the admin tries to call a function on the implementation it will fail with an error that says
                         * "admin cannot fallback to proxy target".
                         *
                         * These properties mean that the admin account can only be used for admin actions like upgrading the proxy or changing
                         * the admin, so it's best if it's a dedicated account that is not used for anything else. This will avoid headaches due
                         * to sudden errors when trying to call a function from the proxy implementation.
                         *
                         * Our recommendation is for the dedicated account to be an instance of the {ProxyAdmin} contract. If set up this way,
                         * you should think of the `ProxyAdmin` instance as the real administrative interface of your proxy.
                         *
                         * NOTE: The real interface of this proxy is that defined in `ITransparentUpgradeableProxy`. This contract does not
                         * inherit from that interface, and instead the admin functions are implicitly implemented using a custom dispatch
                         * mechanism in `_fallback`. Consequently, the compiler will not produce an ABI for this contract. This is necessary to
                         * fully implement transparency without decoding reverts caused by selector clashes between the proxy and the
                         * implementation.
                         *
                         * WARNING: It is not recommended to extend this contract to add additional external functions. If you do so, the compiler
                         * will not check that there are no selector conflicts, due to the note above. A selector clash between any new function
                         * and the functions declared in {ITransparentUpgradeableProxy} will be resolved in favor of the new one. This could
                         * render the admin operations inaccessible, which could prevent upgradeability. Transparency may also be compromised.
                         */
                        contract TransparentUpgradeableProxy is ERC1967Proxy {
                            /**
                             * @dev Initializes an upgradeable proxy managed by `_admin`, backed by the implementation at `_logic`, and
                             * optionally initialized with `_data` as explained in {ERC1967Proxy-constructor}.
                             */
                            constructor(address _logic, address admin_, bytes memory _data) payable ERC1967Proxy(_logic, _data) {
                                _changeAdmin(admin_);
                            }
                            /**
                             * @dev Modifier used internally that will delegate the call to the implementation unless the sender is the admin.
                             *
                             * CAUTION: This modifier is deprecated, as it could cause issues if the modified function has arguments, and the
                             * implementation provides a function with the same selector.
                             */
                            modifier ifAdmin() {
                                if (msg.sender == _getAdmin()) {
                                    _;
                                } else {
                                    _fallback();
                                }
                            }
                            /**
                             * @dev If caller is the admin process the call internally, otherwise transparently fallback to the proxy behavior
                             */
                            function _fallback() internal virtual override {
                                if (msg.sender == _getAdmin()) {
                                    bytes memory ret;
                                    bytes4 selector = msg.sig;
                                    if (selector == ITransparentUpgradeableProxy.upgradeTo.selector) {
                                        ret = _dispatchUpgradeTo();
                                    } else if (selector == ITransparentUpgradeableProxy.upgradeToAndCall.selector) {
                                        ret = _dispatchUpgradeToAndCall();
                                    } else if (selector == ITransparentUpgradeableProxy.changeAdmin.selector) {
                                        ret = _dispatchChangeAdmin();
                                    } else if (selector == ITransparentUpgradeableProxy.admin.selector) {
                                        ret = _dispatchAdmin();
                                    } else if (selector == ITransparentUpgradeableProxy.implementation.selector) {
                                        ret = _dispatchImplementation();
                                    } else {
                                        revert("TransparentUpgradeableProxy: admin cannot fallback to proxy target");
                                    }
                                    assembly {
                                        return(add(ret, 0x20), mload(ret))
                                    }
                                } else {
                                    super._fallback();
                                }
                            }
                            /**
                             * @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 _dispatchAdmin() private returns (bytes memory) {
                                _requireZeroValue();
                                address admin = _getAdmin();
                                return abi.encode(admin);
                            }
                            /**
                             * @dev Returns the current implementation.
                             *
                             * 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 _dispatchImplementation() private returns (bytes memory) {
                                _requireZeroValue();
                                address implementation = _implementation();
                                return abi.encode(implementation);
                            }
                            /**
                             * @dev Changes the admin of the proxy.
                             *
                             * Emits an {AdminChanged} event.
                             */
                            function _dispatchChangeAdmin() private returns (bytes memory) {
                                _requireZeroValue();
                                address newAdmin = abi.decode(msg.data[4:], (address));
                                _changeAdmin(newAdmin);
                                return "";
                            }
                            /**
                             * @dev Upgrade the implementation of the proxy.
                             */
                            function _dispatchUpgradeTo() private returns (bytes memory) {
                                _requireZeroValue();
                                address newImplementation = abi.decode(msg.data[4:], (address));
                                _upgradeToAndCall(newImplementation, bytes(""), false);
                                return "";
                            }
                            /**
                             * @dev Upgrade the implementation of the proxy, and then call a function from the new implementation as specified
                             * by `data`, which should be an encoded function call. This is useful to initialize new storage variables in the
                             * proxied contract.
                             */
                            function _dispatchUpgradeToAndCall() private returns (bytes memory) {
                                (address newImplementation, bytes memory data) = abi.decode(msg.data[4:], (address, bytes));
                                _upgradeToAndCall(newImplementation, data, true);
                                return "";
                            }
                            /**
                             * @dev Returns the current admin.
                             *
                             * CAUTION: This function is deprecated. Use {ERC1967Upgrade-_getAdmin} instead.
                             */
                            function _admin() internal view virtual returns (address) {
                                return _getAdmin();
                            }
                            /**
                             * @dev To keep this contract fully transparent, all `ifAdmin` functions must be payable. This helper is here to
                             * emulate some proxy functions being non-payable while still allowing value to pass through.
                             */
                            function _requireZeroValue() private {
                                require(msg.value == 0);
                            }
                        }
                        // SPDX-License-Identifier: MIT
                        // OpenZeppelin Contracts (last updated v4.7.0) (proxy/ERC1967/ERC1967Proxy.sol)
                        pragma solidity ^0.8.0;
                        import "../Proxy.sol";
                        import "./ERC1967Upgrade.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, ERC1967Upgrade {
                            /**
                             * @dev Initializes the upgradeable proxy with an initial implementation specified by `_logic`.
                             *
                             * If `_data` is nonempty, it's used as data in a delegate call to `_logic`. This will typically be an encoded
                             * function call, and allows initializing the storage of the proxy like a Solidity constructor.
                             */
                            constructor(address _logic, bytes memory _data) payable {
                                _upgradeToAndCall(_logic, _data, false);
                            }
                            /**
                             * @dev Returns the current implementation address.
                             */
                            function _implementation() internal view virtual override returns (address impl) {
                                return ERC1967Upgrade._getImplementation();
                            }
                        }
                        // SPDX-License-Identifier: MIT
                        // OpenZeppelin Contracts (last updated v4.6.0) (proxy/Proxy.sol)
                        pragma solidity ^0.8.0;
                        /**
                         * @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 {
                                _beforeFallback();
                                _delegate(_implementation());
                            }
                            /**
                             * @dev Fallback function that delegates calls to the address returned by `_implementation()`. Will run if no other
                             * function in the contract matches the call data.
                             */
                            fallback() external payable virtual {
                                _fallback();
                            }
                            /**
                             * @dev Fallback function that delegates calls to the address returned by `_implementation()`. Will run if call data
                             * is empty.
                             */
                            receive() external payable virtual {
                                _fallback();
                            }
                            /**
                             * @dev Hook that is called before falling back to the implementation. Can happen as part of a manual `_fallback`
                             * call, or as part of the Solidity `fallback` or `receive` functions.
                             *
                             * If overridden should call `super._beforeFallback()`.
                             */
                            function _beforeFallback() internal virtual {}
                        }
                        // SPDX-License-Identifier: MIT
                        // OpenZeppelin Contracts (last updated v4.9.0) (proxy/ERC1967/ERC1967Upgrade.sol)
                        pragma solidity ^0.8.2;
                        import "../beacon/IBeacon.sol";
                        import "../../interfaces/IERC1967.sol";
                        import "../../interfaces/draft-IERC1822.sol";
                        import "../../utils/Address.sol";
                        import "../../utils/StorageSlot.sol";
                        /**
                         * @dev This abstract contract provides getters and event emitting update functions for
                         * https://eips.ethereum.org/EIPS/eip-1967[EIP1967] slots.
                         *
                         * _Available since v4.1._
                         */
                        abstract contract ERC1967Upgrade is IERC1967 {
                            // This is the keccak-256 hash of "eip1967.proxy.rollback" subtracted by 1
                            bytes32 private constant _ROLLBACK_SLOT = 0x4910fdfa16fed3260ed0e7147f7cc6da11a60208b5b9406d12a635614ffd9143;
                            /**
                             * @dev Storage slot with the address of the current implementation.
                             * This is the keccak-256 hash of "eip1967.proxy.implementation" subtracted by 1, and is
                             * validated in the constructor.
                             */
                            bytes32 internal constant _IMPLEMENTATION_SLOT = 0x360894a13ba1a3210667c828492db98dca3e2076cc3735a920a3ca505d382bbc;
                            /**
                             * @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 {
                                require(Address.isContract(newImplementation), "ERC1967: new implementation is not a contract");
                                StorageSlot.getAddressSlot(_IMPLEMENTATION_SLOT).value = newImplementation;
                            }
                            /**
                             * @dev Perform implementation upgrade
                             *
                             * Emits an {Upgraded} event.
                             */
                            function _upgradeTo(address newImplementation) internal {
                                _setImplementation(newImplementation);
                                emit Upgraded(newImplementation);
                            }
                            /**
                             * @dev Perform implementation upgrade with additional setup call.
                             *
                             * Emits an {Upgraded} event.
                             */
                            function _upgradeToAndCall(address newImplementation, bytes memory data, bool forceCall) internal {
                                _upgradeTo(newImplementation);
                                if (data.length > 0 || forceCall) {
                                    Address.functionDelegateCall(newImplementation, data);
                                }
                            }
                            /**
                             * @dev Perform implementation upgrade with security checks for UUPS proxies, and additional setup call.
                             *
                             * Emits an {Upgraded} event.
                             */
                            function _upgradeToAndCallUUPS(address newImplementation, bytes memory data, bool forceCall) internal {
                                // Upgrades from old implementations will perform a rollback test. This test requires the new
                                // implementation to upgrade back to the old, non-ERC1822 compliant, implementation. Removing
                                // this special case will break upgrade paths from old UUPS implementation to new ones.
                                if (StorageSlot.getBooleanSlot(_ROLLBACK_SLOT).value) {
                                    _setImplementation(newImplementation);
                                } else {
                                    try IERC1822Proxiable(newImplementation).proxiableUUID() returns (bytes32 slot) {
                                        require(slot == _IMPLEMENTATION_SLOT, "ERC1967Upgrade: unsupported proxiableUUID");
                                    } catch {
                                        revert("ERC1967Upgrade: new implementation is not UUPS");
                                    }
                                    _upgradeToAndCall(newImplementation, data, forceCall);
                                }
                            }
                            /**
                             * @dev Storage slot with the admin of the contract.
                             * This is the keccak-256 hash of "eip1967.proxy.admin" subtracted by 1, and is
                             * validated in the constructor.
                             */
                            bytes32 internal constant _ADMIN_SLOT = 0xb53127684a568b3173ae13b9f8a6016e243e63b6e8ee1178d6a717850b5d6103;
                            /**
                             * @dev Returns the current admin.
                             */
                            function _getAdmin() internal view returns (address) {
                                return StorageSlot.getAddressSlot(_ADMIN_SLOT).value;
                            }
                            /**
                             * @dev Stores a new address in the EIP1967 admin slot.
                             */
                            function _setAdmin(address newAdmin) private {
                                require(newAdmin != address(0), "ERC1967: new admin is the zero address");
                                StorageSlot.getAddressSlot(_ADMIN_SLOT).value = newAdmin;
                            }
                            /**
                             * @dev Changes the admin of the proxy.
                             *
                             * Emits an {AdminChanged} event.
                             */
                            function _changeAdmin(address newAdmin) internal {
                                emit AdminChanged(_getAdmin(), newAdmin);
                                _setAdmin(newAdmin);
                            }
                            /**
                             * @dev The storage slot of the UpgradeableBeacon contract which defines the implementation for this proxy.
                             * This is bytes32(uint256(keccak256('eip1967.proxy.beacon')) - 1)) and is validated in the constructor.
                             */
                            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 {
                                require(Address.isContract(newBeacon), "ERC1967: new beacon is not a contract");
                                require(
                                    Address.isContract(IBeacon(newBeacon).implementation()),
                                    "ERC1967: beacon implementation is not a contract"
                                );
                                StorageSlot.getAddressSlot(_BEACON_SLOT).value = newBeacon;
                            }
                            /**
                             * @dev Perform beacon upgrade with additional setup call. Note: This upgrades the address of the beacon, it does
                             * not upgrade the implementation contained in the beacon (see {UpgradeableBeacon-_setImplementation} for that).
                             *
                             * Emits a {BeaconUpgraded} event.
                             */
                            function _upgradeBeaconToAndCall(address newBeacon, bytes memory data, bool forceCall) internal {
                                _setBeacon(newBeacon);
                                emit BeaconUpgraded(newBeacon);
                                if (data.length > 0 || forceCall) {
                                    Address.functionDelegateCall(IBeacon(newBeacon).implementation(), data);
                                }
                            }
                        }
                        // SPDX-License-Identifier: MIT
                        // OpenZeppelin Contracts v4.4.1 (proxy/beacon/IBeacon.sol)
                        pragma solidity ^0.8.0;
                        /**
                         * @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.
                             *
                             * {BeaconProxy} will check that this address is a contract.
                             */
                            function implementation() external view returns (address);
                        }
                        // SPDX-License-Identifier: MIT
                        // OpenZeppelin Contracts (last updated v4.9.0) (interfaces/IERC1967.sol)
                        pragma solidity ^0.8.0;
                        /**
                         * @dev ERC-1967: Proxy Storage Slots. This interface contains the events defined in the ERC.
                         *
                         * _Available since v4.8.3._
                         */
                        interface IERC1967 {
                            /**
                             * @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);
                        }
                        // SPDX-License-Identifier: MIT
                        // OpenZeppelin Contracts (last updated v4.5.0) (interfaces/draft-IERC1822.sol)
                        pragma solidity ^0.8.0;
                        /**
                         * @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 v4.9.0) (utils/Address.sol)
                        pragma solidity ^0.8.1;
                        /**
                         * @dev Collection of functions related to the address type
                         */
                        library Address {
                            /**
                             * @dev Returns true if `account` is a contract.
                             *
                             * [IMPORTANT]
                             * ====
                             * It is unsafe to assume that an address for which this function returns
                             * false is an externally-owned account (EOA) and not a contract.
                             *
                             * Among others, `isContract` will return false for the following
                             * types of addresses:
                             *
                             *  - an externally-owned account
                             *  - a contract in construction
                             *  - an address where a contract will be created
                             *  - an address where a contract lived, but was destroyed
                             *
                             * Furthermore, `isContract` will also return true if the target contract within
                             * the same transaction is already scheduled for destruction by `SELFDESTRUCT`,
                             * which only has an effect at the end of a transaction.
                             * ====
                             *
                             * [IMPORTANT]
                             * ====
                             * You shouldn't rely on `isContract` to protect against flash loan attacks!
                             *
                             * Preventing calls from contracts is highly discouraged. It breaks composability, breaks support for smart wallets
                             * like Gnosis Safe, and does not provide security since it can be circumvented by calling from a contract
                             * constructor.
                             * ====
                             */
                            function isContract(address account) internal view returns (bool) {
                                // This method relies on extcodesize/address.code.length, which returns 0
                                // for contracts in construction, since the code is only stored at the end
                                // of the constructor execution.
                                return account.code.length > 0;
                            }
                            /**
                             * @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.0/security-considerations.html#use-the-checks-effects-interactions-pattern[checks-effects-interactions pattern].
                             */
                            function sendValue(address payable recipient, uint256 amount) internal {
                                require(address(this).balance >= amount, "Address: insufficient balance");
                                (bool success, ) = recipient.call{value: amount}("");
                                require(success, "Address: unable to send value, recipient may have reverted");
                            }
                            /**
                             * @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, it is bubbled up by this
                             * function (like regular Solidity function calls).
                             *
                             * 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.
                             *
                             * _Available since v3.1._
                             */
                            function functionCall(address target, bytes memory data) internal returns (bytes memory) {
                                return functionCallWithValue(target, data, 0, "Address: low-level call failed");
                            }
                            /**
                             * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], but with
                             * `errorMessage` as a fallback revert reason when `target` reverts.
                             *
                             * _Available since v3.1._
                             */
                            function functionCall(
                                address target,
                                bytes memory data,
                                string memory errorMessage
                            ) internal returns (bytes memory) {
                                return functionCallWithValue(target, data, 0, errorMessage);
                            }
                            /**
                             * @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`.
                             *
                             * _Available since v3.1._
                             */
                            function functionCallWithValue(address target, bytes memory data, uint256 value) internal returns (bytes memory) {
                                return functionCallWithValue(target, data, value, "Address: low-level call with value failed");
                            }
                            /**
                             * @dev Same as {xref-Address-functionCallWithValue-address-bytes-uint256-}[`functionCallWithValue`], but
                             * with `errorMessage` as a fallback revert reason when `target` reverts.
                             *
                             * _Available since v3.1._
                             */
                            function functionCallWithValue(
                                address target,
                                bytes memory data,
                                uint256 value,
                                string memory errorMessage
                            ) internal returns (bytes memory) {
                                require(address(this).balance >= value, "Address: insufficient balance for call");
                                (bool success, bytes memory returndata) = target.call{value: value}(data);
                                return verifyCallResultFromTarget(target, success, returndata, errorMessage);
                            }
                            /**
                             * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
                             * but performing a static call.
                             *
                             * _Available since v3.3._
                             */
                            function functionStaticCall(address target, bytes memory data) internal view returns (bytes memory) {
                                return functionStaticCall(target, data, "Address: low-level static call failed");
                            }
                            /**
                             * @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`],
                             * but performing a static call.
                             *
                             * _Available since v3.3._
                             */
                            function functionStaticCall(
                                address target,
                                bytes memory data,
                                string memory errorMessage
                            ) internal view returns (bytes memory) {
                                (bool success, bytes memory returndata) = target.staticcall(data);
                                return verifyCallResultFromTarget(target, success, returndata, errorMessage);
                            }
                            /**
                             * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
                             * but performing a delegate call.
                             *
                             * _Available since v3.4._
                             */
                            function functionDelegateCall(address target, bytes memory data) internal returns (bytes memory) {
                                return functionDelegateCall(target, data, "Address: low-level delegate call failed");
                            }
                            /**
                             * @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`],
                             * but performing a delegate call.
                             *
                             * _Available since v3.4._
                             */
                            function functionDelegateCall(
                                address target,
                                bytes memory data,
                                string memory errorMessage
                            ) internal returns (bytes memory) {
                                (bool success, bytes memory returndata) = target.delegatecall(data);
                                return verifyCallResultFromTarget(target, success, returndata, errorMessage);
                            }
                            /**
                             * @dev Tool to verify that a low level call to smart-contract was successful, and revert (either by bubbling
                             * the revert reason or using the provided one) in case of unsuccessful call or if target was not a contract.
                             *
                             * _Available since v4.8._
                             */
                            function verifyCallResultFromTarget(
                                address target,
                                bool success,
                                bytes memory returndata,
                                string memory errorMessage
                            ) internal view returns (bytes memory) {
                                if (success) {
                                    if (returndata.length == 0) {
                                        // only check isContract if the call was successful and the return data is empty
                                        // otherwise we already know that it was a contract
                                        require(isContract(target), "Address: call to non-contract");
                                    }
                                    return returndata;
                                } else {
                                    _revert(returndata, errorMessage);
                                }
                            }
                            /**
                             * @dev Tool to verify that a low level call was successful, and revert if it wasn't, either by bubbling the
                             * revert reason or using the provided one.
                             *
                             * _Available since v4.3._
                             */
                            function verifyCallResult(
                                bool success,
                                bytes memory returndata,
                                string memory errorMessage
                            ) internal pure returns (bytes memory) {
                                if (success) {
                                    return returndata;
                                } else {
                                    _revert(returndata, errorMessage);
                                }
                            }
                            function _revert(bytes memory returndata, string memory errorMessage) 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(errorMessage);
                                }
                            }
                        }
                        // SPDX-License-Identifier: MIT
                        // OpenZeppelin Contracts (last updated v4.9.0) (utils/StorageSlot.sol)
                        // This file was procedurally generated from scripts/generate/templates/StorageSlot.js.
                        pragma solidity ^0.8.0;
                        /**
                         * @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(Address.isContract(newImplementation), "ERC1967: new implementation is not a contract");
                         *         StorageSlot.getAddressSlot(_IMPLEMENTATION_SLOT).value = newImplementation;
                         *     }
                         * }
                         * ```
                         *
                         * _Available since v4.1 for `address`, `bool`, `bytes32`, `uint256`._
                         * _Available since v4.9 for `string`, `bytes`._
                         */
                        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 6: EigenpieWithdrawManager
                        // SPDX-License-Identifier: GPL-3.0-or-later
                        pragma solidity 0.8.21;
                        import { UtilLib } from "./utils/UtilLib.sol";
                        import { EigenpieConstants } from "./utils/EigenpieConstants.sol";
                        import { SafeERC20 } from "@openzeppelin/contracts/token/ERC20/utils/SafeERC20.sol";
                        import { EigenpieConfigRoleChecker, IEigenpieConfig } from "./utils/EigenpieConfigRoleChecker.sol";
                        import { IMintableERC20 } from "./interfaces/IMintableERC20.sol";
                        import { INodeDelegator } from "./interfaces/INodeDelegator.sol";
                        import { IEigenpieStaking } from "./interfaces/IEigenpieStaking.sol";
                        import { IMLRT } from "./interfaces/IMLRT.sol";
                        import { IEigenpiePreDepositHelper } from "./interfaces/IEigenpiePreDepositHelper.sol";
                        import { IERC20 } from "@openzeppelin/contracts/token/ERC20/IERC20.sol";
                        import { PausableUpgradeable } from "@openzeppelin/contracts-upgradeable/security/PausableUpgradeable.sol";
                        import { ReentrancyGuardUpgradeable } from "@openzeppelin/contracts-upgradeable/security/ReentrancyGuardUpgradeable.sol";
                        import { IEigenpieWithdrawManager } from "./interfaces/IEigenpieWithdrawManager.sol";
                        /// @title EigenpieWithdrawManager - Withdraw Pool Contract for LSTs
                        /// @notice Handles LST asset deposits
                        contract EigenpieWithdrawManager is
                            IEigenpieWithdrawManager,
                            EigenpieConfigRoleChecker,
                            PausableUpgradeable,
                            ReentrancyGuardUpgradeable
                        {
                            using SafeERC20 for IERC20;
                            uint256 public lstWithdrawalDelay; // a buffer time period making sure user able to withdraw teh LST unstake by
                                // Eigenpie
                            uint256 public startTimestamp; // the start timestamp counting epoch
                            uint256 public constant EPOCH_DURATION = 7 days;
                            uint256 public withdrawalscheduleCleanUp; // the threshold to clean up withdra queue length
                            mapping(bytes32 => UserWithdrawalSchedule[]) public withdrawalSchedules; //bytes32 = user + asset
                            mapping(bytes32 => WithdrawalSum) public withdrawalSums; // aggregated withdrawal information // bytes32 = asset + epochTime
                            /// @custom:oz-upgrades-unsafe-allow constructor
                            constructor() {
                                _disableInitializers();
                            }
                            /// @dev Initializes the contract
                            /// @param eigenpieConfigAddr eigenpieConfig address
                            function initialize(
                                address eigenpieConfigAddr,
                                uint256 _lstWithdrawalDelay,
                                uint256 _startTimestamp
                            )
                                external
                                initializer
                            {
                                UtilLib.checkNonZeroAddress(eigenpieConfigAddr);
                                __Pausable_init();
                                __ReentrancyGuard_init();
                                eigenpieConfig = IEigenpieConfig(eigenpieConfigAddr);
                                lstWithdrawalDelay = _lstWithdrawalDelay;
                                startTimestamp = _startTimestamp;
                                withdrawalscheduleCleanUp = 5;
                                emit UpdatedEigenpieConfig(eigenpieConfigAddr);
                            }
                            /*//////////////////////////////////////////////////////////////
                                                    view functions
                            //////////////////////////////////////////////////////////////*/
                            // to get timestamp user able to unstake LST from Eigenpie if they queue withdraw now
                            function nextUserWithdrawalTime() external view returns (uint256) {
                                return startTimestamp + (this.currentEpoch() + 1) * EPOCH_DURATION + lstWithdrawalDelay;
                            }
                            // to get current epoch number
                            function currentEpoch() external view returns (uint256) {
                                return (block.timestamp - startTimestamp) / EPOCH_DURATION + 1;
                            }
                            function getUserQueuedWithdraw(
                                address _user,
                                address[] memory _assets
                            )
                                external
                                view
                                returns (uint256[] memory queuedAmounts, uint256[] memory claimableAmounts, uint256[] memory claimedAmounts)
                            {
                                return _calculateAmounts(_user, _assets);
                            }
                            function userToAssetKey(address _user, address _asset) public pure returns (bytes32) {
                                return _getKey(_user, _asset, 0, true);
                            }
                            function assetEpochKey(address _asset, uint256 _epochTime) public pure returns (bytes32) {
                                return _getKey(address(0), _asset, _epochTime, false);
                            }
                            function getUserWithdrawalSchedules(
                                address user,
                                address[] memory assets
                            )
                                external
                                view
                                returns (uint256[][] memory queuedLstAmounts, uint256[][] memory endTimes)
                            {
                                queuedLstAmounts = new uint256[][](assets.length);
                                endTimes = new uint256[][](assets.length);
                                for (uint256 i = 0; i < assets.length; i++) {
                                    bytes32 key = userToAssetKey(user, assets[i]);
                                    UserWithdrawalSchedule[] memory schedules = withdrawalSchedules[key];
                                    // Initialize arrays to store schedules for the current asset
                                    uint256[] memory assetQueuedAmounts = new uint256[](schedules.length);
                                    uint256[] memory assetEndTimes = new uint256[](schedules.length);
                                    // Iterate through all schedules for the current asset
                                    for (uint256 j = 0; j < schedules.length; j++) {
                                        UserWithdrawalSchedule memory schedule = schedules[j];
                                        assetQueuedAmounts[j] = schedule.queuedWithdrawLSTAmt;
                                        assetEndTimes[j] = schedule.endTime;
                                    }
                                    // Assign arrays to the corresponding row in queuedAmounts and endTimes
                                    queuedLstAmounts[i] = assetQueuedAmounts;
                                    endTimes[i] = assetEndTimes;
                                }
                            }
                            /*//////////////////////////////////////////////////////////////
                                                    Write functions
                            //////////////////////////////////////////////////////////////*/
                            /**
                             * @dev Allows a user to queue for withdrawal of a specific asset.
                             * @param asset The address of the asset to withdraw.
                             * @param mLRTamount The amount of the mLRT Token respective of LST token to withdraw.
                             */
                            function userQueuingForWithdraw(
                                address asset,
                                uint256 mLRTamount
                            )
                                external
                                whenNotPaused
                                nonReentrant
                                onlySupportedAsset(asset)
                            {
                                if (asset == EigenpieConstants.PLATFORM_TOKEN_ADDRESS) revert NativeWithdrawNotSupported();
                                address receipt = eigenpieConfig.mLRTReceiptByAsset(asset);
                                uint256 userReceiptBal = IERC20(receipt).balanceOf(msg.sender);
                                if (mLRTamount > userReceiptBal) revert InvalidAmount();
                                uint256 epochCurr = this.currentEpoch();
                                bytes32 userToAsset = userToAssetKey(msg.sender, asset);
                                bytes32 assetToEpoch = assetEpochKey(asset, epochCurr);
                                uint256 rate = IMLRT(receipt).exchangeRateToLST();
                                uint256 withdrawLSTAmt = (rate * mLRTamount) / 1 ether;
                                uint256 userWithdrawableTime = this.nextUserWithdrawalTime();
                                withdrawalSchedules[userToAsset].push(UserWithdrawalSchedule(mLRTamount, withdrawLSTAmt, 0, userWithdrawableTime));
                                WithdrawalSum storage withdrawalSum = withdrawalSums[assetToEpoch];
                                withdrawalSum.assetTotalToWithdrawAmt += withdrawLSTAmt;
                                withdrawalSum.mLRTTotalToBurn += mLRTamount;
                                IERC20(receipt).safeTransferFrom(msg.sender, address(this), mLRTamount);
                                emit UserQueuingForWithdrawal(msg.sender, asset, mLRTamount, withdrawLSTAmt, epochCurr, userWithdrawableTime);
                            }
                            function userWithdrawAsset(address[] memory assets) external nonReentrant {
                                uint256[] memory claimedWithdrawalSchedules = new uint256[](assets.length);
                                for (uint256 i = 0; i < assets.length;) {
                                    bytes32 userToAsset = userToAssetKey(msg.sender, assets[i]);
                                    UserWithdrawalSchedule[] storage schedules = withdrawalSchedules[userToAsset];
                                    uint256 totalClaimedAmount;
                                    uint256 claimedWithdrawalSchedulesPerAsset;
                                    for (uint256 j = 0; j < schedules.length;) {
                                        UserWithdrawalSchedule storage schedule = schedules[j];
                                        // if claimmable
                                        if (block.timestamp >= schedule.endTime && schedule.claimedAmt == 0) {
                                            claimedWithdrawalSchedulesPerAsset++;
                                            schedule.claimedAmt = schedule.queuedWithdrawLSTAmt;
                                            totalClaimedAmount += schedule.queuedWithdrawLSTAmt;
                                        }
                                        unchecked {++j;}
                                    }
                                    claimedWithdrawalSchedules[i] = claimedWithdrawalSchedulesPerAsset;
                                    if (totalClaimedAmount > 0) {
                                        IERC20(assets[i]).safeTransfer(msg.sender, totalClaimedAmount);
                                        emit AssetWithdrawn(msg.sender, assets[i], totalClaimedAmount);
                                    }
                                    unchecked {++i;}
                                }
                                _cleanUpWithdrawalSchedules(assets, claimedWithdrawalSchedules);
                            }
                            /*//////////////////////////////////////////////////////////////
                                                    Admin functions
                            //////////////////////////////////////////////////////////////*/    
                            // admin to queue Withdraw for aggregated user withdraw reqeust
                            // Huge Assumption!! Admin shuold always fully queue withdraw all aggregated assets in that epoch
                            function queuingWithdraw(
                                address[] memory nodeDelegators,
                                address[][] memory nodeToAssets,
                                uint256[][] memory nodeToAmounts,
                                uint256 epochNumber
                            )
                                external
                                onlyEigenpieManager
                            {
                                if (nodeDelegators.length != nodeToAssets.length || nodeDelegators.length != nodeToAmounts.length) {
                                    revert LengthMismatch();
                                }
                                for (uint256 i = 0; i < nodeDelegators.length;) {
                                    if (nodeToAssets[i].length != nodeToAmounts[i].length) {
                                        revert LengthMismatch();
                                    }
                                    unchecked {++i;}
                                }
                                        
                                for (uint256 i = 0; i < nodeDelegators.length;) {
                                    uint256 assetsLength = nodeToAssets[i].length;
                                    for (uint256 j = 0; j < assetsLength;) {
                                        bytes32 assetToEpochKey = keccak256(abi.encodePacked(nodeToAssets[i][j], epochNumber));
                                        WithdrawalSum storage withdrawalSum = withdrawalSums[assetToEpochKey];
                                        withdrawalSum.assetTotalWithdrawQueued += nodeToAmounts[i][j];
                                        withdrawalSum.mLRTburnt = true;
                                        unchecked {++j;}
                                    }
                                    INodeDelegator(nodeDelegators[i]).queueWithdrawalToEigenLayer(nodeToAssets[i], nodeToAmounts[i]);
                                    unchecked {++i;}
                                }
                                address[] memory supportedAssets = eigenpieConfig.getSupportedAssetList();
                                for (uint256 i = 0; i < supportedAssets.length;) {
                                    bytes32 assetToEpochKey = keccak256(abi.encodePacked(supportedAssets[i], epochNumber));
                                    WithdrawalSum memory withdrawalSum = withdrawalSums[assetToEpochKey];
                                    if (withdrawalSum.assetTotalToWithdrawAmt != withdrawalSum.assetTotalWithdrawQueued) {
                                        revert NotWithdrawAllQueuedRequest();
                                    }
                                    if (withdrawalSum.assetTotalWithdrawQueued > 0) {
                                        address receipt = eigenpieConfig.mLRTReceiptByAsset(supportedAssets[i]);
                                        IMintableERC20(receipt).burnFrom(address(this), withdrawalSum.mLRTTotalToBurn);
                                    }
                                    unchecked {++i;}
                                }
                            }
                            /*//////////////////////////////////////////////////////////////
                                                    Admin functions
                            //////////////////////////////////////////////////////////////*/
                            /// @dev Triggers stopped state. Contract must not be paused.
                            function pause() external onlyEigenpieManager {
                                _pause();
                            }
                            /// @dev Returns to normal state. Contract must be paused
                            function unpause() external onlyDefaultAdmin {
                                _unpause();
                            }
                            function updateWithdrawalScheduleCleanUpThreshold(uint256 _newThreshold) external onlyDefaultAdmin {
                                require(_newThreshold > 0, "New threshold must be greater than zero");
                                withdrawalscheduleCleanUp = _newThreshold;
                                emit VestingWithdrawalCleanUpThresholdUpdated(_newThreshold);
                            }
                            /*//////////////////////////////////////////////////////////////
                                                    Internal functions
                            //////////////////////////////////////////////////////////////*/
                            function _cleanUpWithdrawalSchedules(
                                address[] memory assets,
                                uint256[] memory claimedWithdrawalSchedules
                            )
                                internal
                            {
                                for (uint256 i = 0; i < assets.length;) {
                                    bytes32 userToAsset = userToAssetKey(msg.sender, assets[i]);
                                    UserWithdrawalSchedule[] storage schedules = withdrawalSchedules[userToAsset];
                                    if (claimedWithdrawalSchedules[i] >= withdrawalscheduleCleanUp) {
                                        for (uint256 j = 0; j < schedules.length - claimedWithdrawalSchedules[i];) {
                                            schedules[j] = schedules[j + claimedWithdrawalSchedules[j]];
                                            unchecked {++j;}
                                        }
                                        while (claimedWithdrawalSchedules[i] > 0) {
                                            schedules.pop();
                                            claimedWithdrawalSchedules[i]--;
                                        }
                                    }
                                    unchecked {++i;}
                                }
                            }
                            function _calculateAmounts(
                                address _user,
                                address[] memory _assets
                            )
                                internal
                                view
                                returns (uint256[] memory queuedAmounts, uint256[] memory claimableAmounts, uint256[] memory claimedAmounts)
                            {
                                queuedAmounts = new uint256[](_assets.length);
                                claimableAmounts = new uint256[](_assets.length);
                                claimedAmounts = new uint256[](_assets.length);
                                for (uint256 i = 0; i < _assets.length;) {
                                    bytes32 key = userToAssetKey(_user, _assets[i]);
                                    uint256 totalAmount = 0;
                                    uint256 totalClaimed = 0;
                                    uint256 totalClaimable = 0;
                                    UserWithdrawalSchedule[] memory schedules = withdrawalSchedules[key];
                                    for (uint256 j = 0; j < schedules.length;) {
                                        UserWithdrawalSchedule memory schedule = schedules[j];
                                        bool claimable = (schedule.endTime <= block.timestamp) && schedule.claimedAmt == 0;
                                        if (claimable) totalClaimable += schedule.queuedWithdrawLSTAmt;
                                        totalAmount += schedule.queuedWithdrawLSTAmt;
                                        totalClaimed += schedule.claimedAmt;
                                        unchecked {++j;}
                                    }
                                    queuedAmounts[i] = totalAmount;
                                    claimableAmounts[i] = totalClaimable;
                                    claimedAmounts[i] = totalClaimed;
                                    unchecked {++i;}
                                }
                                return (queuedAmounts, claimableAmounts, claimedAmounts);
                            }
                            function _getKey(address _user, address _asset, uint256 _epochTime, bool _isUser) internal pure returns (bytes32) {
                                if (_isUser) {
                                    return keccak256(abi.encodePacked(_user, _asset));
                                } else {
                                    return keccak256(abi.encodePacked(_asset, _epochTime));
                                }
                            }
                        }
                        // SPDX-License-Identifier: GPL-3.0-or-later
                        pragma solidity 0.8.21;
                        import { EigenpieConstants } from "./EigenpieConstants.sol";
                        /// @title UtilLib - Utility library
                        /// @notice Utility functions
                        library UtilLib {
                            error ZeroAddressNotAllowed();
                            /// @dev zero address check modifier
                            /// @param address_ address to check
                            function checkNonZeroAddress(address address_) internal pure {
                                if (address_ == address(0)) revert ZeroAddressNotAllowed();
                            }
                            function isNativeToken(address addr) internal pure returns (bool) {
                                return addr == EigenpieConstants.PLATFORM_TOKEN_ADDRESS;
                            }
                        }
                        // SPDX-License-Identifier: GPL-3.0-or-later
                        pragma solidity 0.8.21;
                        library EigenpieConstants {
                            //contracts
                            bytes32 public constant EIGENPIE_STAKING = keccak256("EIGENPIE_STAKING");
                            bytes32 public constant EIGEN_STRATEGY_MANAGER = keccak256("EIGEN_STRATEGY_MANAGER");
                            bytes32 public constant EIGEN_DELEGATION_MANAGER = keccak256("EIGEN_DELEGATION_MANAGER");
                            bytes32 public constant PRICE_PROVIDER = keccak256("PRICE_PROVIDER");
                            bytes32 public constant BEACON_DEPOSIT = keccak256("BEACON_DEPOSIT");
                            bytes32 public constant EIGENPOD_MANAGER = keccak256("EIGENPOD_MANAGER");
                            bytes32 public constant EIGENPIE_PREDEPOSITHELPER = keccak256("EIGENPIE_PREDEPOSITHELPER");
                            bytes32 public constant EIGENPIE_REWADR_DISTRIBUTOR = keccak256("EIGENPIE_REWADR_DISTRIBUTOR");
                            bytes32 public constant EIGENPIE_DWR = keccak256("EIGENPIE_DWR");
                            bytes32 public constant SSVNETWORK_ENTRY = keccak256("SSVNETWORK_ENTRY");
                            bytes32 public constant SSV_TOKEN = keccak256("SSV_TOKEN");
                            //Roles
                            bytes32 public constant DEFAULT_ADMIN_ROLE = 0x00;
                            bytes32 public constant MANAGER = keccak256("MANAGER");
                            bytes32 public constant MINTER_ROLE = keccak256("MINTER_ROLE");
                            bytes32 public constant BURNER_ROLE = keccak256("BURNER_ROLE");
                            bytes32 public constant ORACLE_ROLE = keccak256("ORACLE_ROLE");
                            bytes32 public constant ORACLE_ADMIN_ROLE = keccak256("ORACLE_ADMIN_ROLE");
                            bytes32 public constant PRICE_PROVIDER_ROLE = keccak256("PRICE_PROVIDER_ROLE");
                            // For Native Restaking
                            uint256 constant PUBKEY_LENGTH = 48;
                            uint256 constant SIGNATURE_LENGTH = 96;
                            uint256 constant MAX_VALIDATORS = 100;
                            uint256 constant DEPOSIT_AMOUNT = 32 ether;
                            uint256 constant GWEI_TO_WEI = 1e9;
                            uint256 public constant DENOMINATOR = 10_000;
                            address public constant PLATFORM_TOKEN_ADDRESS = 0xeFEfeFEfeFeFEFEFEfefeFeFefEfEfEfeFEFEFEf;
                            bytes32 public constant EIGENPIE_WITHDRAW_MANAGER = keccak256("EIGENPIE_WITHDRAW_MANAGER");
                        }
                        // SPDX-License-Identifier: MIT
                        // OpenZeppelin Contracts (last updated v4.9.3) (token/ERC20/utils/SafeERC20.sol)
                        pragma solidity ^0.8.0;
                        import "../IERC20.sol";
                        import "../extensions/IERC20Permit.sol";
                        import "../../../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 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.encodeWithSelector(token.transfer.selector, 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.encodeWithSelector(token.transferFrom.selector, from, to, value));
                            }
                            /**
                             * @dev Deprecated. This function has issues similar to the ones found in
                             * {IERC20-approve}, and its usage is discouraged.
                             *
                             * Whenever possible, use {safeIncreaseAllowance} and
                             * {safeDecreaseAllowance} instead.
                             */
                            function safeApprove(IERC20 token, address spender, uint256 value) internal {
                                // safeApprove should only be called when setting an initial allowance,
                                // or when resetting it to zero. To increase and decrease it, use
                                // 'safeIncreaseAllowance' and 'safeDecreaseAllowance'
                                require(
                                    (value == 0) || (token.allowance(address(this), spender) == 0),
                                    "SafeERC20: approve from non-zero to non-zero allowance"
                                );
                                _callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, 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);
                                _callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, oldAllowance + value));
                            }
                            /**
                             * @dev Decrease the calling contract's allowance toward `spender` by `value`. If `token` returns no value,
                             * non-reverting calls are assumed to be successful.
                             */
                            function safeDecreaseAllowance(IERC20 token, address spender, uint256 value) internal {
                                unchecked {
                                    uint256 oldAllowance = token.allowance(address(this), spender);
                                    require(oldAllowance >= value, "SafeERC20: decreased allowance below zero");
                                    _callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, oldAllowance - value));
                                }
                            }
                            /**
                             * @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.encodeWithSelector(token.approve.selector, spender, value);
                                if (!_callOptionalReturnBool(token, approvalCall)) {
                                    _callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, 0));
                                    _callOptionalReturn(token, approvalCall);
                                }
                            }
                            /**
                             * @dev Use a ERC-2612 signature to set the `owner` approval toward `spender` on `token`.
                             * Revert on invalid signature.
                             */
                            function safePermit(
                                IERC20Permit token,
                                address owner,
                                address spender,
                                uint256 value,
                                uint256 deadline,
                                uint8 v,
                                bytes32 r,
                                bytes32 s
                            ) internal {
                                uint256 nonceBefore = token.nonces(owner);
                                token.permit(owner, spender, value, deadline, v, r, s);
                                uint256 nonceAfter = token.nonces(owner);
                                require(nonceAfter == nonceBefore + 1, "SafeERC20: permit did not succeed");
                            }
                            /**
                             * @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, "SafeERC20: low-level call failed");
                                require(returndata.length == 0 || abi.decode(returndata, (bool)), "SafeERC20: ERC20 operation did not succeed");
                            }
                            /**
                             * @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.isContract(address(token));
                            }
                        }
                        // SPDX-License-Identifier: GPL-3.0-or-later
                        pragma solidity 0.8.21;
                        import { UtilLib } from "./UtilLib.sol";
                        import { EigenpieConstants } from "./EigenpieConstants.sol";
                        import { IEigenpieConfig } from "../interfaces/IEigenpieConfig.sol";
                        import { IAccessControl } from "@openzeppelin/contracts/access/IAccessControl.sol";
                        /// @title EigenpieConfigRoleChecker - Eigenpie Config Role Checker Contract
                        /// @notice Handles Eigenpie config role checks
                        abstract contract EigenpieConfigRoleChecker {
                            IEigenpieConfig public eigenpieConfig;
                            uint256[49] private __gap; // reserve for upgrade
                            // events
                            event UpdatedEigenpieConfig(address indexed eigenpieConfig);
                            // modifiers
                            modifier onlyRole(bytes32 role) {
                                if (!IAccessControl(address(eigenpieConfig)).hasRole(role, msg.sender)) {
                                    string memory roleStr = string(abi.encodePacked(role));
                                    revert IEigenpieConfig.CallerNotEigenpieConfigAllowedRole(roleStr);
                                }
                                _;
                            }
                            modifier onlyEigenpieManager() {
                                if (!IAccessControl(address(eigenpieConfig)).hasRole(EigenpieConstants.MANAGER, msg.sender)) {
                                    revert IEigenpieConfig.CallerNotEigenpieConfigManager();
                                }
                                _;
                            }
                            modifier onlyPriceProvider() {
                                if (!IAccessControl(address(eigenpieConfig)).hasRole(EigenpieConstants.PRICE_PROVIDER_ROLE, msg.sender)) {
                                    revert IEigenpieConfig.CallerNotEigenpieConfigPriceProvider();
                                }
                                _;
                            }
                            modifier onlyDefaultAdmin() {
                                if (!IAccessControl(address(eigenpieConfig)).hasRole(EigenpieConstants.DEFAULT_ADMIN_ROLE, msg.sender)) {
                                    revert IEigenpieConfig.CallerNotEigenpieConfigAdmin();
                                }
                                _;
                            }
                            modifier onlyOracleAdmin() {
                                if (!IAccessControl(address(eigenpieConfig)).hasRole(EigenpieConstants.ORACLE_ADMIN_ROLE, msg.sender)) {
                                    revert IEigenpieConfig.CallerNotEigenpieConfigOracleAdmin();
                                }
                                _;
                            }
                            modifier onlyOracle() {
                                if (!IAccessControl(address(eigenpieConfig)).hasRole(EigenpieConstants.ORACLE_ROLE, msg.sender)) {
                                    revert IEigenpieConfig.CallerNotEigenpieConfigOracle();
                                }
                                _;
                            }
                            modifier onlyMinter() {
                                if (!IAccessControl(address(eigenpieConfig)).hasRole(EigenpieConstants.MINTER_ROLE, msg.sender)) {
                                    revert IEigenpieConfig.CallerNotEigenpieConfigMinter();
                                }
                                _;
                            }
                            modifier onlyBurner() {
                                if (!IAccessControl(address(eigenpieConfig)).hasRole(EigenpieConstants.BURNER_ROLE, msg.sender)) {
                                    revert IEigenpieConfig.CallerNotEigenpieConfigBurner();
                                }
                                _;
                            }
                            modifier onlySupportedAsset(address asset) {
                                if (!eigenpieConfig.isSupportedAsset(asset)) {
                                    revert IEigenpieConfig.AssetNotSupported();
                                }
                                _;
                            }
                            // setters
                            /// @notice Updates the Eigenpie config contract
                            /// @dev only callable by Eigenpie default
                            /// @param eigenpieConfigAddr the new Eigenpie config contract Address
                            function updateEigenpieConfig(address eigenpieConfigAddr) external virtual onlyDefaultAdmin {
                                UtilLib.checkNonZeroAddress(eigenpieConfigAddr);
                                eigenpieConfig = IEigenpieConfig(eigenpieConfigAddr);
                                emit UpdatedEigenpieConfig(eigenpieConfigAddr);
                            }
                        }
                        // SPDX-License-Identifier: MIT
                        // OpenZeppelin Contracts v4.4.1 (token/ERC20/IERC20.sol)
                        pragma solidity 0.8.21;
                        /**
                         * @dev Interface of the ERC20 standard as defined in the EIP.
                         */
                        interface IMintableERC20 {
                            /**
                             * @dev Returns the amount of tokens in existence.
                             */
                            function totalSupply() external view returns (uint256);
                            /**
                             * @dev Returns the amount of tokens owned by `account`.
                             */
                            function balanceOf(address account) external view returns (uint256);
                            /**
                             * @dev Moves `amount` tokens from the caller's account to `recipient`.
                             *
                             * Returns a boolean value indicating whether the operation succeeded.
                             *
                             * Emits a {Transfer} event.
                             */
                            function transfer(address recipient, uint256 amount) external returns (bool);
                            function symbol() external view returns (string memory);
                            function decimals() external view returns (uint8);
                            /**
                             * @dev Returns the remaining number of tokens that `spender` will be
                             * allowed to spend on behalf of `owner` through {transferFrom}. This is
                             * zero by default.
                             *
                             * This value changes when {approve} or {transferFrom} are called.
                             */
                            function allowance(address owner, address spender) external view returns (uint256);
                            /**
                             * @dev Sets `amount` as the allowance of `spender` over the caller's tokens.
                             *
                             * Returns a boolean value indicating whether the operation succeeded.
                             *
                             * IMPORTANT: Beware that changing an allowance with this method brings the risk
                             * that someone may use both the old and the new allowance by unfortunate
                             * transaction ordering. One possible solution to mitigate this race
                             * condition is to first reduce the spender's allowance to 0 and set the
                             * desired value afterwards:
                             * https://github.com/ethereum/EIPs/issues/20#issuecomment-263524729
                             *
                             * Emits an {Approval} event.
                             */
                            function approve(address spender, uint256 amount) external returns (bool);
                            /**
                             * @dev Moves `amount` tokens from `sender` to `recipient` using the
                             * allowance mechanism. `amount` is then deducted from the caller's
                             * allowance.
                             *
                             * Returns a boolean value indicating whether the operation succeeded.
                             *
                             * Emits a {Transfer} event.
                             */
                            function transferFrom(address sender, address recipient, uint256 amount) external returns (bool);
                            function mint(address, uint256) external;
                            function faucet(uint256) external;
                            function burnFrom(address account, uint256 amount) external;
                            /**
                             * @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);
                        }
                        // SPDX-License-Identifier: GPL-3.0-or-later
                        pragma solidity 0.8.21;
                        import "./eigenlayer/IStrategy.sol";
                        import "./ssvNetwork/ISSVNetworkCore.sol";
                        interface INodeDelegator {
                            // struct
                            struct DepositData {
                                bytes[] publicKeys;
                                bytes[] signatures;
                                bytes32[] depositDataRoots;
                            }
                            struct SSVPayload {
                                uint64[] operatorIds;
                                bytes[] sharesData;
                                uint256 amount;
                                ISSVNetworkCore.Cluster cluster;
                            }
                            // event
                            event AssetDepositIntoStrategy(address indexed asset, address indexed strategy, uint256 depositAmount);
                            event EigenPodCreated(address indexed createdEigenPod);
                            event RewardsForwarded(address indexed destinatino, uint256 amount);
                            event WithdrawalQueuedToEigenLayer(
                                bytes32[] withdrawalRoot,
                                IStrategy[] strategies,
                                address[] assets,
                                uint256[] withdrawalAmounts,
                                uint256 startBlock
                            );
                            event DelegationAddressUpdated(address delegate);
                            event GasSpent(address indexed spender, uint256 gasUsed);
                            event GasRefunded(address indexed receiver, uint256 gasRefund);
                            // errors
                            error TokenTransferFailed();
                            error StrategyIsNotSetForAsset();
                            error NoPubKeysProvided();
                            error EigenPodExisted();
                            error InvalidCall();
                            error InvalidCaller();
                            error AtLeastOneValidator();
                            error MaxValidatorsInput();
                            error PublicKeyNotMatch();
                            error SignaturesNotMatch();
                            error DelegateAddressAlreadySet();
                            // methods
                            function depositAssetIntoStrategy(address asset) external;
                            function maxApproveToEigenStrategyManager(address asset) external;
                            function getAssetBalances() external view returns (address[] memory, uint256[] memory);
                            function getAssetBalance(address asset) external view returns (uint256);
                            function getEthBalance() external view returns (uint256);
                            function createEigenPod() external;
                            function queueWithdrawalToEigenLayer(
                                address[] memory assets,
                                uint256[] memory amount
                            )
                                external;
                        }
                        // SPDX-License-Identifier: GPL-3.0-or-later
                        pragma solidity 0.8.21;
                        interface IEigenpieStaking {
                            //errors
                            error TokenTransferFailed();
                            error InvalidAmountToDeposit();
                            error NotEnoughAssetToTransfer();
                            error MaximumDepositLimitReached();
                            error MaximumNodeDelegatorLimitReached();
                            error InvalidMaximumNodeDelegatorLimit();
                            error MinimumAmountToReceiveNotMet();
                            error InvalidIndex();
                            error NativeTokenTransferFailed();
                            error InvalidCaller();
                            error LengthMismatch();
                            error OnlyWhenPredeposit();
                            //events
                            event MaxNodeDelegatorLimitUpdated(uint256 maxNodeDelegatorLimit);
                            event NodeDelegatorAddedinQueue(address[] nodeDelegatorContracts);
                            event AssetDeposit(
                                address indexed depositor,
                                address indexed asset,
                                uint256 depositAmount,
                                address indexed referral,
                                uint256 mintedAmount,
                                bool isPreDepsoit
                            );
                            event MinAmountToDepositUpdated(uint256 minAmountToDeposit);
                            event PreDepositHelperChanged(address oldPreDepositHelper, address newPreDepositHelper);
                            event PreDepositStatusChanged(bool newIsPreDeposit);
                            struct PoolInfo {
                                address mlrtReceipt;
                            }
                            function depositAsset(address asset, uint256 depositAmount, uint256 minRec, address referral) external payable;
                            function getTotalAssetDeposits(address asset) external view returns (uint256);
                            function getAssetCurrentLimit(address asset) external view returns (uint256);
                            function addNodeDelegatorContractToQueue(address[] calldata nodeDelegatorContract) external;
                            function transferAssetToNodeDelegator(uint256 ndcIndex, address asset, uint256 amount) external;
                            function updateMaxNodeDelegatorLimit(uint256 maxNodeDelegatorLimit) external;
                            function getNodeDelegatorQueue() external view returns (address[] memory);
                            function getAssetDistributionData(address asset)
                                external
                                view
                                returns (uint256 assetLyingInDepositPool, uint256 assetLyingInNDCs, uint256 assetStakedInEigenLayer);
                        }
                        // SPDX-License-Identifier: GPL-3.0-or-later
                        pragma solidity 0.8.21;
                        import { IERC20 } from "@openzeppelin/contracts/token/ERC20/IERC20.sol";
                        interface IMLRT is IERC20 {
                            function updateExchangeRateToLST(uint256 _newRate) external;
                            function exchangeRateToLST() external view returns (uint256);
                            function underlyingAsset() external view returns (address);
                        }
                        // SPDX-License-Identifier: GPL-3.0-or-later
                        pragma solidity 0.8.21;
                        interface IEigenpiePreDepositHelper {
                            function feedUserDeposit(address user, address asset, uint256 mintedAmount) external;
                            function withdraw(
                                uint256 _cycle,
                                address _user,
                                address _asset,
                                address _receipt,
                                uint256 _mlrtAmount
                            ) external;
                            function claimableCycles(uint256 cycle) external returns (bool isClaimmable);
                        }
                        // SPDX-License-Identifier: MIT
                        // OpenZeppelin Contracts (last updated v4.9.0) (token/ERC20/IERC20.sol)
                        pragma solidity ^0.8.0;
                        /**
                         * @dev Interface of the ERC20 standard as defined in the EIP.
                         */
                        interface IERC20 {
                            /**
                             * @dev Emitted when `value` tokens are moved from one account (`from`) to
                             * another (`to`).
                             *
                             * Note that `value` may be zero.
                             */
                            event Transfer(address indexed from, address indexed to, uint256 value);
                            /**
                             * @dev Emitted when the allowance of a `spender` for an `owner` is set by
                             * a call to {approve}. `value` is the new allowance.
                             */
                            event Approval(address indexed owner, address indexed spender, uint256 value);
                            /**
                             * @dev Returns the amount of tokens in existence.
                             */
                            function totalSupply() external view returns (uint256);
                            /**
                             * @dev Returns the amount of tokens owned by `account`.
                             */
                            function balanceOf(address account) external view returns (uint256);
                            /**
                             * @dev Moves `amount` tokens from the caller's account to `to`.
                             *
                             * Returns a boolean value indicating whether the operation succeeded.
                             *
                             * Emits a {Transfer} event.
                             */
                            function transfer(address to, uint256 amount) external returns (bool);
                            /**
                             * @dev Returns the remaining number of tokens that `spender` will be
                             * allowed to spend on behalf of `owner` through {transferFrom}. This is
                             * zero by default.
                             *
                             * This value changes when {approve} or {transferFrom} are called.
                             */
                            function allowance(address owner, address spender) external view returns (uint256);
                            /**
                             * @dev Sets `amount` as the allowance of `spender` over the caller's tokens.
                             *
                             * Returns a boolean value indicating whether the operation succeeded.
                             *
                             * IMPORTANT: Beware that changing an allowance with this method brings the risk
                             * that someone may use both the old and the new allowance by unfortunate
                             * transaction ordering. One possible solution to mitigate this race
                             * condition is to first reduce the spender's allowance to 0 and set the
                             * desired value afterwards:
                             * https://github.com/ethereum/EIPs/issues/20#issuecomment-263524729
                             *
                             * Emits an {Approval} event.
                             */
                            function approve(address spender, uint256 amount) external returns (bool);
                            /**
                             * @dev Moves `amount` tokens from `from` to `to` using the
                             * allowance mechanism. `amount` is then deducted from the caller's
                             * allowance.
                             *
                             * Returns a boolean value indicating whether the operation succeeded.
                             *
                             * Emits a {Transfer} event.
                             */
                            function transferFrom(address from, address to, uint256 amount) external returns (bool);
                        }
                        // SPDX-License-Identifier: MIT
                        // OpenZeppelin Contracts (last updated v4.7.0) (security/Pausable.sol)
                        pragma solidity ^0.8.0;
                        import "../utils/ContextUpgradeable.sol";
                        import "../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 {
                            /**
                             * @dev Emitted when the pause is triggered by `account`.
                             */
                            event Paused(address account);
                            /**
                             * @dev Emitted when the pause is lifted by `account`.
                             */
                            event Unpaused(address account);
                            bool private _paused;
                            /**
                             * @dev Initializes the contract in unpaused state.
                             */
                            function __Pausable_init() internal onlyInitializing {
                                __Pausable_init_unchained();
                            }
                            function __Pausable_init_unchained() internal onlyInitializing {
                                _paused = false;
                            }
                            /**
                             * @dev Modifier to make a function callable only when the contract is not paused.
                             *
                             * Requirements:
                             *
                             * - The contract must not be paused.
                             */
                            modifier whenNotPaused() {
                                _requireNotPaused();
                                _;
                            }
                            /**
                             * @dev Modifier to make a function callable only when the contract is paused.
                             *
                             * Requirements:
                             *
                             * - The contract must be paused.
                             */
                            modifier whenPaused() {
                                _requirePaused();
                                _;
                            }
                            /**
                             * @dev Returns true if the contract is paused, and false otherwise.
                             */
                            function paused() public view virtual returns (bool) {
                                return _paused;
                            }
                            /**
                             * @dev Throws if the contract is paused.
                             */
                            function _requireNotPaused() internal view virtual {
                                require(!paused(), "Pausable: paused");
                            }
                            /**
                             * @dev Throws if the contract is not paused.
                             */
                            function _requirePaused() internal view virtual {
                                require(paused(), "Pausable: not paused");
                            }
                            /**
                             * @dev Triggers stopped state.
                             *
                             * Requirements:
                             *
                             * - The contract must not be paused.
                             */
                            function _pause() internal virtual whenNotPaused {
                                _paused = true;
                                emit Paused(_msgSender());
                            }
                            /**
                             * @dev Returns to normal state.
                             *
                             * Requirements:
                             *
                             * - The contract must be paused.
                             */
                            function _unpause() internal virtual whenPaused {
                                _paused = false;
                                emit Unpaused(_msgSender());
                            }
                            /**
                             * @dev This empty reserved space is put in place to allow future versions to add new
                             * variables without shifting down storage in the inheritance chain.
                             * See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps
                             */
                            uint256[49] private __gap;
                        }
                        // SPDX-License-Identifier: MIT
                        // OpenZeppelin Contracts (last updated v4.9.0) (security/ReentrancyGuard.sol)
                        pragma solidity ^0.8.0;
                        import "../proxy/utils/Initializable.sol";
                        /**
                         * @dev Contract module that helps prevent reentrant calls to a function.
                         *
                         * Inheriting from `ReentrancyGuard` will make the {nonReentrant} modifier
                         * available, which can be applied to functions to make sure there are no nested
                         * (reentrant) calls to them.
                         *
                         * Note that because there is a single `nonReentrant` guard, functions marked as
                         * `nonReentrant` may not call one another. This can be worked around by making
                         * those functions `private`, and then adding `external` `nonReentrant` entry
                         * points to them.
                         *
                         * TIP: If you would like to learn more about reentrancy and alternative ways
                         * to protect against it, check out our blog post
                         * https://blog.openzeppelin.com/reentrancy-after-istanbul/[Reentrancy After Istanbul].
                         */
                        abstract contract ReentrancyGuardUpgradeable is Initializable {
                            // Booleans are more expensive than uint256 or any type that takes up a full
                            // word because each write operation emits an extra SLOAD to first read the
                            // slot's contents, replace the bits taken up by the boolean, and then write
                            // back. This is the compiler's defense against contract upgrades and
                            // pointer aliasing, and it cannot be disabled.
                            // The values being non-zero value makes deployment a bit more expensive,
                            // but in exchange the refund on every call to nonReentrant will be lower in
                            // amount. Since refunds are capped to a percentage of the total
                            // transaction's gas, it is best to keep them low in cases like this one, to
                            // increase the likelihood of the full refund coming into effect.
                            uint256 private constant _NOT_ENTERED = 1;
                            uint256 private constant _ENTERED = 2;
                            uint256 private _status;
                            function __ReentrancyGuard_init() internal onlyInitializing {
                                __ReentrancyGuard_init_unchained();
                            }
                            function __ReentrancyGuard_init_unchained() internal onlyInitializing {
                                _status = _NOT_ENTERED;
                            }
                            /**
                             * @dev Prevents a contract from calling itself, directly or indirectly.
                             * Calling a `nonReentrant` function from another `nonReentrant`
                             * function is not supported. It is possible to prevent this from happening
                             * by making the `nonReentrant` function external, and making it call a
                             * `private` function that does the actual work.
                             */
                            modifier nonReentrant() {
                                _nonReentrantBefore();
                                _;
                                _nonReentrantAfter();
                            }
                            function _nonReentrantBefore() private {
                                // On the first call to nonReentrant, _status will be _NOT_ENTERED
                                require(_status != _ENTERED, "ReentrancyGuard: reentrant call");
                                // Any calls to nonReentrant after this point will fail
                                _status = _ENTERED;
                            }
                            function _nonReentrantAfter() private {
                                // By storing the original value once again, a refund is triggered (see
                                // https://eips.ethereum.org/EIPS/eip-2200)
                                _status = _NOT_ENTERED;
                            }
                            /**
                             * @dev Returns true if the reentrancy guard is currently set to "entered", which indicates there is a
                             * `nonReentrant` function in the call stack.
                             */
                            function _reentrancyGuardEntered() internal view returns (bool) {
                                return _status == _ENTERED;
                            }
                            /**
                             * @dev This empty reserved space is put in place to allow future versions to add new
                             * variables without shifting down storage in the inheritance chain.
                             * See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps
                             */
                            uint256[49] private __gap;
                        }
                        // SPDX-License-Identifier: GPL-3.0-or-later
                        pragma solidity 0.8.21;
                        import { IDelegationManager } from "./eigenlayer/IDelegationManager.sol";
                        import { IERC20 } from "@openzeppelin/contracts/interfaces/IERC20.sol";
                        interface IEigenpieWithdrawManager {
                            struct UserWithdrawalSchedule {
                                uint256 receiptMLRTAmt;
                                uint256 queuedWithdrawLSTAmt;
                                uint256 claimedAmt;
                                uint256 endTime;
                            }
                            struct WithdrawalSum {
                                uint256 assetTotalToWithdrawAmt;
                                uint256 assetTotalWithdrawQueued;
                                uint256 mLRTTotalToBurn;
                                bool mLRTburnt;
                            }
                            //errors
                            error InvalidAmount();
                            error LengthMismatch();
                            error EpochNotYetReached();
                            error NotWithdrawAllQueuedRequest();
                            error NativeWithdrawNotSupported();
                            //events
                            event UserQueuingForWithdrawal(
                                address indexed user,
                                address indexed asset,
                                uint256 mLRTAmount,
                                uint256 LSTAmt,
                                uint256 currentEpoch,
                                uint256 endTime
                            );
                            event AssetWithdrawn(address indexed user, address indexed asset, uint256 LSTAmt);
                            event EpochUpdated(uint256 newEpochTime);
                            event VestingWithdrawalCleanUpThresholdUpdated(uint256 newThreshold);
                        }
                        // SPDX-License-Identifier: MIT
                        // OpenZeppelin Contracts (last updated v4.9.0) (token/ERC20/extensions/IERC20Permit.sol)
                        pragma solidity ^0.8.0;
                        /**
                         * @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.
                         */
                        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].
                             */
                            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 v4.9.0) (utils/Address.sol)
                        pragma solidity ^0.8.1;
                        /**
                         * @dev Collection of functions related to the address type
                         */
                        library Address {
                            /**
                             * @dev Returns true if `account` is a contract.
                             *
                             * [IMPORTANT]
                             * ====
                             * It is unsafe to assume that an address for which this function returns
                             * false is an externally-owned account (EOA) and not a contract.
                             *
                             * Among others, `isContract` will return false for the following
                             * types of addresses:
                             *
                             *  - an externally-owned account
                             *  - a contract in construction
                             *  - an address where a contract will be created
                             *  - an address where a contract lived, but was destroyed
                             *
                             * Furthermore, `isContract` will also return true if the target contract within
                             * the same transaction is already scheduled for destruction by `SELFDESTRUCT`,
                             * which only has an effect at the end of a transaction.
                             * ====
                             *
                             * [IMPORTANT]
                             * ====
                             * You shouldn't rely on `isContract` to protect against flash loan attacks!
                             *
                             * Preventing calls from contracts is highly discouraged. It breaks composability, breaks support for smart wallets
                             * like Gnosis Safe, and does not provide security since it can be circumvented by calling from a contract
                             * constructor.
                             * ====
                             */
                            function isContract(address account) internal view returns (bool) {
                                // This method relies on extcodesize/address.code.length, which returns 0
                                // for contracts in construction, since the code is only stored at the end
                                // of the constructor execution.
                                return account.code.length > 0;
                            }
                            /**
                             * @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.0/security-considerations.html#use-the-checks-effects-interactions-pattern[checks-effects-interactions pattern].
                             */
                            function sendValue(address payable recipient, uint256 amount) internal {
                                require(address(this).balance >= amount, "Address: insufficient balance");
                                (bool success, ) = recipient.call{value: amount}("");
                                require(success, "Address: unable to send value, recipient may have reverted");
                            }
                            /**
                             * @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, it is bubbled up by this
                             * function (like regular Solidity function calls).
                             *
                             * 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.
                             *
                             * _Available since v3.1._
                             */
                            function functionCall(address target, bytes memory data) internal returns (bytes memory) {
                                return functionCallWithValue(target, data, 0, "Address: low-level call failed");
                            }
                            /**
                             * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], but with
                             * `errorMessage` as a fallback revert reason when `target` reverts.
                             *
                             * _Available since v3.1._
                             */
                            function functionCall(
                                address target,
                                bytes memory data,
                                string memory errorMessage
                            ) internal returns (bytes memory) {
                                return functionCallWithValue(target, data, 0, errorMessage);
                            }
                            /**
                             * @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`.
                             *
                             * _Available since v3.1._
                             */
                            function functionCallWithValue(address target, bytes memory data, uint256 value) internal returns (bytes memory) {
                                return functionCallWithValue(target, data, value, "Address: low-level call with value failed");
                            }
                            /**
                             * @dev Same as {xref-Address-functionCallWithValue-address-bytes-uint256-}[`functionCallWithValue`], but
                             * with `errorMessage` as a fallback revert reason when `target` reverts.
                             *
                             * _Available since v3.1._
                             */
                            function functionCallWithValue(
                                address target,
                                bytes memory data,
                                uint256 value,
                                string memory errorMessage
                            ) internal returns (bytes memory) {
                                require(address(this).balance >= value, "Address: insufficient balance for call");
                                (bool success, bytes memory returndata) = target.call{value: value}(data);
                                return verifyCallResultFromTarget(target, success, returndata, errorMessage);
                            }
                            /**
                             * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
                             * but performing a static call.
                             *
                             * _Available since v3.3._
                             */
                            function functionStaticCall(address target, bytes memory data) internal view returns (bytes memory) {
                                return functionStaticCall(target, data, "Address: low-level static call failed");
                            }
                            /**
                             * @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`],
                             * but performing a static call.
                             *
                             * _Available since v3.3._
                             */
                            function functionStaticCall(
                                address target,
                                bytes memory data,
                                string memory errorMessage
                            ) internal view returns (bytes memory) {
                                (bool success, bytes memory returndata) = target.staticcall(data);
                                return verifyCallResultFromTarget(target, success, returndata, errorMessage);
                            }
                            /**
                             * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
                             * but performing a delegate call.
                             *
                             * _Available since v3.4._
                             */
                            function functionDelegateCall(address target, bytes memory data) internal returns (bytes memory) {
                                return functionDelegateCall(target, data, "Address: low-level delegate call failed");
                            }
                            /**
                             * @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`],
                             * but performing a delegate call.
                             *
                             * _Available since v3.4._
                             */
                            function functionDelegateCall(
                                address target,
                                bytes memory data,
                                string memory errorMessage
                            ) internal returns (bytes memory) {
                                (bool success, bytes memory returndata) = target.delegatecall(data);
                                return verifyCallResultFromTarget(target, success, returndata, errorMessage);
                            }
                            /**
                             * @dev Tool to verify that a low level call to smart-contract was successful, and revert (either by bubbling
                             * the revert reason or using the provided one) in case of unsuccessful call or if target was not a contract.
                             *
                             * _Available since v4.8._
                             */
                            function verifyCallResultFromTarget(
                                address target,
                                bool success,
                                bytes memory returndata,
                                string memory errorMessage
                            ) internal view returns (bytes memory) {
                                if (success) {
                                    if (returndata.length == 0) {
                                        // only check isContract if the call was successful and the return data is empty
                                        // otherwise we already know that it was a contract
                                        require(isContract(target), "Address: call to non-contract");
                                    }
                                    return returndata;
                                } else {
                                    _revert(returndata, errorMessage);
                                }
                            }
                            /**
                             * @dev Tool to verify that a low level call was successful, and revert if it wasn't, either by bubbling the
                             * revert reason or using the provided one.
                             *
                             * _Available since v4.3._
                             */
                            function verifyCallResult(
                                bool success,
                                bytes memory returndata,
                                string memory errorMessage
                            ) internal pure returns (bytes memory) {
                                if (success) {
                                    return returndata;
                                } else {
                                    _revert(returndata, errorMessage);
                                }
                            }
                            function _revert(bytes memory returndata, string memory errorMessage) 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(errorMessage);
                                }
                            }
                        }
                        // SPDX-License-Identifier: GPL-3.0-or-later
                        pragma solidity 0.8.21;
                        interface IEigenpieConfig {
                            // Errors
                            error ValueAlreadyInUse();
                            error AssetAlreadySupported();
                            error AssetNotSupported();
                            error CallerNotEigenpieConfigAdmin();
                            error CallerNotEigenpieConfigManager();
                            error CallerNotEigenpieConfigOracle();
                            error CallerNotEigenpieConfigOracleAdmin();
                            error CallerNotEigenpieConfigPriceProvider();
                            error CallerNotEigenpieConfigMinter();
                            error CallerNotEigenpieConfigBurner();
                            error CallerNotEigenpieConfigAllowedRole(string role);
                            // Events
                            event SetContract(bytes32 key, address indexed contractAddr);
                            event AddedNewSupportedAsset(address indexed asset, address indexed receipt, uint256 depositLimit);
                            event ReceiptTokenUpdated(address indexed asset, address indexed receipt);
                            event RemovedSupportedAsset(address indexed asset);
                            event AssetDepositLimitUpdate(address indexed asset, uint256 depositLimit);
                            event AssetStrategyUpdate(address indexed asset, address indexed strategy);
                            event AssetBoostUpdate(address indexed asset, uint256 newBoost);
                            event ReferralUpdate(address indexed me, address indexed myReferral);
                            // methods
                            function baseGasAmountSpent() external returns(uint256);    
                            function assetStrategy(address asset) external view returns (address);
                            function boostByAsset(address) external view returns (uint256);
                            function mLRTReceiptByAsset(address) external view returns (address);
                            function isSupportedAsset(address asset) external view returns (bool);
                            function getContract(bytes32 contractId) external view returns (address);
                            function getSupportedAssetList() external view returns (address[] memory);
                            function depositLimitByAsset(address asset) external view returns (uint256);
                        }
                        // SPDX-License-Identifier: MIT
                        // OpenZeppelin Contracts v4.4.1 (access/IAccessControl.sol)
                        pragma solidity ^0.8.0;
                        /**
                         * @dev External interface of AccessControl declared to support ERC165 detection.
                         */
                        interface IAccessControl {
                            /**
                             * @dev Emitted when `newAdminRole` is set as ``role``'s admin role, replacing `previousAdminRole`
                             *
                             * `DEFAULT_ADMIN_ROLE` is the starting admin for all roles, despite
                             * {RoleAdminChanged} not being emitted signaling this.
                             *
                             * _Available since v3.1._
                             */
                            event RoleAdminChanged(bytes32 indexed role, bytes32 indexed previousAdminRole, bytes32 indexed newAdminRole);
                            /**
                             * @dev Emitted when `account` is granted `role`.
                             *
                             * `sender` is the account that originated the contract call, an admin role
                             * bearer except when using {AccessControl-_setupRole}.
                             */
                            event RoleGranted(bytes32 indexed role, address indexed account, address indexed sender);
                            /**
                             * @dev Emitted when `account` is revoked `role`.
                             *
                             * `sender` is the account that originated the contract call:
                             *   - if using `revokeRole`, it is the admin role bearer
                             *   - if using `renounceRole`, it is the role bearer (i.e. `account`)
                             */
                            event RoleRevoked(bytes32 indexed role, address indexed account, address indexed sender);
                            /**
                             * @dev Returns `true` if `account` has been granted `role`.
                             */
                            function hasRole(bytes32 role, address account) external view returns (bool);
                            /**
                             * @dev Returns the admin role that controls `role`. See {grantRole} and
                             * {revokeRole}.
                             *
                             * To change a role's admin, use {AccessControl-_setRoleAdmin}.
                             */
                            function getRoleAdmin(bytes32 role) external view returns (bytes32);
                            /**
                             * @dev Grants `role` to `account`.
                             *
                             * If `account` had not been already granted `role`, emits a {RoleGranted}
                             * event.
                             *
                             * Requirements:
                             *
                             * - the caller must have ``role``'s admin role.
                             */
                            function grantRole(bytes32 role, address account) external;
                            /**
                             * @dev Revokes `role` from `account`.
                             *
                             * If `account` had been granted `role`, emits a {RoleRevoked} event.
                             *
                             * Requirements:
                             *
                             * - the caller must have ``role``'s admin role.
                             */
                            function revokeRole(bytes32 role, address account) external;
                            /**
                             * @dev Revokes `role` from the calling account.
                             *
                             * Roles are often managed via {grantRole} and {revokeRole}: this function's
                             * purpose is to provide a mechanism for accounts to lose their privileges
                             * if they are compromised (such as when a trusted device is misplaced).
                             *
                             * If the calling account had been granted `role`, emits a {RoleRevoked}
                             * event.
                             *
                             * Requirements:
                             *
                             * - the caller must be `account`.
                             */
                            function renounceRole(bytes32 role, address account) external;
                        }
                        // SPDX-License-Identifier: BUSL-1.1
                        pragma solidity 0.8.21;
                        import "@openzeppelin/contracts/token/ERC20/IERC20.sol";
                        /**
                         * @title Minimal interface for an `Strategy` contract.
                         * @author Layr Labs, Inc.
                         * @notice Terms of Service: https://docs.eigenlayer.xyz/overview/terms-of-service
                         * @notice Custom `Strategy` implementations may expand extensively on this interface.
                         */
                        interface IStrategy {
                            /**
                             * @notice Used to deposit tokens into this Strategy
                             * @param token is the ERC20 token being deposited
                             * @param amount is the amount of token being deposited
                             * @dev This function is only callable by the strategyManager contract. It is invoked inside of the strategyManager's
                             * `depositIntoStrategy` function, and individual share balances are recorded in the strategyManager as well.
                             * @return newShares is the number of new shares issued at the current exchange ratio.
                             */
                            function deposit(IERC20 token, uint256 amount) external returns (uint256);
                            /**
                             * @notice Used to withdraw tokens from this Strategy, to the `recipient`'s address
                             * @param recipient is the address to receive the withdrawn funds
                             * @param token is the ERC20 token being transferred out
                             * @param amountShares is the amount of shares being withdrawn
                             * @dev This function is only callable by the strategyManager contract. It is invoked inside of the strategyManager's
                             * other functions, and individual share balances are recorded in the strategyManager as well.
                             */
                            function withdraw(address recipient, IERC20 token, uint256 amountShares) external;
                            /**
                             * @notice Used to convert a number of shares to the equivalent amount of underlying tokens for this strategy.
                             * @notice In contrast to `sharesToUnderlyingView`, this function **may** make state modifications
                             * @param amountShares is the amount of shares to calculate its conversion into the underlying token
                             * @return The amount of underlying tokens corresponding to the input `amountShares`
                             * @dev Implementation for these functions in particular may vary significantly for different strategies
                             */
                            function sharesToUnderlying(uint256 amountShares) external returns (uint256);
                            /**
                             * @notice Used to convert an amount of underlying tokens to the equivalent amount of shares in this strategy.
                             * @notice In contrast to `underlyingToSharesView`, this function **may** make state modifications
                             * @param amountUnderlying is the amount of `underlyingToken` to calculate its conversion into strategy shares
                             * @return The amount of underlying tokens corresponding to the input `amountShares`
                             * @dev Implementation for these functions in particular may vary significantly for different strategies
                             */
                            function underlyingToShares(uint256 amountUnderlying) external returns (uint256);
                            /**
                             * @notice convenience function for fetching the current underlying value of all of the `user`'s shares in
                             * this strategy. In contrast to `userUnderlyingView`, this function **may** make state modifications
                             */
                            function userUnderlying(address user) external returns (uint256);
                            /**
                             * @notice convenience function for fetching the current total shares of `user` in this strategy, by
                             * querying the `strategyManager` contract
                             */
                            function shares(address user) external view returns (uint256);
                            /**
                             * @notice Used to convert a number of shares to the equivalent amount of underlying tokens for this strategy.
                             * @notice In contrast to `sharesToUnderlying`, this function guarantees no state modifications
                             * @param amountShares is the amount of shares to calculate its conversion into the underlying token
                             * @return The amount of shares corresponding to the input `amountUnderlying`
                             * @dev Implementation for these functions in particular may vary significantly for different strategies
                             */
                            function sharesToUnderlyingView(uint256 amountShares) external view returns (uint256);
                            /**
                             * @notice Used to convert an amount of underlying tokens to the equivalent amount of shares in this strategy.
                             * @notice In contrast to `underlyingToShares`, this function guarantees no state modifications
                             * @param amountUnderlying is the amount of `underlyingToken` to calculate its conversion into strategy shares
                             * @return The amount of shares corresponding to the input `amountUnderlying`
                             * @dev Implementation for these functions in particular may vary significantly for different strategies
                             */
                            function underlyingToSharesView(uint256 amountUnderlying) external view returns (uint256);
                            /**
                             * @notice convenience function for fetching the current underlying value of all of the `user`'s shares in
                             * this strategy. In contrast to `userUnderlying`, this function guarantees no state modifications
                             */
                            function userUnderlyingView(address user) external view returns (uint256);
                            /// @notice The underlying token for shares in this Strategy
                            function underlyingToken() external view returns (IERC20);
                            /// @notice The total number of extant shares in this Strategy
                            function totalShares() external view returns (uint256);
                            /// @notice Returns either a brief string explaining the strategy's goal & purpose, or a link to metadata that explains in more detail.
                            function explanation() external view returns (string memory);
                        }// SPDX-License-Identifier: GPL-3.0-or-later
                        pragma solidity 0.8.21;
                        interface ISSVNetworkCore {
                            /***********/
                            /* Structs */
                            /***********/
                            /// @notice Represents a snapshot of an operator's or a DAO's state at a certain block
                            struct Snapshot {
                                /// @dev The block number when the snapshot was taken
                                uint32 block;
                                /// @dev The last index calculated by the formula index += (currentBlock - block) * fee
                                uint64 index;
                                /// @dev Total accumulated earnings calculated by the formula accumulated + lastIndex * validatorCount
                                uint64 balance;
                            }
                            /// @notice Represents an SSV operator
                            struct Operator {
                                /// @dev The number of validators associated with this operator
                                uint32 validatorCount;
                                /// @dev The fee charged by the operator, set to zero for private operators and cannot be increased once set
                                uint64 fee;
                                /// @dev The address of the operator's owner
                                address owner;
                                /// @dev Whitelisted flag for this operator
                                bool whitelisted;
                                /// @dev The state snapshot of the operator
                                Snapshot snapshot;
                            }
                            /// @notice Represents a request to change an operator's fee
                            struct OperatorFeeChangeRequest {
                                /// @dev The new fee proposed by the operator
                                uint64 fee;
                                /// @dev The time when the approval period for the fee change begins
                                uint64 approvalBeginTime;
                                /// @dev The time when the approval period for the fee change ends
                                uint64 approvalEndTime;
                            }
                            /// @notice Represents a cluster of validators
                            struct Cluster {
                                /// @dev The number of validators in the cluster
                                uint32 validatorCount;
                                /// @dev The index of network fees related to this cluster
                                uint64 networkFeeIndex;
                                /// @dev The last index calculated for the cluster
                                uint64 index;
                                /// @dev Flag indicating whether the cluster is active
                                bool active;
                                /// @dev The balance of the cluster
                                uint256 balance;
                            }
                            /**********/
                            /* Errors */
                            /**********/
                            error CallerNotOwner(); // 0x5cd83192
                            error CallerNotWhitelisted(); // 0x8c6e5d71
                            error FeeTooLow(); // 0x732f9413
                            error FeeExceedsIncreaseLimit(); // 0x958065d9
                            error NoFeeDeclared(); // 0x1d226c30
                            error ApprovalNotWithinTimeframe(); // 0x97e4b518
                            error OperatorDoesNotExist(); // 0x961e3e8c
                            error InsufficientBalance(); // 0xf4d678b8
                            error ValidatorDoesNotExist(); // 0xe51315d2
                            error ClusterNotLiquidatable(); // 0x60300a8d
                            error InvalidPublicKeyLength(); // 0x637297a4
                            error InvalidOperatorIdsLength(); // 0x38186224
                            error ClusterAlreadyEnabled(); // 0x3babafd2
                            error ClusterIsLiquidated(); // 0x95a0cf33
                            error ClusterDoesNotExists(); // 0x185e2b16
                            error IncorrectClusterState(); // 0x12e04c87
                            error UnsortedOperatorsList(); // 0xdd020e25
                            error NewBlockPeriodIsBelowMinimum(); // 0x6e6c9cac
                            error ExceedValidatorLimit(); // 0x6df5ab76
                            error TokenTransferFailed(); // 0x045c4b02
                            error SameFeeChangeNotAllowed(); // 0xc81272f8
                            error FeeIncreaseNotAllowed(); // 0x410a2b6c
                            error NotAuthorized(); // 0xea8e4eb5
                            error OperatorsListNotUnique(); // 0xa5a1ff5d
                            error OperatorAlreadyExists(); // 0x289c9494
                            error TargetModuleDoesNotExist(); // 0x8f9195fb
                            error MaxValueExceeded(); // 0x91aa3017
                            error FeeTooHigh(); // 0xcd4e6167
                            error PublicKeysSharesLengthMismatch(); // 0x9ad467b8
                            error IncorrectValidatorStateWithData(bytes publicKey); // 0x89307938
                            error ValidatorAlreadyExistsWithData(bytes publicKey); // 0x388e7999
                            // legacy errors
                            error ValidatorAlreadyExists(); // 0x8d09a73e
                            error IncorrectValidatorState(); // 0x2feda3c1
                        }// SPDX-License-Identifier: MIT
                        // OpenZeppelin Contracts v4.4.1 (utils/Context.sol)
                        pragma solidity ^0.8.0;
                        import "../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;
                            }
                            /**
                             * @dev This empty reserved space is put in place to allow future versions to add new
                             * variables without shifting down storage in the inheritance chain.
                             * See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps
                             */
                            uint256[50] private __gap;
                        }
                        // SPDX-License-Identifier: MIT
                        // OpenZeppelin Contracts (last updated v4.9.0) (proxy/utils/Initializable.sol)
                        pragma solidity ^0.8.2;
                        import "../../utils/AddressUpgradeable.sol";
                        /**
                         * @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 Indicates that the contract has been initialized.
                             * @custom:oz-retyped-from bool
                             */
                            uint8 private _initialized;
                            /**
                             * @dev Indicates that the contract is in the process of being initialized.
                             */
                            bool private _initializing;
                            /**
                             * @dev Triggered when the contract has been initialized or reinitialized.
                             */
                            event Initialized(uint8 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 functions marked with `initializer` can be nested in the context of a
                             * constructor.
                             *
                             * Emits an {Initialized} event.
                             */
                            modifier initializer() {
                                bool isTopLevelCall = !_initializing;
                                require(
                                    (isTopLevelCall && _initialized < 1) || (!AddressUpgradeable.isContract(address(this)) && _initialized == 1),
                                    "Initializable: contract is already initialized"
                                );
                                _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 255 will prevent any future reinitialization.
                             *
                             * Emits an {Initialized} event.
                             */
                            modifier reinitializer(uint8 version) {
                                require(!_initializing && _initialized < version, "Initializable: contract is already initialized");
                                _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() {
                                require(_initializing, "Initializable: contract is not initializing");
                                _;
                            }
                            /**
                             * @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 {
                                require(!_initializing, "Initializable: contract is initializing");
                                if (_initialized != type(uint8).max) {
                                    _initialized = type(uint8).max;
                                    emit Initialized(type(uint8).max);
                                }
                            }
                            /**
                             * @dev Returns the highest version that has been initialized. See {reinitializer}.
                             */
                            function _getInitializedVersion() internal view returns (uint8) {
                                return _initialized;
                            }
                            /**
                             * @dev Returns `true` if the contract is currently initializing. See {onlyInitializing}.
                             */
                            function _isInitializing() internal view returns (bool) {
                                return _initializing;
                            }
                        }
                        // SPDX-License-Identifier: BUSL-1.1
                        pragma solidity 0.8.21;
                        import "./IStrategy.sol";
                        import "./ISignatureUtils.sol";
                        /**
                         * @title DelegationManager
                         * @author Layr Labs, Inc.
                         * @notice Terms of Service: https://docs.eigenlayer.xyz/overview/terms-of-service
                         * @notice  This is the contract for delegation in EigenLayer. The main functionalities of this contract are
                         * - enabling anyone to register as an operator in EigenLayer
                         * - allowing operators to specify parameters related to stakers who delegate to them
                         * - enabling any staker to delegate its stake to the operator of its choice (a given staker can only delegate to a single operator at a time)
                         * - enabling a staker to undelegate its assets from the operator it is delegated to (performed as part of the withdrawal process, initiated through the StrategyManager)
                         */
                        interface IDelegationManager is ISignatureUtils {
                            // @notice Struct used for storing information about a single operator who has registered with EigenLayer
                            struct OperatorDetails {
                                // @notice address to receive the rewards that the operator earns via serving applications built on EigenLayer.
                                address earningsReceiver;
                                /**
                                 * @notice Address to verify signatures when a staker wishes to delegate to the operator, as well as controlling "forced undelegations".
                                 * @dev Signature verification follows these rules:
                                 * 1) If this address is left as address(0), then any staker will be free to delegate to the operator, i.e. no signature verification will be performed.
                                 * 2) If this address is an EOA (i.e. it has no code), then we follow standard ECDSA signature verification for delegations to the operator.
                                 * 3) If this address is a contract (i.e. it has code) then we forward a call to the contract and verify that it returns the correct EIP-1271 "magic value".
                                 */
                                address delegationApprover;
                                /**
                                 * @notice A minimum delay -- measured in blocks -- enforced between:
                                 * 1) the operator signalling their intent to register for a service, via calling `Slasher.optIntoSlashing`
                                 * and
                                 * 2) the operator completing registration for the service, via the service ultimately calling `Slasher.recordFirstStakeUpdate`
                                 * @dev note that for a specific operator, this value *cannot decrease*, i.e. if the operator wishes to modify their OperatorDetails,
                                 * then they are only allowed to either increase this value or keep it the same.
                                 */
                                uint32 stakerOptOutWindowBlocks;
                            }
                            /**
                             * @notice Abstract struct used in calculating an EIP712 signature for a staker to approve that they (the staker themselves) delegate to a specific operator.
                             * @dev Used in computing the `STAKER_DELEGATION_TYPEHASH` and as a reference in the computation of the stakerDigestHash in the `delegateToBySignature` function.
                             */
                            struct StakerDelegation {
                                // the staker who is delegating
                                address staker;
                                // the operator being delegated to
                                address operator;
                                // the staker's nonce
                                uint256 nonce;
                                // the expiration timestamp (UTC) of the signature
                                uint256 expiry;
                            }
                            /**
                             * @notice Abstract struct used in calculating an EIP712 signature for an operator's delegationApprover to approve that a specific staker delegate to the operator.
                             * @dev Used in computing the `DELEGATION_APPROVAL_TYPEHASH` and as a reference in the computation of the approverDigestHash in the `_delegate` function.
                             */
                            struct DelegationApproval {
                                // the staker who is delegating
                                address staker;
                                // the operator being delegated to
                                address operator;
                                // the operator's provided salt
                                bytes32 salt;
                                // the expiration timestamp (UTC) of the signature
                                uint256 expiry;
                            }
                            /**
                             * Struct type used to specify an existing queued withdrawal. Rather than storing the entire struct, only a hash is stored.
                             * In functions that operate on existing queued withdrawals -- e.g. completeQueuedWithdrawal`, the data is resubmitted and the hash of the submitted
                             * data is computed by `calculateWithdrawalRoot` and checked against the stored hash in order to confirm the integrity of the submitted data.
                             */
                            struct Withdrawal {
                                // The address that originated the Withdrawal
                                address staker;
                                // The address that the staker was delegated to at the time that the Withdrawal was created
                                address delegatedTo;
                                // The address that can complete the Withdrawal + will receive funds when completing the withdrawal
                                address withdrawer;
                                // Nonce used to guarantee that otherwise identical withdrawals have unique hashes
                                uint256 nonce;
                                // Block number when the Withdrawal was created
                                uint32 startBlock;
                                // Array of strategies that the Withdrawal contains
                                IStrategy[] strategies;
                                // Array containing the amount of shares in each Strategy in the `strategies` array
                                uint256[] shares;
                            }
                            struct QueuedWithdrawalParams {
                                // Array of strategies that the QueuedWithdrawal contains
                                IStrategy[] strategies;
                                // Array containing the amount of shares in each Strategy in the `strategies` array
                                uint256[] shares;
                                // The address of the withdrawer
                                address withdrawer;
                            }
                            // @notice Emitted when a new operator registers in EigenLayer and provides their OperatorDetails.
                            event OperatorRegistered(address indexed operator, OperatorDetails operatorDetails);
                            /// @notice Emitted when an operator updates their OperatorDetails to @param newOperatorDetails
                            event OperatorDetailsModified(address indexed operator, OperatorDetails newOperatorDetails);
                            /**
                             * @notice Emitted when @param operator indicates that they are updating their MetadataURI string
                             * @dev Note that these strings are *never stored in storage* and are instead purely emitted in events for off-chain indexing
                             */
                            event OperatorMetadataURIUpdated(address indexed operator, string metadataURI);
                            /// @notice Emitted whenever an operator's shares are increased for a given strategy. Note that shares is the delta in the operator's shares.
                            event OperatorSharesIncreased(address indexed operator, address staker, IStrategy strategy, uint256 shares);
                            /// @notice Emitted whenever an operator's shares are decreased for a given strategy. Note that shares is the delta in the operator's shares.
                            event OperatorSharesDecreased(address indexed operator, address staker, IStrategy strategy, uint256 shares);
                            /// @notice Emitted when @param staker delegates to @param operator.
                            event StakerDelegated(address indexed staker, address indexed operator);
                            /// @notice Emitted when @param staker undelegates from @param operator.
                            event StakerUndelegated(address indexed staker, address indexed operator);
                            /// @notice Emitted when @param staker is undelegated via a call not originating from the staker themself
                            event StakerForceUndelegated(address indexed staker, address indexed operator);
                            /**
                             * @notice Emitted when a new withdrawal is queued.
                             * @param withdrawalRoot Is the hash of the `withdrawal`.
                             * @param withdrawal Is the withdrawal itself.
                             */
                            event WithdrawalQueued(bytes32 withdrawalRoot, Withdrawal withdrawal);
                            /// @notice Emitted when a queued withdrawal is completed
                            event WithdrawalCompleted(bytes32 withdrawalRoot);
                            /// @notice Emitted when a queued withdrawal is *migrated* from the StrategyManager to the DelegationManager
                            event WithdrawalMigrated(bytes32 oldWithdrawalRoot, bytes32 newWithdrawalRoot);
                            
                            /// @notice Emitted when the `minWithdrawalDelayBlocks` variable is modified from `previousValue` to `newValue`.
                            event MinWithdrawalDelayBlocksSet(uint256 previousValue, uint256 newValue);
                            /// @notice Emitted when the `strategyWithdrawalDelayBlocks` variable is modified from `previousValue` to `newValue`.
                            event StrategyWithdrawalDelayBlocksSet(IStrategy strategy, uint256 previousValue, uint256 newValue);
                            /**
                             * @notice Registers the caller as an operator in EigenLayer.
                             * @param registeringOperatorDetails is the `OperatorDetails` for the operator.
                             * @param metadataURI is a URI for the operator's metadata, i.e. a link providing more details on the operator.
                             *
                             * @dev Once an operator is registered, they cannot 'deregister' as an operator, and they will forever be considered "delegated to themself".
                             * @dev This function will revert if the caller attempts to set their `earningsReceiver` to address(0).
                             * @dev Note that the `metadataURI` is *never stored * and is only emitted in the `OperatorMetadataURIUpdated` event
                             */
                            function registerAsOperator(
                                OperatorDetails calldata registeringOperatorDetails,
                                string calldata metadataURI
                            ) external;
                            /**
                             * @notice Updates an operator's stored `OperatorDetails`.
                             * @param newOperatorDetails is the updated `OperatorDetails` for the operator, to replace their current OperatorDetails`.
                             *
                             * @dev The caller must have previously registered as an operator in EigenLayer.
                             * @dev This function will revert if the caller attempts to set their `earningsReceiver` to address(0).
                             */
                            function modifyOperatorDetails(OperatorDetails calldata newOperatorDetails) external;
                            /**
                             * @notice Called by an operator to emit an `OperatorMetadataURIUpdated` event indicating the information has updated.
                             * @param metadataURI The URI for metadata associated with an operator
                             * @dev Note that the `metadataURI` is *never stored * and is only emitted in the `OperatorMetadataURIUpdated` event
                             */
                            function updateOperatorMetadataURI(string calldata metadataURI) external;
                            /**
                             * @notice Caller delegates their stake to an operator.
                             * @param operator The account (`msg.sender`) is delegating its assets to for use in serving applications built on EigenLayer.
                             * @param approverSignatureAndExpiry Verifies the operator approves of this delegation
                             * @param approverSalt A unique single use value tied to an individual signature.
                             * @dev The approverSignatureAndExpiry is used in the event that:
                             *          1) the operator's `delegationApprover` address is set to a non-zero value.
                             *                  AND
                             *          2) neither the operator nor their `delegationApprover` is the `msg.sender`, since in the event that the operator
                             *             or their delegationApprover is the `msg.sender`, then approval is assumed.
                             * @dev In the event that `approverSignatureAndExpiry` is not checked, its content is ignored entirely; it's recommended to use an empty input
                             * in this case to save on complexity + gas costs
                             */
                            function delegateTo(
                                address operator,
                                SignatureWithExpiry memory approverSignatureAndExpiry,
                                bytes32 approverSalt
                            ) external;
                            /**
                             * @notice Caller delegates a staker's stake to an operator with valid signatures from both parties.
                             * @param staker The account delegating stake to an `operator` account
                             * @param operator The account (`staker`) is delegating its assets to for use in serving applications built on EigenLayer.
                             * @param stakerSignatureAndExpiry Signed data from the staker authorizing delegating stake to an operator
                             * @param approverSignatureAndExpiry is a parameter that will be used for verifying that the operator approves of this delegation action in the event that:
                             * @param approverSalt Is a salt used to help guarantee signature uniqueness. Each salt can only be used once by a given approver.
                             *
                             * @dev If `staker` is an EOA, then `stakerSignature` is verified to be a valid ECDSA stakerSignature from `staker`, indicating their intention for this action.
                             * @dev If `staker` is a contract, then `stakerSignature` will be checked according to EIP-1271.
                             * @dev the operator's `delegationApprover` address is set to a non-zero value.
                             * @dev neither the operator nor their `delegationApprover` is the `msg.sender`, since in the event that the operator or their delegationApprover
                             * is the `msg.sender`, then approval is assumed.
                             * @dev This function will revert if the current `block.timestamp` is equal to or exceeds the expiry
                             * @dev In the case that `approverSignatureAndExpiry` is not checked, its content is ignored entirely; it's recommended to use an empty input
                             * in this case to save on complexity + gas costs
                             */
                            function delegateToBySignature(
                                address staker,
                                address operator,
                                SignatureWithExpiry memory stakerSignatureAndExpiry,
                                SignatureWithExpiry memory approverSignatureAndExpiry,
                                bytes32 approverSalt
                            ) external;
                            /**
                             * @notice Undelegates the staker from the operator who they are delegated to. Puts the staker into the "undelegation limbo" mode of the EigenPodManager
                             * and queues a withdrawal of all of the staker's shares in the StrategyManager (to the staker), if necessary.
                             * @param staker The account to be undelegated.
                             * @return withdrawalRoot The root of the newly queued withdrawal, if a withdrawal was queued. Otherwise just bytes32(0).
                             *
                             * @dev Reverts if the `staker` is also an operator, since operators are not allowed to undelegate from themselves.
                             * @dev Reverts if the caller is not the staker, nor the operator who the staker is delegated to, nor the operator's specified "delegationApprover"
                             * @dev Reverts if the `staker` is already undelegated.
                             */
                            function undelegate(address staker) external returns (bytes32[] memory withdrawalRoot);
                            /**
                             * Allows a staker to withdraw some shares. Withdrawn shares/strategies are immediately removed
                             * from the staker. If the staker is delegated, withdrawn shares/strategies are also removed from
                             * their operator.
                             *
                             * All withdrawn shares/strategies are placed in a queue and can be fully withdrawn after a delay.
                             */
                            function queueWithdrawals(
                                QueuedWithdrawalParams[] calldata queuedWithdrawalParams
                            ) external returns (bytes32[] memory);
                            /**
                             * @notice Used to complete the specified `withdrawal`. The caller must match `withdrawal.withdrawer`
                             * @param withdrawal The Withdrawal to complete.
                             * @param tokens Array in which the i-th entry specifies the `token` input to the 'withdraw' function of the i-th Strategy in the `withdrawal.strategies` array.
                             * This input can be provided with zero length if `receiveAsTokens` is set to 'false' (since in that case, this input will be unused)
                             * @param middlewareTimesIndex is the index in the operator that the staker who triggered the withdrawal was delegated to's middleware times array
                             * @param receiveAsTokens If true, the shares specified in the withdrawal will be withdrawn from the specified strategies themselves
                             * and sent to the caller, through calls to `withdrawal.strategies[i].withdraw`. If false, then the shares in the specified strategies
                             * will simply be transferred to the caller directly.
                             * @dev middlewareTimesIndex should be calculated off chain before calling this function by finding the first index that satisfies `slasher.canWithdraw`
                             * @dev beaconChainETHStrategy shares are non-transferrable, so if `receiveAsTokens = false` and `withdrawal.withdrawer != withdrawal.staker`, note that
                             * any beaconChainETHStrategy shares in the `withdrawal` will be _returned to the staker_, rather than transferred to the withdrawer, unlike shares in
                             * any other strategies, which will be transferred to the withdrawer.
                             */
                            function completeQueuedWithdrawal(
                                Withdrawal calldata withdrawal,
                                IERC20[] calldata tokens,
                                uint256 middlewareTimesIndex,
                                bool receiveAsTokens
                            ) external;
                            /**
                             * @notice Array-ified version of `completeQueuedWithdrawal`.
                             * Used to complete the specified `withdrawals`. The function caller must match `withdrawals[...].withdrawer`
                             * @param withdrawals The Withdrawals to complete.
                             * @param tokens Array of tokens for each Withdrawal. See `completeQueuedWithdrawal` for the usage of a single array.
                             * @param middlewareTimesIndexes One index to reference per Withdrawal. See `completeQueuedWithdrawal` for the usage of a single index.
                             * @param receiveAsTokens Whether or not to complete each withdrawal as tokens. See `completeQueuedWithdrawal` for the usage of a single boolean.
                             * @dev See `completeQueuedWithdrawal` for relevant dev tags
                             */
                            function completeQueuedWithdrawals(
                                Withdrawal[] calldata withdrawals,
                                IERC20[][] calldata tokens,
                                uint256[] calldata middlewareTimesIndexes,
                                bool[] calldata receiveAsTokens
                            ) external;
                            /**
                             * @notice Increases a staker's delegated share balance in a strategy.
                             * @param staker The address to increase the delegated shares for their operator.
                             * @param strategy The strategy in which to increase the delegated shares.
                             * @param shares The number of shares to increase.
                             *
                             * @dev *If the staker is actively delegated*, then increases the `staker`'s delegated shares in `strategy` by `shares`. Otherwise does nothing.
                             * @dev Callable only by the StrategyManager or EigenPodManager.
                             */
                            function increaseDelegatedShares(
                                address staker,
                                IStrategy strategy,
                                uint256 shares
                            ) external;
                            /**
                             * @notice Decreases a staker's delegated share balance in a strategy.
                             * @param staker The address to increase the delegated shares for their operator.
                             * @param strategy The strategy in which to decrease the delegated shares.
                             * @param shares The number of shares to decrease.
                             *
                             * @dev *If the staker is actively delegated*, then decreases the `staker`'s delegated shares in `strategy` by `shares`. Otherwise does nothing.
                             * @dev Callable only by the StrategyManager or EigenPodManager.
                             */
                            function decreaseDelegatedShares(
                                address staker,
                                IStrategy strategy,
                                uint256 shares
                            ) external;
                            /**
                             * @notice returns the address of the operator that `staker` is delegated to.
                             * @notice Mapping: staker => operator whom the staker is currently delegated to.
                             * @dev Note that returning address(0) indicates that the staker is not actively delegated to any operator.
                             */
                            function delegatedTo(address staker) external view returns (address);
                            /**
                             * @notice Returns the OperatorDetails struct associated with an `operator`.
                             */
                            function operatorDetails(address operator) external view returns (OperatorDetails memory);
                            /*
                             * @notice Returns the earnings receiver address for an operator
                             */
                            function earningsReceiver(address operator) external view returns (address);
                            /**
                             * @notice Returns the delegationApprover account for an operator
                             */
                            function delegationApprover(address operator) external view returns (address);
                            /**
                             * @notice Returns the stakerOptOutWindowBlocks for an operator
                             */
                            function stakerOptOutWindowBlocks(address operator) external view returns (uint256);
                            /**
                             * @notice Given array of strategies, returns array of shares for the operator
                             */
                            function getOperatorShares(
                                address operator,
                                IStrategy[] memory strategies
                            ) external view returns (uint256[] memory);
                            /**
                             * @notice Given a list of strategies, return the minimum number of blocks that must pass to withdraw
                             * from all the inputted strategies. Return value is >= minWithdrawalDelayBlocks as this is the global min withdrawal delay.
                             * @param strategies The strategies to check withdrawal delays for
                             */
                            function getWithdrawalDelay(IStrategy[] calldata strategies) external view returns (uint256);
                            /**
                             * @notice returns the total number of shares in `strategy` that are delegated to `operator`.
                             * @notice Mapping: operator => strategy => total number of shares in the strategy delegated to the operator.
                             * @dev By design, the following invariant should hold for each Strategy:
                             * (operator's shares in delegation manager) = sum (shares above zero of all stakers delegated to operator)
                             * = sum (delegateable shares of all stakers delegated to the operator)
                             */
                            function operatorShares(address operator, IStrategy strategy) external view returns (uint256);
                            /**
                             * @notice Returns 'true' if `staker` *is* actively delegated, and 'false' otherwise.
                             */
                            function isDelegated(address staker) external view returns (bool);
                            /**
                             * @notice Returns true is an operator has previously registered for delegation.
                             */
                            function isOperator(address operator) external view returns (bool);
                            /// @notice Mapping: staker => number of signed delegation nonces (used in `delegateToBySignature`) from the staker that the contract has already checked
                            function stakerNonce(address staker) external view returns (uint256);
                            /**
                             * @notice Mapping: delegationApprover => 32-byte salt => whether or not the salt has already been used by the delegationApprover.
                             * @dev Salts are used in the `delegateTo` and `delegateToBySignature` functions. Note that these functions only process the delegationApprover's
                             * signature + the provided salt if the operator being delegated to has specified a nonzero address as their `delegationApprover`.
                             */
                            function delegationApproverSaltIsSpent(address _delegationApprover, bytes32 salt) external view returns (bool);
                            /**
                             * @notice Minimum delay enforced by this contract for completing queued withdrawals. Measured in blocks, and adjustable by this contract's owner,
                             * up to a maximum of `MAX_WITHDRAWAL_DELAY_BLOCKS`. Minimum value is 0 (i.e. no delay enforced).
                             * Note that strategies each have a separate withdrawal delay, which can be greater than this value. So the minimum number of blocks that must pass
                             * to withdraw a strategy is MAX(minWithdrawalDelayBlocks, strategyWithdrawalDelayBlocks[strategy])
                             */
                            function minWithdrawalDelayBlocks() external view returns (uint256);
                            /**
                             * @notice Minimum delay enforced by this contract per Strategy for completing queued withdrawals. Measured in blocks, and adjustable by this contract's owner,
                             * up to a maximum of `MAX_WITHDRAWAL_DELAY_BLOCKS`. Minimum value is 0 (i.e. no delay enforced).
                             */
                            function strategyWithdrawalDelayBlocks(IStrategy strategy) external view returns (uint256);
                            /**
                             * @notice Calculates the digestHash for a `staker` to sign to delegate to an `operator`
                             * @param staker The signing staker
                             * @param operator The operator who is being delegated to
                             * @param expiry The desired expiry time of the staker's signature
                             */
                            function calculateCurrentStakerDelegationDigestHash(
                                address staker,
                                address operator,
                                uint256 expiry
                            ) external view returns (bytes32);
                            /**
                             * @notice Calculates the digest hash to be signed and used in the `delegateToBySignature` function
                             * @param staker The signing staker
                             * @param _stakerNonce The nonce of the staker. In practice we use the staker's current nonce, stored at `stakerNonce[staker]`
                             * @param operator The operator who is being delegated to
                             * @param expiry The desired expiry time of the staker's signature
                             */
                            function calculateStakerDelegationDigestHash(
                                address staker,
                                uint256 _stakerNonce,
                                address operator,
                                uint256 expiry
                            ) external view returns (bytes32);
                            /**
                             * @notice Calculates the digest hash to be signed by the operator's delegationApprove and used in the `delegateTo` and `delegateToBySignature` functions.
                             * @param staker The account delegating their stake
                             * @param operator The account receiving delegated stake
                             * @param _delegationApprover the operator's `delegationApprover` who will be signing the delegationHash (in general)
                             * @param approverSalt A unique and single use value associated with the approver signature.
                             * @param expiry Time after which the approver's signature becomes invalid
                             */
                            function calculateDelegationApprovalDigestHash(
                                address staker,
                                address operator,
                                address _delegationApprover,
                                bytes32 approverSalt,
                                uint256 expiry
                            ) external view returns (bytes32);
                            /// @notice The EIP-712 typehash for the contract's domain
                            function DOMAIN_TYPEHASH() external view returns (bytes32);
                            /// @notice The EIP-712 typehash for the StakerDelegation struct used by the contract
                            function STAKER_DELEGATION_TYPEHASH() external view returns (bytes32);
                            /// @notice The EIP-712 typehash for the DelegationApproval struct used by the contract
                            function DELEGATION_APPROVAL_TYPEHASH() external view returns (bytes32);
                            /**
                             * @notice Getter function for the current EIP-712 domain separator for this contract.
                             *
                             * @dev The domain separator will change in the event of a fork that changes the ChainID.
                             * @dev By introducing a domain separator the DApp developers are guaranteed that there can be no signature collision.
                             * for more detailed information please read EIP-712.
                             */
                            function domainSeparator() external view returns (bytes32);
                            
                            /// @notice Mapping: staker => cumulative number of queued withdrawals they have ever initiated.
                            /// @dev This only increments (doesn't decrement), and is used to help ensure that otherwise identical withdrawals have unique hashes.
                            function cumulativeWithdrawalsQueued(address staker) external view returns (uint256);
                            /// @notice Returns the keccak256 hash of `withdrawal`.
                            function calculateWithdrawalRoot(Withdrawal memory withdrawal) external pure returns (bytes32);
                            function getDelegatableShares(address staker) external view returns (IStrategy[] memory, uint256[] memory);
                        }// SPDX-License-Identifier: MIT
                        // OpenZeppelin Contracts v4.4.1 (interfaces/IERC20.sol)
                        pragma solidity ^0.8.0;
                        import "../token/ERC20/IERC20.sol";
                        // SPDX-License-Identifier: MIT
                        // OpenZeppelin Contracts (last updated v4.9.0) (utils/Address.sol)
                        pragma solidity ^0.8.1;
                        /**
                         * @dev Collection of functions related to the address type
                         */
                        library AddressUpgradeable {
                            /**
                             * @dev Returns true if `account` is a contract.
                             *
                             * [IMPORTANT]
                             * ====
                             * It is unsafe to assume that an address for which this function returns
                             * false is an externally-owned account (EOA) and not a contract.
                             *
                             * Among others, `isContract` will return false for the following
                             * types of addresses:
                             *
                             *  - an externally-owned account
                             *  - a contract in construction
                             *  - an address where a contract will be created
                             *  - an address where a contract lived, but was destroyed
                             *
                             * Furthermore, `isContract` will also return true if the target contract within
                             * the same transaction is already scheduled for destruction by `SELFDESTRUCT`,
                             * which only has an effect at the end of a transaction.
                             * ====
                             *
                             * [IMPORTANT]
                             * ====
                             * You shouldn't rely on `isContract` to protect against flash loan attacks!
                             *
                             * Preventing calls from contracts is highly discouraged. It breaks composability, breaks support for smart wallets
                             * like Gnosis Safe, and does not provide security since it can be circumvented by calling from a contract
                             * constructor.
                             * ====
                             */
                            function isContract(address account) internal view returns (bool) {
                                // This method relies on extcodesize/address.code.length, which returns 0
                                // for contracts in construction, since the code is only stored at the end
                                // of the constructor execution.
                                return account.code.length > 0;
                            }
                            /**
                             * @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.0/security-considerations.html#use-the-checks-effects-interactions-pattern[checks-effects-interactions pattern].
                             */
                            function sendValue(address payable recipient, uint256 amount) internal {
                                require(address(this).balance >= amount, "Address: insufficient balance");
                                (bool success, ) = recipient.call{value: amount}("");
                                require(success, "Address: unable to send value, recipient may have reverted");
                            }
                            /**
                             * @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, it is bubbled up by this
                             * function (like regular Solidity function calls).
                             *
                             * 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.
                             *
                             * _Available since v3.1._
                             */
                            function functionCall(address target, bytes memory data) internal returns (bytes memory) {
                                return functionCallWithValue(target, data, 0, "Address: low-level call failed");
                            }
                            /**
                             * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], but with
                             * `errorMessage` as a fallback revert reason when `target` reverts.
                             *
                             * _Available since v3.1._
                             */
                            function functionCall(
                                address target,
                                bytes memory data,
                                string memory errorMessage
                            ) internal returns (bytes memory) {
                                return functionCallWithValue(target, data, 0, errorMessage);
                            }
                            /**
                             * @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`.
                             *
                             * _Available since v3.1._
                             */
                            function functionCallWithValue(address target, bytes memory data, uint256 value) internal returns (bytes memory) {
                                return functionCallWithValue(target, data, value, "Address: low-level call with value failed");
                            }
                            /**
                             * @dev Same as {xref-Address-functionCallWithValue-address-bytes-uint256-}[`functionCallWithValue`], but
                             * with `errorMessage` as a fallback revert reason when `target` reverts.
                             *
                             * _Available since v3.1._
                             */
                            function functionCallWithValue(
                                address target,
                                bytes memory data,
                                uint256 value,
                                string memory errorMessage
                            ) internal returns (bytes memory) {
                                require(address(this).balance >= value, "Address: insufficient balance for call");
                                (bool success, bytes memory returndata) = target.call{value: value}(data);
                                return verifyCallResultFromTarget(target, success, returndata, errorMessage);
                            }
                            /**
                             * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
                             * but performing a static call.
                             *
                             * _Available since v3.3._
                             */
                            function functionStaticCall(address target, bytes memory data) internal view returns (bytes memory) {
                                return functionStaticCall(target, data, "Address: low-level static call failed");
                            }
                            /**
                             * @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`],
                             * but performing a static call.
                             *
                             * _Available since v3.3._
                             */
                            function functionStaticCall(
                                address target,
                                bytes memory data,
                                string memory errorMessage
                            ) internal view returns (bytes memory) {
                                (bool success, bytes memory returndata) = target.staticcall(data);
                                return verifyCallResultFromTarget(target, success, returndata, errorMessage);
                            }
                            /**
                             * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
                             * but performing a delegate call.
                             *
                             * _Available since v3.4._
                             */
                            function functionDelegateCall(address target, bytes memory data) internal returns (bytes memory) {
                                return functionDelegateCall(target, data, "Address: low-level delegate call failed");
                            }
                            /**
                             * @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`],
                             * but performing a delegate call.
                             *
                             * _Available since v3.4._
                             */
                            function functionDelegateCall(
                                address target,
                                bytes memory data,
                                string memory errorMessage
                            ) internal returns (bytes memory) {
                                (bool success, bytes memory returndata) = target.delegatecall(data);
                                return verifyCallResultFromTarget(target, success, returndata, errorMessage);
                            }
                            /**
                             * @dev Tool to verify that a low level call to smart-contract was successful, and revert (either by bubbling
                             * the revert reason or using the provided one) in case of unsuccessful call or if target was not a contract.
                             *
                             * _Available since v4.8._
                             */
                            function verifyCallResultFromTarget(
                                address target,
                                bool success,
                                bytes memory returndata,
                                string memory errorMessage
                            ) internal view returns (bytes memory) {
                                if (success) {
                                    if (returndata.length == 0) {
                                        // only check isContract if the call was successful and the return data is empty
                                        // otherwise we already know that it was a contract
                                        require(isContract(target), "Address: call to non-contract");
                                    }
                                    return returndata;
                                } else {
                                    _revert(returndata, errorMessage);
                                }
                            }
                            /**
                             * @dev Tool to verify that a low level call was successful, and revert if it wasn't, either by bubbling the
                             * revert reason or using the provided one.
                             *
                             * _Available since v4.3._
                             */
                            function verifyCallResult(
                                bool success,
                                bytes memory returndata,
                                string memory errorMessage
                            ) internal pure returns (bytes memory) {
                                if (success) {
                                    return returndata;
                                } else {
                                    _revert(returndata, errorMessage);
                                }
                            }
                            function _revert(bytes memory returndata, string memory errorMessage) 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(errorMessage);
                                }
                            }
                        }
                        // SPDX-License-Identifier: BUSL-1.1
                        pragma solidity 0.8.21;
                        /**
                         * @title The interface for common signature utilities.
                         * @author Layr Labs, Inc.
                         * @notice Terms of Service: https://docs.eigenlayer.xyz/overview/terms-of-service
                         */
                        interface ISignatureUtils {
                            // @notice Struct that bundles together a signature and an expiration time for the signature. Used primarily for stack management.
                            struct SignatureWithExpiry {
                                // the signature itself, formatted as a single bytes object
                                bytes signature;
                                // the expiration timestamp (UTC) of the signature
                                uint256 expiry;
                            }
                            // @notice Struct that bundles together a signature, a salt for uniqueness, and an expiration time for the signature. Used primarily for stack management.
                            struct SignatureWithSaltAndExpiry {
                                // the signature itself, formatted as a single bytes object
                                bytes signature;
                                // the salt used to generate the signature
                                bytes32 salt;
                                // the expiration timestamp (UTC) of the signature
                                uint256 expiry;
                            }
                        }

                        File 4 of 6: TransparentUpgradeableProxy
                        // SPDX-License-Identifier: MIT
                        // OpenZeppelin Contracts (last updated v4.9.0) (proxy/transparent/TransparentUpgradeableProxy.sol)
                        pragma solidity ^0.8.0;
                        import "../ERC1967/ERC1967Proxy.sol";
                        /**
                         * @dev Interface for {TransparentUpgradeableProxy}. In order to implement transparency, {TransparentUpgradeableProxy}
                         * does not implement this interface directly, and some of its functions are implemented by an internal dispatch
                         * mechanism. The compiler is unaware that these functions are implemented by {TransparentUpgradeableProxy} and will not
                         * include them in the ABI so this interface must be used to interact with it.
                         */
                        interface ITransparentUpgradeableProxy is IERC1967 {
                            function admin() external view returns (address);
                            function implementation() external view returns (address);
                            function changeAdmin(address) external;
                            function upgradeTo(address) external;
                            function upgradeToAndCall(address, bytes memory) external payable;
                        }
                        /**
                         * @dev This contract implements a proxy that is upgradeable by an admin.
                         *
                         * To avoid https://medium.com/nomic-labs-blog/malicious-backdoors-in-ethereum-proxies-62629adf3357[proxy selector
                         * clashing], which can potentially be used in an attack, this contract uses the
                         * https://blog.openzeppelin.com/the-transparent-proxy-pattern/[transparent proxy pattern]. This pattern implies two
                         * things that go hand in hand:
                         *
                         * 1. If any account other than the admin calls the proxy, the call will be forwarded to the implementation, even if
                         * that call matches one of the admin functions exposed by the proxy itself.
                         * 2. If the admin calls the proxy, it can access the admin functions, but its calls will never be forwarded to the
                         * implementation. If the admin tries to call a function on the implementation it will fail with an error that says
                         * "admin cannot fallback to proxy target".
                         *
                         * These properties mean that the admin account can only be used for admin actions like upgrading the proxy or changing
                         * the admin, so it's best if it's a dedicated account that is not used for anything else. This will avoid headaches due
                         * to sudden errors when trying to call a function from the proxy implementation.
                         *
                         * Our recommendation is for the dedicated account to be an instance of the {ProxyAdmin} contract. If set up this way,
                         * you should think of the `ProxyAdmin` instance as the real administrative interface of your proxy.
                         *
                         * NOTE: The real interface of this proxy is that defined in `ITransparentUpgradeableProxy`. This contract does not
                         * inherit from that interface, and instead the admin functions are implicitly implemented using a custom dispatch
                         * mechanism in `_fallback`. Consequently, the compiler will not produce an ABI for this contract. This is necessary to
                         * fully implement transparency without decoding reverts caused by selector clashes between the proxy and the
                         * implementation.
                         *
                         * WARNING: It is not recommended to extend this contract to add additional external functions. If you do so, the compiler
                         * will not check that there are no selector conflicts, due to the note above. A selector clash between any new function
                         * and the functions declared in {ITransparentUpgradeableProxy} will be resolved in favor of the new one. This could
                         * render the admin operations inaccessible, which could prevent upgradeability. Transparency may also be compromised.
                         */
                        contract TransparentUpgradeableProxy is ERC1967Proxy {
                            /**
                             * @dev Initializes an upgradeable proxy managed by `_admin`, backed by the implementation at `_logic`, and
                             * optionally initialized with `_data` as explained in {ERC1967Proxy-constructor}.
                             */
                            constructor(address _logic, address admin_, bytes memory _data) payable ERC1967Proxy(_logic, _data) {
                                _changeAdmin(admin_);
                            }
                            /**
                             * @dev Modifier used internally that will delegate the call to the implementation unless the sender is the admin.
                             *
                             * CAUTION: This modifier is deprecated, as it could cause issues if the modified function has arguments, and the
                             * implementation provides a function with the same selector.
                             */
                            modifier ifAdmin() {
                                if (msg.sender == _getAdmin()) {
                                    _;
                                } else {
                                    _fallback();
                                }
                            }
                            /**
                             * @dev If caller is the admin process the call internally, otherwise transparently fallback to the proxy behavior
                             */
                            function _fallback() internal virtual override {
                                if (msg.sender == _getAdmin()) {
                                    bytes memory ret;
                                    bytes4 selector = msg.sig;
                                    if (selector == ITransparentUpgradeableProxy.upgradeTo.selector) {
                                        ret = _dispatchUpgradeTo();
                                    } else if (selector == ITransparentUpgradeableProxy.upgradeToAndCall.selector) {
                                        ret = _dispatchUpgradeToAndCall();
                                    } else if (selector == ITransparentUpgradeableProxy.changeAdmin.selector) {
                                        ret = _dispatchChangeAdmin();
                                    } else if (selector == ITransparentUpgradeableProxy.admin.selector) {
                                        ret = _dispatchAdmin();
                                    } else if (selector == ITransparentUpgradeableProxy.implementation.selector) {
                                        ret = _dispatchImplementation();
                                    } else {
                                        revert("TransparentUpgradeableProxy: admin cannot fallback to proxy target");
                                    }
                                    assembly {
                                        return(add(ret, 0x20), mload(ret))
                                    }
                                } else {
                                    super._fallback();
                                }
                            }
                            /**
                             * @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 _dispatchAdmin() private returns (bytes memory) {
                                _requireZeroValue();
                                address admin = _getAdmin();
                                return abi.encode(admin);
                            }
                            /**
                             * @dev Returns the current implementation.
                             *
                             * 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 _dispatchImplementation() private returns (bytes memory) {
                                _requireZeroValue();
                                address implementation = _implementation();
                                return abi.encode(implementation);
                            }
                            /**
                             * @dev Changes the admin of the proxy.
                             *
                             * Emits an {AdminChanged} event.
                             */
                            function _dispatchChangeAdmin() private returns (bytes memory) {
                                _requireZeroValue();
                                address newAdmin = abi.decode(msg.data[4:], (address));
                                _changeAdmin(newAdmin);
                                return "";
                            }
                            /**
                             * @dev Upgrade the implementation of the proxy.
                             */
                            function _dispatchUpgradeTo() private returns (bytes memory) {
                                _requireZeroValue();
                                address newImplementation = abi.decode(msg.data[4:], (address));
                                _upgradeToAndCall(newImplementation, bytes(""), false);
                                return "";
                            }
                            /**
                             * @dev Upgrade the implementation of the proxy, and then call a function from the new implementation as specified
                             * by `data`, which should be an encoded function call. This is useful to initialize new storage variables in the
                             * proxied contract.
                             */
                            function _dispatchUpgradeToAndCall() private returns (bytes memory) {
                                (address newImplementation, bytes memory data) = abi.decode(msg.data[4:], (address, bytes));
                                _upgradeToAndCall(newImplementation, data, true);
                                return "";
                            }
                            /**
                             * @dev Returns the current admin.
                             *
                             * CAUTION: This function is deprecated. Use {ERC1967Upgrade-_getAdmin} instead.
                             */
                            function _admin() internal view virtual returns (address) {
                                return _getAdmin();
                            }
                            /**
                             * @dev To keep this contract fully transparent, all `ifAdmin` functions must be payable. This helper is here to
                             * emulate some proxy functions being non-payable while still allowing value to pass through.
                             */
                            function _requireZeroValue() private {
                                require(msg.value == 0);
                            }
                        }
                        // SPDX-License-Identifier: MIT
                        // OpenZeppelin Contracts (last updated v4.7.0) (proxy/ERC1967/ERC1967Proxy.sol)
                        pragma solidity ^0.8.0;
                        import "../Proxy.sol";
                        import "./ERC1967Upgrade.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, ERC1967Upgrade {
                            /**
                             * @dev Initializes the upgradeable proxy with an initial implementation specified by `_logic`.
                             *
                             * If `_data` is nonempty, it's used as data in a delegate call to `_logic`. This will typically be an encoded
                             * function call, and allows initializing the storage of the proxy like a Solidity constructor.
                             */
                            constructor(address _logic, bytes memory _data) payable {
                                _upgradeToAndCall(_logic, _data, false);
                            }
                            /**
                             * @dev Returns the current implementation address.
                             */
                            function _implementation() internal view virtual override returns (address impl) {
                                return ERC1967Upgrade._getImplementation();
                            }
                        }
                        // SPDX-License-Identifier: MIT
                        // OpenZeppelin Contracts (last updated v4.6.0) (proxy/Proxy.sol)
                        pragma solidity ^0.8.0;
                        /**
                         * @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 {
                                _beforeFallback();
                                _delegate(_implementation());
                            }
                            /**
                             * @dev Fallback function that delegates calls to the address returned by `_implementation()`. Will run if no other
                             * function in the contract matches the call data.
                             */
                            fallback() external payable virtual {
                                _fallback();
                            }
                            /**
                             * @dev Fallback function that delegates calls to the address returned by `_implementation()`. Will run if call data
                             * is empty.
                             */
                            receive() external payable virtual {
                                _fallback();
                            }
                            /**
                             * @dev Hook that is called before falling back to the implementation. Can happen as part of a manual `_fallback`
                             * call, or as part of the Solidity `fallback` or `receive` functions.
                             *
                             * If overridden should call `super._beforeFallback()`.
                             */
                            function _beforeFallback() internal virtual {}
                        }
                        // SPDX-License-Identifier: MIT
                        // OpenZeppelin Contracts (last updated v4.9.0) (proxy/ERC1967/ERC1967Upgrade.sol)
                        pragma solidity ^0.8.2;
                        import "../beacon/IBeacon.sol";
                        import "../../interfaces/IERC1967.sol";
                        import "../../interfaces/draft-IERC1822.sol";
                        import "../../utils/Address.sol";
                        import "../../utils/StorageSlot.sol";
                        /**
                         * @dev This abstract contract provides getters and event emitting update functions for
                         * https://eips.ethereum.org/EIPS/eip-1967[EIP1967] slots.
                         *
                         * _Available since v4.1._
                         */
                        abstract contract ERC1967Upgrade is IERC1967 {
                            // This is the keccak-256 hash of "eip1967.proxy.rollback" subtracted by 1
                            bytes32 private constant _ROLLBACK_SLOT = 0x4910fdfa16fed3260ed0e7147f7cc6da11a60208b5b9406d12a635614ffd9143;
                            /**
                             * @dev Storage slot with the address of the current implementation.
                             * This is the keccak-256 hash of "eip1967.proxy.implementation" subtracted by 1, and is
                             * validated in the constructor.
                             */
                            bytes32 internal constant _IMPLEMENTATION_SLOT = 0x360894a13ba1a3210667c828492db98dca3e2076cc3735a920a3ca505d382bbc;
                            /**
                             * @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 {
                                require(Address.isContract(newImplementation), "ERC1967: new implementation is not a contract");
                                StorageSlot.getAddressSlot(_IMPLEMENTATION_SLOT).value = newImplementation;
                            }
                            /**
                             * @dev Perform implementation upgrade
                             *
                             * Emits an {Upgraded} event.
                             */
                            function _upgradeTo(address newImplementation) internal {
                                _setImplementation(newImplementation);
                                emit Upgraded(newImplementation);
                            }
                            /**
                             * @dev Perform implementation upgrade with additional setup call.
                             *
                             * Emits an {Upgraded} event.
                             */
                            function _upgradeToAndCall(address newImplementation, bytes memory data, bool forceCall) internal {
                                _upgradeTo(newImplementation);
                                if (data.length > 0 || forceCall) {
                                    Address.functionDelegateCall(newImplementation, data);
                                }
                            }
                            /**
                             * @dev Perform implementation upgrade with security checks for UUPS proxies, and additional setup call.
                             *
                             * Emits an {Upgraded} event.
                             */
                            function _upgradeToAndCallUUPS(address newImplementation, bytes memory data, bool forceCall) internal {
                                // Upgrades from old implementations will perform a rollback test. This test requires the new
                                // implementation to upgrade back to the old, non-ERC1822 compliant, implementation. Removing
                                // this special case will break upgrade paths from old UUPS implementation to new ones.
                                if (StorageSlot.getBooleanSlot(_ROLLBACK_SLOT).value) {
                                    _setImplementation(newImplementation);
                                } else {
                                    try IERC1822Proxiable(newImplementation).proxiableUUID() returns (bytes32 slot) {
                                        require(slot == _IMPLEMENTATION_SLOT, "ERC1967Upgrade: unsupported proxiableUUID");
                                    } catch {
                                        revert("ERC1967Upgrade: new implementation is not UUPS");
                                    }
                                    _upgradeToAndCall(newImplementation, data, forceCall);
                                }
                            }
                            /**
                             * @dev Storage slot with the admin of the contract.
                             * This is the keccak-256 hash of "eip1967.proxy.admin" subtracted by 1, and is
                             * validated in the constructor.
                             */
                            bytes32 internal constant _ADMIN_SLOT = 0xb53127684a568b3173ae13b9f8a6016e243e63b6e8ee1178d6a717850b5d6103;
                            /**
                             * @dev Returns the current admin.
                             */
                            function _getAdmin() internal view returns (address) {
                                return StorageSlot.getAddressSlot(_ADMIN_SLOT).value;
                            }
                            /**
                             * @dev Stores a new address in the EIP1967 admin slot.
                             */
                            function _setAdmin(address newAdmin) private {
                                require(newAdmin != address(0), "ERC1967: new admin is the zero address");
                                StorageSlot.getAddressSlot(_ADMIN_SLOT).value = newAdmin;
                            }
                            /**
                             * @dev Changes the admin of the proxy.
                             *
                             * Emits an {AdminChanged} event.
                             */
                            function _changeAdmin(address newAdmin) internal {
                                emit AdminChanged(_getAdmin(), newAdmin);
                                _setAdmin(newAdmin);
                            }
                            /**
                             * @dev The storage slot of the UpgradeableBeacon contract which defines the implementation for this proxy.
                             * This is bytes32(uint256(keccak256('eip1967.proxy.beacon')) - 1)) and is validated in the constructor.
                             */
                            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 {
                                require(Address.isContract(newBeacon), "ERC1967: new beacon is not a contract");
                                require(
                                    Address.isContract(IBeacon(newBeacon).implementation()),
                                    "ERC1967: beacon implementation is not a contract"
                                );
                                StorageSlot.getAddressSlot(_BEACON_SLOT).value = newBeacon;
                            }
                            /**
                             * @dev Perform beacon upgrade with additional setup call. Note: This upgrades the address of the beacon, it does
                             * not upgrade the implementation contained in the beacon (see {UpgradeableBeacon-_setImplementation} for that).
                             *
                             * Emits a {BeaconUpgraded} event.
                             */
                            function _upgradeBeaconToAndCall(address newBeacon, bytes memory data, bool forceCall) internal {
                                _setBeacon(newBeacon);
                                emit BeaconUpgraded(newBeacon);
                                if (data.length > 0 || forceCall) {
                                    Address.functionDelegateCall(IBeacon(newBeacon).implementation(), data);
                                }
                            }
                        }
                        // SPDX-License-Identifier: MIT
                        // OpenZeppelin Contracts v4.4.1 (proxy/beacon/IBeacon.sol)
                        pragma solidity ^0.8.0;
                        /**
                         * @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.
                             *
                             * {BeaconProxy} will check that this address is a contract.
                             */
                            function implementation() external view returns (address);
                        }
                        // SPDX-License-Identifier: MIT
                        // OpenZeppelin Contracts (last updated v4.9.0) (interfaces/IERC1967.sol)
                        pragma solidity ^0.8.0;
                        /**
                         * @dev ERC-1967: Proxy Storage Slots. This interface contains the events defined in the ERC.
                         *
                         * _Available since v4.8.3._
                         */
                        interface IERC1967 {
                            /**
                             * @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);
                        }
                        // SPDX-License-Identifier: MIT
                        // OpenZeppelin Contracts (last updated v4.5.0) (interfaces/draft-IERC1822.sol)
                        pragma solidity ^0.8.0;
                        /**
                         * @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 v4.9.0) (utils/Address.sol)
                        pragma solidity ^0.8.1;
                        /**
                         * @dev Collection of functions related to the address type
                         */
                        library Address {
                            /**
                             * @dev Returns true if `account` is a contract.
                             *
                             * [IMPORTANT]
                             * ====
                             * It is unsafe to assume that an address for which this function returns
                             * false is an externally-owned account (EOA) and not a contract.
                             *
                             * Among others, `isContract` will return false for the following
                             * types of addresses:
                             *
                             *  - an externally-owned account
                             *  - a contract in construction
                             *  - an address where a contract will be created
                             *  - an address where a contract lived, but was destroyed
                             *
                             * Furthermore, `isContract` will also return true if the target contract within
                             * the same transaction is already scheduled for destruction by `SELFDESTRUCT`,
                             * which only has an effect at the end of a transaction.
                             * ====
                             *
                             * [IMPORTANT]
                             * ====
                             * You shouldn't rely on `isContract` to protect against flash loan attacks!
                             *
                             * Preventing calls from contracts is highly discouraged. It breaks composability, breaks support for smart wallets
                             * like Gnosis Safe, and does not provide security since it can be circumvented by calling from a contract
                             * constructor.
                             * ====
                             */
                            function isContract(address account) internal view returns (bool) {
                                // This method relies on extcodesize/address.code.length, which returns 0
                                // for contracts in construction, since the code is only stored at the end
                                // of the constructor execution.
                                return account.code.length > 0;
                            }
                            /**
                             * @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.0/security-considerations.html#use-the-checks-effects-interactions-pattern[checks-effects-interactions pattern].
                             */
                            function sendValue(address payable recipient, uint256 amount) internal {
                                require(address(this).balance >= amount, "Address: insufficient balance");
                                (bool success, ) = recipient.call{value: amount}("");
                                require(success, "Address: unable to send value, recipient may have reverted");
                            }
                            /**
                             * @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, it is bubbled up by this
                             * function (like regular Solidity function calls).
                             *
                             * 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.
                             *
                             * _Available since v3.1._
                             */
                            function functionCall(address target, bytes memory data) internal returns (bytes memory) {
                                return functionCallWithValue(target, data, 0, "Address: low-level call failed");
                            }
                            /**
                             * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], but with
                             * `errorMessage` as a fallback revert reason when `target` reverts.
                             *
                             * _Available since v3.1._
                             */
                            function functionCall(
                                address target,
                                bytes memory data,
                                string memory errorMessage
                            ) internal returns (bytes memory) {
                                return functionCallWithValue(target, data, 0, errorMessage);
                            }
                            /**
                             * @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`.
                             *
                             * _Available since v3.1._
                             */
                            function functionCallWithValue(address target, bytes memory data, uint256 value) internal returns (bytes memory) {
                                return functionCallWithValue(target, data, value, "Address: low-level call with value failed");
                            }
                            /**
                             * @dev Same as {xref-Address-functionCallWithValue-address-bytes-uint256-}[`functionCallWithValue`], but
                             * with `errorMessage` as a fallback revert reason when `target` reverts.
                             *
                             * _Available since v3.1._
                             */
                            function functionCallWithValue(
                                address target,
                                bytes memory data,
                                uint256 value,
                                string memory errorMessage
                            ) internal returns (bytes memory) {
                                require(address(this).balance >= value, "Address: insufficient balance for call");
                                (bool success, bytes memory returndata) = target.call{value: value}(data);
                                return verifyCallResultFromTarget(target, success, returndata, errorMessage);
                            }
                            /**
                             * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
                             * but performing a static call.
                             *
                             * _Available since v3.3._
                             */
                            function functionStaticCall(address target, bytes memory data) internal view returns (bytes memory) {
                                return functionStaticCall(target, data, "Address: low-level static call failed");
                            }
                            /**
                             * @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`],
                             * but performing a static call.
                             *
                             * _Available since v3.3._
                             */
                            function functionStaticCall(
                                address target,
                                bytes memory data,
                                string memory errorMessage
                            ) internal view returns (bytes memory) {
                                (bool success, bytes memory returndata) = target.staticcall(data);
                                return verifyCallResultFromTarget(target, success, returndata, errorMessage);
                            }
                            /**
                             * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
                             * but performing a delegate call.
                             *
                             * _Available since v3.4._
                             */
                            function functionDelegateCall(address target, bytes memory data) internal returns (bytes memory) {
                                return functionDelegateCall(target, data, "Address: low-level delegate call failed");
                            }
                            /**
                             * @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`],
                             * but performing a delegate call.
                             *
                             * _Available since v3.4._
                             */
                            function functionDelegateCall(
                                address target,
                                bytes memory data,
                                string memory errorMessage
                            ) internal returns (bytes memory) {
                                (bool success, bytes memory returndata) = target.delegatecall(data);
                                return verifyCallResultFromTarget(target, success, returndata, errorMessage);
                            }
                            /**
                             * @dev Tool to verify that a low level call to smart-contract was successful, and revert (either by bubbling
                             * the revert reason or using the provided one) in case of unsuccessful call or if target was not a contract.
                             *
                             * _Available since v4.8._
                             */
                            function verifyCallResultFromTarget(
                                address target,
                                bool success,
                                bytes memory returndata,
                                string memory errorMessage
                            ) internal view returns (bytes memory) {
                                if (success) {
                                    if (returndata.length == 0) {
                                        // only check isContract if the call was successful and the return data is empty
                                        // otherwise we already know that it was a contract
                                        require(isContract(target), "Address: call to non-contract");
                                    }
                                    return returndata;
                                } else {
                                    _revert(returndata, errorMessage);
                                }
                            }
                            /**
                             * @dev Tool to verify that a low level call was successful, and revert if it wasn't, either by bubbling the
                             * revert reason or using the provided one.
                             *
                             * _Available since v4.3._
                             */
                            function verifyCallResult(
                                bool success,
                                bytes memory returndata,
                                string memory errorMessage
                            ) internal pure returns (bytes memory) {
                                if (success) {
                                    return returndata;
                                } else {
                                    _revert(returndata, errorMessage);
                                }
                            }
                            function _revert(bytes memory returndata, string memory errorMessage) 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(errorMessage);
                                }
                            }
                        }
                        // SPDX-License-Identifier: MIT
                        // OpenZeppelin Contracts (last updated v4.9.0) (utils/StorageSlot.sol)
                        // This file was procedurally generated from scripts/generate/templates/StorageSlot.js.
                        pragma solidity ^0.8.0;
                        /**
                         * @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(Address.isContract(newImplementation), "ERC1967: new implementation is not a contract");
                         *         StorageSlot.getAddressSlot(_IMPLEMENTATION_SLOT).value = newImplementation;
                         *     }
                         * }
                         * ```
                         *
                         * _Available since v4.1 for `address`, `bool`, `bytes32`, `uint256`._
                         * _Available since v4.9 for `string`, `bytes`._
                         */
                        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 5 of 6: EigenpieConfig
                        // SPDX-License-Identifier: GPL-3.0-or-later
                        pragma solidity 0.8.21;
                        import { UtilLib } from "./utils/UtilLib.sol";
                        import { EigenpieConstants } from "./utils/EigenpieConstants.sol";
                        import { IEigenpieConfig } from "./interfaces/IEigenpieConfig.sol";
                        import { IMLRT } from "./interfaces/IMLRT.sol";
                        import { IStrategy } from "./interfaces/IStrategy.sol";
                        import { AccessControlUpgradeable } from "@openzeppelin/contracts-upgradeable/access/AccessControlUpgradeable.sol";
                        /// @title EigenpieConfig - Eigenpie Config Contract
                        /// @notice Handles Eigenpie configuration
                        contract EigenpieConfig is IEigenpieConfig, AccessControlUpgradeable {
                            mapping(bytes32 contractKey => address contractAddress) public contractMap;
                            mapping(address token => bool isSupported) public isSupportedAsset;
                            mapping(address token => uint256 amount) public depositLimitByAsset;
                            mapping(address token => address mLRTReceipt) public mLRTReceiptByAsset;
                            mapping(address token => uint256 boost) public boostByAsset;
                            mapping(address token => address strategy) public override assetStrategy;
                            address[] public supportedAssetList;
                            /// @custom:oz-upgrades-unsafe-allow constructor
                            constructor() {
                                _disableInitializers();
                            }
                            /*/////////////////// ERROR /////////////////////*/
                            error InvalidAsset();
                            /*/////////////////// END /////////////////////*/
                            modifier onlySupportedAsset(address asset) {
                                if (!isSupportedAsset[asset]) {
                                    revert AssetNotSupported();
                                }
                                _;
                            }
                            /// @dev Initializes the contract
                            /// @param admin Admin address
                            function initialize(address admin) external initializer {
                                UtilLib.checkNonZeroAddress(admin);
                                __AccessControl_init();
                                _grantRole(DEFAULT_ADMIN_ROLE, admin);
                            }
                            /// @dev Adds a new supported asset
                            /// @param asset Asset address
                            /// @param mLRTReceipt MLRT receipt
                            /// @param depositLimit Deposit limit for the asset
                            function addNewSupportedAsset(
                                address asset,
                                address mLRTReceipt,
                                uint256 depositLimit
                            )
                                external
                                onlyRole(EigenpieConstants.MANAGER)
                            {
                                _addNewSupportedAsset(asset, mLRTReceipt, depositLimit);
                            }
                            /// @dev Adds a new supported asset
                            /// @param asset Asset address
                            /// @param mLRTReceipt MLRT receipt
                            function updateReceiptToken(
                                address asset,
                                address mLRTReceipt
                            )
                                external
                                onlyRole(EigenpieConstants.DEFAULT_ADMIN_ROLE)
                            {
                                if (!isSupportedAsset[asset]) {
                                    revert AssetNotSupported();
                                }
                                if (asset != IMLRT(mLRTReceipt).underlyingAsset()) revert InvalidAsset();
                                mLRTReceiptByAsset[asset] = mLRTReceipt;
                                emit ReceiptTokenUpdated(asset, mLRTReceipt);
                            }
                            /// @dev private function to add a new supported asset
                            /// @param asset Asset address
                            /// @param depositLimit Deposit limit for the asset
                            function _addNewSupportedAsset(address asset, address mLRTReceipt, uint256 depositLimit) private {
                                UtilLib.checkNonZeroAddress(asset);
                                UtilLib.checkNonZeroAddress(mLRTReceipt);
                                if (isSupportedAsset[asset]) {
                                    revert AssetAlreadySupported();
                                }
                                if (asset != IMLRT(mLRTReceipt).underlyingAsset()) revert InvalidAsset();
                                isSupportedAsset[asset] = true;
                                mLRTReceiptByAsset[asset] = mLRTReceipt;
                                supportedAssetList.push(asset);
                                depositLimitByAsset[asset] = depositLimit;
                                boostByAsset[asset] = EigenpieConstants.DENOMINATOR;
                                emit AddedNewSupportedAsset(asset, mLRTReceipt, depositLimit);
                            }
                            /// @dev Updates the deposit limit for an asset
                            /// @param asset Asset address
                            /// @param depositLimit New deposit limit
                            function updateAssetDepositLimit(
                                address asset,
                                uint256 depositLimit
                            )
                                external
                                onlyRole(EigenpieConstants.MANAGER)
                                onlySupportedAsset(asset)
                            {
                                depositLimitByAsset[asset] = depositLimit;
                                emit AssetDepositLimitUpdate(asset, depositLimit);
                            }
                            /// @dev Updates the strategy for an asset
                            /// @param asset Asset address
                            /// @param strategy New strategy address
                            function updateAssetStrategy(
                                address asset,
                                address strategy
                            )
                                external
                                onlyRole(DEFAULT_ADMIN_ROLE)
                                onlySupportedAsset(asset)
                            {
                                UtilLib.checkNonZeroAddress(strategy);
                                if (assetStrategy[asset] == strategy) {
                                    revert ValueAlreadyInUse();
                                }
                                if (asset != address(IStrategy(strategy).underlyingToken())) revert InvalidAsset();
                                assetStrategy[asset] = strategy;
                                emit AssetStrategyUpdate(asset, strategy);
                            }
                            /// @dev Updates the point boost for an asset
                            /// @param asset Asset address
                            /// @param boost point boost effect
                            function updateAssetBoost(
                                address asset,
                                uint256 boost
                            )
                                external
                                onlyRole(DEFAULT_ADMIN_ROLE)
                                onlySupportedAsset(asset)
                            {
                                boostByAsset[asset] = boost;
                                emit AssetBoostUpdate(asset, boost);
                            }
                            /*//////////////////////////////////////////////////////////////
                                                    GETTERS
                            //////////////////////////////////////////////////////////////*/
                            function getContract(bytes32 contractKey) external view override returns (address) {
                                return contractMap[contractKey];
                            }
                            function getSupportedAssetList() external view override returns (address[] memory) {
                                return supportedAssetList;
                            }
                            /*//////////////////////////////////////////////////////////////
                                                    SETTERS
                            //////////////////////////////////////////////////////////////*/
                            function setContract(bytes32 contractKey, address contractAddress) external onlyRole(DEFAULT_ADMIN_ROLE) {
                                _setContract(contractKey, contractAddress);
                            }
                            /// @dev private function to set a contract
                            /// @param key Contract key
                            /// @param val Contract address
                            function _setContract(bytes32 key, address val) private {
                                UtilLib.checkNonZeroAddress(val);
                                if (contractMap[key] == val) {
                                    revert ValueAlreadyInUse();
                                }
                                contractMap[key] = val;
                                emit SetContract(key, val);
                            }
                        }
                        // SPDX-License-Identifier: GPL-3.0-or-later
                        pragma solidity 0.8.21;
                        /// @title UtilLib - Utility library
                        /// @notice Utility functions
                        library UtilLib {
                            error ZeroAddressNotAllowed();
                            /// @dev zero address check modifier
                            /// @param address_ address to check
                            function checkNonZeroAddress(address address_) internal pure {
                                if (address_ == address(0)) revert ZeroAddressNotAllowed();
                            }
                        }
                        // SPDX-License-Identifier: GPL-3.0-or-later
                        pragma solidity 0.8.21;
                        library EigenpieConstants {
                            //contracts
                            bytes32 public constant EIGENPIE_STAKING = keccak256("EIGENPIE_STAKING");
                            bytes32 public constant EIGEN_STRATEGY_MANAGER = keccak256("EIGEN_STRATEGY_MANAGER");
                            bytes32 public constant PRICE_PROVIDER = keccak256("PRICE_PROVIDER");
                            //Roles
                            bytes32 public constant DEFAULT_ADMIN_ROLE = 0x00;
                            bytes32 public constant MANAGER = keccak256("MANAGER");
                            bytes32 public constant MINTER_ROLE = keccak256("MINTER_ROLE");
                            bytes32 public constant BURNER_ROLE = keccak256("BURNER_ROLE");
                            bytes32 public constant ORACLE_ROLE = keccak256("ORACLE_ROLE");
                            bytes32 public constant ORACLE_ADMIN_ROLE = keccak256("ORACLE_ADMIN_ROLE");
                            bytes32 public constant PRICE_PROVIDER_ROLE = keccak256("PRICE_PROVIDER_ROLE");
                            uint256 public constant DENOMINATOR = 10_000;
                        }
                        // SPDX-License-Identifier: GPL-3.0-or-later
                        pragma solidity 0.8.21;
                        interface IEigenpieConfig {
                            // Errors
                            error ValueAlreadyInUse();
                            error AssetAlreadySupported();
                            error AssetNotSupported();
                            error CallerNotEigenpieConfigAdmin();
                            error CallerNotEigenpieConfigManager();
                            error CallerNotEigenpieConfigOracle();
                            error CallerNotEigenpieConfigOracleAdmin();
                            error CallerNotEigenpieConfigPriceProvider();
                            error CallerNotEigenpieConfigMinter();
                            error CallerNotEigenpieConfigBurner();
                            error CallerNotEigenpieConfigAllowedRole(string role);
                            // Events
                            event SetContract(bytes32 key, address indexed contractAddr);
                            event AddedNewSupportedAsset(address indexed asset, address indexed receipt, uint256 depositLimit);
                            event ReceiptTokenUpdated(address indexed asset, address indexed receipt);
                            event RemovedSupportedAsset(address indexed asset);
                            event AssetDepositLimitUpdate(address indexed asset, uint256 depositLimit);
                            event AssetStrategyUpdate(address indexed asset, address indexed strategy);
                            event AssetBoostUpdate(address indexed asset, uint256 newBoost);
                            event ReferralUpdate(address indexed me, address indexed myReferral);
                            // methods
                            function assetStrategy(address asset) external view returns (address);
                            function boostByAsset(address) external view returns (uint256);
                            function mLRTReceiptByAsset(address) external view returns (address);
                            function isSupportedAsset(address asset) external view returns (bool);
                            function getContract(bytes32 contractId) external view returns (address);
                            function getSupportedAssetList() external view returns (address[] memory);
                            function depositLimitByAsset(address asset) external view returns (uint256);
                        }
                        // SPDX-License-Identifier: GPL-3.0-or-later
                        pragma solidity 0.8.21;
                        import { IERC20 } from "@openzeppelin/contracts/token/ERC20/IERC20.sol";
                        interface IMLRT is IERC20 {
                            function updateExchangeRateToLST(uint256 _newRate) external;
                            function exchangeRateToLST() external view returns (uint256);
                            function underlyingAsset() external view returns (address);
                        }
                        // SPDX-License-Identifier: BUSL-1.1
                        pragma solidity >=0.5.0;
                        import "@openzeppelin/contracts/token/ERC20/IERC20.sol";
                        /**
                         * @title Minimal interface for an `Strategy` contract.
                         * @author Layr Labs, Inc.
                         * @notice Terms of Service: https://docs.eigenlayer.xyz/overview/terms-of-service
                         * @notice Custom `Strategy` implementations may expand extensively on this interface.
                         */
                        interface IStrategy {
                            /**
                             * @notice Used to deposit tokens into this Strategy
                             * @param token is the ERC20 token being deposited
                             * @param amount is the amount of token being deposited
                             * @dev This function is only callable by the strategyManager contract. It is invoked inside of the
                             * strategyManager's
                             * `depositIntoStrategy` function, and individual share balances are recorded in the strategyManager as well.
                             * @return newShares is the number of new shares issued at the current exchange ratio.
                             */
                            function deposit(IERC20 token, uint256 amount) external returns (uint256);
                            /**
                             * @notice Used to withdraw tokens from this Strategy, to the `depositor`'s address
                             * @param depositor is the address to receive the withdrawn funds
                             * @param token is the ERC20 token being transferred out
                             * @param amountShares is the amount of shares being withdrawn
                             * @dev This function is only callable by the strategyManager contract. It is invoked inside of the
                             * strategyManager's
                             * other functions, and individual share balances are recorded in the strategyManager as well.
                             */
                            function withdraw(address depositor, IERC20 token, uint256 amountShares) external;
                            /**
                             * @notice Used to convert a number of shares to the equivalent amount of underlying tokens for this strategy.
                             * @notice In contrast to `sharesToUnderlyingView`, this function **may** make state modifications
                             * @param amountShares is the amount of shares to calculate its conversion into the underlying token
                             * @return The amount of underlying tokens corresponding to the input `amountShares`
                             * @dev Implementation for these functions in particular may vary significantly for different strategies
                             */
                            function sharesToUnderlying(uint256 amountShares) external returns (uint256);
                            /**
                             * @notice Used to convert an amount of underlying tokens to the equivalent amount of shares in this strategy.
                             * @notice In contrast to `underlyingToSharesView`, this function **may** make state modifications
                             * @param amountUnderlying is the amount of `underlyingToken` to calculate its conversion into strategy shares
                             * @return The amount of underlying tokens corresponding to the input `amountShares`
                             * @dev Implementation for these functions in particular may vary significantly for different strategies
                             */
                            function underlyingToShares(uint256 amountUnderlying) external returns (uint256);
                            /**
                             * @notice convenience function for fetching the current underlying value of all of the `user`'s shares in
                             * this strategy. In contrast to `userUnderlyingView`, this function **may** make state modifications
                             */
                            function userUnderlying(address user) external returns (uint256);
                            /**
                             * @notice Used to convert a number of shares to the equivalent amount of underlying tokens for this strategy.
                             * @notice In contrast to `sharesToUnderlying`, this function guarantees no state modifications
                             * @param amountShares is the amount of shares to calculate its conversion into the underlying token
                             * @return The amount of shares corresponding to the input `amountUnderlying`
                             * @dev Implementation for these functions in particular may vary significantly for different strategies
                             */
                            function sharesToUnderlyingView(uint256 amountShares) external view returns (uint256);
                            /**
                             * @notice Used to convert an amount of underlying tokens to the equivalent amount of shares in this strategy.
                             * @notice In contrast to `underlyingToShares`, this function guarantees no state modifications
                             * @param amountUnderlying is the amount of `underlyingToken` to calculate its conversion into strategy shares
                             * @return The amount of shares corresponding to the input `amountUnderlying`
                             * @dev Implementation for these functions in particular may vary significantly for different strategies
                             */
                            function underlyingToSharesView(uint256 amountUnderlying) external view returns (uint256);
                            /**
                             * @notice convenience function for fetching the current underlying value of all of the `user`'s shares in
                             * this strategy. In contrast to `userUnderlying`, this function guarantees no state modifications
                             */
                            function userUnderlyingView(address user) external view returns (uint256);
                            /// @notice The underlying token for shares in this Strategy
                            function underlyingToken() external view returns (IERC20);
                            /// @notice The total number of extant shares in this Strategy
                            function totalShares() external view returns (uint256);
                            /// @notice Returns either a brief string explaining the strategy's goal & purpose, or a link to metadata that
                            /// explains in more detail.
                            function explanation() external view returns (string memory);
                        }
                        // SPDX-License-Identifier: MIT
                        // OpenZeppelin Contracts (last updated v4.9.0) (access/AccessControl.sol)
                        pragma solidity ^0.8.0;
                        import "./IAccessControlUpgradeable.sol";
                        import "../utils/ContextUpgradeable.sol";
                        import "../utils/StringsUpgradeable.sol";
                        import "../utils/introspection/ERC165Upgradeable.sol";
                        import "../proxy/utils/Initializable.sol";
                        /**
                         * @dev Contract module that allows children to implement role-based access
                         * control mechanisms. This is a lightweight version that doesn't allow enumerating role
                         * members except through off-chain means by accessing the contract event logs. Some
                         * applications may benefit from on-chain enumerability, for those cases see
                         * {AccessControlEnumerable}.
                         *
                         * Roles are referred to by their `bytes32` identifier. These should be exposed
                         * in the external API and be unique. The best way to achieve this is by
                         * using `public constant` hash digests:
                         *
                         * ```solidity
                         * bytes32 public constant MY_ROLE = keccak256("MY_ROLE");
                         * ```
                         *
                         * Roles can be used to represent a set of permissions. To restrict access to a
                         * function call, use {hasRole}:
                         *
                         * ```solidity
                         * function foo() public {
                         *     require(hasRole(MY_ROLE, msg.sender));
                         *     ...
                         * }
                         * ```
                         *
                         * Roles can be granted and revoked dynamically via the {grantRole} and
                         * {revokeRole} functions. Each role has an associated admin role, and only
                         * accounts that have a role's admin role can call {grantRole} and {revokeRole}.
                         *
                         * By default, the admin role for all roles is `DEFAULT_ADMIN_ROLE`, which means
                         * that only accounts with this role will be able to grant or revoke other
                         * roles. More complex role relationships can be created by using
                         * {_setRoleAdmin}.
                         *
                         * WARNING: The `DEFAULT_ADMIN_ROLE` is also its own admin: it has permission to
                         * grant and revoke this role. Extra precautions should be taken to secure
                         * accounts that have been granted it. We recommend using {AccessControlDefaultAdminRules}
                         * to enforce additional security measures for this role.
                         */
                        abstract contract AccessControlUpgradeable is Initializable, ContextUpgradeable, IAccessControlUpgradeable, ERC165Upgradeable {
                            function __AccessControl_init() internal onlyInitializing {
                            }
                            function __AccessControl_init_unchained() internal onlyInitializing {
                            }
                            struct RoleData {
                                mapping(address => bool) members;
                                bytes32 adminRole;
                            }
                            mapping(bytes32 => RoleData) private _roles;
                            bytes32 public constant DEFAULT_ADMIN_ROLE = 0x00;
                            /**
                             * @dev Modifier that checks that an account has a specific role. Reverts
                             * with a standardized message including the required role.
                             *
                             * The format of the revert reason is given by the following regular expression:
                             *
                             *  /^AccessControl: account (0x[0-9a-f]{40}) is missing role (0x[0-9a-f]{64})$/
                             *
                             * _Available since v4.1._
                             */
                            modifier onlyRole(bytes32 role) {
                                _checkRole(role);
                                _;
                            }
                            /**
                             * @dev See {IERC165-supportsInterface}.
                             */
                            function supportsInterface(bytes4 interfaceId) public view virtual override returns (bool) {
                                return interfaceId == type(IAccessControlUpgradeable).interfaceId || super.supportsInterface(interfaceId);
                            }
                            /**
                             * @dev Returns `true` if `account` has been granted `role`.
                             */
                            function hasRole(bytes32 role, address account) public view virtual override returns (bool) {
                                return _roles[role].members[account];
                            }
                            /**
                             * @dev Revert with a standard message if `_msgSender()` is missing `role`.
                             * Overriding this function changes the behavior of the {onlyRole} modifier.
                             *
                             * Format of the revert message is described in {_checkRole}.
                             *
                             * _Available since v4.6._
                             */
                            function _checkRole(bytes32 role) internal view virtual {
                                _checkRole(role, _msgSender());
                            }
                            /**
                             * @dev Revert with a standard message if `account` is missing `role`.
                             *
                             * The format of the revert reason is given by the following regular expression:
                             *
                             *  /^AccessControl: account (0x[0-9a-f]{40}) is missing role (0x[0-9a-f]{64})$/
                             */
                            function _checkRole(bytes32 role, address account) internal view virtual {
                                if (!hasRole(role, account)) {
                                    revert(
                                        string(
                                            abi.encodePacked(
                                                "AccessControl: account ",
                                                StringsUpgradeable.toHexString(account),
                                                " is missing role ",
                                                StringsUpgradeable.toHexString(uint256(role), 32)
                                            )
                                        )
                                    );
                                }
                            }
                            /**
                             * @dev Returns the admin role that controls `role`. See {grantRole} and
                             * {revokeRole}.
                             *
                             * To change a role's admin, use {_setRoleAdmin}.
                             */
                            function getRoleAdmin(bytes32 role) public view virtual override returns (bytes32) {
                                return _roles[role].adminRole;
                            }
                            /**
                             * @dev Grants `role` to `account`.
                             *
                             * If `account` had not been already granted `role`, emits a {RoleGranted}
                             * event.
                             *
                             * Requirements:
                             *
                             * - the caller must have ``role``'s admin role.
                             *
                             * May emit a {RoleGranted} event.
                             */
                            function grantRole(bytes32 role, address account) public virtual override onlyRole(getRoleAdmin(role)) {
                                _grantRole(role, account);
                            }
                            /**
                             * @dev Revokes `role` from `account`.
                             *
                             * If `account` had been granted `role`, emits a {RoleRevoked} event.
                             *
                             * Requirements:
                             *
                             * - the caller must have ``role``'s admin role.
                             *
                             * May emit a {RoleRevoked} event.
                             */
                            function revokeRole(bytes32 role, address account) public virtual override onlyRole(getRoleAdmin(role)) {
                                _revokeRole(role, account);
                            }
                            /**
                             * @dev Revokes `role` from the calling account.
                             *
                             * Roles are often managed via {grantRole} and {revokeRole}: this function's
                             * purpose is to provide a mechanism for accounts to lose their privileges
                             * if they are compromised (such as when a trusted device is misplaced).
                             *
                             * If the calling account had been revoked `role`, emits a {RoleRevoked}
                             * event.
                             *
                             * Requirements:
                             *
                             * - the caller must be `account`.
                             *
                             * May emit a {RoleRevoked} event.
                             */
                            function renounceRole(bytes32 role, address account) public virtual override {
                                require(account == _msgSender(), "AccessControl: can only renounce roles for self");
                                _revokeRole(role, account);
                            }
                            /**
                             * @dev Grants `role` to `account`.
                             *
                             * If `account` had not been already granted `role`, emits a {RoleGranted}
                             * event. Note that unlike {grantRole}, this function doesn't perform any
                             * checks on the calling account.
                             *
                             * May emit a {RoleGranted} event.
                             *
                             * [WARNING]
                             * ====
                             * This function should only be called from the constructor when setting
                             * up the initial roles for the system.
                             *
                             * Using this function in any other way is effectively circumventing the admin
                             * system imposed by {AccessControl}.
                             * ====
                             *
                             * NOTE: This function is deprecated in favor of {_grantRole}.
                             */
                            function _setupRole(bytes32 role, address account) internal virtual {
                                _grantRole(role, account);
                            }
                            /**
                             * @dev Sets `adminRole` as ``role``'s admin role.
                             *
                             * Emits a {RoleAdminChanged} event.
                             */
                            function _setRoleAdmin(bytes32 role, bytes32 adminRole) internal virtual {
                                bytes32 previousAdminRole = getRoleAdmin(role);
                                _roles[role].adminRole = adminRole;
                                emit RoleAdminChanged(role, previousAdminRole, adminRole);
                            }
                            /**
                             * @dev Grants `role` to `account`.
                             *
                             * Internal function without access restriction.
                             *
                             * May emit a {RoleGranted} event.
                             */
                            function _grantRole(bytes32 role, address account) internal virtual {
                                if (!hasRole(role, account)) {
                                    _roles[role].members[account] = true;
                                    emit RoleGranted(role, account, _msgSender());
                                }
                            }
                            /**
                             * @dev Revokes `role` from `account`.
                             *
                             * Internal function without access restriction.
                             *
                             * May emit a {RoleRevoked} event.
                             */
                            function _revokeRole(bytes32 role, address account) internal virtual {
                                if (hasRole(role, account)) {
                                    _roles[role].members[account] = false;
                                    emit RoleRevoked(role, account, _msgSender());
                                }
                            }
                            /**
                             * @dev This empty reserved space is put in place to allow future versions to add new
                             * variables without shifting down storage in the inheritance chain.
                             * See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps
                             */
                            uint256[49] private __gap;
                        }
                        // SPDX-License-Identifier: MIT
                        // OpenZeppelin Contracts (last updated v4.9.0) (token/ERC20/IERC20.sol)
                        pragma solidity ^0.8.0;
                        /**
                         * @dev Interface of the ERC20 standard as defined in the EIP.
                         */
                        interface IERC20 {
                            /**
                             * @dev Emitted when `value` tokens are moved from one account (`from`) to
                             * another (`to`).
                             *
                             * Note that `value` may be zero.
                             */
                            event Transfer(address indexed from, address indexed to, uint256 value);
                            /**
                             * @dev Emitted when the allowance of a `spender` for an `owner` is set by
                             * a call to {approve}. `value` is the new allowance.
                             */
                            event Approval(address indexed owner, address indexed spender, uint256 value);
                            /**
                             * @dev Returns the amount of tokens in existence.
                             */
                            function totalSupply() external view returns (uint256);
                            /**
                             * @dev Returns the amount of tokens owned by `account`.
                             */
                            function balanceOf(address account) external view returns (uint256);
                            /**
                             * @dev Moves `amount` tokens from the caller's account to `to`.
                             *
                             * Returns a boolean value indicating whether the operation succeeded.
                             *
                             * Emits a {Transfer} event.
                             */
                            function transfer(address to, uint256 amount) external returns (bool);
                            /**
                             * @dev Returns the remaining number of tokens that `spender` will be
                             * allowed to spend on behalf of `owner` through {transferFrom}. This is
                             * zero by default.
                             *
                             * This value changes when {approve} or {transferFrom} are called.
                             */
                            function allowance(address owner, address spender) external view returns (uint256);
                            /**
                             * @dev Sets `amount` as the allowance of `spender` over the caller's tokens.
                             *
                             * Returns a boolean value indicating whether the operation succeeded.
                             *
                             * IMPORTANT: Beware that changing an allowance with this method brings the risk
                             * that someone may use both the old and the new allowance by unfortunate
                             * transaction ordering. One possible solution to mitigate this race
                             * condition is to first reduce the spender's allowance to 0 and set the
                             * desired value afterwards:
                             * https://github.com/ethereum/EIPs/issues/20#issuecomment-263524729
                             *
                             * Emits an {Approval} event.
                             */
                            function approve(address spender, uint256 amount) external returns (bool);
                            /**
                             * @dev Moves `amount` tokens from `from` to `to` using the
                             * allowance mechanism. `amount` is then deducted from the caller's
                             * allowance.
                             *
                             * Returns a boolean value indicating whether the operation succeeded.
                             *
                             * Emits a {Transfer} event.
                             */
                            function transferFrom(address from, address to, uint256 amount) external returns (bool);
                        }
                        // SPDX-License-Identifier: MIT
                        // OpenZeppelin Contracts v4.4.1 (access/IAccessControl.sol)
                        pragma solidity ^0.8.0;
                        /**
                         * @dev External interface of AccessControl declared to support ERC165 detection.
                         */
                        interface IAccessControlUpgradeable {
                            /**
                             * @dev Emitted when `newAdminRole` is set as ``role``'s admin role, replacing `previousAdminRole`
                             *
                             * `DEFAULT_ADMIN_ROLE` is the starting admin for all roles, despite
                             * {RoleAdminChanged} not being emitted signaling this.
                             *
                             * _Available since v3.1._
                             */
                            event RoleAdminChanged(bytes32 indexed role, bytes32 indexed previousAdminRole, bytes32 indexed newAdminRole);
                            /**
                             * @dev Emitted when `account` is granted `role`.
                             *
                             * `sender` is the account that originated the contract call, an admin role
                             * bearer except when using {AccessControl-_setupRole}.
                             */
                            event RoleGranted(bytes32 indexed role, address indexed account, address indexed sender);
                            /**
                             * @dev Emitted when `account` is revoked `role`.
                             *
                             * `sender` is the account that originated the contract call:
                             *   - if using `revokeRole`, it is the admin role bearer
                             *   - if using `renounceRole`, it is the role bearer (i.e. `account`)
                             */
                            event RoleRevoked(bytes32 indexed role, address indexed account, address indexed sender);
                            /**
                             * @dev Returns `true` if `account` has been granted `role`.
                             */
                            function hasRole(bytes32 role, address account) external view returns (bool);
                            /**
                             * @dev Returns the admin role that controls `role`. See {grantRole} and
                             * {revokeRole}.
                             *
                             * To change a role's admin, use {AccessControl-_setRoleAdmin}.
                             */
                            function getRoleAdmin(bytes32 role) external view returns (bytes32);
                            /**
                             * @dev Grants `role` to `account`.
                             *
                             * If `account` had not been already granted `role`, emits a {RoleGranted}
                             * event.
                             *
                             * Requirements:
                             *
                             * - the caller must have ``role``'s admin role.
                             */
                            function grantRole(bytes32 role, address account) external;
                            /**
                             * @dev Revokes `role` from `account`.
                             *
                             * If `account` had been granted `role`, emits a {RoleRevoked} event.
                             *
                             * Requirements:
                             *
                             * - the caller must have ``role``'s admin role.
                             */
                            function revokeRole(bytes32 role, address account) external;
                            /**
                             * @dev Revokes `role` from the calling account.
                             *
                             * Roles are often managed via {grantRole} and {revokeRole}: this function's
                             * purpose is to provide a mechanism for accounts to lose their privileges
                             * if they are compromised (such as when a trusted device is misplaced).
                             *
                             * If the calling account had been granted `role`, emits a {RoleRevoked}
                             * event.
                             *
                             * Requirements:
                             *
                             * - the caller must be `account`.
                             */
                            function renounceRole(bytes32 role, address account) external;
                        }
                        // SPDX-License-Identifier: MIT
                        // OpenZeppelin Contracts v4.4.1 (utils/Context.sol)
                        pragma solidity ^0.8.0;
                        import "../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;
                            }
                            /**
                             * @dev This empty reserved space is put in place to allow future versions to add new
                             * variables without shifting down storage in the inheritance chain.
                             * See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps
                             */
                            uint256[50] private __gap;
                        }
                        // SPDX-License-Identifier: MIT
                        // OpenZeppelin Contracts (last updated v4.9.0) (utils/Strings.sol)
                        pragma solidity ^0.8.0;
                        import "./math/MathUpgradeable.sol";
                        import "./math/SignedMathUpgradeable.sol";
                        /**
                         * @dev String operations.
                         */
                        library StringsUpgradeable {
                            bytes16 private constant _SYMBOLS = "0123456789abcdef";
                            uint8 private constant _ADDRESS_LENGTH = 20;
                            /**
                             * @dev Converts a `uint256` to its ASCII `string` decimal representation.
                             */
                            function toString(uint256 value) internal pure returns (string memory) {
                                unchecked {
                                    uint256 length = MathUpgradeable.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), _SYMBOLS))
                                        }
                                        value /= 10;
                                        if (value == 0) break;
                                    }
                                    return buffer;
                                }
                            }
                            /**
                             * @dev Converts a `int256` to its ASCII `string` decimal representation.
                             */
                            function toString(int256 value) internal pure returns (string memory) {
                                return string(abi.encodePacked(value < 0 ? "-" : "", toString(SignedMathUpgradeable.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, MathUpgradeable.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) {
                                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] = _SYMBOLS[value & 0xf];
                                    value >>= 4;
                                }
                                require(value == 0, "Strings: hex length insufficient");
                                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 keccak256(bytes(a)) == keccak256(bytes(b));
                            }
                        }
                        // SPDX-License-Identifier: MIT
                        // OpenZeppelin Contracts v4.4.1 (utils/introspection/ERC165.sol)
                        pragma solidity ^0.8.0;
                        import "./IERC165Upgradeable.sol";
                        import "../../proxy/utils/Initializable.sol";
                        /**
                         * @dev Implementation of the {IERC165} interface.
                         *
                         * Contracts that want to implement ERC165 should inherit from this contract and override {supportsInterface} to check
                         * for the additional interface id that will be supported. For example:
                         *
                         * ```solidity
                         * function supportsInterface(bytes4 interfaceId) public view virtual override returns (bool) {
                         *     return interfaceId == type(MyInterface).interfaceId || super.supportsInterface(interfaceId);
                         * }
                         * ```
                         *
                         * Alternatively, {ERC165Storage} provides an easier to use but more expensive implementation.
                         */
                        abstract contract ERC165Upgradeable is Initializable, IERC165Upgradeable {
                            function __ERC165_init() internal onlyInitializing {
                            }
                            function __ERC165_init_unchained() internal onlyInitializing {
                            }
                            /**
                             * @dev See {IERC165-supportsInterface}.
                             */
                            function supportsInterface(bytes4 interfaceId) public view virtual override returns (bool) {
                                return interfaceId == type(IERC165Upgradeable).interfaceId;
                            }
                            /**
                             * @dev This empty reserved space is put in place to allow future versions to add new
                             * variables without shifting down storage in the inheritance chain.
                             * See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps
                             */
                            uint256[50] private __gap;
                        }
                        // SPDX-License-Identifier: MIT
                        // OpenZeppelin Contracts (last updated v4.9.0) (proxy/utils/Initializable.sol)
                        pragma solidity ^0.8.2;
                        import "../../utils/AddressUpgradeable.sol";
                        /**
                         * @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 Indicates that the contract has been initialized.
                             * @custom:oz-retyped-from bool
                             */
                            uint8 private _initialized;
                            /**
                             * @dev Indicates that the contract is in the process of being initialized.
                             */
                            bool private _initializing;
                            /**
                             * @dev Triggered when the contract has been initialized or reinitialized.
                             */
                            event Initialized(uint8 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 functions marked with `initializer` can be nested in the context of a
                             * constructor.
                             *
                             * Emits an {Initialized} event.
                             */
                            modifier initializer() {
                                bool isTopLevelCall = !_initializing;
                                require(
                                    (isTopLevelCall && _initialized < 1) || (!AddressUpgradeable.isContract(address(this)) && _initialized == 1),
                                    "Initializable: contract is already initialized"
                                );
                                _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 255 will prevent any future reinitialization.
                             *
                             * Emits an {Initialized} event.
                             */
                            modifier reinitializer(uint8 version) {
                                require(!_initializing && _initialized < version, "Initializable: contract is already initialized");
                                _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() {
                                require(_initializing, "Initializable: contract is not initializing");
                                _;
                            }
                            /**
                             * @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 {
                                require(!_initializing, "Initializable: contract is initializing");
                                if (_initialized != type(uint8).max) {
                                    _initialized = type(uint8).max;
                                    emit Initialized(type(uint8).max);
                                }
                            }
                            /**
                             * @dev Returns the highest version that has been initialized. See {reinitializer}.
                             */
                            function _getInitializedVersion() internal view returns (uint8) {
                                return _initialized;
                            }
                            /**
                             * @dev Returns `true` if the contract is currently initializing. See {onlyInitializing}.
                             */
                            function _isInitializing() internal view returns (bool) {
                                return _initializing;
                            }
                        }
                        // SPDX-License-Identifier: MIT
                        // OpenZeppelin Contracts (last updated v4.9.0) (utils/math/Math.sol)
                        pragma solidity ^0.8.0;
                        /**
                         * @dev Standard math utilities missing in the Solidity language.
                         */
                        library MathUpgradeable {
                            enum Rounding {
                                Down, // Toward negative infinity
                                Up, // Toward infinity
                                Zero // Toward zero
                            }
                            /**
                             * @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 up instead
                             * of rounding down.
                             */
                            function ceilDiv(uint256 a, uint256 b) internal pure returns (uint256) {
                                // (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; // Least significant 256 bits of the product
                                    uint256 prod1; // Most significant 256 bits of the product
                                    assembly {
                                        let mm := mulmod(x, y, not(0))
                                        prod0 := mul(x, y)
                                        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.
                                    require(denominator > prod1, "Math: mulDiv overflow");
                                    ///////////////////////////////////////////////
                                    // 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.
                                    // Does not overflow because the denominator cannot be zero at this stage in the function.
                                    uint256 twos = denominator & (~denominator + 1);
                                    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 (rounding == Rounding.Up && 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 down.
                             *
                             * 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 + (rounding == Rounding.Up && result * result < a ? 1 : 0);
                                }
                            }
                            /**
                             * @dev Return the log in base 2, rounded down, of a positive value.
                             * 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 + (rounding == Rounding.Up && 1 << result < value ? 1 : 0);
                                }
                            }
                            /**
                             * @dev Return the log in base 10, rounded down, of a positive value.
                             * 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 + (rounding == Rounding.Up && 10 ** result < value ? 1 : 0);
                                }
                            }
                            /**
                             * @dev Return the log in base 256, rounded down, of a positive value.
                             * 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 + (rounding == Rounding.Up && 1 << (result << 3) < value ? 1 : 0);
                                }
                            }
                        }
                        // SPDX-License-Identifier: MIT
                        // OpenZeppelin Contracts (last updated v4.8.0) (utils/math/SignedMath.sol)
                        pragma solidity ^0.8.0;
                        /**
                         * @dev Standard signed math utilities missing in the Solidity language.
                         */
                        library SignedMathUpgradeable {
                            /**
                             * @dev Returns the largest of two signed numbers.
                             */
                            function max(int256 a, int256 b) internal pure returns (int256) {
                                return a > b ? a : b;
                            }
                            /**
                             * @dev Returns the smallest of two signed numbers.
                             */
                            function min(int256 a, int256 b) internal pure returns (int256) {
                                return a < b ? a : b;
                            }
                            /**
                             * @dev Returns the average of two signed numbers without overflow.
                             * The result is rounded towards zero.
                             */
                            function average(int256 a, int256 b) internal pure returns (int256) {
                                // Formula from the book "Hacker's Delight"
                                int256 x = (a & b) + ((a ^ b) >> 1);
                                return x + (int256(uint256(x) >> 255) & (a ^ b));
                            }
                            /**
                             * @dev Returns the absolute unsigned value of a signed value.
                             */
                            function abs(int256 n) internal pure returns (uint256) {
                                unchecked {
                                    // must be unchecked in order to support `n = type(int256).min`
                                    return uint256(n >= 0 ? n : -n);
                                }
                            }
                        }
                        // SPDX-License-Identifier: MIT
                        // OpenZeppelin Contracts v4.4.1 (utils/introspection/IERC165.sol)
                        pragma solidity ^0.8.0;
                        /**
                         * @dev Interface of the ERC165 standard, as defined in the
                         * https://eips.ethereum.org/EIPS/eip-165[EIP].
                         *
                         * Implementers can declare support of contract interfaces, which can then be
                         * queried by others ({ERC165Checker}).
                         *
                         * For an implementation, see {ERC165}.
                         */
                        interface IERC165Upgradeable {
                            /**
                             * @dev Returns true if this contract implements the interface defined by
                             * `interfaceId`. See the corresponding
                             * https://eips.ethereum.org/EIPS/eip-165#how-interfaces-are-identified[EIP section]
                             * to learn more about how these ids are created.
                             *
                             * This function call must use less than 30 000 gas.
                             */
                            function supportsInterface(bytes4 interfaceId) external view returns (bool);
                        }
                        // SPDX-License-Identifier: MIT
                        // OpenZeppelin Contracts (last updated v4.9.0) (utils/Address.sol)
                        pragma solidity ^0.8.1;
                        /**
                         * @dev Collection of functions related to the address type
                         */
                        library AddressUpgradeable {
                            /**
                             * @dev Returns true if `account` is a contract.
                             *
                             * [IMPORTANT]
                             * ====
                             * It is unsafe to assume that an address for which this function returns
                             * false is an externally-owned account (EOA) and not a contract.
                             *
                             * Among others, `isContract` will return false for the following
                             * types of addresses:
                             *
                             *  - an externally-owned account
                             *  - a contract in construction
                             *  - an address where a contract will be created
                             *  - an address where a contract lived, but was destroyed
                             *
                             * Furthermore, `isContract` will also return true if the target contract within
                             * the same transaction is already scheduled for destruction by `SELFDESTRUCT`,
                             * which only has an effect at the end of a transaction.
                             * ====
                             *
                             * [IMPORTANT]
                             * ====
                             * You shouldn't rely on `isContract` to protect against flash loan attacks!
                             *
                             * Preventing calls from contracts is highly discouraged. It breaks composability, breaks support for smart wallets
                             * like Gnosis Safe, and does not provide security since it can be circumvented by calling from a contract
                             * constructor.
                             * ====
                             */
                            function isContract(address account) internal view returns (bool) {
                                // This method relies on extcodesize/address.code.length, which returns 0
                                // for contracts in construction, since the code is only stored at the end
                                // of the constructor execution.
                                return account.code.length > 0;
                            }
                            /**
                             * @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.0/security-considerations.html#use-the-checks-effects-interactions-pattern[checks-effects-interactions pattern].
                             */
                            function sendValue(address payable recipient, uint256 amount) internal {
                                require(address(this).balance >= amount, "Address: insufficient balance");
                                (bool success, ) = recipient.call{value: amount}("");
                                require(success, "Address: unable to send value, recipient may have reverted");
                            }
                            /**
                             * @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, it is bubbled up by this
                             * function (like regular Solidity function calls).
                             *
                             * 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.
                             *
                             * _Available since v3.1._
                             */
                            function functionCall(address target, bytes memory data) internal returns (bytes memory) {
                                return functionCallWithValue(target, data, 0, "Address: low-level call failed");
                            }
                            /**
                             * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], but with
                             * `errorMessage` as a fallback revert reason when `target` reverts.
                             *
                             * _Available since v3.1._
                             */
                            function functionCall(
                                address target,
                                bytes memory data,
                                string memory errorMessage
                            ) internal returns (bytes memory) {
                                return functionCallWithValue(target, data, 0, errorMessage);
                            }
                            /**
                             * @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`.
                             *
                             * _Available since v3.1._
                             */
                            function functionCallWithValue(address target, bytes memory data, uint256 value) internal returns (bytes memory) {
                                return functionCallWithValue(target, data, value, "Address: low-level call with value failed");
                            }
                            /**
                             * @dev Same as {xref-Address-functionCallWithValue-address-bytes-uint256-}[`functionCallWithValue`], but
                             * with `errorMessage` as a fallback revert reason when `target` reverts.
                             *
                             * _Available since v3.1._
                             */
                            function functionCallWithValue(
                                address target,
                                bytes memory data,
                                uint256 value,
                                string memory errorMessage
                            ) internal returns (bytes memory) {
                                require(address(this).balance >= value, "Address: insufficient balance for call");
                                (bool success, bytes memory returndata) = target.call{value: value}(data);
                                return verifyCallResultFromTarget(target, success, returndata, errorMessage);
                            }
                            /**
                             * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
                             * but performing a static call.
                             *
                             * _Available since v3.3._
                             */
                            function functionStaticCall(address target, bytes memory data) internal view returns (bytes memory) {
                                return functionStaticCall(target, data, "Address: low-level static call failed");
                            }
                            /**
                             * @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`],
                             * but performing a static call.
                             *
                             * _Available since v3.3._
                             */
                            function functionStaticCall(
                                address target,
                                bytes memory data,
                                string memory errorMessage
                            ) internal view returns (bytes memory) {
                                (bool success, bytes memory returndata) = target.staticcall(data);
                                return verifyCallResultFromTarget(target, success, returndata, errorMessage);
                            }
                            /**
                             * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
                             * but performing a delegate call.
                             *
                             * _Available since v3.4._
                             */
                            function functionDelegateCall(address target, bytes memory data) internal returns (bytes memory) {
                                return functionDelegateCall(target, data, "Address: low-level delegate call failed");
                            }
                            /**
                             * @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`],
                             * but performing a delegate call.
                             *
                             * _Available since v3.4._
                             */
                            function functionDelegateCall(
                                address target,
                                bytes memory data,
                                string memory errorMessage
                            ) internal returns (bytes memory) {
                                (bool success, bytes memory returndata) = target.delegatecall(data);
                                return verifyCallResultFromTarget(target, success, returndata, errorMessage);
                            }
                            /**
                             * @dev Tool to verify that a low level call to smart-contract was successful, and revert (either by bubbling
                             * the revert reason or using the provided one) in case of unsuccessful call or if target was not a contract.
                             *
                             * _Available since v4.8._
                             */
                            function verifyCallResultFromTarget(
                                address target,
                                bool success,
                                bytes memory returndata,
                                string memory errorMessage
                            ) internal view returns (bytes memory) {
                                if (success) {
                                    if (returndata.length == 0) {
                                        // only check isContract if the call was successful and the return data is empty
                                        // otherwise we already know that it was a contract
                                        require(isContract(target), "Address: call to non-contract");
                                    }
                                    return returndata;
                                } else {
                                    _revert(returndata, errorMessage);
                                }
                            }
                            /**
                             * @dev Tool to verify that a low level call was successful, and revert if it wasn't, either by bubbling the
                             * revert reason or using the provided one.
                             *
                             * _Available since v4.3._
                             */
                            function verifyCallResult(
                                bool success,
                                bytes memory returndata,
                                string memory errorMessage
                            ) internal pure returns (bytes memory) {
                                if (success) {
                                    return returndata;
                                } else {
                                    _revert(returndata, errorMessage);
                                }
                            }
                            function _revert(bytes memory returndata, string memory errorMessage) 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(errorMessage);
                                }
                            }
                        }
                        

                        File 6 of 6: MLRT
                        // SPDX-License-Identifier: GPL-3.0-or-later
                        pragma solidity 0.8.21;
                        import { UtilLib } from "../utils/UtilLib.sol";
                        import { EigenpieConfigRoleChecker, IEigenpieConfig, EigenpieConstants } from "../utils/EigenpieConfigRoleChecker.sol";
                        import { IMLRT } from "../interfaces/IMLRT.sol";
                        import { IPriceProvider } from "../interfaces/IPriceProvider.sol";
                        import { ERC20Upgradeable, Initializable } from "@openzeppelin/contracts-upgradeable/token/ERC20/ERC20Upgradeable.sol";
                        import { PausableUpgradeable } from "@openzeppelin/contracts-upgradeable/security/PausableUpgradeable.sol";
                        /// @title mLRT token Contract. A generic token contract for Liquid Wrapper for staked LST through Eigenpie
                        /// @author Eigenpie Team
                        /// @notice The ERC20 contract for the mLRT token
                        contract MLRT is Initializable, EigenpieConfigRoleChecker, ERC20Upgradeable, PausableUpgradeable {
                            address public underlyingAsset;
                            uint256 public exchangeRateToLST;
                            event LSTExchangeRateUpdated(address indexed caller, uint256 newExchangeRate);
                            /// @custom:oz-upgrades-unsafe-allow constructor
                            constructor() {
                                _disableInitializers();
                            }
                            /// @dev Initializes the contract
                            /// @param asset underlying asset
                            /// @param eigenpieConfigAddr Eigenpie config address
                            /// @param name name of the MLRT token
                            /// @param symbol symbol of the MLRT token
                            function initialize(
                                address asset,
                                address eigenpieConfigAddr,
                                string calldata name,
                                string calldata symbol
                            )
                                external
                                initializer
                            {
                                UtilLib.checkNonZeroAddress(asset);
                                UtilLib.checkNonZeroAddress(eigenpieConfigAddr);
                                __ERC20_init(name, symbol);
                                __Pausable_init();
                                eigenpieConfig = IEigenpieConfig(eigenpieConfigAddr);
                                underlyingAsset = asset;
                                exchangeRateToLST = 1 ether;
                                emit UpdatedEigenpieConfig(eigenpieConfigAddr);
                            }
                            /*//////////////////////////////////////////////////////////////
                                                    view functions
                            //////////////////////////////////////////////////////////////*/
                            function exchangeRateToNative() external view returns (uint256) {
                                address priceProvider = eigenpieConfig.getContract(EigenpieConstants.PRICE_PROVIDER);
                                uint256 exchangeRateETH = IPriceProvider(priceProvider).getAssetPrice(underlyingAsset);
                                return exchangeRateToLST * exchangeRateETH / 1 ether;
                            }
                            /*//////////////////////////////////////////////////////////////
                                                    Write functions
                            //////////////////////////////////////////////////////////////*/
                            /// @notice Mints EGETH when called by an authorized caller
                            /// @param to the account to mint to
                            /// @param amount the amount of EGETH to mint
                            function mint(address to, uint256 amount) external onlyMinter whenNotPaused {
                                _mint(to, amount);
                            }
                            /// @notice Burns EGETH when called by an authorized caller
                            /// @param account the account to burn from
                            /// @param amount the amount of EGETH to burn
                            function burnFrom(address account, uint256 amount) external onlyBurner whenNotPaused {
                                _burn(account, amount);
                            }
                            /// @dev Triggers stopped state.
                            /// @dev Only callable by Eigenpie config manager. Contract must NOT be paused.
                            function pause() external onlyEigenpieManager {
                                _pause();
                            }
                            /// @notice Returns to normal state.
                            /// @dev Only callable by the admin. Contract must be paused
                            function unpause() external onlyDefaultAdmin {
                                _unpause();
                            }
                            /// @notice Updates the MLRT to LST rate
                            /// @dev only callable by the Price Provider
                            /// @param _newRate the new Eigenpie config contract
                            function updateExchangeRateToLST(uint256 _newRate) external onlyPriceProvider {
                                exchangeRateToLST = _newRate;
                                emit LSTExchangeRateUpdated(msg.sender, _newRate);
                            }
                        }
                        // SPDX-License-Identifier: GPL-3.0-or-later
                        pragma solidity 0.8.21;
                        /// @title UtilLib - Utility library
                        /// @notice Utility functions
                        library UtilLib {
                            error ZeroAddressNotAllowed();
                            /// @dev zero address check modifier
                            /// @param address_ address to check
                            function checkNonZeroAddress(address address_) internal pure {
                                if (address_ == address(0)) revert ZeroAddressNotAllowed();
                            }
                        }
                        // SPDX-License-Identifier: GPL-3.0-or-later
                        pragma solidity 0.8.21;
                        import { UtilLib } from "./UtilLib.sol";
                        import { EigenpieConstants } from "./EigenpieConstants.sol";
                        import { IEigenpieConfig } from "../interfaces/IEigenpieConfig.sol";
                        import { IAccessControl } from "@openzeppelin/contracts/access/IAccessControl.sol";
                        /// @title EigenpieConfigRoleChecker - Eigenpie Config Role Checker Contract
                        /// @notice Handles Eigenpie config role checks
                        abstract contract EigenpieConfigRoleChecker {
                            IEigenpieConfig public eigenpieConfig;
                            uint256[49] private __gap; // reserve for upgrade
                            // events
                            event UpdatedEigenpieConfig(address indexed eigenpieConfig);
                            // modifiers
                            modifier onlyRole(bytes32 role) {
                                if (!IAccessControl(address(eigenpieConfig)).hasRole(role, msg.sender)) {
                                    string memory roleStr = string(abi.encodePacked(role));
                                    revert IEigenpieConfig.CallerNotEigenpieConfigAllowedRole(roleStr);
                                }
                                _;
                            }
                            modifier onlyEigenpieManager() {
                                if (!IAccessControl(address(eigenpieConfig)).hasRole(EigenpieConstants.MANAGER, msg.sender)) {
                                    revert IEigenpieConfig.CallerNotEigenpieConfigManager();
                                }
                                _;
                            }
                            modifier onlyPriceProvider() {
                                if (!IAccessControl(address(eigenpieConfig)).hasRole(EigenpieConstants.PRICE_PROVIDER_ROLE, msg.sender)) {
                                    revert IEigenpieConfig.CallerNotEigenpieConfigPriceProvider();
                                }
                                _;
                            }
                            modifier onlyDefaultAdmin() {
                                if (!IAccessControl(address(eigenpieConfig)).hasRole(EigenpieConstants.DEFAULT_ADMIN_ROLE, msg.sender)) {
                                    revert IEigenpieConfig.CallerNotEigenpieConfigAdmin();
                                }
                                _;
                            }
                            modifier onlyOracleAdmin() {
                                if (!IAccessControl(address(eigenpieConfig)).hasRole(EigenpieConstants.ORACLE_ADMIN_ROLE, msg.sender)) {
                                    revert IEigenpieConfig.CallerNotEigenpieConfigOracleAdmin();
                                }
                                _;
                            }
                            modifier onlyOracle() {
                                if (!IAccessControl(address(eigenpieConfig)).hasRole(EigenpieConstants.ORACLE_ROLE, msg.sender)) {
                                    revert IEigenpieConfig.CallerNotEigenpieConfigOracle();
                                }
                                _;
                            }
                            modifier onlyMinter() {
                                if (!IAccessControl(address(eigenpieConfig)).hasRole(EigenpieConstants.MINTER_ROLE, msg.sender)) {
                                    revert IEigenpieConfig.CallerNotEigenpieConfigMinter();
                                }
                                _;
                            }
                            modifier onlyBurner() {
                                if (!IAccessControl(address(eigenpieConfig)).hasRole(EigenpieConstants.BURNER_ROLE, msg.sender)) {
                                    revert IEigenpieConfig.CallerNotEigenpieConfigBurner();
                                }
                                _;
                            }        
                            modifier onlySupportedAsset(address asset) {
                                if (!eigenpieConfig.isSupportedAsset(asset)) {
                                    revert IEigenpieConfig.AssetNotSupported();
                                }
                                _;
                            }
                            // setters
                            /// @notice Updates the Eigenpie config contract
                            /// @dev only callable by Eigenpie default
                            /// @param eigenpieConfigAddr the new Eigenpie config contract Address
                            function updateEigenpieConfig(address eigenpieConfigAddr) external virtual onlyDefaultAdmin {
                                UtilLib.checkNonZeroAddress(eigenpieConfigAddr);
                                eigenpieConfig = IEigenpieConfig(eigenpieConfigAddr);
                                emit UpdatedEigenpieConfig(eigenpieConfigAddr);
                            }
                        }
                        // SPDX-License-Identifier: GPL-3.0-or-later
                        pragma solidity 0.8.21;
                        import { IERC20 } from "@openzeppelin/contracts/token/ERC20/IERC20.sol";
                        interface IMLRT is IERC20 {
                            function updateExchangeRateToLST(uint256 _newRate) external;
                            function exchangeRateToLST() external view returns (uint256);
                            function underlyingAsset() external view returns (address);
                        }
                        // SPDX-License-Identifier: GPL-3.0-or-later
                        pragma solidity 0.8.21;
                        interface IPriceProvider {
                            // errors
                            error NewRateTooHigh();
                            error NewRateTooLow();
                            error UpdateTooFrequently();
                            // events
                            event AssetPriceAdapterUpdate(address indexed asset, address indexed priceOracle);
                            event ExchangeRateUpdate(address indexed asset, address indexed receipt, uint256 newExchangeRate);
                            event RateUpdateCeilingUpdate(address indexed caller, uint256 newRateLimit, uint256 newWindowLimit);
                            // methods
                            function getAssetPrice(address asset) external view returns (uint256);
                            function assetPriceOracle(address asset) external view returns (address);
                        }
                        // SPDX-License-Identifier: MIT
                        // OpenZeppelin Contracts (last updated v4.9.0) (token/ERC20/ERC20.sol)
                        pragma solidity ^0.8.0;
                        import "./IERC20Upgradeable.sol";
                        import "./extensions/IERC20MetadataUpgradeable.sol";
                        import "../../utils/ContextUpgradeable.sol";
                        import "../../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}.
                         * For a generic mechanism see {ERC20PresetMinterPauser}.
                         *
                         * TIP: For a detailed writeup see our guide
                         * https://forum.openzeppelin.com/t/how-to-implement-erc20-supply-mechanisms/226[How
                         * to implement supply mechanisms].
                         *
                         * The default value of {decimals} is 18. To change this, you should override
                         * this function so it returns a different value.
                         *
                         * We have followed general OpenZeppelin Contracts guidelines: functions revert
                         * instead returning `false` on failure. This behavior is nonetheless
                         * conventional and does not conflict with the expectations of ERC20
                         * applications.
                         *
                         * Additionally, an {Approval} event is emitted on calls to {transferFrom}.
                         * This allows applications to reconstruct the allowance for all accounts just
                         * by listening to said events. Other implementations of the EIP may not emit
                         * these events, as it isn't required by the specification.
                         *
                         * Finally, the non-standard {decreaseAllowance} and {increaseAllowance}
                         * functions have been added to mitigate the well-known issues around setting
                         * allowances. See {IERC20-approve}.
                         */
                        contract ERC20Upgradeable is Initializable, ContextUpgradeable, IERC20Upgradeable, IERC20MetadataUpgradeable {
                            mapping(address => uint256) private _balances;
                            mapping(address => mapping(address => uint256)) private _allowances;
                            uint256 private _totalSupply;
                            string private _name;
                            string private _symbol;
                            /**
                             * @dev Sets the values for {name} and {symbol}.
                             *
                             * All two of these values are immutable: they can only be set once during
                             * construction.
                             */
                            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 {
                                _name = name_;
                                _symbol = symbol_;
                            }
                            /**
                             * @dev Returns the name of the token.
                             */
                            function name() public view virtual override returns (string memory) {
                                return _name;
                            }
                            /**
                             * @dev Returns the symbol of the token, usually a shorter version of the
                             * name.
                             */
                            function symbol() public view virtual override returns (string memory) {
                                return _symbol;
                            }
                            /**
                             * @dev Returns the number of decimals used to get its user representation.
                             * For example, if `decimals` equals `2`, a balance of `505` tokens should
                             * be displayed to a user as `5.05` (`505 / 10 ** 2`).
                             *
                             * Tokens usually opt for a value of 18, imitating the relationship between
                             * Ether and Wei. This is the default value returned by this function, unless
                             * it's overridden.
                             *
                             * NOTE: This information is only used for _display_ purposes: it in
                             * no way affects any of the arithmetic of the contract, including
                             * {IERC20-balanceOf} and {IERC20-transfer}.
                             */
                            function decimals() public view virtual override returns (uint8) {
                                return 18;
                            }
                            /**
                             * @dev See {IERC20-totalSupply}.
                             */
                            function totalSupply() public view virtual override returns (uint256) {
                                return _totalSupply;
                            }
                            /**
                             * @dev See {IERC20-balanceOf}.
                             */
                            function balanceOf(address account) public view virtual override returns (uint256) {
                                return _balances[account];
                            }
                            /**
                             * @dev See {IERC20-transfer}.
                             *
                             * Requirements:
                             *
                             * - `to` cannot be the zero address.
                             * - the caller must have a balance of at least `amount`.
                             */
                            function transfer(address to, uint256 amount) public virtual override returns (bool) {
                                address owner = _msgSender();
                                _transfer(owner, to, amount);
                                return true;
                            }
                            /**
                             * @dev See {IERC20-allowance}.
                             */
                            function allowance(address owner, address spender) public view virtual override returns (uint256) {
                                return _allowances[owner][spender];
                            }
                            /**
                             * @dev See {IERC20-approve}.
                             *
                             * NOTE: If `amount` is the maximum `uint256`, the allowance is not updated on
                             * `transferFrom`. This is semantically equivalent to an infinite approval.
                             *
                             * Requirements:
                             *
                             * - `spender` cannot be the zero address.
                             */
                            function approve(address spender, uint256 amount) public virtual override returns (bool) {
                                address owner = _msgSender();
                                _approve(owner, spender, amount);
                                return true;
                            }
                            /**
                             * @dev See {IERC20-transferFrom}.
                             *
                             * Emits an {Approval} event indicating the updated allowance. This is not
                             * required by the EIP. See the note at the beginning of {ERC20}.
                             *
                             * NOTE: Does not update the allowance if the current allowance
                             * is the maximum `uint256`.
                             *
                             * Requirements:
                             *
                             * - `from` and `to` cannot be the zero address.
                             * - `from` must have a balance of at least `amount`.
                             * - the caller must have allowance for ``from``'s tokens of at least
                             * `amount`.
                             */
                            function transferFrom(address from, address to, uint256 amount) public virtual override returns (bool) {
                                address spender = _msgSender();
                                _spendAllowance(from, spender, amount);
                                _transfer(from, to, amount);
                                return true;
                            }
                            /**
                             * @dev Atomically increases the allowance granted to `spender` by the caller.
                             *
                             * This is an alternative to {approve} that can be used as a mitigation for
                             * problems described in {IERC20-approve}.
                             *
                             * Emits an {Approval} event indicating the updated allowance.
                             *
                             * Requirements:
                             *
                             * - `spender` cannot be the zero address.
                             */
                            function increaseAllowance(address spender, uint256 addedValue) public virtual returns (bool) {
                                address owner = _msgSender();
                                _approve(owner, spender, allowance(owner, spender) + addedValue);
                                return true;
                            }
                            /**
                             * @dev Atomically decreases the allowance granted to `spender` by the caller.
                             *
                             * This is an alternative to {approve} that can be used as a mitigation for
                             * problems described in {IERC20-approve}.
                             *
                             * Emits an {Approval} event indicating the updated allowance.
                             *
                             * Requirements:
                             *
                             * - `spender` cannot be the zero address.
                             * - `spender` must have allowance for the caller of at least
                             * `subtractedValue`.
                             */
                            function decreaseAllowance(address spender, uint256 subtractedValue) public virtual returns (bool) {
                                address owner = _msgSender();
                                uint256 currentAllowance = allowance(owner, spender);
                                require(currentAllowance >= subtractedValue, "ERC20: decreased allowance below zero");
                                unchecked {
                                    _approve(owner, spender, currentAllowance - subtractedValue);
                                }
                                return true;
                            }
                            /**
                             * @dev Moves `amount` of tokens from `from` to `to`.
                             *
                             * This internal function is equivalent to {transfer}, and can be used to
                             * e.g. implement automatic token fees, slashing mechanisms, etc.
                             *
                             * Emits a {Transfer} event.
                             *
                             * Requirements:
                             *
                             * - `from` cannot be the zero address.
                             * - `to` cannot be the zero address.
                             * - `from` must have a balance of at least `amount`.
                             */
                            function _transfer(address from, address to, uint256 amount) internal virtual {
                                require(from != address(0), "ERC20: transfer from the zero address");
                                require(to != address(0), "ERC20: transfer to the zero address");
                                _beforeTokenTransfer(from, to, amount);
                                uint256 fromBalance = _balances[from];
                                require(fromBalance >= amount, "ERC20: transfer amount exceeds balance");
                                unchecked {
                                    _balances[from] = fromBalance - amount;
                                    // Overflow not possible: the sum of all balances is capped by totalSupply, and the sum is preserved by
                                    // decrementing then incrementing.
                                    _balances[to] += amount;
                                }
                                emit Transfer(from, to, amount);
                                _afterTokenTransfer(from, to, amount);
                            }
                            /** @dev Creates `amount` tokens and assigns them to `account`, increasing
                             * the total supply.
                             *
                             * Emits a {Transfer} event with `from` set to the zero address.
                             *
                             * Requirements:
                             *
                             * - `account` cannot be the zero address.
                             */
                            function _mint(address account, uint256 amount) internal virtual {
                                require(account != address(0), "ERC20: mint to the zero address");
                                _beforeTokenTransfer(address(0), account, amount);
                                _totalSupply += amount;
                                unchecked {
                                    // Overflow not possible: balance + amount is at most totalSupply + amount, which is checked above.
                                    _balances[account] += amount;
                                }
                                emit Transfer(address(0), account, amount);
                                _afterTokenTransfer(address(0), account, amount);
                            }
                            /**
                             * @dev Destroys `amount` tokens from `account`, reducing the
                             * total supply.
                             *
                             * Emits a {Transfer} event with `to` set to the zero address.
                             *
                             * Requirements:
                             *
                             * - `account` cannot be the zero address.
                             * - `account` must have at least `amount` tokens.
                             */
                            function _burn(address account, uint256 amount) internal virtual {
                                require(account != address(0), "ERC20: burn from the zero address");
                                _beforeTokenTransfer(account, address(0), amount);
                                uint256 accountBalance = _balances[account];
                                require(accountBalance >= amount, "ERC20: burn amount exceeds balance");
                                unchecked {
                                    _balances[account] = accountBalance - amount;
                                    // Overflow not possible: amount <= accountBalance <= totalSupply.
                                    _totalSupply -= amount;
                                }
                                emit Transfer(account, address(0), amount);
                                _afterTokenTransfer(account, address(0), amount);
                            }
                            /**
                             * @dev Sets `amount` as the allowance of `spender` over the `owner` s tokens.
                             *
                             * This internal function is equivalent to `approve`, and can be used to
                             * e.g. set automatic allowances for certain subsystems, etc.
                             *
                             * Emits an {Approval} event.
                             *
                             * Requirements:
                             *
                             * - `owner` cannot be the zero address.
                             * - `spender` cannot be the zero address.
                             */
                            function _approve(address owner, address spender, uint256 amount) internal virtual {
                                require(owner != address(0), "ERC20: approve from the zero address");
                                require(spender != address(0), "ERC20: approve to the zero address");
                                _allowances[owner][spender] = amount;
                                emit Approval(owner, spender, amount);
                            }
                            /**
                             * @dev Updates `owner` s allowance for `spender` based on spent `amount`.
                             *
                             * Does not update the allowance amount in case of infinite allowance.
                             * Revert if not enough allowance is available.
                             *
                             * Might emit an {Approval} event.
                             */
                            function _spendAllowance(address owner, address spender, uint256 amount) internal virtual {
                                uint256 currentAllowance = allowance(owner, spender);
                                if (currentAllowance != type(uint256).max) {
                                    require(currentAllowance >= amount, "ERC20: insufficient allowance");
                                    unchecked {
                                        _approve(owner, spender, currentAllowance - amount);
                                    }
                                }
                            }
                            /**
                             * @dev Hook that is called before any transfer of tokens. This includes
                             * minting and burning.
                             *
                             * Calling conditions:
                             *
                             * - when `from` and `to` are both non-zero, `amount` of ``from``'s tokens
                             * will be transferred to `to`.
                             * - when `from` is zero, `amount` tokens will be minted for `to`.
                             * - when `to` is zero, `amount` of ``from``'s tokens will be burned.
                             * - `from` and `to` are never both zero.
                             *
                             * To learn more about hooks, head to xref:ROOT:extending-contracts.adoc#using-hooks[Using Hooks].
                             */
                            function _beforeTokenTransfer(address from, address to, uint256 amount) internal virtual {}
                            /**
                             * @dev Hook that is called after any transfer of tokens. This includes
                             * minting and burning.
                             *
                             * Calling conditions:
                             *
                             * - when `from` and `to` are both non-zero, `amount` of ``from``'s tokens
                             * has been transferred to `to`.
                             * - when `from` is zero, `amount` tokens have been minted for `to`.
                             * - when `to` is zero, `amount` of ``from``'s tokens have been burned.
                             * - `from` and `to` are never both zero.
                             *
                             * To learn more about hooks, head to xref:ROOT:extending-contracts.adoc#using-hooks[Using Hooks].
                             */
                            function _afterTokenTransfer(address from, address to, uint256 amount) internal virtual {}
                            /**
                             * @dev This empty reserved space is put in place to allow future versions to add new
                             * variables without shifting down storage in the inheritance chain.
                             * See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps
                             */
                            uint256[45] private __gap;
                        }
                        // SPDX-License-Identifier: MIT
                        // OpenZeppelin Contracts (last updated v4.7.0) (security/Pausable.sol)
                        pragma solidity ^0.8.0;
                        import "../utils/ContextUpgradeable.sol";
                        import "../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 {
                            /**
                             * @dev Emitted when the pause is triggered by `account`.
                             */
                            event Paused(address account);
                            /**
                             * @dev Emitted when the pause is lifted by `account`.
                             */
                            event Unpaused(address account);
                            bool private _paused;
                            /**
                             * @dev Initializes the contract in unpaused state.
                             */
                            function __Pausable_init() internal onlyInitializing {
                                __Pausable_init_unchained();
                            }
                            function __Pausable_init_unchained() internal onlyInitializing {
                                _paused = false;
                            }
                            /**
                             * @dev Modifier to make a function callable only when the contract is not paused.
                             *
                             * Requirements:
                             *
                             * - The contract must not be paused.
                             */
                            modifier whenNotPaused() {
                                _requireNotPaused();
                                _;
                            }
                            /**
                             * @dev Modifier to make a function callable only when the contract is paused.
                             *
                             * Requirements:
                             *
                             * - The contract must be paused.
                             */
                            modifier whenPaused() {
                                _requirePaused();
                                _;
                            }
                            /**
                             * @dev Returns true if the contract is paused, and false otherwise.
                             */
                            function paused() public view virtual returns (bool) {
                                return _paused;
                            }
                            /**
                             * @dev Throws if the contract is paused.
                             */
                            function _requireNotPaused() internal view virtual {
                                require(!paused(), "Pausable: paused");
                            }
                            /**
                             * @dev Throws if the contract is not paused.
                             */
                            function _requirePaused() internal view virtual {
                                require(paused(), "Pausable: not paused");
                            }
                            /**
                             * @dev Triggers stopped state.
                             *
                             * Requirements:
                             *
                             * - The contract must not be paused.
                             */
                            function _pause() internal virtual whenNotPaused {
                                _paused = true;
                                emit Paused(_msgSender());
                            }
                            /**
                             * @dev Returns to normal state.
                             *
                             * Requirements:
                             *
                             * - The contract must be paused.
                             */
                            function _unpause() internal virtual whenPaused {
                                _paused = false;
                                emit Unpaused(_msgSender());
                            }
                            /**
                             * @dev This empty reserved space is put in place to allow future versions to add new
                             * variables without shifting down storage in the inheritance chain.
                             * See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps
                             */
                            uint256[49] private __gap;
                        }
                        // SPDX-License-Identifier: GPL-3.0-or-later
                        pragma solidity 0.8.21;
                        library EigenpieConstants {
                            //contracts
                            bytes32 public constant EIGENPIE_STAKING = keccak256("EIGENPIE_STAKING");
                            bytes32 public constant EIGEN_STRATEGY_MANAGER = keccak256("EIGEN_STRATEGY_MANAGER");
                            bytes32 public constant PRICE_PROVIDER = keccak256("PRICE_PROVIDER");
                            //Roles
                            bytes32 public constant DEFAULT_ADMIN_ROLE = 0x00;
                            bytes32 public constant MANAGER = keccak256("MANAGER");
                            bytes32 public constant MINTER_ROLE = keccak256("MINTER_ROLE");
                            bytes32 public constant BURNER_ROLE = keccak256("BURNER_ROLE");
                            bytes32 public constant ORACLE_ROLE = keccak256("ORACLE_ROLE");
                            bytes32 public constant ORACLE_ADMIN_ROLE = keccak256("ORACLE_ADMIN_ROLE");
                            bytes32 public constant PRICE_PROVIDER_ROLE = keccak256("PRICE_PROVIDER_ROLE");
                            uint256 public constant DENOMINATOR = 10_000;
                        }
                        // SPDX-License-Identifier: GPL-3.0-or-later
                        pragma solidity 0.8.21;
                        interface IEigenpieConfig {
                            // Errors
                            error ValueAlreadyInUse();
                            error AssetAlreadySupported();
                            error AssetNotSupported();
                            error CallerNotEigenpieConfigAdmin();
                            error CallerNotEigenpieConfigManager();
                            error CallerNotEigenpieConfigOracle();
                            error CallerNotEigenpieConfigOracleAdmin();
                            error CallerNotEigenpieConfigPriceProvider();
                            error CallerNotEigenpieConfigMinter();
                            error CallerNotEigenpieConfigBurner();
                            error CallerNotEigenpieConfigAllowedRole(string role);
                            // Events
                            event SetContract(bytes32 key, address indexed contractAddr);
                            event AddedNewSupportedAsset(address indexed asset, address indexed receipt, uint256 depositLimit);
                            event ReceiptTokenUpdated(address indexed asset, address indexed receipt);
                            event RemovedSupportedAsset(address indexed asset);
                            event AssetDepositLimitUpdate(address indexed asset, uint256 depositLimit);
                            event AssetStrategyUpdate(address indexed asset, address indexed strategy);
                            event AssetBoostUpdate(address indexed asset, uint256 newBoost);
                            event ReferralUpdate(address indexed me, address indexed myReferral);
                            // methods
                            function assetStrategy(address asset) external view returns (address);
                            function boostByAsset(address) external view returns (uint256);
                            function mLRTReceiptByAsset(address) external view returns (address);
                            function isSupportedAsset(address asset) external view returns (bool);
                            function getContract(bytes32 contractId) external view returns (address);
                            function getSupportedAssetList() external view returns (address[] memory);
                            function depositLimitByAsset(address asset) external view returns (uint256);
                        }
                        // SPDX-License-Identifier: MIT
                        // OpenZeppelin Contracts v4.4.1 (access/IAccessControl.sol)
                        pragma solidity ^0.8.0;
                        /**
                         * @dev External interface of AccessControl declared to support ERC165 detection.
                         */
                        interface IAccessControl {
                            /**
                             * @dev Emitted when `newAdminRole` is set as ``role``'s admin role, replacing `previousAdminRole`
                             *
                             * `DEFAULT_ADMIN_ROLE` is the starting admin for all roles, despite
                             * {RoleAdminChanged} not being emitted signaling this.
                             *
                             * _Available since v3.1._
                             */
                            event RoleAdminChanged(bytes32 indexed role, bytes32 indexed previousAdminRole, bytes32 indexed newAdminRole);
                            /**
                             * @dev Emitted when `account` is granted `role`.
                             *
                             * `sender` is the account that originated the contract call, an admin role
                             * bearer except when using {AccessControl-_setupRole}.
                             */
                            event RoleGranted(bytes32 indexed role, address indexed account, address indexed sender);
                            /**
                             * @dev Emitted when `account` is revoked `role`.
                             *
                             * `sender` is the account that originated the contract call:
                             *   - if using `revokeRole`, it is the admin role bearer
                             *   - if using `renounceRole`, it is the role bearer (i.e. `account`)
                             */
                            event RoleRevoked(bytes32 indexed role, address indexed account, address indexed sender);
                            /**
                             * @dev Returns `true` if `account` has been granted `role`.
                             */
                            function hasRole(bytes32 role, address account) external view returns (bool);
                            /**
                             * @dev Returns the admin role that controls `role`. See {grantRole} and
                             * {revokeRole}.
                             *
                             * To change a role's admin, use {AccessControl-_setRoleAdmin}.
                             */
                            function getRoleAdmin(bytes32 role) external view returns (bytes32);
                            /**
                             * @dev Grants `role` to `account`.
                             *
                             * If `account` had not been already granted `role`, emits a {RoleGranted}
                             * event.
                             *
                             * Requirements:
                             *
                             * - the caller must have ``role``'s admin role.
                             */
                            function grantRole(bytes32 role, address account) external;
                            /**
                             * @dev Revokes `role` from `account`.
                             *
                             * If `account` had been granted `role`, emits a {RoleRevoked} event.
                             *
                             * Requirements:
                             *
                             * - the caller must have ``role``'s admin role.
                             */
                            function revokeRole(bytes32 role, address account) external;
                            /**
                             * @dev Revokes `role` from the calling account.
                             *
                             * Roles are often managed via {grantRole} and {revokeRole}: this function's
                             * purpose is to provide a mechanism for accounts to lose their privileges
                             * if they are compromised (such as when a trusted device is misplaced).
                             *
                             * If the calling account had been granted `role`, emits a {RoleRevoked}
                             * event.
                             *
                             * Requirements:
                             *
                             * - the caller must be `account`.
                             */
                            function renounceRole(bytes32 role, address account) external;
                        }
                        // SPDX-License-Identifier: MIT
                        // OpenZeppelin Contracts (last updated v4.9.0) (token/ERC20/IERC20.sol)
                        pragma solidity ^0.8.0;
                        /**
                         * @dev Interface of the ERC20 standard as defined in the EIP.
                         */
                        interface IERC20 {
                            /**
                             * @dev Emitted when `value` tokens are moved from one account (`from`) to
                             * another (`to`).
                             *
                             * Note that `value` may be zero.
                             */
                            event Transfer(address indexed from, address indexed to, uint256 value);
                            /**
                             * @dev Emitted when the allowance of a `spender` for an `owner` is set by
                             * a call to {approve}. `value` is the new allowance.
                             */
                            event Approval(address indexed owner, address indexed spender, uint256 value);
                            /**
                             * @dev Returns the amount of tokens in existence.
                             */
                            function totalSupply() external view returns (uint256);
                            /**
                             * @dev Returns the amount of tokens owned by `account`.
                             */
                            function balanceOf(address account) external view returns (uint256);
                            /**
                             * @dev Moves `amount` tokens from the caller's account to `to`.
                             *
                             * Returns a boolean value indicating whether the operation succeeded.
                             *
                             * Emits a {Transfer} event.
                             */
                            function transfer(address to, uint256 amount) external returns (bool);
                            /**
                             * @dev Returns the remaining number of tokens that `spender` will be
                             * allowed to spend on behalf of `owner` through {transferFrom}. This is
                             * zero by default.
                             *
                             * This value changes when {approve} or {transferFrom} are called.
                             */
                            function allowance(address owner, address spender) external view returns (uint256);
                            /**
                             * @dev Sets `amount` as the allowance of `spender` over the caller's tokens.
                             *
                             * Returns a boolean value indicating whether the operation succeeded.
                             *
                             * IMPORTANT: Beware that changing an allowance with this method brings the risk
                             * that someone may use both the old and the new allowance by unfortunate
                             * transaction ordering. One possible solution to mitigate this race
                             * condition is to first reduce the spender's allowance to 0 and set the
                             * desired value afterwards:
                             * https://github.com/ethereum/EIPs/issues/20#issuecomment-263524729
                             *
                             * Emits an {Approval} event.
                             */
                            function approve(address spender, uint256 amount) external returns (bool);
                            /**
                             * @dev Moves `amount` tokens from `from` to `to` using the
                             * allowance mechanism. `amount` is then deducted from the caller's
                             * allowance.
                             *
                             * Returns a boolean value indicating whether the operation succeeded.
                             *
                             * Emits a {Transfer} event.
                             */
                            function transferFrom(address from, address to, uint256 amount) external returns (bool);
                        }
                        // SPDX-License-Identifier: MIT
                        // OpenZeppelin Contracts (last updated v4.9.0) (token/ERC20/IERC20.sol)
                        pragma solidity ^0.8.0;
                        /**
                         * @dev Interface of the ERC20 standard as defined in the EIP.
                         */
                        interface IERC20Upgradeable {
                            /**
                             * @dev Emitted when `value` tokens are moved from one account (`from`) to
                             * another (`to`).
                             *
                             * Note that `value` may be zero.
                             */
                            event Transfer(address indexed from, address indexed to, uint256 value);
                            /**
                             * @dev Emitted when the allowance of a `spender` for an `owner` is set by
                             * a call to {approve}. `value` is the new allowance.
                             */
                            event Approval(address indexed owner, address indexed spender, uint256 value);
                            /**
                             * @dev Returns the amount of tokens in existence.
                             */
                            function totalSupply() external view returns (uint256);
                            /**
                             * @dev Returns the amount of tokens owned by `account`.
                             */
                            function balanceOf(address account) external view returns (uint256);
                            /**
                             * @dev Moves `amount` tokens from the caller's account to `to`.
                             *
                             * Returns a boolean value indicating whether the operation succeeded.
                             *
                             * Emits a {Transfer} event.
                             */
                            function transfer(address to, uint256 amount) external returns (bool);
                            /**
                             * @dev Returns the remaining number of tokens that `spender` will be
                             * allowed to spend on behalf of `owner` through {transferFrom}. This is
                             * zero by default.
                             *
                             * This value changes when {approve} or {transferFrom} are called.
                             */
                            function allowance(address owner, address spender) external view returns (uint256);
                            /**
                             * @dev Sets `amount` as the allowance of `spender` over the caller's tokens.
                             *
                             * Returns a boolean value indicating whether the operation succeeded.
                             *
                             * IMPORTANT: Beware that changing an allowance with this method brings the risk
                             * that someone may use both the old and the new allowance by unfortunate
                             * transaction ordering. One possible solution to mitigate this race
                             * condition is to first reduce the spender's allowance to 0 and set the
                             * desired value afterwards:
                             * https://github.com/ethereum/EIPs/issues/20#issuecomment-263524729
                             *
                             * Emits an {Approval} event.
                             */
                            function approve(address spender, uint256 amount) external returns (bool);
                            /**
                             * @dev Moves `amount` tokens from `from` to `to` using the
                             * allowance mechanism. `amount` is then deducted from the caller's
                             * allowance.
                             *
                             * Returns a boolean value indicating whether the operation succeeded.
                             *
                             * Emits a {Transfer} event.
                             */
                            function transferFrom(address from, address to, uint256 amount) external returns (bool);
                        }
                        // SPDX-License-Identifier: MIT
                        // OpenZeppelin Contracts v4.4.1 (token/ERC20/extensions/IERC20Metadata.sol)
                        pragma solidity ^0.8.0;
                        import "../IERC20Upgradeable.sol";
                        /**
                         * @dev Interface for the optional metadata functions from the ERC20 standard.
                         *
                         * _Available since v4.1._
                         */
                        interface IERC20MetadataUpgradeable is IERC20Upgradeable {
                            /**
                             * @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 v4.4.1 (utils/Context.sol)
                        pragma solidity ^0.8.0;
                        import "../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;
                            }
                            /**
                             * @dev This empty reserved space is put in place to allow future versions to add new
                             * variables without shifting down storage in the inheritance chain.
                             * See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps
                             */
                            uint256[50] private __gap;
                        }
                        // SPDX-License-Identifier: MIT
                        // OpenZeppelin Contracts (last updated v4.9.0) (proxy/utils/Initializable.sol)
                        pragma solidity ^0.8.2;
                        import "../../utils/AddressUpgradeable.sol";
                        /**
                         * @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 Indicates that the contract has been initialized.
                             * @custom:oz-retyped-from bool
                             */
                            uint8 private _initialized;
                            /**
                             * @dev Indicates that the contract is in the process of being initialized.
                             */
                            bool private _initializing;
                            /**
                             * @dev Triggered when the contract has been initialized or reinitialized.
                             */
                            event Initialized(uint8 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 functions marked with `initializer` can be nested in the context of a
                             * constructor.
                             *
                             * Emits an {Initialized} event.
                             */
                            modifier initializer() {
                                bool isTopLevelCall = !_initializing;
                                require(
                                    (isTopLevelCall && _initialized < 1) || (!AddressUpgradeable.isContract(address(this)) && _initialized == 1),
                                    "Initializable: contract is already initialized"
                                );
                                _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 255 will prevent any future reinitialization.
                             *
                             * Emits an {Initialized} event.
                             */
                            modifier reinitializer(uint8 version) {
                                require(!_initializing && _initialized < version, "Initializable: contract is already initialized");
                                _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() {
                                require(_initializing, "Initializable: contract is not initializing");
                                _;
                            }
                            /**
                             * @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 {
                                require(!_initializing, "Initializable: contract is initializing");
                                if (_initialized != type(uint8).max) {
                                    _initialized = type(uint8).max;
                                    emit Initialized(type(uint8).max);
                                }
                            }
                            /**
                             * @dev Returns the highest version that has been initialized. See {reinitializer}.
                             */
                            function _getInitializedVersion() internal view returns (uint8) {
                                return _initialized;
                            }
                            /**
                             * @dev Returns `true` if the contract is currently initializing. See {onlyInitializing}.
                             */
                            function _isInitializing() internal view returns (bool) {
                                return _initializing;
                            }
                        }
                        // SPDX-License-Identifier: MIT
                        // OpenZeppelin Contracts (last updated v4.9.0) (utils/Address.sol)
                        pragma solidity ^0.8.1;
                        /**
                         * @dev Collection of functions related to the address type
                         */
                        library AddressUpgradeable {
                            /**
                             * @dev Returns true if `account` is a contract.
                             *
                             * [IMPORTANT]
                             * ====
                             * It is unsafe to assume that an address for which this function returns
                             * false is an externally-owned account (EOA) and not a contract.
                             *
                             * Among others, `isContract` will return false for the following
                             * types of addresses:
                             *
                             *  - an externally-owned account
                             *  - a contract in construction
                             *  - an address where a contract will be created
                             *  - an address where a contract lived, but was destroyed
                             *
                             * Furthermore, `isContract` will also return true if the target contract within
                             * the same transaction is already scheduled for destruction by `SELFDESTRUCT`,
                             * which only has an effect at the end of a transaction.
                             * ====
                             *
                             * [IMPORTANT]
                             * ====
                             * You shouldn't rely on `isContract` to protect against flash loan attacks!
                             *
                             * Preventing calls from contracts is highly discouraged. It breaks composability, breaks support for smart wallets
                             * like Gnosis Safe, and does not provide security since it can be circumvented by calling from a contract
                             * constructor.
                             * ====
                             */
                            function isContract(address account) internal view returns (bool) {
                                // This method relies on extcodesize/address.code.length, which returns 0
                                // for contracts in construction, since the code is only stored at the end
                                // of the constructor execution.
                                return account.code.length > 0;
                            }
                            /**
                             * @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.0/security-considerations.html#use-the-checks-effects-interactions-pattern[checks-effects-interactions pattern].
                             */
                            function sendValue(address payable recipient, uint256 amount) internal {
                                require(address(this).balance >= amount, "Address: insufficient balance");
                                (bool success, ) = recipient.call{value: amount}("");
                                require(success, "Address: unable to send value, recipient may have reverted");
                            }
                            /**
                             * @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, it is bubbled up by this
                             * function (like regular Solidity function calls).
                             *
                             * 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.
                             *
                             * _Available since v3.1._
                             */
                            function functionCall(address target, bytes memory data) internal returns (bytes memory) {
                                return functionCallWithValue(target, data, 0, "Address: low-level call failed");
                            }
                            /**
                             * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], but with
                             * `errorMessage` as a fallback revert reason when `target` reverts.
                             *
                             * _Available since v3.1._
                             */
                            function functionCall(
                                address target,
                                bytes memory data,
                                string memory errorMessage
                            ) internal returns (bytes memory) {
                                return functionCallWithValue(target, data, 0, errorMessage);
                            }
                            /**
                             * @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`.
                             *
                             * _Available since v3.1._
                             */
                            function functionCallWithValue(address target, bytes memory data, uint256 value) internal returns (bytes memory) {
                                return functionCallWithValue(target, data, value, "Address: low-level call with value failed");
                            }
                            /**
                             * @dev Same as {xref-Address-functionCallWithValue-address-bytes-uint256-}[`functionCallWithValue`], but
                             * with `errorMessage` as a fallback revert reason when `target` reverts.
                             *
                             * _Available since v3.1._
                             */
                            function functionCallWithValue(
                                address target,
                                bytes memory data,
                                uint256 value,
                                string memory errorMessage
                            ) internal returns (bytes memory) {
                                require(address(this).balance >= value, "Address: insufficient balance for call");
                                (bool success, bytes memory returndata) = target.call{value: value}(data);
                                return verifyCallResultFromTarget(target, success, returndata, errorMessage);
                            }
                            /**
                             * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
                             * but performing a static call.
                             *
                             * _Available since v3.3._
                             */
                            function functionStaticCall(address target, bytes memory data) internal view returns (bytes memory) {
                                return functionStaticCall(target, data, "Address: low-level static call failed");
                            }
                            /**
                             * @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`],
                             * but performing a static call.
                             *
                             * _Available since v3.3._
                             */
                            function functionStaticCall(
                                address target,
                                bytes memory data,
                                string memory errorMessage
                            ) internal view returns (bytes memory) {
                                (bool success, bytes memory returndata) = target.staticcall(data);
                                return verifyCallResultFromTarget(target, success, returndata, errorMessage);
                            }
                            /**
                             * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
                             * but performing a delegate call.
                             *
                             * _Available since v3.4._
                             */
                            function functionDelegateCall(address target, bytes memory data) internal returns (bytes memory) {
                                return functionDelegateCall(target, data, "Address: low-level delegate call failed");
                            }
                            /**
                             * @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`],
                             * but performing a delegate call.
                             *
                             * _Available since v3.4._
                             */
                            function functionDelegateCall(
                                address target,
                                bytes memory data,
                                string memory errorMessage
                            ) internal returns (bytes memory) {
                                (bool success, bytes memory returndata) = target.delegatecall(data);
                                return verifyCallResultFromTarget(target, success, returndata, errorMessage);
                            }
                            /**
                             * @dev Tool to verify that a low level call to smart-contract was successful, and revert (either by bubbling
                             * the revert reason or using the provided one) in case of unsuccessful call or if target was not a contract.
                             *
                             * _Available since v4.8._
                             */
                            function verifyCallResultFromTarget(
                                address target,
                                bool success,
                                bytes memory returndata,
                                string memory errorMessage
                            ) internal view returns (bytes memory) {
                                if (success) {
                                    if (returndata.length == 0) {
                                        // only check isContract if the call was successful and the return data is empty
                                        // otherwise we already know that it was a contract
                                        require(isContract(target), "Address: call to non-contract");
                                    }
                                    return returndata;
                                } else {
                                    _revert(returndata, errorMessage);
                                }
                            }
                            /**
                             * @dev Tool to verify that a low level call was successful, and revert if it wasn't, either by bubbling the
                             * revert reason or using the provided one.
                             *
                             * _Available since v4.3._
                             */
                            function verifyCallResult(
                                bool success,
                                bytes memory returndata,
                                string memory errorMessage
                            ) internal pure returns (bytes memory) {
                                if (success) {
                                    return returndata;
                                } else {
                                    _revert(returndata, errorMessage);
                                }
                            }
                            function _revert(bytes memory returndata, string memory errorMessage) 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(errorMessage);
                                }
                            }
                        }