ETH Price: $1,945.53 (-1.35%)

Transaction Decoder

Block:
24114217 at Dec-28-2025 10:47:47 PM +UTC
Transaction Fee:
0.00000553761083512 ETH $0.01
Gas Used:
153,712 Gas / 0.036025885 Gwei

Emitted Events:

697 ERC1967Proxy.0xddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef( 0xddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef, 0x0000000000000000000000003bbf5b5e873543dc90bcaee9bc98bd8ccd06e60f, 0x0000000000000000000000007e13876b92f1a62c599c231f783f682e96b91761, 0000000000000000000000000000000000000000000000540cb2a15fe2266c77 )
698 LinkToken.0xddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef( 0xddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef, 0x0000000000000000000000007e13876b92f1a62c599c231f783f682e96b91761, 0x0000000000000000000000003bbf5b5e873543dc90bcaee9bc98bd8ccd06e60f, 0000000000000000000000000000000000000000000000543a3cf1e2487c23bf )
699 0x7e13876b92f1a62c599c231f783f682e96b91761.0x8b3e96f2b889fa771c53c981b40daf005f63f637f1869f707052d15a3dd97140( 0x8b3e96f2b889fa771c53c981b40daf005f63f637f1869f707052d15a3dd97140, 0x0000000000000000000000003bbf5b5e873543dc90bcaee9bc98bd8ccd06e60f, 0000000000000000000000000000000000000000000000000000000000000001, 0000000000000000000000000000000000000000000000540cb2a15fe2266c76, 0000000000000000000000000000000000000000000000000000000000000000, 0000000000000000000000000000000000000000000000543a3cf1e2487c23bf )

Account State Difference:

  Address   Before After State Difference Code
0x3BBF5B5e...CcD06e60f
0.023854909951614561 Eth
Nonce: 462
0.023849372340779441 Eth
Nonce: 463
0.00000553761083512
0x51491077...4EcF986CA
0x7E13876B...E96B91761
0xb8b295df...26fc43cD5
(BuilderNet)
169.739818495569078801 Eth169.739818495569386225 Eth0.000000000000307424

Execution Trace

Curve: stLINK-ng Pool.3df02124( )
  • Null: 0x000...004.00000000( )
  • Null: 0x000...004.00000000( )
  • LinkToken.balanceOf( _owner=0x7E13876B92F1a62C599C231f783f682E96B91761 ) => ( balance=338720851851782009824788 )
  • ERC1967Proxy.70a08231( )
    • StakingPool.balanceOf( _account=0x7E13876B92F1a62C599C231f783f682E96B91761 ) => ( 145153230963982078568796 )
    • Null: 0x000...004.00000000( )
    • Null: 0x000...004.00000000( )
    • Null: 0x000...004.00000000( )
    • Null: 0x000...004.00000000( )
    • Null: 0x000...004.00000000( )
    • Null: 0x000...004.00000000( )
    • ERC1967Proxy.70a08231( )
      • StakingPool.balanceOf( _account=0x7E13876B92F1a62C599C231f783f682E96B91761 ) => ( 145153230963982078568796 )
      • ERC1967Proxy.23b872dd( )
        • StakingPool.transferFrom( from=0x3BBF5B5e873543dc90bCaEe9BC98bd8CcD06e60f, to=0x7E13876B92F1a62C599C231f783f682E96B91761, amount=1550441473299100101751 ) => ( True )
        • ERC1967Proxy.70a08231( )
          • StakingPool.balanceOf( _account=0x7E13876B92F1a62C599C231f783f682E96B91761 ) => ( 146703672437281178670546 )
          • Null: 0x000...004.00000000( )
          • Null: 0x000...004.00000000( )
          • Null: 0x000...004.00000000( )
          • Null: 0x000...004.00000000( )
          • Null: 0x000...004.00000000( )
          • Null: 0x000...004.00000000( )
          • Null: 0x000...004.00000000( )
          • Null: 0x000...004.00000000( )
          • Null: 0x000...004.00000000( )
          • Null: 0x000...004.00000000( )
          • Null: 0x000...004.00000000( )
          • Null: 0x000...004.00000000( )
          • Null: 0x000...004.00000000( )
          • LinkToken.balanceOf( _owner=0x7E13876B92F1a62C599C231f783f682E96B91761 ) => ( balance=338720851851782009824788 )
          • LinkToken.transfer( _to=0x3BBF5B5e873543dc90bCaEe9BC98bd8CcD06e60f, _value=1553722997098585793471 ) => ( success=True )
            File 1 of 3: ERC1967Proxy
            // SPDX-License-Identifier: MIT
            pragma solidity ^0.8.0;
            import "@openzeppelin/contracts/proxy/beacon/BeaconProxy.sol";
            import "@openzeppelin/contracts/proxy/beacon/UpgradeableBeacon.sol";
            import "@openzeppelin/contracts/proxy/ERC1967/ERC1967Proxy.sol";
            import "@openzeppelin/contracts/proxy/transparent/TransparentUpgradeableProxy.sol";
            import "@openzeppelin/contracts/proxy/transparent/ProxyAdmin.sol";
            // Kept for backwards compatibility with older versions of Hardhat and Truffle plugins.
            contract AdminUpgradeabilityProxy is TransparentUpgradeableProxy {
                constructor(address logic, address admin, bytes memory data) payable TransparentUpgradeableProxy(logic, admin, data) {}
            }
            // SPDX-License-Identifier: MIT
            pragma solidity ^0.8.0;
            import "./IBeacon.sol";
            import "../Proxy.sol";
            import "../ERC1967/ERC1967Upgrade.sol";
            /**
             * @dev This contract implements a proxy that gets the implementation address for each call from a {UpgradeableBeacon}.
             *
             * The beacon address is stored in storage slot `uint256(keccak256('eip1967.proxy.beacon')) - 1`, so that it doesn't
             * conflict with the storage layout of the implementation behind the proxy.
             *
             * _Available since v3.4._
             */
            contract BeaconProxy is Proxy, ERC1967Upgrade {
                /**
                 * @dev Initializes the proxy with `beacon`.
                 *
                 * If `data` is nonempty, it's used as data in a delegate call to the implementation returned by the beacon. This
                 * will typically be an encoded function call, and allows initializating the storage of the proxy like a Solidity
                 * constructor.
                 *
                 * Requirements:
                 *
                 * - `beacon` must be a contract with the interface {IBeacon}.
                 */
                constructor(address beacon, bytes memory data) payable {
                    assert(_BEACON_SLOT == bytes32(uint256(keccak256("eip1967.proxy.beacon")) - 1));
                    _upgradeBeaconToAndCall(beacon, data, false);
                }
                /**
                 * @dev Returns the current beacon address.
                 */
                function _beacon() internal view virtual returns (address) {
                    return _getBeacon();
                }
                /**
                 * @dev Returns the current implementation address of the associated beacon.
                 */
                function _implementation() internal view virtual override returns (address) {
                    return IBeacon(_getBeacon()).implementation();
                }
                /**
                 * @dev Changes the proxy to use a new beacon. Deprecated: see {_upgradeBeaconToAndCall}.
                 *
                 * If `data` is nonempty, it's used as data in a delegate call to the implementation returned by the beacon.
                 *
                 * Requirements:
                 *
                 * - `beacon` must be a contract.
                 * - The implementation returned by `beacon` must be a contract.
                 */
                function _setBeacon(address beacon, bytes memory data) internal virtual {
                    _upgradeBeaconToAndCall(beacon, data, false);
                }
            }
            // SPDX-License-Identifier: MIT
            pragma solidity ^0.8.0;
            import "./IBeacon.sol";
            import "../../access/Ownable.sol";
            import "../../utils/Address.sol";
            /**
             * @dev This contract is used in conjunction with one or more instances of {BeaconProxy} to determine their
             * implementation contract, which is where they will delegate all function calls.
             *
             * An owner is able to change the implementation the beacon points to, thus upgrading the proxies that use this beacon.
             */
            contract UpgradeableBeacon is IBeacon, Ownable {
                address private _implementation;
                /**
                 * @dev Emitted when the implementation returned by the beacon is changed.
                 */
                event Upgraded(address indexed implementation);
                /**
                 * @dev Sets the address of the initial implementation, and the deployer account as the owner who can upgrade the
                 * beacon.
                 */
                constructor(address implementation_) {
                    _setImplementation(implementation_);
                }
                /**
                 * @dev Returns the current implementation address.
                 */
                function implementation() public view virtual override returns (address) {
                    return _implementation;
                }
                /**
                 * @dev Upgrades the beacon to a new implementation.
                 *
                 * Emits an {Upgraded} event.
                 *
                 * Requirements:
                 *
                 * - msg.sender must be the owner of the contract.
                 * - `newImplementation` must be a contract.
                 */
                function upgradeTo(address newImplementation) public virtual onlyOwner {
                    _setImplementation(newImplementation);
                    emit Upgraded(newImplementation);
                }
                /**
                 * @dev Sets the implementation contract address for this beacon
                 *
                 * Requirements:
                 *
                 * - `newImplementation` must be a contract.
                 */
                function _setImplementation(address newImplementation) private {
                    require(Address.isContract(newImplementation), "UpgradeableBeacon: implementation is not a contract");
                    _implementation = newImplementation;
                }
            }
            // SPDX-License-Identifier: MIT
            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 initializating the storage of the proxy like a Solidity constructor.
                 */
                constructor(address _logic, bytes memory _data) payable {
                    assert(_IMPLEMENTATION_SLOT == bytes32(uint256(keccak256("eip1967.proxy.implementation")) - 1));
                    _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
            pragma solidity ^0.8.0;
            import "../ERC1967/ERC1967Proxy.sol";
            /**
             * @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.
             */
            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) {
                    assert(_ADMIN_SLOT == bytes32(uint256(keccak256("eip1967.proxy.admin")) - 1));
                    _changeAdmin(admin_);
                }
                /**
                 * @dev Modifier used internally that will delegate the call to the implementation unless the sender is the admin.
                 */
                modifier ifAdmin() {
                    if (msg.sender == _getAdmin()) {
                        _;
                    } else {
                        _fallback();
                    }
                }
                /**
                 * @dev Returns the current admin.
                 *
                 * NOTE: Only the admin can call this function. See {ProxyAdmin-getProxyAdmin}.
                 *
                 * 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 admin() external ifAdmin returns (address admin_) {
                    admin_ = _getAdmin();
                }
                /**
                 * @dev Returns the current implementation.
                 *
                 * NOTE: Only the admin can call this function. See {ProxyAdmin-getProxyImplementation}.
                 *
                 * TIP: To get this value clients can read directly from the storage slot shown below (specified by EIP1967) using the
                 * https://eth.wiki/json-rpc/API#eth_getstorageat[`eth_getStorageAt`] RPC call.
                 * `0x360894a13ba1a3210667c828492db98dca3e2076cc3735a920a3ca505d382bbc`
                 */
                function implementation() external ifAdmin returns (address implementation_) {
                    implementation_ = _implementation();
                }
                /**
                 * @dev Changes the admin of the proxy.
                 *
                 * Emits an {AdminChanged} event.
                 *
                 * NOTE: Only the admin can call this function. See {ProxyAdmin-changeProxyAdmin}.
                 */
                function changeAdmin(address newAdmin) external virtual ifAdmin {
                    _changeAdmin(newAdmin);
                }
                /**
                 * @dev Upgrade the implementation of the proxy.
                 *
                 * NOTE: Only the admin can call this function. See {ProxyAdmin-upgrade}.
                 */
                function upgradeTo(address newImplementation) external ifAdmin {
                    _upgradeToAndCall(newImplementation, bytes(""), false);
                }
                /**
                 * @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.
                 *
                 * NOTE: Only the admin can call this function. See {ProxyAdmin-upgradeAndCall}.
                 */
                function upgradeToAndCall(address newImplementation, bytes calldata data) external payable ifAdmin {
                    _upgradeToAndCall(newImplementation, data, true);
                }
                /**
                 * @dev Returns the current admin.
                 */
                function _admin() internal view virtual returns (address) {
                    return _getAdmin();
                }
                /**
                 * @dev Makes sure the admin cannot access the fallback function. See {Proxy-_beforeFallback}.
                 */
                function _beforeFallback() internal virtual override {
                    require(msg.sender != _getAdmin(), "TransparentUpgradeableProxy: admin cannot fallback to proxy target");
                    super._beforeFallback();
                }
            }
            // SPDX-License-Identifier: MIT
            pragma solidity ^0.8.0;
            import "./TransparentUpgradeableProxy.sol";
            import "../../access/Ownable.sol";
            /**
             * @dev This is an auxiliary contract meant to be assigned as the admin of a {TransparentUpgradeableProxy}. For an
             * explanation of why you would want to use this see the documentation for {TransparentUpgradeableProxy}.
             */
            contract ProxyAdmin is Ownable {
                /**
                 * @dev Returns the current implementation of `proxy`.
                 *
                 * Requirements:
                 *
                 * - This contract must be the admin of `proxy`.
                 */
                function getProxyImplementation(TransparentUpgradeableProxy proxy) public view virtual returns (address) {
                    // We need to manually run the static call since the getter cannot be flagged as view
                    // bytes4(keccak256("implementation()")) == 0x5c60da1b
                    (bool success, bytes memory returndata) = address(proxy).staticcall(hex"5c60da1b");
                    require(success);
                    return abi.decode(returndata, (address));
                }
                /**
                 * @dev Returns the current admin of `proxy`.
                 *
                 * Requirements:
                 *
                 * - This contract must be the admin of `proxy`.
                 */
                function getProxyAdmin(TransparentUpgradeableProxy proxy) public view virtual returns (address) {
                    // We need to manually run the static call since the getter cannot be flagged as view
                    // bytes4(keccak256("admin()")) == 0xf851a440
                    (bool success, bytes memory returndata) = address(proxy).staticcall(hex"f851a440");
                    require(success);
                    return abi.decode(returndata, (address));
                }
                /**
                 * @dev Changes the admin of `proxy` to `newAdmin`.
                 *
                 * Requirements:
                 *
                 * - This contract must be the current admin of `proxy`.
                 */
                function changeProxyAdmin(TransparentUpgradeableProxy proxy, address newAdmin) public virtual onlyOwner {
                    proxy.changeAdmin(newAdmin);
                }
                /**
                 * @dev Upgrades `proxy` to `implementation`. See {TransparentUpgradeableProxy-upgradeTo}.
                 *
                 * Requirements:
                 *
                 * - This contract must be the admin of `proxy`.
                 */
                function upgrade(TransparentUpgradeableProxy proxy, address implementation) public virtual onlyOwner {
                    proxy.upgradeTo(implementation);
                }
                /**
                 * @dev Upgrades `proxy` to `implementation` and calls a function on the new implementation. See
                 * {TransparentUpgradeableProxy-upgradeToAndCall}.
                 *
                 * Requirements:
                 *
                 * - This contract must be the admin of `proxy`.
                 */
                function upgradeAndCall(TransparentUpgradeableProxy proxy, address implementation, bytes memory data) public payable virtual onlyOwner {
                    proxy.upgradeToAndCall{value: msg.value}(implementation, data);
                }
            }
            // SPDX-License-Identifier: MIT
            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
            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 internall call site, it will return directly to the external caller.
                 */
                function _delegate(address implementation) internal virtual {
                    // solhint-disable-next-line no-inline-assembly
                    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 overriden 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 internall 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 overriden should call `super._beforeFallback()`.
                 */
                function _beforeFallback() internal virtual {
                }
            }
            // SPDX-License-Identifier: MIT
            pragma solidity ^0.8.2;
            import "../beacon/IBeacon.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._
             *
             * @custom:oz-upgrades-unsafe-allow delegatecall
             */
            abstract contract ERC1967Upgrade {
                // 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 Emitted when the implementation is upgraded.
                 */
                event Upgraded(address indexed implementation);
                /**
                 * @dev Returns the current implementation address.
                 */
                function _getImplementation() internal view returns (address) {
                    return StorageSlot.getAddressSlot(_IMPLEMENTATION_SLOT).value;
                }
                /**
                 * @dev Stores a new address in the EIP1967 implementation slot.
                 */
                function _setImplementation(address newImplementation) private {
                    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 {
                    _setImplementation(newImplementation);
                    emit Upgraded(newImplementation);
                    if (data.length > 0 || forceCall) {
                        Address.functionDelegateCall(newImplementation, data);
                    }
                }
                /**
                 * @dev Perform implementation upgrade with security checks for UUPS proxies, and additional setup call.
                 *
                 * Emits an {Upgraded} event.
                 */
                function _upgradeToAndCallSecure(address newImplementation, bytes memory data, bool forceCall) internal {
                    address oldImplementation = _getImplementation();
                    // Initial upgrade and setup call
                    _setImplementation(newImplementation);
                    if (data.length > 0 || forceCall) {
                        Address.functionDelegateCall(newImplementation, data);
                    }
                    // Perform rollback test if not already in progress
                    StorageSlot.BooleanSlot storage rollbackTesting = StorageSlot.getBooleanSlot(_ROLLBACK_SLOT);
                    if (!rollbackTesting.value) {
                        // Trigger rollback using upgradeTo from the new implementation
                        rollbackTesting.value = true;
                        Address.functionDelegateCall(
                            newImplementation,
                            abi.encodeWithSignature(
                                "upgradeTo(address)",
                                oldImplementation
                            )
                        );
                        rollbackTesting.value = false;
                        // Check rollback was effective
                        require(oldImplementation == _getImplementation(), "ERC1967Upgrade: upgrade breaks further upgrades");
                        // Finally reset to the new implementation and log the upgrade
                        _setImplementation(newImplementation);
                        emit Upgraded(newImplementation);
                    }
                }
                /**
                 * @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);
                    }
                }
                /**
                 * @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 Emitted when the admin account has changed.
                 */
                event AdminChanged(address previousAdmin, address newAdmin);
                /**
                 * @dev Returns the current admin.
                 */
                function _getAdmin() internal view returns (address) {
                    return StorageSlot.getAddressSlot(_ADMIN_SLOT).value;
                }
                /**
                 * @dev Stores a new address in the EIP1967 admin slot.
                 */
                function _setAdmin(address newAdmin) private {
                    require(newAdmin != address(0), "ERC1967: new admin is the zero address");
                    StorageSlot.getAddressSlot(_ADMIN_SLOT).value = newAdmin;
                }
                /**
                 * @dev Changes the admin of the proxy.
                 *
                 * Emits an {AdminChanged} event.
                 */
                function _changeAdmin(address newAdmin) internal {
                    emit AdminChanged(_getAdmin(), newAdmin);
                    _setAdmin(newAdmin);
                }
                /**
                 * @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 Emitted when the beacon is upgraded.
                 */
                event BeaconUpgraded(address indexed beacon);
                /**
                 * @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;
                }
            }
            // SPDX-License-Identifier: MIT
            pragma solidity ^0.8.0;
            /**
             * @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
                 * ====
                 */
                function isContract(address account) internal view returns (bool) {
                    // This method relies on extcodesize, which returns 0 for contracts in
                    // construction, since the code is only stored at the end of the
                    // constructor execution.
                    uint256 size;
                    // solhint-disable-next-line no-inline-assembly
                    assembly { size := extcodesize(account) }
                    return size > 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://diligence.consensys.net/posts/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.5.11/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");
                    // solhint-disable-next-line avoid-low-level-calls, avoid-call-value
                    (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 functionCall(target, data, "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");
                    require(isContract(target), "Address: call to non-contract");
                    // solhint-disable-next-line avoid-low-level-calls
                    (bool success, bytes memory returndata) = target.call{ value: value }(data);
                    return _verifyCallResult(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) {
                    require(isContract(target), "Address: static call to non-contract");
                    // solhint-disable-next-line avoid-low-level-calls
                    (bool success, bytes memory returndata) = target.staticcall(data);
                    return _verifyCallResult(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) {
                    require(isContract(target), "Address: delegate call to non-contract");
                    // solhint-disable-next-line avoid-low-level-calls
                    (bool success, bytes memory returndata) = target.delegatecall(data);
                    return _verifyCallResult(success, returndata, errorMessage);
                }
                function _verifyCallResult(bool success, bytes memory returndata, string memory errorMessage) private pure returns(bytes memory) {
                    if (success) {
                        return returndata;
                    } else {
                        // 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
                            // solhint-disable-next-line no-inline-assembly
                            assembly {
                                let returndata_size := mload(returndata)
                                revert(add(32, returndata), returndata_size)
                            }
                        } else {
                            revert(errorMessage);
                        }
                    }
                }
            }
            // SPDX-License-Identifier: MIT
            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:
             * ```
             * 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`, and `uint256`._
             */
            library StorageSlot {
                struct AddressSlot {
                    address value;
                }
                struct BooleanSlot {
                    bool value;
                }
                struct Bytes32Slot {
                    bytes32 value;
                }
                struct Uint256Slot {
                    uint256 value;
                }
                /**
                 * @dev Returns an `AddressSlot` with member `value` located at `slot`.
                 */
                function getAddressSlot(bytes32 slot) internal pure returns (AddressSlot storage r) {
                    assembly {
                        r.slot := slot
                    }
                }
                /**
                 * @dev Returns an `BooleanSlot` with member `value` located at `slot`.
                 */
                function getBooleanSlot(bytes32 slot) internal pure returns (BooleanSlot storage r) {
                    assembly {
                        r.slot := slot
                    }
                }
                /**
                 * @dev Returns an `Bytes32Slot` with member `value` located at `slot`.
                 */
                function getBytes32Slot(bytes32 slot) internal pure returns (Bytes32Slot storage r) {
                    assembly {
                        r.slot := slot
                    }
                }
                /**
                 * @dev Returns an `Uint256Slot` with member `value` located at `slot`.
                 */
                function getUint256Slot(bytes32 slot) internal pure returns (Uint256Slot storage r) {
                    assembly {
                        r.slot := slot
                    }
                }
            }
            // SPDX-License-Identifier: MIT
            pragma solidity ^0.8.0;
            import "../utils/Context.sol";
            /**
             * @dev Contract module which provides a basic access control mechanism, where
             * there is an account (an owner) that can be granted exclusive access to
             * specific functions.
             *
             * By default, the owner account will be the one that deploys the contract. This
             * can later be changed with {transferOwnership}.
             *
             * This module is used through inheritance. It will make available the modifier
             * `onlyOwner`, which can be applied to your functions to restrict their use to
             * the owner.
             */
            abstract contract Ownable is Context {
                address private _owner;
                event OwnershipTransferred(address indexed previousOwner, address indexed newOwner);
                /**
                 * @dev Initializes the contract setting the deployer as the initial owner.
                 */
                constructor () {
                    address msgSender = _msgSender();
                    _owner = msgSender;
                    emit OwnershipTransferred(address(0), msgSender);
                }
                /**
                 * @dev Returns the address of the current owner.
                 */
                function owner() public view virtual returns (address) {
                    return _owner;
                }
                /**
                 * @dev Throws if called by any account other than the owner.
                 */
                modifier onlyOwner() {
                    require(owner() == _msgSender(), "Ownable: caller is not the owner");
                    _;
                }
                /**
                 * @dev Leaves the contract without owner. It will not be possible to call
                 * `onlyOwner` functions anymore. Can only be called by the current owner.
                 *
                 * NOTE: Renouncing ownership will leave the contract without an owner,
                 * thereby removing any functionality that is only available to the owner.
                 */
                function renounceOwnership() public virtual onlyOwner {
                    emit OwnershipTransferred(_owner, address(0));
                    _owner = address(0);
                }
                /**
                 * @dev Transfers ownership of the contract to a new account (`newOwner`).
                 * Can only be called by the current owner.
                 */
                function transferOwnership(address newOwner) public virtual onlyOwner {
                    require(newOwner != address(0), "Ownable: new owner is the zero address");
                    emit OwnershipTransferred(_owner, newOwner);
                    _owner = newOwner;
                }
            }
            // SPDX-License-Identifier: MIT
            pragma solidity ^0.8.0;
            /*
             * @dev Provides information about the current execution context, including the
             * sender of the transaction and its data. While these are generally available
             * via msg.sender and msg.data, they should not be accessed in such a direct
             * manner, since when dealing with meta-transactions the account sending and
             * paying for execution may not be the actual sender (as far as an application
             * is concerned).
             *
             * This contract is only required for intermediate, library-like contracts.
             */
            abstract contract Context {
                function _msgSender() internal view virtual returns (address) {
                    return msg.sender;
                }
                function _msgData() internal view virtual returns (bytes calldata) {
                    this; // silence state mutability warning without generating bytecode - see https://github.com/ethereum/solidity/issues/2691
                    return msg.data;
                }
            }
            

            File 2 of 3: LinkToken
            pragma solidity ^0.4.16;
            
            
            /**
             * @title SafeMath
             * @dev Math operations with safety checks that throw on error
             */
            library SafeMath {
              function mul(uint256 a, uint256 b) internal constant returns (uint256) {
                uint256 c = a * b;
                assert(a == 0 || c / a == b);
                return c;
              }
            
              function div(uint256 a, uint256 b) internal constant returns (uint256) {
                // assert(b > 0); // Solidity automatically throws when dividing by 0
                uint256 c = a / b;
                // assert(a == b * c + a % b); // There is no case in which this doesn't hold
                return c;
              }
            
              function sub(uint256 a, uint256 b) internal constant returns (uint256) {
                assert(b <= a);
                return a - b;
              }
            
              function add(uint256 a, uint256 b) internal constant returns (uint256) {
                uint256 c = a + b;
                assert(c >= a);
                return c;
              }
            }
            
            
            /**
             * @title ERC20Basic
             * @dev Simpler version of ERC20 interface
             * @dev see https://github.com/ethereum/EIPs/issues/179
             */
            contract ERC20Basic {
              uint256 public totalSupply;
              function balanceOf(address who) constant returns (uint256);
              function transfer(address to, uint256 value) returns (bool);
              event Transfer(address indexed from, address indexed to, uint256 value);
            }
            /**
             * @title ERC20 interface
             * @dev see https://github.com/ethereum/EIPs/issues/20
             */
            contract ERC20 is ERC20Basic {
              function allowance(address owner, address spender) constant returns (uint256);
              function transferFrom(address from, address to, uint256 value) returns (bool);
              function approve(address spender, uint256 value) returns (bool);
              event Approval(address indexed owner, address indexed spender, uint256 value);
            }
            
            contract ERC677 is ERC20 {
              function transferAndCall(address to, uint value, bytes data) returns (bool success);
            
              event Transfer(address indexed from, address indexed to, uint value, bytes data);
            }
            
            contract ERC677Receiver {
              function onTokenTransfer(address _sender, uint _value, bytes _data);
            }
            
            /**
             * @title Basic token
             * @dev Basic version of StandardToken, with no allowances. 
             */
            contract BasicToken is ERC20Basic {
              using SafeMath for uint256;
            
              mapping(address => uint256) balances;
            
              /**
              * @dev transfer token for a specified address
              * @param _to The address to transfer to.
              * @param _value The amount to be transferred.
              */
              function transfer(address _to, uint256 _value) returns (bool) {
                balances[msg.sender] = balances[msg.sender].sub(_value);
                balances[_to] = balances[_to].add(_value);
                Transfer(msg.sender, _to, _value);
                return true;
              }
            
              /**
              * @dev Gets the balance of the specified address.
              * @param _owner The address to query the the balance of. 
              * @return An uint256 representing the amount owned by the passed address.
              */
              function balanceOf(address _owner) constant returns (uint256 balance) {
                return balances[_owner];
              }
            
            }
            
            
            /**
             * @title Standard ERC20 token
             *
             * @dev Implementation of the basic standard token.
             * @dev https://github.com/ethereum/EIPs/issues/20
             * @dev Based on code by FirstBlood: https://github.com/Firstbloodio/token/blob/master/smart_contract/FirstBloodToken.sol
             */
            contract StandardToken is ERC20, BasicToken {
            
              mapping (address => mapping (address => uint256)) allowed;
            
            
              /**
               * @dev Transfer tokens from one address to another
               * @param _from address The address which you want to send tokens from
               * @param _to address The address which you want to transfer to
               * @param _value uint256 the amount of tokens to be transferred
               */
              function transferFrom(address _from, address _to, uint256 _value) returns (bool) {
                var _allowance = allowed[_from][msg.sender];
            
                // Check is not needed because sub(_allowance, _value) will already throw if this condition is not met
                // require (_value <= _allowance);
            
                balances[_from] = balances[_from].sub(_value);
                balances[_to] = balances[_to].add(_value);
                allowed[_from][msg.sender] = _allowance.sub(_value);
                Transfer(_from, _to, _value);
                return true;
              }
            
              /**
               * @dev Approve the passed address to spend the specified amount of tokens on behalf of msg.sender.
               * @param _spender The address which will spend the funds.
               * @param _value The amount of tokens to be spent.
               */
              function approve(address _spender, uint256 _value) returns (bool) {
                allowed[msg.sender][_spender] = _value;
                Approval(msg.sender, _spender, _value);
                return true;
              }
            
              /**
               * @dev Function to check the amount of tokens that an owner allowed to a spender.
               * @param _owner address The address which owns the funds.
               * @param _spender address The address which will spend the funds.
               * @return A uint256 specifying the amount of tokens still available for the spender.
               */
              function allowance(address _owner, address _spender) constant returns (uint256 remaining) {
                return allowed[_owner][_spender];
              }
              
                /*
               * approve should be called when allowed[_spender] == 0. To increment
               * allowed value is better to use this function to avoid 2 calls (and wait until 
               * the first transaction is mined)
               * From MonolithDAO Token.sol
               */
              function increaseApproval (address _spender, uint _addedValue) 
                returns (bool success) {
                allowed[msg.sender][_spender] = allowed[msg.sender][_spender].add(_addedValue);
                Approval(msg.sender, _spender, allowed[msg.sender][_spender]);
                return true;
              }
            
              function decreaseApproval (address _spender, uint _subtractedValue) 
                returns (bool success) {
                uint oldValue = allowed[msg.sender][_spender];
                if (_subtractedValue > oldValue) {
                  allowed[msg.sender][_spender] = 0;
                } else {
                  allowed[msg.sender][_spender] = oldValue.sub(_subtractedValue);
                }
                Approval(msg.sender, _spender, allowed[msg.sender][_spender]);
                return true;
              }
            
            }
            
            contract ERC677Token is ERC677 {
            
              /**
              * @dev transfer token to a contract address with additional data if the recipient is a contact.
              * @param _to The address to transfer to.
              * @param _value The amount to be transferred.
              * @param _data The extra data to be passed to the receiving contract.
              */
              function transferAndCall(address _to, uint _value, bytes _data)
                public
                returns (bool success)
              {
                super.transfer(_to, _value);
                Transfer(msg.sender, _to, _value, _data);
                if (isContract(_to)) {
                  contractFallback(_to, _value, _data);
                }
                return true;
              }
            
            
              // PRIVATE
            
              function contractFallback(address _to, uint _value, bytes _data)
                private
              {
                ERC677Receiver receiver = ERC677Receiver(_to);
                receiver.onTokenTransfer(msg.sender, _value, _data);
              }
            
              function isContract(address _addr)
                private
                returns (bool hasCode)
              {
                uint length;
                assembly { length := extcodesize(_addr) }
                return length > 0;
              }
            
            }
            
            contract LinkToken is StandardToken, ERC677Token {
            
              uint public constant totalSupply = 10**27;
              string public constant name = 'ChainLink Token';
              uint8 public constant decimals = 18;
              string public constant symbol = 'LINK';
            
              function LinkToken()
                public
              {
                balances[msg.sender] = totalSupply;
              }
            
              /**
              * @dev transfer token to a specified address with additional data if the recipient is a contract.
              * @param _to The address to transfer to.
              * @param _value The amount to be transferred.
              * @param _data The extra data to be passed to the receiving contract.
              */
              function transferAndCall(address _to, uint _value, bytes _data)
                public
                validRecipient(_to)
                returns (bool success)
              {
                return super.transferAndCall(_to, _value, _data);
              }
            
              /**
              * @dev transfer token to a specified address.
              * @param _to The address to transfer to.
              * @param _value The amount to be transferred.
              */
              function transfer(address _to, uint _value)
                public
                validRecipient(_to)
                returns (bool success)
              {
                return super.transfer(_to, _value);
              }
            
              /**
               * @dev Approve the passed address to spend the specified amount of tokens on behalf of msg.sender.
               * @param _spender The address which will spend the funds.
               * @param _value The amount of tokens to be spent.
               */
              function approve(address _spender, uint256 _value)
                public
                validRecipient(_spender)
                returns (bool)
              {
                return super.approve(_spender,  _value);
              }
            
              /**
               * @dev Transfer tokens from one address to another
               * @param _from address The address which you want to send tokens from
               * @param _to address The address which you want to transfer to
               * @param _value uint256 the amount of tokens to be transferred
               */
              function transferFrom(address _from, address _to, uint256 _value)
                public
                validRecipient(_to)
                returns (bool)
              {
                return super.transferFrom(_from, _to, _value);
              }
            
            
              // MODIFIERS
            
              modifier validRecipient(address _recipient) {
                require(_recipient != address(0) && _recipient != address(this));
                _;
              }
            
            }

            File 3 of 3: StakingPool
            // SPDX-License-Identifier: MIT
            // OpenZeppelin Contracts (last updated v4.9.0) (access/Ownable.sol)
            pragma solidity ^0.8.0;
            import "../utils/ContextUpgradeable.sol";
            import {Initializable} from "../proxy/utils/Initializable.sol";
            /**
             * @dev Contract module which provides a basic access control mechanism, where
             * there is an account (an owner) that can be granted exclusive access to
             * specific functions.
             *
             * By default, the owner account will be the one that deploys the contract. This
             * can later be changed with {transferOwnership}.
             *
             * This module is used through inheritance. It will make available the modifier
             * `onlyOwner`, which can be applied to your functions to restrict their use to
             * the owner.
             */
            abstract contract OwnableUpgradeable is Initializable, ContextUpgradeable {
                address private _owner;
                event OwnershipTransferred(address indexed previousOwner, address indexed newOwner);
                /**
                 * @dev Initializes the contract setting the deployer as the initial owner.
                 */
                function __Ownable_init() internal onlyInitializing {
                    __Ownable_init_unchained();
                }
                function __Ownable_init_unchained() internal onlyInitializing {
                    _transferOwnership(_msgSender());
                }
                /**
                 * @dev Throws if called by any account other than the owner.
                 */
                modifier onlyOwner() {
                    _checkOwner();
                    _;
                }
                /**
                 * @dev Returns the address of the current owner.
                 */
                function owner() public view virtual returns (address) {
                    return _owner;
                }
                /**
                 * @dev Throws if the sender is not the owner.
                 */
                function _checkOwner() internal view virtual {
                    require(owner() == _msgSender(), "Ownable: caller is not the owner");
                }
                /**
                 * @dev Leaves the contract without owner. It will not be possible to call
                 * `onlyOwner` functions. Can only be called by the current owner.
                 *
                 * NOTE: Renouncing ownership will leave the contract without an owner,
                 * thereby disabling any functionality that is only available to the owner.
                 */
                function renounceOwnership() public virtual onlyOwner {
                    _transferOwnership(address(0));
                }
                /**
                 * @dev Transfers ownership of the contract to a new account (`newOwner`).
                 * Can only be called by the current owner.
                 */
                function transferOwnership(address newOwner) public virtual onlyOwner {
                    require(newOwner != address(0), "Ownable: new owner is the zero address");
                    _transferOwnership(newOwner);
                }
                /**
                 * @dev Transfers ownership of the contract to a new account (`newOwner`).
                 * Internal function without access restriction.
                 */
                function _transferOwnership(address newOwner) internal virtual {
                    address oldOwner = _owner;
                    _owner = newOwner;
                    emit OwnershipTransferred(oldOwner, newOwner);
                }
                /**
                 * @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.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 IERC1822ProxiableUpgradeable {
                /**
                 * @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) (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 IERC1967Upgradeable {
                /**
                 * @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 v4.4.1 (proxy/beacon/IBeacon.sol)
            pragma solidity ^0.8.0;
            /**
             * @dev This is the interface that {BeaconProxy} expects of its beacon.
             */
            interface IBeaconUpgradeable {
                /**
                 * @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) (proxy/ERC1967/ERC1967Upgrade.sol)
            pragma solidity ^0.8.2;
            import "../beacon/IBeaconUpgradeable.sol";
            import "../../interfaces/IERC1967Upgradeable.sol";
            import "../../interfaces/draft-IERC1822Upgradeable.sol";
            import "../../utils/AddressUpgradeable.sol";
            import "../../utils/StorageSlotUpgradeable.sol";
            import {Initializable} from "../utils/Initializable.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 ERC1967UpgradeUpgradeable is Initializable, IERC1967Upgradeable {
                // 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;
                function __ERC1967Upgrade_init() internal onlyInitializing {
                }
                function __ERC1967Upgrade_init_unchained() internal onlyInitializing {
                }
                /**
                 * @dev Returns the current implementation address.
                 */
                function _getImplementation() internal view returns (address) {
                    return StorageSlotUpgradeable.getAddressSlot(_IMPLEMENTATION_SLOT).value;
                }
                /**
                 * @dev Stores a new address in the EIP1967 implementation slot.
                 */
                function _setImplementation(address newImplementation) private {
                    require(AddressUpgradeable.isContract(newImplementation), "ERC1967: new implementation is not a contract");
                    StorageSlotUpgradeable.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) {
                        AddressUpgradeable.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 (StorageSlotUpgradeable.getBooleanSlot(_ROLLBACK_SLOT).value) {
                        _setImplementation(newImplementation);
                    } else {
                        try IERC1822ProxiableUpgradeable(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 StorageSlotUpgradeable.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");
                    StorageSlotUpgradeable.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 StorageSlotUpgradeable.getAddressSlot(_BEACON_SLOT).value;
                }
                /**
                 * @dev Stores a new beacon in the EIP1967 beacon slot.
                 */
                function _setBeacon(address newBeacon) private {
                    require(AddressUpgradeable.isContract(newBeacon), "ERC1967: new beacon is not a contract");
                    require(
                        AddressUpgradeable.isContract(IBeaconUpgradeable(newBeacon).implementation()),
                        "ERC1967: beacon implementation is not a contract"
                    );
                    StorageSlotUpgradeable.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) {
                        AddressUpgradeable.functionDelegateCall(IBeaconUpgradeable(newBeacon).implementation(), 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) (proxy/utils/UUPSUpgradeable.sol)
            pragma solidity ^0.8.0;
            import "../../interfaces/draft-IERC1822Upgradeable.sol";
            import "../ERC1967/ERC1967UpgradeUpgradeable.sol";
            import {Initializable} from "./Initializable.sol";
            /**
             * @dev An upgradeability mechanism designed for UUPS proxies. The functions included here can perform an upgrade of an
             * {ERC1967Proxy}, when this contract is set as the implementation behind such a proxy.
             *
             * A security mechanism ensures that an upgrade does not turn off upgradeability accidentally, although this risk is
             * reinstated if the upgrade retains upgradeability but removes the security mechanism, e.g. by replacing
             * `UUPSUpgradeable` with a custom implementation of upgrades.
             *
             * The {_authorizeUpgrade} function must be overridden to include access restriction to the upgrade mechanism.
             *
             * _Available since v4.1._
             */
            abstract contract UUPSUpgradeable is Initializable, IERC1822ProxiableUpgradeable, ERC1967UpgradeUpgradeable {
                /// @custom:oz-upgrades-unsafe-allow state-variable-immutable state-variable-assignment
                address private immutable __self = address(this);
                /**
                 * @dev Check that the execution is being performed through a delegatecall call and that the execution context is
                 * a proxy contract with an implementation (as defined in ERC1967) pointing to self. This should only be the case
                 * for UUPS and transparent proxies that are using the current contract as their implementation. Execution of a
                 * function through ERC1167 minimal proxies (clones) would not normally pass this test, but is not guaranteed to
                 * fail.
                 */
                modifier onlyProxy() {
                    require(address(this) != __self, "Function must be called through delegatecall");
                    require(_getImplementation() == __self, "Function must be called through active proxy");
                    _;
                }
                /**
                 * @dev Check that the execution is not being performed through a delegate call. This allows a function to be
                 * callable on the implementing contract but not through proxies.
                 */
                modifier notDelegated() {
                    require(address(this) == __self, "UUPSUpgradeable: must not be called through delegatecall");
                    _;
                }
                function __UUPSUpgradeable_init() internal onlyInitializing {
                }
                function __UUPSUpgradeable_init_unchained() internal onlyInitializing {
                }
                /**
                 * @dev Implementation of the ERC1822 {proxiableUUID} function. This returns the storage slot used by the
                 * implementation. It is used to validate the implementation's compatibility when performing an upgrade.
                 *
                 * IMPORTANT: A proxy pointing at a proxiable contract should not be considered proxiable itself, because this risks
                 * bricking a proxy that upgrades to it, by delegating to itself until out of gas. Thus it is critical that this
                 * function revert if invoked through a proxy. This is guaranteed by the `notDelegated` modifier.
                 */
                function proxiableUUID() external view virtual override notDelegated returns (bytes32) {
                    return _IMPLEMENTATION_SLOT;
                }
                /**
                 * @dev Upgrade the implementation of the proxy to `newImplementation`.
                 *
                 * Calls {_authorizeUpgrade}.
                 *
                 * Emits an {Upgraded} event.
                 *
                 * @custom:oz-upgrades-unsafe-allow-reachable delegatecall
                 */
                function upgradeTo(address newImplementation) public virtual onlyProxy {
                    _authorizeUpgrade(newImplementation);
                    _upgradeToAndCallUUPS(newImplementation, new bytes(0), false);
                }
                /**
                 * @dev Upgrade the implementation of the proxy to `newImplementation`, and subsequently execute the function call
                 * encoded in `data`.
                 *
                 * Calls {_authorizeUpgrade}.
                 *
                 * Emits an {Upgraded} event.
                 *
                 * @custom:oz-upgrades-unsafe-allow-reachable delegatecall
                 */
                function upgradeToAndCall(address newImplementation, bytes memory data) public payable virtual onlyProxy {
                    _authorizeUpgrade(newImplementation);
                    _upgradeToAndCallUUPS(newImplementation, data, true);
                }
                /**
                 * @dev Function that should revert when `msg.sender` is not authorized to upgrade the contract. Called by
                 * {upgradeTo} and {upgradeToAndCall}.
                 *
                 * Normally, this function will use an xref:access.adoc[access control] modifier such as {Ownable-onlyOwner}.
                 *
                 * ```solidity
                 * function _authorizeUpgrade(address) internal override onlyOwner {}
                 * ```
                 */
                function _authorizeUpgrade(address newImplementation) 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[50] private __gap;
            }
            // 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 {Initializable} from "../../proxy/utils/Initializable.sol";
            /**
             * @dev Implementation of the {IERC20} interface.
             *
             * This implementation is agnostic to the way tokens are created. This means
             * that a supply mechanism has to be added in a derived contract using {_mint}.
             * 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 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 (last updated v4.9.4) (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.
             *
             * ==== Security Considerations
             *
             * There are two important considerations concerning the use of `permit`. The first is that a valid permit signature
             * expresses an allowance, and it should not be assumed to convey additional meaning. In particular, it should not be
             * considered as an intention to spend the allowance in any specific way. The second is that because permits have
             * built-in replay protection and can be submitted by anyone, they can be frontrun. A protocol that uses permits should
             * take this into consideration and allow a `permit` call to fail. Combining these two aspects, a pattern that may be
             * generally recommended is:
             *
             * ```solidity
             * function doThingWithPermit(..., uint256 value, uint256 deadline, uint8 v, bytes32 r, bytes32 s) public {
             *     try token.permit(msg.sender, address(this), value, deadline, v, r, s) {} catch {}
             *     doThing(..., value);
             * }
             *
             * function doThing(..., uint256 value) public {
             *     token.safeTransferFrom(msg.sender, address(this), value);
             *     ...
             * }
             * ```
             *
             * Observe that: 1) `msg.sender` is used as the owner, leaving no ambiguity as to the signer intent, and 2) the use of
             * `try/catch` allows the permit to fail and makes the code tolerant to frontrunning. (See also
             * {SafeERC20-safeTransferFrom}).
             *
             * Additionally, note that smart contract wallets (such as Argent or Safe) are not able to produce permit signatures, so
             * contracts should have entry points that don't rely on permit.
             */
            interface IERC20PermitUpgradeable {
                /**
                 * @dev Sets `value` as the allowance of `spender` over ``owner``'s tokens,
                 * given ``owner``'s signed approval.
                 *
                 * IMPORTANT: The same issues {IERC20-approve} has related to transaction
                 * ordering also apply here.
                 *
                 * Emits an {Approval} event.
                 *
                 * Requirements:
                 *
                 * - `spender` cannot be the zero address.
                 * - `deadline` must be a timestamp in the future.
                 * - `v`, `r` and `s` must be a valid `secp256k1` signature from `owner`
                 * over the EIP712-formatted function arguments.
                 * - the signature must use ``owner``'s current nonce (see {nonces}).
                 *
                 * For more information on the signature format, see the
                 * https://eips.ethereum.org/EIPS/eip-2612#specification[relevant EIP
                 * section].
                 *
                 * CAUTION: See Security Considerations above.
                 */
                function permit(
                    address owner,
                    address spender,
                    uint256 value,
                    uint256 deadline,
                    uint8 v,
                    bytes32 r,
                    bytes32 s
                ) external;
                /**
                 * @dev Returns the current nonce for `owner`. This value must be
                 * included whenever a signature is generated for {permit}.
                 *
                 * Every successful call to {permit} increases ``owner``'s nonce by one. This
                 * prevents a signature from being used multiple times.
                 */
                function nonces(address owner) external view returns (uint256);
                /**
                 * @dev Returns the domain separator used in the encoding of the signature for {permit}, as defined by {EIP712}.
                 */
                // solhint-disable-next-line func-name-mixedcase
                function DOMAIN_SEPARATOR() external view returns (bytes32);
            }
            // SPDX-License-Identifier: MIT
            // OpenZeppelin Contracts (last updated 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 (last updated v4.9.3) (token/ERC20/utils/SafeERC20.sol)
            pragma solidity ^0.8.0;
            import "../IERC20Upgradeable.sol";
            import "../extensions/IERC20PermitUpgradeable.sol";
            import "../../../utils/AddressUpgradeable.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 SafeERC20Upgradeable {
                using AddressUpgradeable 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(IERC20Upgradeable 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(IERC20Upgradeable 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(IERC20Upgradeable 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(IERC20Upgradeable 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(IERC20Upgradeable 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(IERC20Upgradeable 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(
                    IERC20PermitUpgradeable 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(IERC20Upgradeable 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(IERC20Upgradeable 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))) && AddressUpgradeable.isContract(address(token));
                }
            }
            // 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: MIT
            // OpenZeppelin Contracts (last updated v4.9.4) (utils/Context.sol)
            pragma solidity ^0.8.0;
            import {Initializable} from "../proxy/utils/Initializable.sol";
            /**
             * @dev Provides information about the current execution context, including the
             * sender of the transaction and its data. While these are generally available
             * via msg.sender and msg.data, they should not be accessed in such a direct
             * manner, since when dealing with meta-transactions the account sending and
             * paying for execution may not be the actual sender (as far as an application
             * is concerned).
             *
             * This contract is only required for intermediate, library-like contracts.
             */
            abstract contract ContextUpgradeable is Initializable {
                function __Context_init() internal onlyInitializing {
                }
                function __Context_init_unchained() internal onlyInitializing {
                }
                function _msgSender() internal view virtual returns (address) {
                    return msg.sender;
                }
                function _msgData() internal view virtual returns (bytes calldata) {
                    return msg.data;
                }
                function _contextSuffixLength() internal view virtual returns (uint256) {
                    return 0;
                }
                /**
                 * @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/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 StorageSlotUpgradeable {
                struct AddressSlot {
                    address value;
                }
                struct BooleanSlot {
                    bool value;
                }
                struct Bytes32Slot {
                    bytes32 value;
                }
                struct Uint256Slot {
                    uint256 value;
                }
                struct StringSlot {
                    string value;
                }
                struct BytesSlot {
                    bytes value;
                }
                /**
                 * @dev Returns an `AddressSlot` with member `value` located at `slot`.
                 */
                function getAddressSlot(bytes32 slot) internal pure returns (AddressSlot storage r) {
                    /// @solidity memory-safe-assembly
                    assembly {
                        r.slot := slot
                    }
                }
                /**
                 * @dev Returns an `BooleanSlot` with member `value` located at `slot`.
                 */
                function getBooleanSlot(bytes32 slot) internal pure returns (BooleanSlot storage r) {
                    /// @solidity memory-safe-assembly
                    assembly {
                        r.slot := slot
                    }
                }
                /**
                 * @dev Returns an `Bytes32Slot` with member `value` located at `slot`.
                 */
                function getBytes32Slot(bytes32 slot) internal pure returns (Bytes32Slot storage r) {
                    /// @solidity memory-safe-assembly
                    assembly {
                        r.slot := slot
                    }
                }
                /**
                 * @dev Returns an `Uint256Slot` with member `value` located at `slot`.
                 */
                function getUint256Slot(bytes32 slot) internal pure returns (Uint256Slot storage r) {
                    /// @solidity memory-safe-assembly
                    assembly {
                        r.slot := slot
                    }
                }
                /**
                 * @dev Returns an `StringSlot` with member `value` located at `slot`.
                 */
                function getStringSlot(bytes32 slot) internal pure returns (StringSlot storage r) {
                    /// @solidity memory-safe-assembly
                    assembly {
                        r.slot := slot
                    }
                }
                /**
                 * @dev Returns an `StringSlot` representation of the string storage pointer `store`.
                 */
                function getStringSlot(string storage store) internal pure returns (StringSlot storage r) {
                    /// @solidity memory-safe-assembly
                    assembly {
                        r.slot := store.slot
                    }
                }
                /**
                 * @dev Returns an `BytesSlot` with member `value` located at `slot`.
                 */
                function getBytesSlot(bytes32 slot) internal pure returns (BytesSlot storage r) {
                    /// @solidity memory-safe-assembly
                    assembly {
                        r.slot := slot
                    }
                }
                /**
                 * @dev Returns an `BytesSlot` representation of the bytes storage pointer `store`.
                 */
                function getBytesSlot(bytes storage store) internal pure returns (BytesSlot storage r) {
                    /// @solidity memory-safe-assembly
                    assembly {
                        r.slot := store.slot
                    }
                }
            }
            // SPDX-License-Identifier: GPL-3.0
            pragma solidity 0.8.15;
            import "@openzeppelin/contracts-upgradeable/proxy/utils/UUPSUpgradeable.sol";
            import "@openzeppelin/contracts-upgradeable/access/OwnableUpgradeable.sol";
            import "../tokens/base/ERC677Upgradeable.sol";
            /**
             * @title Staking Rewards Pool
             * @notice Handles staking and reward distribution for a single asset
             * @dev Rewards can be positive or negative (user balances can increase and decrease)
             */
            abstract contract StakingRewardsPool is ERC677Upgradeable, UUPSUpgradeable, OwnableUpgradeable {
                // used to prevent vault inflation attack
                uint256 private constant DEAD_SHARES = 10 ** 3;
                // address of staking asset token
                IERC20Upgradeable public token;
                // mapping of staker address to their shares balance
                mapping(address => uint256) private shares;
                // total number of shares minted
                uint256 public totalShares;
                /**
                 * @notice Initializes the contract
                 * @param _token address of staking asset token
                 * @param _liquidTokenName name of liquid staking token
                 * @param _liquidTokenSymbol symbol of liquid staking token
                 */
                function __StakingRewardsPool_init(
                    address _token,
                    string memory _liquidTokenName,
                    string memory _liquidTokenSymbol
                ) public onlyInitializing {
                    __ERC677_init(_liquidTokenName, _liquidTokenSymbol, 0);
                    __UUPSUpgradeable_init();
                    __Ownable_init();
                    token = IERC20Upgradeable(_token);
                }
                /**
                 * @notice Returns the total supply of liquid staking tokens
                 * @return total supply
                 */
                function totalSupply() public view override returns (uint256) {
                    return _totalStaked();
                }
                /**
                 * @notice Returns an account's LST balance
                 * @param _account account address
                 * @return account's balance
                 */
                function balanceOf(address _account) public view override returns (uint256) {
                    uint256 balance = getStakeByShares(shares[_account]);
                    if (balance < 100) {
                        return 0;
                    } else {
                        return balance;
                    }
                }
                /**
                 * @notice Returns an account's share balance
                 * @param _account account address
                 * @return account's share balance
                 */
                function sharesOf(address _account) public view returns (uint256) {
                    return shares[_account];
                }
                /**
                 * @notice Returns the amount of shares that corresponds to an LST amount
                 * @param _amount staked amount
                 * @return amount of shares
                 */
                function getSharesByStake(uint256 _amount) public view returns (uint256) {
                    uint256 totalStaked = _totalStaked();
                    if (totalStaked == 0) {
                        return _amount;
                    } else {
                        return (_amount * totalShares) / totalStaked;
                    }
                }
                /**
                 * @notice Returns the amount of LST that corresponds to an amount of shares
                 * @param _amount shares amount
                 * @return amount of LST
                 */
                function getStakeByShares(uint256 _amount) public view returns (uint256) {
                    if (totalShares == 0) {
                        return _amount;
                    } else {
                        return (_amount * _totalStaked()) / totalShares;
                    }
                }
                /**
                 * @notice Transfers shares from sender to another account
                 * @param _recipient account to transfer to
                 * @param _sharesAmount amount of shares to transfer
                 */
                function transferShares(address _recipient, uint256 _sharesAmount) external returns (bool) {
                    _transferShares(msg.sender, _recipient, _sharesAmount);
                    return true;
                }
                /**
                 * @notice Transfers shares from one account to another
                 * @param _sender account to transfer from
                 * @param _recipient account to transfer to
                 * @param _sharesAmount amount of shares to transfer
                 */
                function transferSharesFrom(
                    address _sender,
                    address _recipient,
                    uint256 _sharesAmount
                ) external returns (bool) {
                    uint256 tokensAmount = getStakeByShares(_sharesAmount);
                    _spendAllowance(_sender, msg.sender, tokensAmount);
                    _transferShares(_sender, _recipient, _sharesAmount);
                    return true;
                }
                /**
                 * @notice Returns the total amount of asset tokens staked in the pool
                 * @return total staked amount
                 */
                function _totalStaked() internal view virtual returns (uint256);
                /**
                 * @notice Transfers an LST balance from one account to another
                 * @param _sender account to transfer from
                 * @param _recipient account to transfer to
                 * @param _amount amount to transfer
                 */
                function _transfer(address _sender, address _recipient, uint256 _amount) internal override {
                    uint256 sharesToTransfer = getSharesByStake(_amount);
                    require(_sender != address(0), "Transfer from the zero address");
                    require(_recipient != address(0), "Transfer to the zero address");
                    require(shares[_sender] >= sharesToTransfer, "Transfer amount exceeds balance");
                    shares[_sender] -= sharesToTransfer;
                    shares[_recipient] += sharesToTransfer;
                    emit Transfer(_sender, _recipient, _amount);
                }
                /**
                 * @notice Transfers shares from one account to another
                 * @param _sender account to transfer from
                 * @param _recipient account to transfer to
                 * @param _sharesAmount amount of shares to transfer
                 */
                function _transferShares(address _sender, address _recipient, uint256 _sharesAmount) internal {
                    require(_sender != address(0), "Transfer from the zero address");
                    require(_recipient != address(0), "Transfer to the zero address");
                    require(shares[_sender] >= _sharesAmount, "Transfer amount exceeds balance");
                    shares[_sender] -= _sharesAmount;
                    shares[_recipient] += _sharesAmount;
                    emit Transfer(_sender, _recipient, getStakeByShares(_sharesAmount));
                }
                /**
                 * @notice Mints new shares to an account
                 * @dev takes an LST amount and calculates the amount of shares it corresponds to
                 * @param _recipient account to mint shares for
                 * @param _amount stake amount
                 */
                function _mint(address _recipient, uint256 _amount) internal override {
                    uint256 sharesToMint = getSharesByStake(_amount);
                    _mintShares(_recipient, sharesToMint);
                    emit Transfer(address(0), _recipient, _amount);
                }
                /**
                 * @notice Mints new shares to an account
                 * @param _recipient account to mint shares for
                 * @param _amount shares amount
                 */
                function _mintShares(address _recipient, uint256 _amount) internal {
                    require(_recipient != address(0), "Mint to the zero address");
                    if (totalShares == 0) {
                        shares[address(0)] = DEAD_SHARES;
                        totalShares = DEAD_SHARES;
                        _amount -= DEAD_SHARES;
                    }
                    totalShares += _amount;
                    shares[_recipient] += _amount;
                }
                /**
                 * @notice Burns shares belonging to an account
                 * @dev takes an LST amount and calculates the amount of shares it corresponds to
                 * @param _account account to burn shares for
                 * @param _amount LST amount
                 */
                function _burn(address _account, uint256 _amount) internal override {
                    uint256 sharesToBurn = getSharesByStake(_amount);
                    require(_account != address(0), "Burn from the zero address");
                    require(shares[_account] >= sharesToBurn, "Burn amount exceeds balance");
                    totalShares -= sharesToBurn;
                    shares[_account] -= sharesToBurn;
                    emit Transfer(_account, address(0), _amount);
                }
                /**
                 * @dev Checks authorization for contract upgrades
                 */
                function _authorizeUpgrade(address) internal override onlyOwner {}
            }
            // SPDX-License-Identifier: GPL-3.0
            pragma solidity 0.8.15;
            interface IERC677Receiver {
                function onTokenTransfer(address _sender, uint256 _value, bytes calldata _data) external;
            }
            // SPDX-License-Identifier: GPL-3.0
            pragma solidity 0.8.15;
            interface IStrategy {
                function deposit(uint256 _amount, bytes calldata _data) external;
                function withdraw(uint256 _amount, bytes calldata _data) external;
                function updateDeposits(
                    bytes calldata _data
                ) external returns (int256 depositChange, address[] memory receivers, uint256[] memory amounts);
                function getTotalDeposits() external view returns (uint256);
                function getMaxDeposits() external view returns (uint256);
                function getMinDeposits() external view returns (uint256);
                function canDeposit() external view returns (uint256);
                function canWithdraw() external view returns (uint256);
                function getDepositChange() external view returns (int256);
            }
            // SPDX-License-Identifier: GPL-3.0
            pragma solidity 0.8.15;
            import "@openzeppelin/contracts-upgradeable/token/ERC20/utils/SafeERC20Upgradeable.sol";
            import "./base/StakingRewardsPool.sol";
            import "./interfaces/IStrategy.sol";
            /**
             * @title Staking Pool
             * @notice Allows users to stake asset tokens and receive liquid staking tokens 1:1, then deposits staked
             * asset tokens into strategy contracts
             */
            contract StakingPool is StakingRewardsPool {
                using SafeERC20Upgradeable for IERC20Upgradeable;
                struct Fee {
                    // address to receive fee
                    address receiver;
                    // value of fee in basis points
                    uint256 basisPoints;
                }
                // list of all strategies controlled by pool
                address[] private strategies;
                // total number of tokens staked in the pool
                uint256 public totalStaked;
                // max number of tokens that can sit in the pool outside of a strategy
                uint256 public unusedDepositLimit;
                // list of fees that are paid on rewards
                Fee[] private fees;
                // address of priority pool
                address public priorityPool;
                // address of rebase controller
                address public rebaseController;
                uint16 private poolIndex; // deprecated
                event UpdateStrategyRewards(
                    address indexed account,
                    uint256 totalStaked,
                    int rewardsAmount,
                    uint256 totalFees
                );
                event Burn(address indexed account, uint256 amount);
                event DonateTokens(address indexed sender, uint256 amount);
                error SenderNotAuthorized();
                error InvalidDeposit();
                error NothingStaked();
                /// @custom:oz-upgrades-unsafe-allow constructor
                constructor() {
                    _disableInitializers();
                }
                /**
                 * @notice Initializes the contract
                 * @param _token address of asset token
                 * @param _liquidTokenName name of liquid staking token
                 * @param _liquidTokenSymbol symbol of liquid staking token
                 * @param _fees list of fees that are paid on rewards
                 * @param _unusedDepositLimit max number of tokens that can sit in the pool outside of a strategy
                 */
                function initialize(
                    address _token,
                    string memory _liquidTokenName,
                    string memory _liquidTokenSymbol,
                    Fee[] memory _fees,
                    uint256 _unusedDepositLimit
                ) public initializer {
                    __StakingRewardsPool_init(_token, _liquidTokenName, _liquidTokenSymbol);
                    for (uint256 i = 0; i < _fees.length; i++) {
                        fees.push(_fees[i]);
                    }
                    require(_totalFeesBasisPoints() <= 4000, "Total fees must be <= 40%");
                    unusedDepositLimit = _unusedDepositLimit;
                }
                /**
                 * @notice Reverts if sender is not priority pool
                 */
                modifier onlyPriorityPool() {
                    if (msg.sender != priorityPool) revert SenderNotAuthorized();
                    _;
                }
                /**
                 * @notice Returns a list of all strategies
                 * @return list of strategies
                 */
                function getStrategies() external view returns (address[] memory) {
                    return strategies;
                }
                /**
                 * @notice Returns a list of all fees
                 * @return list of fees
                 */
                function getFees() external view returns (Fee[] memory) {
                    return fees;
                }
                /**
                 * @notice Stakes asset tokens and mints liquid staking tokens
                 * @dev will deposit unused deposits and new deposits into strategies
                 * @param _account account to stake for
                 * @param _amount amount to stake
                 * @param _data list of deposit data passed to strategies
                 **/
                function deposit(
                    address _account,
                    uint256 _amount,
                    bytes[] calldata _data
                ) external onlyPriorityPool {
                    require(strategies.length > 0, "Must be > 0 strategies to stake");
                    uint256 startingBalance = token.balanceOf(address(this));
                    if (_amount > 0) {
                        token.safeTransferFrom(msg.sender, address(this), _amount);
                        _depositLiquidity(_data);
                        _mint(_account, _amount);
                        totalStaked += _amount;
                    } else {
                        _depositLiquidity(_data);
                    }
                    uint256 endingBalance = token.balanceOf(address(this));
                    if (endingBalance > startingBalance && endingBalance > unusedDepositLimit)
                        revert InvalidDeposit();
                }
                /**
                 * @notice Withdraws asset tokens and burns liquid staking tokens
                 * @dev will withdraw from strategies if not enough liquidity
                 * @param _account account to withdraw for
                 * @param _receiver address to receive withdrawal
                 * @param _amount amount to withdraw
                 * @param _data list of withdrawal data passed to strategies
                 **/
                function withdraw(
                    address _account,
                    address _receiver,
                    uint256 _amount,
                    bytes[] calldata _data
                ) external onlyPriorityPool {
                    uint256 toWithdraw = _amount;
                    if (_amount == type(uint256).max) {
                        toWithdraw = balanceOf(_account);
                    }
                    uint256 balance = token.balanceOf(address(this));
                    if (toWithdraw > balance) {
                        _withdrawLiquidity(toWithdraw - balance, _data);
                    }
                    require(
                        token.balanceOf(address(this)) >= toWithdraw,
                        "Not enough liquidity available to withdraw"
                    );
                    _burn(_account, toWithdraw);
                    totalStaked -= toWithdraw;
                    token.safeTransfer(_receiver, toWithdraw);
                }
                /**
                 * @notice Manually deposits asset tokens into a specific strategy
                 * @param _index index of strategy
                 * @param _amount amount to deposit
                 * @param _data deposit data passed to strategy
                 **/
                function strategyDeposit(
                    uint256 _index,
                    uint256 _amount,
                    bytes calldata _data
                ) external onlyOwner {
                    require(_index < strategies.length, "Strategy does not exist");
                    IStrategy(strategies[_index]).deposit(_amount, _data);
                }
                /**
                 * @notice Manually withdraws asset tokens from a strategy
                 * @param _index index of strategy
                 * @param _amount amount to withdraw
                 * @param _data withdrawal data passed to strategy
                 **/
                function strategyWithdraw(
                    uint256 _index,
                    uint256 _amount,
                    bytes calldata _data
                ) external onlyOwner {
                    require(_index < strategies.length, "Strategy does not exist");
                    IStrategy(strategies[_index]).withdraw(_amount, _data);
                }
                /**
                 * @notice Returns the maximum amount of tokens that the pool can hold
                 * @return maximum deposit limit
                 **/
                function getMaxDeposits() public view returns (uint256) {
                    uint256 max;
                    for (uint256 i = 0; i < strategies.length; i++) {
                        uint strategyMax = IStrategy(strategies[i]).getMaxDeposits();
                        if (strategyMax >= type(uint256).max - max) {
                            return type(uint256).max;
                        }
                        max += strategyMax;
                    }
                    return max;
                }
                /**
                 * @notice Returns the minimum amount of tokens that must remain in the pool
                 * @return minimum deposit limit
                 */
                function getMinDeposits() public view returns (uint256) {
                    uint256 min;
                    for (uint256 i = 0; i < strategies.length; i++) {
                        IStrategy strategy = IStrategy(strategies[i]);
                        min += strategy.getMinDeposits();
                    }
                    return min;
                }
                /**
                 * @notice Returns the amount of unused asset tokens sitting in this pool outside a strategy
                 * @dev these tokens earn no yield and will be deposited ASAP on the next call to _depositLiquidity
                 * @return amount of unused tokens
                 */
                function getUnusedDeposits() external view returns (uint256) {
                    return token.balanceOf(address(this));
                }
                /**
                 * @notice Returns the sum of available deposit room across all strategies
                 * @dev does not account for unused deposits sitting in pool
                 * @return strategy deposit room
                 */
                function getStrategyDepositRoom() external view returns (uint256) {
                    uint256 depositRoom;
                    for (uint256 i = 0; i < strategies.length; ++i) {
                        uint strategyDepositRoom = IStrategy(strategies[i]).canDeposit();
                        if (strategyDepositRoom >= type(uint256).max - depositRoom) {
                            return type(uint256).max;
                        }
                        depositRoom += strategyDepositRoom;
                    }
                    return depositRoom;
                }
                /**
                 * @notice Returns the total available deposit room for this pool
                 * @dev accounts for unused deposits sitting in pool
                 * @return available deposit room
                 */
                function canDeposit() external view returns (uint256) {
                    uint256 max = getMaxDeposits();
                    if (max <= totalStaked) {
                        return 0;
                    } else {
                        return max - totalStaked;
                    }
                }
                /**
                 * @notice Returns the total available withdrawal room for this pool
                 * @return available withdrawal room
                 */
                function canWithdraw() external view returns (uint256) {
                    uint256 min = getMinDeposits();
                    if (min >= totalStaked) {
                        return 0;
                    } else {
                        return totalStaked - min;
                    }
                }
                /**
                 * @notice Adds a new strategy
                 * @param _strategy address of strategy
                 **/
                function addStrategy(address _strategy) external onlyOwner {
                    require(!_strategyExists(_strategy), "Strategy already exists");
                    token.safeApprove(_strategy, type(uint256).max);
                    strategies.push(_strategy);
                }
                /**
                 * @notice Removes an existing strategy
                 * @param _index index of strategy
                 * @param _strategyUpdateData update data passed to strategy
                 * @param _strategyWithdrawalData withdrawal data passed to strategy
                 **/
                function removeStrategy(
                    uint256 _index,
                    bytes memory _strategyUpdateData,
                    bytes calldata _strategyWithdrawalData
                ) external onlyOwner {
                    require(_index < strategies.length, "Strategy does not exist");
                    uint256[] memory idxs = new uint256[](1);
                    idxs[0] = _index;
                    _updateStrategyRewards(idxs, _strategyUpdateData);
                    IStrategy strategy = IStrategy(strategies[_index]);
                    uint256 totalStrategyDeposits = strategy.getTotalDeposits();
                    if (totalStrategyDeposits > 0) {
                        strategy.withdraw(totalStrategyDeposits, _strategyWithdrawalData);
                    }
                    for (uint256 i = _index; i < strategies.length - 1; i++) {
                        strategies[i] = strategies[i + 1];
                    }
                    strategies.pop();
                    token.safeApprove(address(strategy), 0);
                }
                /**
                 * @notice Reorders strategies
                 * @param _newOrder list containing strategy indexes in a new order
                 **/
                function reorderStrategies(uint256[] calldata _newOrder) external onlyOwner {
                    require(_newOrder.length == strategies.length, "newOrder.length must = strategies.length");
                    address[] memory strategyAddresses = new address[](strategies.length);
                    for (uint256 i = 0; i < strategies.length; i++) {
                        strategyAddresses[i] = strategies[i];
                    }
                    for (uint256 i = 0; i < strategies.length; i++) {
                        require(strategyAddresses[_newOrder[i]] != address(0), "all indices must be valid");
                        strategies[i] = strategyAddresses[_newOrder[i]];
                        strategyAddresses[_newOrder[i]] = address(0);
                    }
                }
                /*
                 * @notice Adds a new fee
                 * @param _receiver receiver of fee
                 * @param _feeBasisPoints fee in basis points
                 **/
                function addFee(address _receiver, uint256 _feeBasisPoints) external onlyOwner {
                    uint256[] memory strategyIdxs = new uint256[](strategies.length);
                    for (uint256 i = 0; i < strategyIdxs.length; ++i) {
                        strategyIdxs[i] = i;
                    }
                    _updateStrategyRewards(strategyIdxs, "");
                    fees.push(Fee(_receiver, _feeBasisPoints));
                    require(_totalFeesBasisPoints() <= 4000, "Total fees must be <= 40%");
                }
                /**
                 * @notice Updates an existing fee
                 * @param _index index of fee
                 * @param _receiver receiver of fee
                 * @param _feeBasisPoints fee in basis points
                 **/
                function updateFee(
                    uint256 _index,
                    address _receiver,
                    uint256 _feeBasisPoints
                ) external onlyOwner {
                    require(_index < fees.length, "Fee does not exist");
                    uint256[] memory strategyIdxs = new uint256[](strategies.length);
                    for (uint256 i = 0; i < strategyIdxs.length; ++i) {
                        strategyIdxs[i] = i;
                    }
                    _updateStrategyRewards(strategyIdxs, "");
                    if (_feeBasisPoints == 0) {
                        fees[_index] = fees[fees.length - 1];
                        fees.pop();
                    } else {
                        fees[_index].receiver = _receiver;
                        fees[_index].basisPoints = _feeBasisPoints;
                    }
                    require(_totalFeesBasisPoints() <= 4000, "Total fees must be <= 40%");
                }
                /**
                 * @notice Returns the amount of rewards earned since the last call to updateStrategyRewards
                 * @param _strategyIdxs indexes of strategies to sum rewards for
                 * @return total rewards
                 **/
                function getStrategyRewards(uint256[] calldata _strategyIdxs) external view returns (int256) {
                    int256 totalRewards;
                    for (uint256 i = 0; i < _strategyIdxs.length; i++) {
                        IStrategy strategy = IStrategy(strategies[_strategyIdxs[i]]);
                        totalRewards += strategy.getDepositChange();
                    }
                    return totalRewards;
                }
                /**
                 * @notice Distributes rewards/fees based on balance changes in strategies since the last update
                 * @param _strategyIdxs indexes of strategies to update rewards for
                 * @param _data update data passed to each strategy
                 **/
                function updateStrategyRewards(uint256[] memory _strategyIdxs, bytes memory _data) external {
                    if (msg.sender != rebaseController && !_strategyExists(msg.sender))
                        revert SenderNotAuthorized();
                    _updateStrategyRewards(_strategyIdxs, _data);
                }
                /**
                 * @notice Burns the senders liquid staking tokens, effectively donating their underlying stake to the pool
                 * @param _amount amount to burn
                 **/
                function burn(uint256 _amount) external {
                    _burn(msg.sender, _amount);
                    emit Burn(msg.sender, _amount);
                }
                /**
                 * @notice Deposits asset tokens into the pool without minting liquid staking tokens,
                 * effectively donating them to the pool
                 * @param _amount amount to deposit
                 **/
                function donateTokens(uint256 _amount) external {
                    if (totalStaked == 0) revert NothingStaked();
                    token.safeTransferFrom(msg.sender, address(this), _amount);
                    totalStaked += _amount;
                    emit DonateTokens(msg.sender, _amount);
                }
                /**
                 * @notice Sets the maximum amount of unused deposits that can sit in the pool
                 * @param _unusedDepositLimit maximum amount of unused deposits
                 **/
                function setUnusedDepositLimit(uint256 _unusedDepositLimit) external onlyOwner {
                    unusedDepositLimit = _unusedDepositLimit;
                }
                /**
                 * @notice Sets the priority pool
                 * @param _priorityPool address of priority pool
                 **/
                function setPriorityPool(address _priorityPool) external onlyOwner {
                    priorityPool = _priorityPool;
                }
                /**
                 * @notice Sets the rebase controller
                 * @dev this address has sole authority to update rewards
                 * @param _rebaseController address of rebase controller
                 **/
                function setRebaseController(address _rebaseController) external onlyOwner {
                    rebaseController = _rebaseController;
                }
                /**
                 * @notice Returns the total amount of asset tokens staked in the pool
                 * @return the total staked amount
                 */
                function _totalStaked() internal view override returns (uint256) {
                    return totalStaked;
                }
                /**
                 * @notice Deposits available liquidity into strategies
                 * @dev deposits into strategies in ascending order, only moving to the next once the current is full
                 * @param _data list of deposit data passed to strategies
                 **/
                function _depositLiquidity(bytes[] calldata _data) private {
                    uint256 toDeposit = token.balanceOf(address(this));
                    if (toDeposit > 0) {
                        for (uint256 i = 0; i < strategies.length; i++) {
                            IStrategy strategy = IStrategy(strategies[i]);
                            uint256 strategyCanDeposit = strategy.canDeposit();
                            if (strategyCanDeposit >= toDeposit) {
                                strategy.deposit(toDeposit, _data[i]);
                                break;
                            } else if (strategyCanDeposit > 0) {
                                strategy.deposit(strategyCanDeposit, _data[i]);
                                toDeposit = token.balanceOf(address(this));
                            }
                        }
                    }
                }
                /**
                 * @notice Withdraws liquidity from strategies
                 * @dev withdraws from strategies in descending order only moving to the next once once the current is empty
                 * @param _amount amount to withdraw
                 * @param _data list of withdrawal data passed to strategies
                 **/
                function _withdrawLiquidity(uint256 _amount, bytes[] calldata _data) private {
                    uint256 toWithdraw = _amount;
                    uint256 balance = token.balanceOf(address(this));
                    for (uint256 i = strategies.length; i > 0; i--) {
                        IStrategy strategy = IStrategy(strategies[i - 1]);
                        uint256 strategyCanWithdrawdraw = strategy.canWithdraw();
                        if (strategyCanWithdrawdraw >= toWithdraw) {
                            strategy.withdraw(toWithdraw, _data[i - 1]);
                            break;
                        } else if (strategyCanWithdrawdraw > 0) {
                            strategy.withdraw(strategyCanWithdrawdraw, _data[i - 1]);
                            uint256 withdrawn = token.balanceOf(address(this)) - balance;
                            balance += withdrawn;
                            toWithdraw -= withdrawn;
                        }
                    }
                }
                /**
                 * @notice Distributes rewards/fees based on balance changes in strategies since the last update
                 * @param _strategyIdxs indexes of strategies to update rewards for
                 * @param _data update data passed to each strategy
                 **/
                function _updateStrategyRewards(uint256[] memory _strategyIdxs, bytes memory _data) private {
                    int256 totalRewards;
                    uint256 totalFeeAmounts;
                    uint256 totalFeeCount;
                    address[][] memory receivers = new address[][](strategies.length + 1);
                    uint256[][] memory feeAmounts = new uint256[][](strategies.length + 1);
                    // sum up rewards and fees across strategies
                    for (uint256 i = 0; i < _strategyIdxs.length; ++i) {
                        IStrategy strategy = IStrategy(strategies[_strategyIdxs[i]]);
                        (
                            int256 depositChange,
                            address[] memory strategyReceivers,
                            uint256[] memory strategyFeeAmounts
                        ) = strategy.updateDeposits(_data);
                        totalRewards += depositChange;
                        if (strategyReceivers.length != 0) {
                            receivers[i] = strategyReceivers;
                            feeAmounts[i] = strategyFeeAmounts;
                            totalFeeCount += receivers[i].length;
                            for (uint256 j = 0; j < strategyReceivers.length; ++j) {
                                totalFeeAmounts += strategyFeeAmounts[j];
                            }
                        }
                    }
                    // update totalStaked if there was a net change in deposits
                    if (totalRewards != 0) {
                        totalStaked = uint256(int256(totalStaked) + totalRewards);
                    }
                    // calulate fees if net positive rewards were earned
                    if (totalRewards > 0) {
                        receivers[receivers.length - 1] = new address[](fees.length);
                        feeAmounts[feeAmounts.length - 1] = new uint256[](fees.length);
                        totalFeeCount += fees.length;
                        for (uint256 i = 0; i < fees.length; i++) {
                            receivers[receivers.length - 1][i] = fees[i].receiver;
                            feeAmounts[feeAmounts.length - 1][i] =
                                (uint256(totalRewards) * fees[i].basisPoints) /
                                10000;
                            totalFeeAmounts += feeAmounts[feeAmounts.length - 1][i];
                        }
                    }
                    // safety check
                    if (totalFeeAmounts >= totalStaked) {
                        totalFeeAmounts = 0;
                    }
                    // distribute fees to receivers if there are any
                    if (totalFeeAmounts > 0) {
                        uint256 sharesToMint = (totalFeeAmounts * totalShares) /
                            (totalStaked - totalFeeAmounts);
                        _mintShares(address(this), sharesToMint);
                        uint256 feesPaidCount;
                        for (uint256 i = 0; i < receivers.length; i++) {
                            for (uint256 j = 0; j < receivers[i].length; j++) {
                                if (feesPaidCount == totalFeeCount - 1) {
                                    transferAndCallFrom(
                                        address(this),
                                        receivers[i][j],
                                        balanceOf(address(this)),
                                        "0x"
                                    );
                                } else {
                                    transferAndCallFrom(address(this), receivers[i][j], feeAmounts[i][j], "0x");
                                    feesPaidCount++;
                                }
                            }
                        }
                    }
                    emit UpdateStrategyRewards(msg.sender, totalStaked, totalRewards, totalFeeAmounts);
                }
                /**
                 * @notice Returns the sum of all fees
                 * @return sum of fees in basis points
                 **/
                function _totalFeesBasisPoints() private view returns (uint256) {
                    uint256 totalFees;
                    for (uint i = 0; i < fees.length; i++) {
                        totalFees += fees[i].basisPoints;
                    }
                    return totalFees;
                }
                /**
                 * @notice Returns whether or not a strategy exists
                 * @param _strategy address of strategy
                 * @return true if strategy exists, false otherwise
                 **/
                function _strategyExists(address _strategy) private view returns (bool) {
                    for (uint256 i = 0; i < strategies.length; i++) {
                        if (strategies[i] == _strategy) {
                            return true;
                        }
                    }
                    return false;
                }
            }
            // SPDX-License-Identifier: GPL-3.0
            pragma solidity 0.8.15;
            import "@openzeppelin/contracts-upgradeable/token/ERC20/ERC20Upgradeable.sol";
            import "../../interfaces/IERC677Receiver.sol";
            contract ERC677Upgradeable is ERC20Upgradeable {
                function __ERC677_init(
                    string memory _tokenName,
                    string memory _tokenSymbol,
                    uint256 _totalSupply
                ) public onlyInitializing {
                    __ERC20_init(_tokenName, _tokenSymbol);
                    if (_totalSupply != 0) {
                        _mint(msg.sender, _totalSupply * (10 ** uint256(decimals())));
                    }
                }
                function transferAndCall(
                    address _to,
                    uint256 _value,
                    bytes memory _data
                ) public returns (bool) {
                    super.transfer(_to, _value);
                    if (isContract(_to)) {
                        contractFallback(msg.sender, _to, _value, _data);
                    }
                    return true;
                }
                function transferAndCallFrom(
                    address _sender,
                    address _to,
                    uint256 _value,
                    bytes memory _data
                ) internal returns (bool) {
                    _transfer(_sender, _to, _value);
                    if (isContract(_to)) {
                        contractFallback(_sender, _to, _value, _data);
                    }
                    return true;
                }
                function contractFallback(
                    address _sender,
                    address _to,
                    uint256 _value,
                    bytes memory _data
                ) internal {
                    IERC677Receiver receiver = IERC677Receiver(_to);
                    receiver.onTokenTransfer(_sender, _value, _data);
                }
                function isContract(address _addr) internal view returns (bool hasCode) {
                    uint256 length;
                    assembly {
                        length := extcodesize(_addr)
                    }
                    return length > 0;
                }
            }