ETH Price: $2,128.50 (+2.22%)

Transaction Decoder

Block:
19692301 at Apr-19-2024 09:56:47 PM +UTC
Transaction Fee:
0.001786426211519053 ETH $3.80
Gas Used:
253,879 Gas / 7.036526107 Gwei

Emitted Events:

173 EntryPoint.BeforeExecution( )
174 TransparentUpgradeableProxy.0xddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef( 0xddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef, 0x0000000000000000000000002ddb11443bd9ceb92d4951a05f55eb7096eb53d3, 0x0000000000000000000000009ef22cc19ea80606aba40e1e5f4e5553473da6bc, 0000000000000000000000000000000000000000000000b4a3939e53bee28000 )
175 TransparentUpgradeableProxy.0xddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef( 0xddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef, 0x0000000000000000000000002ddb11443bd9ceb92d4951a05f55eb7096eb53d3, 0x000000000000000000000000b9577e83a6d9a6de35047aa066e3758221fe0da2, 000000000000000000000000000000000000000000000000e86139660c898000 )
176 TransparentUpgradeableProxy.0x4ec90e965519d92681267467f775ada5bd214aa92c0dc93d90a5e880ce9ed026( 0x4ec90e965519d92681267467f775ada5bd214aa92c0dc93d90a5e880ce9ed026, 0000000000000000000000000000000000000000000000000000000000000009, 0000000000000000000000009ef22cc19ea80606aba40e1e5f4e5553473da6bc, 0000000000000000000000000000000000000000000000b58bf4d7b9cb6c0000 )
177 EntryPoint.UserOperationEvent( userOpHash=751A092A09294E2E75B53FA338839E4027EAAC326D1E1CA72497B06AB72C7477, sender=0x9ef22cc19ea80606aba40e1e5f4e5553473da6bc, paymaster=VerifyingPaymaster, nonce=22, success=True, actualGasCost=1822854307174992, actualGasUsed=259056 )

Account State Difference:

  Address   Before After State Difference Code
0x2DDB1144...096EB53d3
0x47bdBD8f...Bb9305f18 1.118309445167639695 Eth1.120714445167639695 Eth0.002405
0x5FF137D4...a026d2789
(Entry Point 0.6.0)
47.03014310792754355 Eth47.028320253620368558 Eth0.001822854307174992
0x8E3A5942...BF94672f4
(beaverbuild)
17.287223857793324013 Eth17.287224111672324013 Eth0.000000253879
0x9eF22cc1...3473Da6BC
0.084970015296247949 Eth0.082565015296247949 Eth0.002405
0xFd72Ae8F...fd99Eb0AC
(Stackup: Bundler 3)
2.126488078959514708 Eth
Nonce: 3341
2.126524507055170647 Eth
Nonce: 3342
0.000036428095655939

Execution Trace

EntryPoint.handleOps( ops=, beneficiary=0xFd72Ae8Ff5CC18849D83f13A252A0D8fd99Eb0AC )
  • 0x9ef22cc19ea80606aba40e1e5f4e5553473da6bc.3a871cdd( )
    • 0x8253291a17d3beb95fadab2751d52b324d22ef2d.3a871cdd( )
      • Null: 0x000...001.751a092a( )
      • Null: 0x000...001.9aa049e5( )
      • VerifyingPaymaster.validatePaymasterUserOp( userOp=[{name:sender, type:address, order:1, indexed:false, value:0x9eF22cc19eA80606ABA40e1e5F4E5553473Da6BC, valueString:0x9eF22cc19eA80606ABA40e1e5F4E5553473Da6BC}, {name:nonce, type:uint256, order:2, indexed:false, value:22, valueString:22}, {name:initCode, type:bytes, order:3, indexed:false, value:0x, valueString:0x}, {name:callData, type:bytes, order:4, indexed:false, value:0x940D3C600000000000000000000000008AE01FCF7C655655FF2C6EF907B8B4718AB4E17C00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000080000000000000000000000000000000000000000000000000000000000000000100000000000000000000000000000000000000000000000000000000000003A48D80FF0A0000000000000000000000000000000000000000000000000000000000000020000000000000000000000000000000000000000000000000000000000000034E0047BDBD8FD3B73C47C36C8B218347FB3BB9305F1800000000000000000000000000000000000000000000000000088B55B2AF50000000000000000000000000000000000000000000000000000000000000000000002DDB11443BD9CEB92D4951A05F55EB7096EB53D3000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000002A42E7BA6EF00000000000000000000000000000000000000000000000000000000000000090000000000000000000000009EF22CC19EA80606ABA40E1E5F4E5553473DA6BC0000000000000000000000000000000000000000000000B58BF4D7B9CB6C00000000000000000000000000000000000000000000000000000000000000000080000000000000000000000000000000000000000000000000000000000000001077010425AB518A2EE957C0564871BAB7BD28AF05CFCE68FE7D7414ACA920BCFC9C2694DC9B55344E20FFF8AB74401E5E2305BCEBD35FA47D16ED59F8FB5A7AC964AE296223B0E6C7B4A0E025B0C953A9AB44AC159CC1BC903FE95557EC76167D10439E074E9CF5A996F71F9DC02047D3DFADD22A80078B7720221A29FB6B9145CEE48D9727596FF604675F51C9B6BF31C26AFA3140F1A17982BD7026F10AACD964043AEED019A7FF10B91A9EF10EBD93665BDB07D9C0CC4914B259C00F629EF84572AFC5BA4FCC9E6D084A7F7B40CF912B53C574C6C98A72240A12769459472127AB5338C08E1F50FFA811131DA52C08E9F11152148B4830786AA3EDF927322DB2A3D05DD813CEBE14EFA66EF09C634A2ADA3BDAA1F5026E7F1BEACE2D458B7A1FD451A3D82AEFA3EFF76DB2F1B817583E0BFF133609F72CBA5F380B51B13ADA56D7B7D7C1B2175B252715623BF36CBB65AA93612F3E661489EFA1A03708B1192E198AAB0F7BF9159774A3626969B432891EA0EE16CE2BE922E982164F38F255F9AB054459886200656E5AFE20BC87C9C0D5BC34C996D75EBAA9028D94189CC21509A1C85BCE8CC5D63C72FB79F89D438CE5F8172349DD52CA4DB2F35E4C68FAD7FBAB98199AE8B823187350647C74AA8945A5AE0772F306A996672DBE069C7120512B815D45DBD9C84E486F616A2AAABA7B8A86FBD73FA5CD0586D415AD737B00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000, valueString:0x940D3C600000000000000000000000008AE01FCF7C655655FF2C6EF907B8B4718AB4E17C00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000080000000000000000000000000000000000000000000000000000000000000000100000000000000000000000000000000000000000000000000000000000003A48D80FF0A0000000000000000000000000000000000000000000000000000000000000020000000000000000000000000000000000000000000000000000000000000034E0047BDBD8FD3B73C47C36C8B218347FB3BB9305F1800000000000000000000000000000000000000000000000000088B55B2AF50000000000000000000000000000000000000000000000000000000000000000000002DDB11443BD9CEB92D4951A05F55EB7096EB53D3000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000002A42E7BA6EF00000000000000000000000000000000000000000000000000000000000000090000000000000000000000009EF22CC19EA80606ABA40E1E5F4E5553473DA6BC0000000000000000000000000000000000000000000000B58BF4D7B9CB6C00000000000000000000000000000000000000000000000000000000000000000080000000000000000000000000000000000000000000000000000000000000001077010425AB518A2EE957C0564871BAB7BD28AF05CFCE68FE7D7414ACA920BCFC9C2694DC9B55344E20FFF8AB74401E5E2305BCEBD35FA47D16ED59F8FB5A7AC964AE296223B0E6C7B4A0E025B0C953A9AB44AC159CC1BC903FE95557EC76167D10439E074E9CF5A996F71F9DC02047D3DFADD22A80078B7720221A29FB6B9145CEE48D9727596FF604675F51C9B6BF31C26AFA3140F1A17982BD7026F10AACD964043AEED019A7FF10B91A9EF10EBD93665BDB07D9C0CC4914B259C00F629EF84572AFC5BA4FCC9E6D084A7F7B40CF912B53C574C6C98A72240A12769459472127AB5338C08E1F50FFA811131DA52C08E9F11152148B4830786AA3EDF927322DB2A3D05DD813CEBE14EFA66EF09C634A2ADA3BDAA1F5026E7F1BEACE2D458B7A1FD451A3D82AEFA3EFF76DB2F1B817583E0BFF133609F72CBA5F380B51B13ADA56D7B7D7C1B2175B252715623BF36CBB65AA93612F3E661489EFA1A03708B1192E198AAB0F7BF9159774A3626969B432891EA0EE16CE2BE922E982164F38F255F9AB054459886200656E5AFE20BC87C9C0D5BC34C996D75EBAA9028D94189CC21509A1C85BCE8CC5D63C72FB79F89D438CE5F8172349DD52CA4DB2F35E4C68FAD7FBAB98199AE8B823187350647C74AA8945A5AE0772F306A996672DBE069C7120512B815D45DBD9C84E486F616A2AAABA7B8A86FBD73FA5CD0586D415AD737B00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000}, {name:callGasLimit, type:uint256, order:5, indexed:false, value:158343, valueString:158343}, {name:verificationGasLimit, type:uint256, order:6, indexed:false, value:91551, valueString:91551}, {name:preVerificationGas, type:uint256, order:7, indexed:false, value:64679, valueString:64679}, {name:maxFeePerGas, type:uint256, order:8, indexed:false, value:9477709660, valueString:9477709660}, {name:maxPriorityFeePerGas, type:uint256, order:9, indexed:false, value:1000000, valueString:1000000}, {name:paymasterAndData, type:bytes, order:10, indexed:false, value:0x9D6AC51B972544251FCC0F2902E633E3F9BD3F29000000000000000000000000000000000000000000000000000000006622F6940000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000006A9B93CA5C207C59D4A08AAC7F637BCF0CA80157BCFBC7B8801DEA8DF818EB861F5BD5E69ED2FBE835B5D7EA1F30555B94B8CEA5B3A9E089C0EFB905233AB0201B, valueString:0x9D6AC51B972544251FCC0F2902E633E3F9BD3F29000000000000000000000000000000000000000000000000000000006622F6940000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000006A9B93CA5C207C59D4A08AAC7F637BCF0CA80157BCFBC7B8801DEA8DF818EB861F5BD5E69ED2FBE835B5D7EA1F30555B94B8CEA5B3A9E089C0EFB905233AB0201B}, {name:signature, type:bytes, order:11, indexed:false, value:0xCDA21FBE630D6B7C9627EE28C7AB313E9417399B9B5349E0CA655AE0905E8CCE207984913E83648EA8243DE5E2E9FC9D676BDA365AD5AE7585028BB8975715361C, valueString:0xCDA21FBE630D6B7C9627EE28C7AB313E9417399B9B5349E0CA655AE0905E8CCE207984913E83648EA8243DE5E2E9FC9D676BDA365AD5AE7585028BB8975715361C}], userOpHash=751A092A09294E2E75B53FA338839E4027EAAC326D1E1CA72497B06AB72C7477, maxCost=4716819155040500 ) => ( context=0x, validationData=2504381531546825506580642525143379377330304829124121722880 )
        • Null: 0x000...001.a328beb0( )
        • EntryPoint.innerHandleOp( callData=0x940D3C600000000000000000000000008AE01FCF7C655655FF2C6EF907B8B4718AB4E17C00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000080000000000000000000000000000000000000000000000000000000000000000100000000000000000000000000000000000000000000000000000000000003A48D80FF0A0000000000000000000000000000000000000000000000000000000000000020000000000000000000000000000000000000000000000000000000000000034E0047BDBD8FD3B73C47C36C8B218347FB3BB9305F1800000000000000000000000000000000000000000000000000088B55B2AF50000000000000000000000000000000000000000000000000000000000000000000002DDB11443BD9CEB92D4951A05F55EB7096EB53D3000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000002A42E7BA6EF00000000000000000000000000000000000000000000000000000000000000090000000000000000000000009EF22CC19EA80606ABA40E1E5F4E5553473DA6BC0000000000000000000000000000000000000000000000B58BF4D7B9CB6C00000000000000000000000000000000000000000000000000000000000000000080000000000000000000000000000000000000000000000000000000000000001077010425AB518A2EE957C0564871BAB7BD28AF05CFCE68FE7D7414ACA920BCFC9C2694DC9B55344E20FFF8AB74401E5E2305BCEBD35FA47D16ED59F8FB5A7AC964AE296223B0E6C7B4A0E025B0C953A9AB44AC159CC1BC903FE95557EC76167D10439E074E9CF5A996F71F9DC02047D3DFADD22A80078B7720221A29FB6B9145CEE48D9727596FF604675F51C9B6BF31C26AFA3140F1A17982BD7026F10AACD964043AEED019A7FF10B91A9EF10EBD93665BDB07D9C0CC4914B259C00F629EF84572AFC5BA4FCC9E6D084A7F7B40CF912B53C574C6C98A72240A12769459472127AB5338C08E1F50FFA811131DA52C08E9F11152148B4830786AA3EDF927322DB2A3D05DD813CEBE14EFA66EF09C634A2ADA3BDAA1F5026E7F1BEACE2D458B7A1FD451A3D82AEFA3EFF76DB2F1B817583E0BFF133609F72CBA5F380B51B13ADA56D7B7D7C1B2175B252715623BF36CBB65AA93612F3E661489EFA1A03708B1192E198AAB0F7BF9159774A3626969B432891EA0EE16CE2BE922E982164F38F255F9AB054459886200656E5AFE20BC87C9C0D5BC34C996D75EBAA9028D94189CC21509A1C85BCE8CC5D63C72FB79F89D438CE5F8172349DD52CA4DB2F35E4C68FAD7FBAB98199AE8B823187350647C74AA8945A5AE0772F306A996672DBE069C7120512B815D45DBD9C84E486F616A2AAABA7B8A86FBD73FA5CD0586D415AD737B00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000, opInfo=[{name:mUserOp, type:tuple, order:1, indexed:false, value:[{name:sender, type:address, order:1, indexed:false, value:0x9eF22cc19eA80606ABA40e1e5F4E5553473Da6BC, valueString:0x9eF22cc19eA80606ABA40e1e5F4E5553473Da6BC}, {name:nonce, type:uint256, order:2, indexed:false, value:22, valueString:22}, {name:callGasLimit, type:uint256, order:3, indexed:false, value:158343, valueString:158343}, {name:verificationGasLimit, type:uint256, order:4, indexed:false, value:91551, valueString:91551}, {name:preVerificationGas, type:uint256, order:5, indexed:false, value:64679, valueString:64679}, {name:paymaster, type:address, order:6, indexed:false, value:0x9D6AC51b972544251Fcc0F2902e633E3f9BD3f29, valueString:0x9D6AC51b972544251Fcc0F2902e633E3f9BD3f29}, {name:maxFeePerGas, type:uint256, order:7, indexed:false, value:9477709660, valueString:9477709660}, {name:maxPriorityFeePerGas, type:uint256, order:8, indexed:false, value:1000000, valueString:1000000}], valueString:[{name:sender, type:address, order:1, indexed:false, value:0x9eF22cc19eA80606ABA40e1e5F4E5553473Da6BC, valueString:0x9eF22cc19eA80606ABA40e1e5F4E5553473Da6BC}, {name:nonce, type:uint256, order:2, indexed:false, value:22, valueString:22}, {name:callGasLimit, type:uint256, order:3, indexed:false, value:158343, valueString:158343}, {name:verificationGasLimit, type:uint256, order:4, indexed:false, value:91551, valueString:91551}, {name:preVerificationGas, type:uint256, order:5, indexed:false, value:64679, valueString:64679}, {name:paymaster, type:address, order:6, indexed:false, value:0x9D6AC51b972544251Fcc0F2902e633E3f9BD3f29, valueString:0x9D6AC51b972544251Fcc0F2902e633E3f9BD3f29}, {name:maxFeePerGas, type:uint256, order:7, indexed:false, value:9477709660, valueString:9477709660}, {name:maxPriorityFeePerGas, type:uint256, order:8, indexed:false, value:1000000, valueString:1000000}]}, {name:userOpHash, type:bytes32, order:2, indexed:false, value:751A092A09294E2E75B53FA338839E4027EAAC326D1E1CA72497B06AB72C7477, valueString:751A092A09294E2E75B53FA338839E4027EAAC326D1E1CA72497B06AB72C7477}, {name:prefund, type:uint256, order:3, indexed:false, value:4716819155040500, valueString:4716819155040500}, {name:contextOffset, type:uint256, order:4, indexed:false, value:1216, valueString:1216}, {name:preOpGas, type:uint256, order:5, indexed:false, value:118656, valueString:118656}], context=0x ) => ( actualGasCost=1822854307174992 )
          • 0x9ef22cc19ea80606aba40e1e5f4e5553473da6bc.940d3c60( )
            • 0x8253291a17d3beb95fadab2751d52b324d22ef2d.940d3c60( )
              • MultiSend.multiSend( transactions=0x0047BDBD8FD3B73C47C36C8B218347FB3BB9305F1800000000000000000000000000000000000000000000000000088B55B2AF50000000000000000000000000000000000000000000000000000000000000000000002DDB11443BD9CEB92D4951A05F55EB7096EB53D3000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000002A42E7BA6EF00000000000000000000000000000000000000000000000000000000000000090000000000000000000000009EF22CC19EA80606ABA40E1E5F4E5553473DA6BC0000000000000000000000000000000000000000000000B58BF4D7B9CB6C00000000000000000000000000000000000000000000000000000000000000000080000000000000000000000000000000000000000000000000000000000000001077010425AB518A2EE957C0564871BAB7BD28AF05CFCE68FE7D7414ACA920BCFC9C2694DC9B55344E20FFF8AB74401E5E2305BCEBD35FA47D16ED59F8FB5A7AC964AE296223B0E6C7B4A0E025B0C953A9AB44AC159CC1BC903FE95557EC76167D10439E074E9CF5A996F71F9DC02047D3DFADD22A80078B7720221A29FB6B9145CEE48D9727596FF604675F51C9B6BF31C26AFA3140F1A17982BD7026F10AACD964043AEED019A7FF10B91A9EF10EBD93665BDB07D9C0CC4914B259C00F629EF84572AFC5BA4FCC9E6D084A7F7B40CF912B53C574C6C98A72240A12769459472127AB5338C08E1F50FFA811131DA52C08E9F11152148B4830786AA3EDF927322DB2A3D05DD813CEBE14EFA66EF09C634A2ADA3BDAA1F5026E7F1BEACE2D458B7A1FD451A3D82AEFA3EFF76DB2F1B817583E0BFF133609F72CBA5F380B51B13ADA56D7B7D7C1B2175B252715623BF36CBB65AA93612F3E661489EFA1A03708B1192E198AAB0F7BF9159774A3626969B432891EA0EE16CE2BE922E982164F38F255F9AB054459886200656E5AFE20BC87C9C0D5BC34C996D75EBAA9028D94189CC21509A1C85BCE8CC5D63C72FB79F89D438CE5F8172349DD52CA4DB2F35E4C68FAD7FBAB98199AE8B823187350647C74AA8945A5AE0772F306A996672DBE069C7120512B815D45DBD9C84E486F616A2AAABA7B8A86FBD73FA5CD0586D415AD737B )
                • ETH 0.002405 0x47bdbd8fd3b73c47c36c8b218347fb3bb9305f18.CALL( )
                • TransparentUpgradeableProxy.2e7ba6ef( )
                  • MerkleDistributor.claim( )
                  • ETH 0.001822854307174992 Stackup: Bundler 3.CALL( )
                    handleOps[EntryPoint (ln:137)]
                    File 1 of 6: EntryPoint
                    // SPDX-License-Identifier: MIT
                    // OpenZeppelin Contracts (last updated v4.8.0) (security/ReentrancyGuard.sol)
                    pragma solidity ^0.8.0;
                    /**
                     * @dev Contract module that helps prevent reentrant calls to a function.
                     *
                     * Inheriting from `ReentrancyGuard` will make the {nonReentrant} modifier
                     * available, which can be applied to functions to make sure there are no nested
                     * (reentrant) calls to them.
                     *
                     * Note that because there is a single `nonReentrant` guard, functions marked as
                     * `nonReentrant` may not call one another. This can be worked around by making
                     * those functions `private`, and then adding `external` `nonReentrant` entry
                     * points to them.
                     *
                     * TIP: If you would like to learn more about reentrancy and alternative ways
                     * to protect against it, check out our blog post
                     * https://blog.openzeppelin.com/reentrancy-after-istanbul/[Reentrancy After Istanbul].
                     */
                    abstract contract ReentrancyGuard {
                        // Booleans are more expensive than uint256 or any type that takes up a full
                        // word because each write operation emits an extra SLOAD to first read the
                        // slot's contents, replace the bits taken up by the boolean, and then write
                        // back. This is the compiler's defense against contract upgrades and
                        // pointer aliasing, and it cannot be disabled.
                        // The values being non-zero value makes deployment a bit more expensive,
                        // but in exchange the refund on every call to nonReentrant will be lower in
                        // amount. Since refunds are capped to a percentage of the total
                        // transaction's gas, it is best to keep them low in cases like this one, to
                        // increase the likelihood of the full refund coming into effect.
                        uint256 private constant _NOT_ENTERED = 1;
                        uint256 private constant _ENTERED = 2;
                        uint256 private _status;
                        constructor() {
                            _status = _NOT_ENTERED;
                        }
                        /**
                         * @dev Prevents a contract from calling itself, directly or indirectly.
                         * Calling a `nonReentrant` function from another `nonReentrant`
                         * function is not supported. It is possible to prevent this from happening
                         * by making the `nonReentrant` function external, and making it call a
                         * `private` function that does the actual work.
                         */
                        modifier nonReentrant() {
                            _nonReentrantBefore();
                            _;
                            _nonReentrantAfter();
                        }
                        function _nonReentrantBefore() private {
                            // On the first call to nonReentrant, _status will be _NOT_ENTERED
                            require(_status != _ENTERED, "ReentrancyGuard: reentrant call");
                            // Any calls to nonReentrant after this point will fail
                            _status = _ENTERED;
                        }
                        function _nonReentrantAfter() private {
                            // By storing the original value once again, a refund is triggered (see
                            // https://eips.ethereum.org/EIPS/eip-2200)
                            _status = _NOT_ENTERED;
                        }
                    }
                    /**
                     ** Account-Abstraction (EIP-4337) singleton EntryPoint implementation.
                     ** Only one instance required on each chain.
                     **/
                    // SPDX-License-Identifier: GPL-3.0
                    pragma solidity ^0.8.12;
                    /* solhint-disable avoid-low-level-calls */
                    /* solhint-disable no-inline-assembly */
                    import "../interfaces/IAccount.sol";
                    import "../interfaces/IPaymaster.sol";
                    import "../interfaces/IEntryPoint.sol";
                    import "../utils/Exec.sol";
                    import "./StakeManager.sol";
                    import "./SenderCreator.sol";
                    import "./Helpers.sol";
                    import "./NonceManager.sol";
                    import "@openzeppelin/contracts/security/ReentrancyGuard.sol";
                    contract EntryPoint is IEntryPoint, StakeManager, NonceManager, ReentrancyGuard {
                        using UserOperationLib for UserOperation;
                        SenderCreator private immutable senderCreator = new SenderCreator();
                        // internal value used during simulation: need to query aggregator.
                        address private constant SIMULATE_FIND_AGGREGATOR = address(1);
                        // marker for inner call revert on out of gas
                        bytes32 private constant INNER_OUT_OF_GAS = hex'deaddead';
                        uint256 private constant REVERT_REASON_MAX_LEN = 2048;
                        /**
                         * for simulation purposes, validateUserOp (and validatePaymasterUserOp) must return this value
                         * in case of signature failure, instead of revert.
                         */
                        uint256 public constant SIG_VALIDATION_FAILED = 1;
                        /**
                         * compensate the caller's beneficiary address with the collected fees of all UserOperations.
                         * @param beneficiary the address to receive the fees
                         * @param amount amount to transfer.
                         */
                        function _compensate(address payable beneficiary, uint256 amount) internal {
                            require(beneficiary != address(0), "AA90 invalid beneficiary");
                            (bool success,) = beneficiary.call{value : amount}("");
                            require(success, "AA91 failed send to beneficiary");
                        }
                        /**
                         * execute a user op
                         * @param opIndex index into the opInfo array
                         * @param userOp the userOp to execute
                         * @param opInfo the opInfo filled by validatePrepayment for this userOp.
                         * @return collected the total amount this userOp paid.
                         */
                        function _executeUserOp(uint256 opIndex, UserOperation calldata userOp, UserOpInfo memory opInfo) private returns (uint256 collected) {
                            uint256 preGas = gasleft();
                            bytes memory context = getMemoryBytesFromOffset(opInfo.contextOffset);
                            try this.innerHandleOp(userOp.callData, opInfo, context) returns (uint256 _actualGasCost) {
                                collected = _actualGasCost;
                            } catch {
                                bytes32 innerRevertCode;
                                assembly {
                                    returndatacopy(0, 0, 32)
                                    innerRevertCode := mload(0)
                                }
                                // handleOps was called with gas limit too low. abort entire bundle.
                                if (innerRevertCode == INNER_OUT_OF_GAS) {
                                    //report paymaster, since if it is not deliberately caused by the bundler,
                                    // it must be a revert caused by paymaster.
                                    revert FailedOp(opIndex, "AA95 out of gas");
                                }
                                uint256 actualGas = preGas - gasleft() + opInfo.preOpGas;
                                collected = _handlePostOp(opIndex, IPaymaster.PostOpMode.postOpReverted, opInfo, context, actualGas);
                            }
                        }
                        /**
                         * Execute a batch of UserOperations.
                         * no signature aggregator is used.
                         * if any account requires an aggregator (that is, it returned an aggregator when
                         * performing simulateValidation), then handleAggregatedOps() must be used instead.
                         * @param ops the operations to execute
                         * @param beneficiary the address to receive the fees
                         */
                        function handleOps(UserOperation[] calldata ops, address payable beneficiary) public nonReentrant {
                            uint256 opslen = ops.length;
                            UserOpInfo[] memory opInfos = new UserOpInfo[](opslen);
                        unchecked {
                            for (uint256 i = 0; i < opslen; i++) {
                                UserOpInfo memory opInfo = opInfos[i];
                                (uint256 validationData, uint256 pmValidationData) = _validatePrepayment(i, ops[i], opInfo);
                                _validateAccountAndPaymasterValidationData(i, validationData, pmValidationData, address(0));
                            }
                            uint256 collected = 0;
                            emit BeforeExecution();
                            for (uint256 i = 0; i < opslen; i++) {
                                collected += _executeUserOp(i, ops[i], opInfos[i]);
                            }
                            _compensate(beneficiary, collected);
                        } //unchecked
                        }
                        /**
                         * Execute a batch of UserOperation with Aggregators
                         * @param opsPerAggregator the operations to execute, grouped by aggregator (or address(0) for no-aggregator accounts)
                         * @param beneficiary the address to receive the fees
                         */
                        function handleAggregatedOps(
                            UserOpsPerAggregator[] calldata opsPerAggregator,
                            address payable beneficiary
                        ) public nonReentrant {
                            uint256 opasLen = opsPerAggregator.length;
                            uint256 totalOps = 0;
                            for (uint256 i = 0; i < opasLen; i++) {
                                UserOpsPerAggregator calldata opa = opsPerAggregator[i];
                                UserOperation[] calldata ops = opa.userOps;
                                IAggregator aggregator = opa.aggregator;
                                //address(1) is special marker of "signature error"
                                require(address(aggregator) != address(1), "AA96 invalid aggregator");
                                if (address(aggregator) != address(0)) {
                                    // solhint-disable-next-line no-empty-blocks
                                    try aggregator.validateSignatures(ops, opa.signature) {}
                                    catch {
                                        revert SignatureValidationFailed(address(aggregator));
                                    }
                                }
                                totalOps += ops.length;
                            }
                            UserOpInfo[] memory opInfos = new UserOpInfo[](totalOps);
                            emit BeforeExecution();
                            uint256 opIndex = 0;
                            for (uint256 a = 0; a < opasLen; a++) {
                                UserOpsPerAggregator calldata opa = opsPerAggregator[a];
                                UserOperation[] calldata ops = opa.userOps;
                                IAggregator aggregator = opa.aggregator;
                                uint256 opslen = ops.length;
                                for (uint256 i = 0; i < opslen; i++) {
                                    UserOpInfo memory opInfo = opInfos[opIndex];
                                    (uint256 validationData, uint256 paymasterValidationData) = _validatePrepayment(opIndex, ops[i], opInfo);
                                    _validateAccountAndPaymasterValidationData(i, validationData, paymasterValidationData, address(aggregator));
                                    opIndex++;
                                }
                            }
                            uint256 collected = 0;
                            opIndex = 0;
                            for (uint256 a = 0; a < opasLen; a++) {
                                UserOpsPerAggregator calldata opa = opsPerAggregator[a];
                                emit SignatureAggregatorChanged(address(opa.aggregator));
                                UserOperation[] calldata ops = opa.userOps;
                                uint256 opslen = ops.length;
                                for (uint256 i = 0; i < opslen; i++) {
                                    collected += _executeUserOp(opIndex, ops[i], opInfos[opIndex]);
                                    opIndex++;
                                }
                            }
                            emit SignatureAggregatorChanged(address(0));
                            _compensate(beneficiary, collected);
                        }
                        /// @inheritdoc IEntryPoint
                        function simulateHandleOp(UserOperation calldata op, address target, bytes calldata targetCallData) external override {
                            UserOpInfo memory opInfo;
                            _simulationOnlyValidations(op);
                            (uint256 validationData, uint256 paymasterValidationData) = _validatePrepayment(0, op, opInfo);
                            ValidationData memory data = _intersectTimeRange(validationData, paymasterValidationData);
                            numberMarker();
                            uint256 paid = _executeUserOp(0, op, opInfo);
                            numberMarker();
                            bool targetSuccess;
                            bytes memory targetResult;
                            if (target != address(0)) {
                                (targetSuccess, targetResult) = target.call(targetCallData);
                            }
                            revert ExecutionResult(opInfo.preOpGas, paid, data.validAfter, data.validUntil, targetSuccess, targetResult);
                        }
                        // A memory copy of UserOp static fields only.
                        // Excluding: callData, initCode and signature. Replacing paymasterAndData with paymaster.
                        struct MemoryUserOp {
                            address sender;
                            uint256 nonce;
                            uint256 callGasLimit;
                            uint256 verificationGasLimit;
                            uint256 preVerificationGas;
                            address paymaster;
                            uint256 maxFeePerGas;
                            uint256 maxPriorityFeePerGas;
                        }
                        struct UserOpInfo {
                            MemoryUserOp mUserOp;
                            bytes32 userOpHash;
                            uint256 prefund;
                            uint256 contextOffset;
                            uint256 preOpGas;
                        }
                        /**
                         * inner function to handle a UserOperation.
                         * Must be declared "external" to open a call context, but it can only be called by handleOps.
                         */
                        function innerHandleOp(bytes memory callData, UserOpInfo memory opInfo, bytes calldata context) external returns (uint256 actualGasCost) {
                            uint256 preGas = gasleft();
                            require(msg.sender == address(this), "AA92 internal call only");
                            MemoryUserOp memory mUserOp = opInfo.mUserOp;
                            uint callGasLimit = mUserOp.callGasLimit;
                        unchecked {
                            // handleOps was called with gas limit too low. abort entire bundle.
                            if (gasleft() < callGasLimit + mUserOp.verificationGasLimit + 5000) {
                                assembly {
                                    mstore(0, INNER_OUT_OF_GAS)
                                    revert(0, 32)
                                }
                            }
                        }
                            IPaymaster.PostOpMode mode = IPaymaster.PostOpMode.opSucceeded;
                            if (callData.length > 0) {
                                bool success = Exec.call(mUserOp.sender, 0, callData, callGasLimit);
                                if (!success) {
                                    bytes memory result = Exec.getReturnData(REVERT_REASON_MAX_LEN);
                                    if (result.length > 0) {
                                        emit UserOperationRevertReason(opInfo.userOpHash, mUserOp.sender, mUserOp.nonce, result);
                                    }
                                    mode = IPaymaster.PostOpMode.opReverted;
                                }
                            }
                        unchecked {
                            uint256 actualGas = preGas - gasleft() + opInfo.preOpGas;
                            //note: opIndex is ignored (relevant only if mode==postOpReverted, which is only possible outside of innerHandleOp)
                            return _handlePostOp(0, mode, opInfo, context, actualGas);
                        }
                        }
                        /**
                         * generate a request Id - unique identifier for this request.
                         * the request ID is a hash over the content of the userOp (except the signature), the entrypoint and the chainid.
                         */
                        function getUserOpHash(UserOperation calldata userOp) public view returns (bytes32) {
                            return keccak256(abi.encode(userOp.hash(), address(this), block.chainid));
                        }
                        /**
                         * copy general fields from userOp into the memory opInfo structure.
                         */
                        function _copyUserOpToMemory(UserOperation calldata userOp, MemoryUserOp memory mUserOp) internal pure {
                            mUserOp.sender = userOp.sender;
                            mUserOp.nonce = userOp.nonce;
                            mUserOp.callGasLimit = userOp.callGasLimit;
                            mUserOp.verificationGasLimit = userOp.verificationGasLimit;
                            mUserOp.preVerificationGas = userOp.preVerificationGas;
                            mUserOp.maxFeePerGas = userOp.maxFeePerGas;
                            mUserOp.maxPriorityFeePerGas = userOp.maxPriorityFeePerGas;
                            bytes calldata paymasterAndData = userOp.paymasterAndData;
                            if (paymasterAndData.length > 0) {
                                require(paymasterAndData.length >= 20, "AA93 invalid paymasterAndData");
                                mUserOp.paymaster = address(bytes20(paymasterAndData[: 20]));
                            } else {
                                mUserOp.paymaster = address(0);
                            }
                        }
                        /**
                         * Simulate a call to account.validateUserOp and paymaster.validatePaymasterUserOp.
                         * @dev this method always revert. Successful result is ValidationResult error. other errors are failures.
                         * @dev The node must also verify it doesn't use banned opcodes, and that it doesn't reference storage outside the account's data.
                         * @param userOp the user operation to validate.
                         */
                        function simulateValidation(UserOperation calldata userOp) external {
                            UserOpInfo memory outOpInfo;
                            _simulationOnlyValidations(userOp);
                            (uint256 validationData, uint256 paymasterValidationData) = _validatePrepayment(0, userOp, outOpInfo);
                            StakeInfo memory paymasterInfo = _getStakeInfo(outOpInfo.mUserOp.paymaster);
                            StakeInfo memory senderInfo = _getStakeInfo(outOpInfo.mUserOp.sender);
                            StakeInfo memory factoryInfo;
                            {
                                bytes calldata initCode = userOp.initCode;
                                address factory = initCode.length >= 20 ? address(bytes20(initCode[0 : 20])) : address(0);
                                factoryInfo = _getStakeInfo(factory);
                            }
                            ValidationData memory data = _intersectTimeRange(validationData, paymasterValidationData);
                            address aggregator = data.aggregator;
                            bool sigFailed = aggregator == address(1);
                            ReturnInfo memory returnInfo = ReturnInfo(outOpInfo.preOpGas, outOpInfo.prefund,
                                sigFailed, data.validAfter, data.validUntil, getMemoryBytesFromOffset(outOpInfo.contextOffset));
                            if (aggregator != address(0) && aggregator != address(1)) {
                                AggregatorStakeInfo memory aggregatorInfo = AggregatorStakeInfo(aggregator, _getStakeInfo(aggregator));
                                revert ValidationResultWithAggregation(returnInfo, senderInfo, factoryInfo, paymasterInfo, aggregatorInfo);
                            }
                            revert ValidationResult(returnInfo, senderInfo, factoryInfo, paymasterInfo);
                        }
                        function _getRequiredPrefund(MemoryUserOp memory mUserOp) internal pure returns (uint256 requiredPrefund) {
                        unchecked {
                            //when using a Paymaster, the verificationGasLimit is used also to as a limit for the postOp call.
                            // our security model might call postOp eventually twice
                            uint256 mul = mUserOp.paymaster != address(0) ? 3 : 1;
                            uint256 requiredGas = mUserOp.callGasLimit + mUserOp.verificationGasLimit * mul + mUserOp.preVerificationGas;
                            requiredPrefund = requiredGas * mUserOp.maxFeePerGas;
                        }
                        }
                        // create the sender's contract if needed.
                        function _createSenderIfNeeded(uint256 opIndex, UserOpInfo memory opInfo, bytes calldata initCode) internal {
                            if (initCode.length != 0) {
                                address sender = opInfo.mUserOp.sender;
                                if (sender.code.length != 0) revert FailedOp(opIndex, "AA10 sender already constructed");
                                address sender1 = senderCreator.createSender{gas : opInfo.mUserOp.verificationGasLimit}(initCode);
                                if (sender1 == address(0)) revert FailedOp(opIndex, "AA13 initCode failed or OOG");
                                if (sender1 != sender) revert FailedOp(opIndex, "AA14 initCode must return sender");
                                if (sender1.code.length == 0) revert FailedOp(opIndex, "AA15 initCode must create sender");
                                address factory = address(bytes20(initCode[0 : 20]));
                                emit AccountDeployed(opInfo.userOpHash, sender, factory, opInfo.mUserOp.paymaster);
                            }
                        }
                        /**
                         * Get counterfactual sender address.
                         *  Calculate the sender contract address that will be generated by the initCode and salt in the UserOperation.
                         * this method always revert, and returns the address in SenderAddressResult error
                         * @param initCode the constructor code to be passed into the UserOperation.
                         */
                        function getSenderAddress(bytes calldata initCode) public {
                            address sender = senderCreator.createSender(initCode);
                            revert SenderAddressResult(sender);
                        }
                        function _simulationOnlyValidations(UserOperation calldata userOp) internal view {
                            // solhint-disable-next-line no-empty-blocks
                            try this._validateSenderAndPaymaster(userOp.initCode, userOp.sender, userOp.paymasterAndData) {}
                            catch Error(string memory revertReason) {
                                if (bytes(revertReason).length != 0) {
                                    revert FailedOp(0, revertReason);
                                }
                            }
                        }
                        /**
                        * Called only during simulation.
                        * This function always reverts to prevent warm/cold storage differentiation in simulation vs execution.
                        */
                        function _validateSenderAndPaymaster(bytes calldata initCode, address sender, bytes calldata paymasterAndData) external view {
                            if (initCode.length == 0 && sender.code.length == 0) {
                                // it would revert anyway. but give a meaningful message
                                revert("AA20 account not deployed");
                            }
                            if (paymasterAndData.length >= 20) {
                                address paymaster = address(bytes20(paymasterAndData[0 : 20]));
                                if (paymaster.code.length == 0) {
                                    // it would revert anyway. but give a meaningful message
                                    revert("AA30 paymaster not deployed");
                                }
                            }
                            // always revert
                            revert("");
                        }
                        /**
                         * call account.validateUserOp.
                         * revert (with FailedOp) in case validateUserOp reverts, or account didn't send required prefund.
                         * decrement account's deposit if needed
                         */
                        function _validateAccountPrepayment(uint256 opIndex, UserOperation calldata op, UserOpInfo memory opInfo, uint256 requiredPrefund)
                        internal returns (uint256 gasUsedByValidateAccountPrepayment, uint256 validationData) {
                        unchecked {
                            uint256 preGas = gasleft();
                            MemoryUserOp memory mUserOp = opInfo.mUserOp;
                            address sender = mUserOp.sender;
                            _createSenderIfNeeded(opIndex, opInfo, op.initCode);
                            address paymaster = mUserOp.paymaster;
                            numberMarker();
                            uint256 missingAccountFunds = 0;
                            if (paymaster == address(0)) {
                                uint256 bal = balanceOf(sender);
                                missingAccountFunds = bal > requiredPrefund ? 0 : requiredPrefund - bal;
                            }
                            try IAccount(sender).validateUserOp{gas : mUserOp.verificationGasLimit}(op, opInfo.userOpHash, missingAccountFunds)
                            returns (uint256 _validationData) {
                                validationData = _validationData;
                            } catch Error(string memory revertReason) {
                                revert FailedOp(opIndex, string.concat("AA23 reverted: ", revertReason));
                            } catch {
                                revert FailedOp(opIndex, "AA23 reverted (or OOG)");
                            }
                            if (paymaster == address(0)) {
                                DepositInfo storage senderInfo = deposits[sender];
                                uint256 deposit = senderInfo.deposit;
                                if (requiredPrefund > deposit) {
                                    revert FailedOp(opIndex, "AA21 didn't pay prefund");
                                }
                                senderInfo.deposit = uint112(deposit - requiredPrefund);
                            }
                            gasUsedByValidateAccountPrepayment = preGas - gasleft();
                        }
                        }
                        /**
                         * In case the request has a paymaster:
                         * Validate paymaster has enough deposit.
                         * Call paymaster.validatePaymasterUserOp.
                         * Revert with proper FailedOp in case paymaster reverts.
                         * Decrement paymaster's deposit
                         */
                        function _validatePaymasterPrepayment(uint256 opIndex, UserOperation calldata op, UserOpInfo memory opInfo, uint256 requiredPreFund, uint256 gasUsedByValidateAccountPrepayment)
                        internal returns (bytes memory context, uint256 validationData) {
                        unchecked {
                            MemoryUserOp memory mUserOp = opInfo.mUserOp;
                            uint256 verificationGasLimit = mUserOp.verificationGasLimit;
                            require(verificationGasLimit > gasUsedByValidateAccountPrepayment, "AA41 too little verificationGas");
                            uint256 gas = verificationGasLimit - gasUsedByValidateAccountPrepayment;
                            address paymaster = mUserOp.paymaster;
                            DepositInfo storage paymasterInfo = deposits[paymaster];
                            uint256 deposit = paymasterInfo.deposit;
                            if (deposit < requiredPreFund) {
                                revert FailedOp(opIndex, "AA31 paymaster deposit too low");
                            }
                            paymasterInfo.deposit = uint112(deposit - requiredPreFund);
                            try IPaymaster(paymaster).validatePaymasterUserOp{gas : gas}(op, opInfo.userOpHash, requiredPreFund) returns (bytes memory _context, uint256 _validationData){
                                context = _context;
                                validationData = _validationData;
                            } catch Error(string memory revertReason) {
                                revert FailedOp(opIndex, string.concat("AA33 reverted: ", revertReason));
                            } catch {
                                revert FailedOp(opIndex, "AA33 reverted (or OOG)");
                            }
                        }
                        }
                        /**
                         * revert if either account validationData or paymaster validationData is expired
                         */
                        function _validateAccountAndPaymasterValidationData(uint256 opIndex, uint256 validationData, uint256 paymasterValidationData, address expectedAggregator) internal view {
                            (address aggregator, bool outOfTimeRange) = _getValidationData(validationData);
                            if (expectedAggregator != aggregator) {
                                revert FailedOp(opIndex, "AA24 signature error");
                            }
                            if (outOfTimeRange) {
                                revert FailedOp(opIndex, "AA22 expired or not due");
                            }
                            //pmAggregator is not a real signature aggregator: we don't have logic to handle it as address.
                            // non-zero address means that the paymaster fails due to some signature check (which is ok only during estimation)
                            address pmAggregator;
                            (pmAggregator, outOfTimeRange) = _getValidationData(paymasterValidationData);
                            if (pmAggregator != address(0)) {
                                revert FailedOp(opIndex, "AA34 signature error");
                            }
                            if (outOfTimeRange) {
                                revert FailedOp(opIndex, "AA32 paymaster expired or not due");
                            }
                        }
                        function _getValidationData(uint256 validationData) internal view returns (address aggregator, bool outOfTimeRange) {
                            if (validationData == 0) {
                                return (address(0), false);
                            }
                            ValidationData memory data = _parseValidationData(validationData);
                            // solhint-disable-next-line not-rely-on-time
                            outOfTimeRange = block.timestamp > data.validUntil || block.timestamp < data.validAfter;
                            aggregator = data.aggregator;
                        }
                        /**
                         * validate account and paymaster (if defined).
                         * also make sure total validation doesn't exceed verificationGasLimit
                         * this method is called off-chain (simulateValidation()) and on-chain (from handleOps)
                         * @param opIndex the index of this userOp into the "opInfos" array
                         * @param userOp the userOp to validate
                         */
                        function _validatePrepayment(uint256 opIndex, UserOperation calldata userOp, UserOpInfo memory outOpInfo)
                        private returns (uint256 validationData, uint256 paymasterValidationData) {
                            uint256 preGas = gasleft();
                            MemoryUserOp memory mUserOp = outOpInfo.mUserOp;
                            _copyUserOpToMemory(userOp, mUserOp);
                            outOpInfo.userOpHash = getUserOpHash(userOp);
                            // validate all numeric values in userOp are well below 128 bit, so they can safely be added
                            // and multiplied without causing overflow
                            uint256 maxGasValues = mUserOp.preVerificationGas | mUserOp.verificationGasLimit | mUserOp.callGasLimit |
                            userOp.maxFeePerGas | userOp.maxPriorityFeePerGas;
                            require(maxGasValues <= type(uint120).max, "AA94 gas values overflow");
                            uint256 gasUsedByValidateAccountPrepayment;
                            (uint256 requiredPreFund) = _getRequiredPrefund(mUserOp);
                            (gasUsedByValidateAccountPrepayment, validationData) = _validateAccountPrepayment(opIndex, userOp, outOpInfo, requiredPreFund);
                            if (!_validateAndUpdateNonce(mUserOp.sender, mUserOp.nonce)) {
                                revert FailedOp(opIndex, "AA25 invalid account nonce");
                            }
                            //a "marker" where account opcode validation is done and paymaster opcode validation is about to start
                            // (used only by off-chain simulateValidation)
                            numberMarker();
                            bytes memory context;
                            if (mUserOp.paymaster != address(0)) {
                                (context, paymasterValidationData) = _validatePaymasterPrepayment(opIndex, userOp, outOpInfo, requiredPreFund, gasUsedByValidateAccountPrepayment);
                            }
                        unchecked {
                            uint256 gasUsed = preGas - gasleft();
                            if (userOp.verificationGasLimit < gasUsed) {
                                revert FailedOp(opIndex, "AA40 over verificationGasLimit");
                            }
                            outOpInfo.prefund = requiredPreFund;
                            outOpInfo.contextOffset = getOffsetOfMemoryBytes(context);
                            outOpInfo.preOpGas = preGas - gasleft() + userOp.preVerificationGas;
                        }
                        }
                        /**
                         * process post-operation.
                         * called just after the callData is executed.
                         * if a paymaster is defined and its validation returned a non-empty context, its postOp is called.
                         * the excess amount is refunded to the account (or paymaster - if it was used in the request)
                         * @param opIndex index in the batch
                         * @param mode - whether is called from innerHandleOp, or outside (postOpReverted)
                         * @param opInfo userOp fields and info collected during validation
                         * @param context the context returned in validatePaymasterUserOp
                         * @param actualGas the gas used so far by this user operation
                         */
                        function _handlePostOp(uint256 opIndex, IPaymaster.PostOpMode mode, UserOpInfo memory opInfo, bytes memory context, uint256 actualGas) private returns (uint256 actualGasCost) {
                            uint256 preGas = gasleft();
                        unchecked {
                            address refundAddress;
                            MemoryUserOp memory mUserOp = opInfo.mUserOp;
                            uint256 gasPrice = getUserOpGasPrice(mUserOp);
                            address paymaster = mUserOp.paymaster;
                            if (paymaster == address(0)) {
                                refundAddress = mUserOp.sender;
                            } else {
                                refundAddress = paymaster;
                                if (context.length > 0) {
                                    actualGasCost = actualGas * gasPrice;
                                    if (mode != IPaymaster.PostOpMode.postOpReverted) {
                                        IPaymaster(paymaster).postOp{gas : mUserOp.verificationGasLimit}(mode, context, actualGasCost);
                                    } else {
                                        // solhint-disable-next-line no-empty-blocks
                                        try IPaymaster(paymaster).postOp{gas : mUserOp.verificationGasLimit}(mode, context, actualGasCost) {}
                                        catch Error(string memory reason) {
                                            revert FailedOp(opIndex, string.concat("AA50 postOp reverted: ", reason));
                                        }
                                        catch {
                                            revert FailedOp(opIndex, "AA50 postOp revert");
                                        }
                                    }
                                }
                            }
                            actualGas += preGas - gasleft();
                            actualGasCost = actualGas * gasPrice;
                            if (opInfo.prefund < actualGasCost) {
                                revert FailedOp(opIndex, "AA51 prefund below actualGasCost");
                            }
                            uint256 refund = opInfo.prefund - actualGasCost;
                            _incrementDeposit(refundAddress, refund);
                            bool success = mode == IPaymaster.PostOpMode.opSucceeded;
                            emit UserOperationEvent(opInfo.userOpHash, mUserOp.sender, mUserOp.paymaster, mUserOp.nonce, success, actualGasCost, actualGas);
                        } // unchecked
                        }
                        /**
                         * the gas price this UserOp agrees to pay.
                         * relayer/block builder might submit the TX with higher priorityFee, but the user should not
                         */
                        function getUserOpGasPrice(MemoryUserOp memory mUserOp) internal view returns (uint256) {
                        unchecked {
                            uint256 maxFeePerGas = mUserOp.maxFeePerGas;
                            uint256 maxPriorityFeePerGas = mUserOp.maxPriorityFeePerGas;
                            if (maxFeePerGas == maxPriorityFeePerGas) {
                                //legacy mode (for networks that don't support basefee opcode)
                                return maxFeePerGas;
                            }
                            return min(maxFeePerGas, maxPriorityFeePerGas + block.basefee);
                        }
                        }
                        function min(uint256 a, uint256 b) internal pure returns (uint256) {
                            return a < b ? a : b;
                        }
                        function getOffsetOfMemoryBytes(bytes memory data) internal pure returns (uint256 offset) {
                            assembly {offset := data}
                        }
                        function getMemoryBytesFromOffset(uint256 offset) internal pure returns (bytes memory data) {
                            assembly {data := offset}
                        }
                        //place the NUMBER opcode in the code.
                        // this is used as a marker during simulation, as this OP is completely banned from the simulated code of the
                        // account and paymaster.
                        function numberMarker() internal view {
                            assembly {mstore(0, number())}
                        }
                    }
                    // SPDX-License-Identifier: GPL-3.0
                    pragma solidity ^0.8.12;
                    /* solhint-disable no-inline-assembly */
                    /**
                     * returned data from validateUserOp.
                     * validateUserOp returns a uint256, with is created by `_packedValidationData` and parsed by `_parseValidationData`
                     * @param aggregator - address(0) - the account validated the signature by itself.
                     *              address(1) - the account failed to validate the signature.
                     *              otherwise - this is an address of a signature aggregator that must be used to validate the signature.
                     * @param validAfter - this UserOp is valid only after this timestamp.
                     * @param validaUntil - this UserOp is valid only up to this timestamp.
                     */
                        struct ValidationData {
                            address aggregator;
                            uint48 validAfter;
                            uint48 validUntil;
                        }
                    //extract sigFailed, validAfter, validUntil.
                    // also convert zero validUntil to type(uint48).max
                        function _parseValidationData(uint validationData) pure returns (ValidationData memory data) {
                            address aggregator = address(uint160(validationData));
                            uint48 validUntil = uint48(validationData >> 160);
                            if (validUntil == 0) {
                                validUntil = type(uint48).max;
                            }
                            uint48 validAfter = uint48(validationData >> (48 + 160));
                            return ValidationData(aggregator, validAfter, validUntil);
                        }
                    // intersect account and paymaster ranges.
                        function _intersectTimeRange(uint256 validationData, uint256 paymasterValidationData) pure returns (ValidationData memory) {
                            ValidationData memory accountValidationData = _parseValidationData(validationData);
                            ValidationData memory pmValidationData = _parseValidationData(paymasterValidationData);
                            address aggregator = accountValidationData.aggregator;
                            if (aggregator == address(0)) {
                                aggregator = pmValidationData.aggregator;
                            }
                            uint48 validAfter = accountValidationData.validAfter;
                            uint48 validUntil = accountValidationData.validUntil;
                            uint48 pmValidAfter = pmValidationData.validAfter;
                            uint48 pmValidUntil = pmValidationData.validUntil;
                            if (validAfter < pmValidAfter) validAfter = pmValidAfter;
                            if (validUntil > pmValidUntil) validUntil = pmValidUntil;
                            return ValidationData(aggregator, validAfter, validUntil);
                        }
                    /**
                     * helper to pack the return value for validateUserOp
                     * @param data - the ValidationData to pack
                     */
                        function _packValidationData(ValidationData memory data) pure returns (uint256) {
                            return uint160(data.aggregator) | (uint256(data.validUntil) << 160) | (uint256(data.validAfter) << (160 + 48));
                        }
                    /**
                     * helper to pack the return value for validateUserOp, when not using an aggregator
                     * @param sigFailed - true for signature failure, false for success
                     * @param validUntil last timestamp this UserOperation is valid (or zero for infinite)
                     * @param validAfter first timestamp this UserOperation is valid
                     */
                        function _packValidationData(bool sigFailed, uint48 validUntil, uint48 validAfter) pure returns (uint256) {
                            return (sigFailed ? 1 : 0) | (uint256(validUntil) << 160) | (uint256(validAfter) << (160 + 48));
                        }
                    /**
                     * keccak function over calldata.
                     * @dev copy calldata into memory, do keccak and drop allocated memory. Strangely, this is more efficient than letting solidity do it.
                     */
                        function calldataKeccak(bytes calldata data) pure returns (bytes32 ret) {
                            assembly {
                                let mem := mload(0x40)
                                let len := data.length
                                calldatacopy(mem, data.offset, len)
                                ret := keccak256(mem, len)
                            }
                        }
                    // SPDX-License-Identifier: GPL-3.0
                    pragma solidity ^0.8.12;
                    import "../interfaces/IEntryPoint.sol";
                    /**
                     * nonce management functionality
                     */
                    contract NonceManager is INonceManager {
                        /**
                         * The next valid sequence number for a given nonce key.
                         */
                        mapping(address => mapping(uint192 => uint256)) public nonceSequenceNumber;
                        function getNonce(address sender, uint192 key)
                        public view override returns (uint256 nonce) {
                            return nonceSequenceNumber[sender][key] | (uint256(key) << 64);
                        }
                        // allow an account to manually increment its own nonce.
                        // (mainly so that during construction nonce can be made non-zero,
                        // to "absorb" the gas cost of first nonce increment to 1st transaction (construction),
                        // not to 2nd transaction)
                        function incrementNonce(uint192 key) public override {
                            nonceSequenceNumber[msg.sender][key]++;
                        }
                        /**
                         * validate nonce uniqueness for this account.
                         * called just after validateUserOp()
                         */
                        function _validateAndUpdateNonce(address sender, uint256 nonce) internal returns (bool) {
                            uint192 key = uint192(nonce >> 64);
                            uint64 seq = uint64(nonce);
                            return nonceSequenceNumber[sender][key]++ == seq;
                        }
                    }
                    // SPDX-License-Identifier: GPL-3.0
                    pragma solidity ^0.8.12;
                    /**
                     * helper contract for EntryPoint, to call userOp.initCode from a "neutral" address,
                     * which is explicitly not the entryPoint itself.
                     */
                    contract SenderCreator {
                        /**
                         * call the "initCode" factory to create and return the sender account address
                         * @param initCode the initCode value from a UserOp. contains 20 bytes of factory address, followed by calldata
                         * @return sender the returned address of the created account, or zero address on failure.
                         */
                        function createSender(bytes calldata initCode) external returns (address sender) {
                            address factory = address(bytes20(initCode[0 : 20]));
                            bytes memory initCallData = initCode[20 :];
                            bool success;
                            /* solhint-disable no-inline-assembly */
                            assembly {
                                success := call(gas(), factory, 0, add(initCallData, 0x20), mload(initCallData), 0, 32)
                                sender := mload(0)
                            }
                            if (!success) {
                                sender = address(0);
                            }
                        }
                    }
                    // SPDX-License-Identifier: GPL-3.0-only
                    pragma solidity ^0.8.12;
                    import "../interfaces/IStakeManager.sol";
                    /* solhint-disable avoid-low-level-calls */
                    /* solhint-disable not-rely-on-time */
                    /**
                     * manage deposits and stakes.
                     * deposit is just a balance used to pay for UserOperations (either by a paymaster or an account)
                     * stake is value locked for at least "unstakeDelay" by a paymaster.
                     */
                    abstract contract StakeManager is IStakeManager {
                        /// maps paymaster to their deposits and stakes
                        mapping(address => DepositInfo) public deposits;
                        /// @inheritdoc IStakeManager
                        function getDepositInfo(address account) public view returns (DepositInfo memory info) {
                            return deposits[account];
                        }
                        // internal method to return just the stake info
                        function _getStakeInfo(address addr) internal view returns (StakeInfo memory info) {
                            DepositInfo storage depositInfo = deposits[addr];
                            info.stake = depositInfo.stake;
                            info.unstakeDelaySec = depositInfo.unstakeDelaySec;
                        }
                        /// return the deposit (for gas payment) of the account
                        function balanceOf(address account) public view returns (uint256) {
                            return deposits[account].deposit;
                        }
                        receive() external payable {
                            depositTo(msg.sender);
                        }
                        function _incrementDeposit(address account, uint256 amount) internal {
                            DepositInfo storage info = deposits[account];
                            uint256 newAmount = info.deposit + amount;
                            require(newAmount <= type(uint112).max, "deposit overflow");
                            info.deposit = uint112(newAmount);
                        }
                        /**
                         * add to the deposit of the given account
                         */
                        function depositTo(address account) public payable {
                            _incrementDeposit(account, msg.value);
                            DepositInfo storage info = deposits[account];
                            emit Deposited(account, info.deposit);
                        }
                        /**
                         * add to the account's stake - amount and delay
                         * any pending unstake is first cancelled.
                         * @param unstakeDelaySec the new lock duration before the deposit can be withdrawn.
                         */
                        function addStake(uint32 unstakeDelaySec) public payable {
                            DepositInfo storage info = deposits[msg.sender];
                            require(unstakeDelaySec > 0, "must specify unstake delay");
                            require(unstakeDelaySec >= info.unstakeDelaySec, "cannot decrease unstake time");
                            uint256 stake = info.stake + msg.value;
                            require(stake > 0, "no stake specified");
                            require(stake <= type(uint112).max, "stake overflow");
                            deposits[msg.sender] = DepositInfo(
                                info.deposit,
                                true,
                                uint112(stake),
                                unstakeDelaySec,
                                0
                            );
                            emit StakeLocked(msg.sender, stake, unstakeDelaySec);
                        }
                        /**
                         * attempt to unlock the stake.
                         * the value can be withdrawn (using withdrawStake) after the unstake delay.
                         */
                        function unlockStake() external {
                            DepositInfo storage info = deposits[msg.sender];
                            require(info.unstakeDelaySec != 0, "not staked");
                            require(info.staked, "already unstaking");
                            uint48 withdrawTime = uint48(block.timestamp) + info.unstakeDelaySec;
                            info.withdrawTime = withdrawTime;
                            info.staked = false;
                            emit StakeUnlocked(msg.sender, withdrawTime);
                        }
                        /**
                         * withdraw from the (unlocked) stake.
                         * must first call unlockStake and wait for the unstakeDelay to pass
                         * @param withdrawAddress the address to send withdrawn value.
                         */
                        function withdrawStake(address payable withdrawAddress) external {
                            DepositInfo storage info = deposits[msg.sender];
                            uint256 stake = info.stake;
                            require(stake > 0, "No stake to withdraw");
                            require(info.withdrawTime > 0, "must call unlockStake() first");
                            require(info.withdrawTime <= block.timestamp, "Stake withdrawal is not due");
                            info.unstakeDelaySec = 0;
                            info.withdrawTime = 0;
                            info.stake = 0;
                            emit StakeWithdrawn(msg.sender, withdrawAddress, stake);
                            (bool success,) = withdrawAddress.call{value : stake}("");
                            require(success, "failed to withdraw stake");
                        }
                        /**
                         * withdraw from the deposit.
                         * @param withdrawAddress the address to send withdrawn value.
                         * @param withdrawAmount the amount to withdraw.
                         */
                        function withdrawTo(address payable withdrawAddress, uint256 withdrawAmount) external {
                            DepositInfo storage info = deposits[msg.sender];
                            require(withdrawAmount <= info.deposit, "Withdraw amount too large");
                            info.deposit = uint112(info.deposit - withdrawAmount);
                            emit Withdrawn(msg.sender, withdrawAddress, withdrawAmount);
                            (bool success,) = withdrawAddress.call{value : withdrawAmount}("");
                            require(success, "failed to withdraw");
                        }
                    }
                    // SPDX-License-Identifier: GPL-3.0
                    pragma solidity ^0.8.12;
                    import "./UserOperation.sol";
                    interface IAccount {
                        /**
                         * Validate user's signature and nonce
                         * the entryPoint will make the call to the recipient only if this validation call returns successfully.
                         * signature failure should be reported by returning SIG_VALIDATION_FAILED (1).
                         * This allows making a "simulation call" without a valid signature
                         * Other failures (e.g. nonce mismatch, or invalid signature format) should still revert to signal failure.
                         *
                         * @dev Must validate caller is the entryPoint.
                         *      Must validate the signature and nonce
                         * @param userOp the operation that is about to be executed.
                         * @param userOpHash hash of the user's request data. can be used as the basis for signature.
                         * @param missingAccountFunds missing funds on the account's deposit in the entrypoint.
                         *      This is the minimum amount to transfer to the sender(entryPoint) to be able to make the call.
                         *      The excess is left as a deposit in the entrypoint, for future calls.
                         *      can be withdrawn anytime using "entryPoint.withdrawTo()"
                         *      In case there is a paymaster in the request (or the current deposit is high enough), this value will be zero.
                         * @return validationData packaged ValidationData structure. use `_packValidationData` and `_unpackValidationData` to encode and decode
                         *      <20-byte> sigAuthorizer - 0 for valid signature, 1 to mark signature failure,
                         *         otherwise, an address of an "authorizer" contract.
                         *      <6-byte> validUntil - last timestamp this operation is valid. 0 for "indefinite"
                         *      <6-byte> validAfter - first timestamp this operation is valid
                         *      If an account doesn't use time-range, it is enough to return SIG_VALIDATION_FAILED value (1) for signature failure.
                         *      Note that the validation code cannot use block.timestamp (or block.number) directly.
                         */
                        function validateUserOp(UserOperation calldata userOp, bytes32 userOpHash, uint256 missingAccountFunds)
                        external returns (uint256 validationData);
                    }
                    // SPDX-License-Identifier: GPL-3.0
                    pragma solidity ^0.8.12;
                    import "./UserOperation.sol";
                    /**
                     * Aggregated Signatures validator.
                     */
                    interface IAggregator {
                        /**
                         * validate aggregated signature.
                         * revert if the aggregated signature does not match the given list of operations.
                         */
                        function validateSignatures(UserOperation[] calldata userOps, bytes calldata signature) external view;
                        /**
                         * validate signature of a single userOp
                         * This method is should be called by bundler after EntryPoint.simulateValidation() returns (reverts) with ValidationResultWithAggregation
                         * First it validates the signature over the userOp. Then it returns data to be used when creating the handleOps.
                         * @param userOp the userOperation received from the user.
                         * @return sigForUserOp the value to put into the signature field of the userOp when calling handleOps.
                         *    (usually empty, unless account and aggregator support some kind of "multisig"
                         */
                        function validateUserOpSignature(UserOperation calldata userOp)
                        external view returns (bytes memory sigForUserOp);
                        /**
                         * aggregate multiple signatures into a single value.
                         * This method is called off-chain to calculate the signature to pass with handleOps()
                         * bundler MAY use optimized custom code perform this aggregation
                         * @param userOps array of UserOperations to collect the signatures from.
                         * @return aggregatedSignature the aggregated signature
                         */
                        function aggregateSignatures(UserOperation[] calldata userOps) external view returns (bytes memory aggregatedSignature);
                    }
                    /**
                     ** Account-Abstraction (EIP-4337) singleton EntryPoint implementation.
                     ** Only one instance required on each chain.
                     **/
                    // SPDX-License-Identifier: GPL-3.0
                    pragma solidity ^0.8.12;
                    /* solhint-disable avoid-low-level-calls */
                    /* solhint-disable no-inline-assembly */
                    /* solhint-disable reason-string */
                    import "./UserOperation.sol";
                    import "./IStakeManager.sol";
                    import "./IAggregator.sol";
                    import "./INonceManager.sol";
                    interface IEntryPoint is IStakeManager, INonceManager {
                        /***
                         * An event emitted after each successful request
                         * @param userOpHash - unique identifier for the request (hash its entire content, except signature).
                         * @param sender - the account that generates this request.
                         * @param paymaster - if non-null, the paymaster that pays for this request.
                         * @param nonce - the nonce value from the request.
                         * @param success - true if the sender transaction succeeded, false if reverted.
                         * @param actualGasCost - actual amount paid (by account or paymaster) for this UserOperation.
                         * @param actualGasUsed - total gas used by this UserOperation (including preVerification, creation, validation and execution).
                         */
                        event UserOperationEvent(bytes32 indexed userOpHash, address indexed sender, address indexed paymaster, uint256 nonce, bool success, uint256 actualGasCost, uint256 actualGasUsed);
                        /**
                         * account "sender" was deployed.
                         * @param userOpHash the userOp that deployed this account. UserOperationEvent will follow.
                         * @param sender the account that is deployed
                         * @param factory the factory used to deploy this account (in the initCode)
                         * @param paymaster the paymaster used by this UserOp
                         */
                        event AccountDeployed(bytes32 indexed userOpHash, address indexed sender, address factory, address paymaster);
                        /**
                         * An event emitted if the UserOperation "callData" reverted with non-zero length
                         * @param userOpHash the request unique identifier.
                         * @param sender the sender of this request
                         * @param nonce the nonce used in the request
                         * @param revertReason - the return bytes from the (reverted) call to "callData".
                         */
                        event UserOperationRevertReason(bytes32 indexed userOpHash, address indexed sender, uint256 nonce, bytes revertReason);
                        /**
                         * an event emitted by handleOps(), before starting the execution loop.
                         * any event emitted before this event, is part of the validation.
                         */
                        event BeforeExecution();
                        /**
                         * signature aggregator used by the following UserOperationEvents within this bundle.
                         */
                        event SignatureAggregatorChanged(address indexed aggregator);
                        /**
                         * a custom revert error of handleOps, to identify the offending op.
                         *  NOTE: if simulateValidation passes successfully, there should be no reason for handleOps to fail on it.
                         *  @param opIndex - index into the array of ops to the failed one (in simulateValidation, this is always zero)
                         *  @param reason - revert reason
                         *      The string starts with a unique code "AAmn", where "m" is "1" for factory, "2" for account and "3" for paymaster issues,
                         *      so a failure can be attributed to the correct entity.
                         *   Should be caught in off-chain handleOps simulation and not happen on-chain.
                         *   Useful for mitigating DoS attempts against batchers or for troubleshooting of factory/account/paymaster reverts.
                         */
                        error FailedOp(uint256 opIndex, string reason);
                        /**
                         * error case when a signature aggregator fails to verify the aggregated signature it had created.
                         */
                        error SignatureValidationFailed(address aggregator);
                        /**
                         * Successful result from simulateValidation.
                         * @param returnInfo gas and time-range returned values
                         * @param senderInfo stake information about the sender
                         * @param factoryInfo stake information about the factory (if any)
                         * @param paymasterInfo stake information about the paymaster (if any)
                         */
                        error ValidationResult(ReturnInfo returnInfo,
                            StakeInfo senderInfo, StakeInfo factoryInfo, StakeInfo paymasterInfo);
                        /**
                         * Successful result from simulateValidation, if the account returns a signature aggregator
                         * @param returnInfo gas and time-range returned values
                         * @param senderInfo stake information about the sender
                         * @param factoryInfo stake information about the factory (if any)
                         * @param paymasterInfo stake information about the paymaster (if any)
                         * @param aggregatorInfo signature aggregation info (if the account requires signature aggregator)
                         *      bundler MUST use it to verify the signature, or reject the UserOperation
                         */
                        error ValidationResultWithAggregation(ReturnInfo returnInfo,
                            StakeInfo senderInfo, StakeInfo factoryInfo, StakeInfo paymasterInfo,
                            AggregatorStakeInfo aggregatorInfo);
                        /**
                         * return value of getSenderAddress
                         */
                        error SenderAddressResult(address sender);
                        /**
                         * return value of simulateHandleOp
                         */
                        error ExecutionResult(uint256 preOpGas, uint256 paid, uint48 validAfter, uint48 validUntil, bool targetSuccess, bytes targetResult);
                        //UserOps handled, per aggregator
                        struct UserOpsPerAggregator {
                            UserOperation[] userOps;
                            // aggregator address
                            IAggregator aggregator;
                            // aggregated signature
                            bytes signature;
                        }
                        /**
                         * Execute a batch of UserOperation.
                         * no signature aggregator is used.
                         * if any account requires an aggregator (that is, it returned an aggregator when
                         * performing simulateValidation), then handleAggregatedOps() must be used instead.
                         * @param ops the operations to execute
                         * @param beneficiary the address to receive the fees
                         */
                        function handleOps(UserOperation[] calldata ops, address payable beneficiary) external;
                        /**
                         * Execute a batch of UserOperation with Aggregators
                         * @param opsPerAggregator the operations to execute, grouped by aggregator (or address(0) for no-aggregator accounts)
                         * @param beneficiary the address to receive the fees
                         */
                        function handleAggregatedOps(
                            UserOpsPerAggregator[] calldata opsPerAggregator,
                            address payable beneficiary
                        ) external;
                        /**
                         * generate a request Id - unique identifier for this request.
                         * the request ID is a hash over the content of the userOp (except the signature), the entrypoint and the chainid.
                         */
                        function getUserOpHash(UserOperation calldata userOp) external view returns (bytes32);
                        /**
                         * Simulate a call to account.validateUserOp and paymaster.validatePaymasterUserOp.
                         * @dev this method always revert. Successful result is ValidationResult error. other errors are failures.
                         * @dev The node must also verify it doesn't use banned opcodes, and that it doesn't reference storage outside the account's data.
                         * @param userOp the user operation to validate.
                         */
                        function simulateValidation(UserOperation calldata userOp) external;
                        /**
                         * gas and return values during simulation
                         * @param preOpGas the gas used for validation (including preValidationGas)
                         * @param prefund the required prefund for this operation
                         * @param sigFailed validateUserOp's (or paymaster's) signature check failed
                         * @param validAfter - first timestamp this UserOp is valid (merging account and paymaster time-range)
                         * @param validUntil - last timestamp this UserOp is valid (merging account and paymaster time-range)
                         * @param paymasterContext returned by validatePaymasterUserOp (to be passed into postOp)
                         */
                        struct ReturnInfo {
                            uint256 preOpGas;
                            uint256 prefund;
                            bool sigFailed;
                            uint48 validAfter;
                            uint48 validUntil;
                            bytes paymasterContext;
                        }
                        /**
                         * returned aggregated signature info.
                         * the aggregator returned by the account, and its current stake.
                         */
                        struct AggregatorStakeInfo {
                            address aggregator;
                            StakeInfo stakeInfo;
                        }
                        /**
                         * Get counterfactual sender address.
                         *  Calculate the sender contract address that will be generated by the initCode and salt in the UserOperation.
                         * this method always revert, and returns the address in SenderAddressResult error
                         * @param initCode the constructor code to be passed into the UserOperation.
                         */
                        function getSenderAddress(bytes memory initCode) external;
                        /**
                         * simulate full execution of a UserOperation (including both validation and target execution)
                         * this method will always revert with "ExecutionResult".
                         * it performs full validation of the UserOperation, but ignores signature error.
                         * an optional target address is called after the userop succeeds, and its value is returned
                         * (before the entire call is reverted)
                         * Note that in order to collect the the success/failure of the target call, it must be executed
                         * with trace enabled to track the emitted events.
                         * @param op the UserOperation to simulate
                         * @param target if nonzero, a target address to call after userop simulation. If called, the targetSuccess and targetResult
                         *        are set to the return from that call.
                         * @param targetCallData callData to pass to target address
                         */
                        function simulateHandleOp(UserOperation calldata op, address target, bytes calldata targetCallData) external;
                    }
                    // SPDX-License-Identifier: GPL-3.0
                    pragma solidity ^0.8.12;
                    interface INonceManager {
                        /**
                         * Return the next nonce for this sender.
                         * Within a given key, the nonce values are sequenced (starting with zero, and incremented by one on each userop)
                         * But UserOp with different keys can come with arbitrary order.
                         *
                         * @param sender the account address
                         * @param key the high 192 bit of the nonce
                         * @return nonce a full nonce to pass for next UserOp with this sender.
                         */
                        function getNonce(address sender, uint192 key)
                        external view returns (uint256 nonce);
                        /**
                         * Manually increment the nonce of the sender.
                         * This method is exposed just for completeness..
                         * Account does NOT need to call it, neither during validation, nor elsewhere,
                         * as the EntryPoint will update the nonce regardless.
                         * Possible use-case is call it with various keys to "initialize" their nonces to one, so that future
                         * UserOperations will not pay extra for the first transaction with a given key.
                         */
                        function incrementNonce(uint192 key) external;
                    }
                    // SPDX-License-Identifier: GPL-3.0
                    pragma solidity ^0.8.12;
                    import "./UserOperation.sol";
                    /**
                     * the interface exposed by a paymaster contract, who agrees to pay the gas for user's operations.
                     * a paymaster must hold a stake to cover the required entrypoint stake and also the gas for the transaction.
                     */
                    interface IPaymaster {
                        enum PostOpMode {
                            opSucceeded, // user op succeeded
                            opReverted, // user op reverted. still has to pay for gas.
                            postOpReverted //user op succeeded, but caused postOp to revert. Now it's a 2nd call, after user's op was deliberately reverted.
                        }
                        /**
                         * payment validation: check if paymaster agrees to pay.
                         * Must verify sender is the entryPoint.
                         * Revert to reject this request.
                         * Note that bundlers will reject this method if it changes the state, unless the paymaster is trusted (whitelisted)
                         * The paymaster pre-pays using its deposit, and receive back a refund after the postOp method returns.
                         * @param userOp the user operation
                         * @param userOpHash hash of the user's request data.
                         * @param maxCost the maximum cost of this transaction (based on maximum gas and gas price from userOp)
                         * @return context value to send to a postOp
                         *      zero length to signify postOp is not required.
                         * @return validationData signature and time-range of this operation, encoded the same as the return value of validateUserOperation
                         *      <20-byte> sigAuthorizer - 0 for valid signature, 1 to mark signature failure,
                         *         otherwise, an address of an "authorizer" contract.
                         *      <6-byte> validUntil - last timestamp this operation is valid. 0 for "indefinite"
                         *      <6-byte> validAfter - first timestamp this operation is valid
                         *      Note that the validation code cannot use block.timestamp (or block.number) directly.
                         */
                        function validatePaymasterUserOp(UserOperation calldata userOp, bytes32 userOpHash, uint256 maxCost)
                        external returns (bytes memory context, uint256 validationData);
                        /**
                         * post-operation handler.
                         * Must verify sender is the entryPoint
                         * @param mode enum with the following options:
                         *      opSucceeded - user operation succeeded.
                         *      opReverted  - user op reverted. still has to pay for gas.
                         *      postOpReverted - user op succeeded, but caused postOp (in mode=opSucceeded) to revert.
                         *                       Now this is the 2nd call, after user's op was deliberately reverted.
                         * @param context - the context value returned by validatePaymasterUserOp
                         * @param actualGasCost - actual gas used so far (without this postOp call).
                         */
                        function postOp(PostOpMode mode, bytes calldata context, uint256 actualGasCost) external;
                    }
                    // SPDX-License-Identifier: GPL-3.0-only
                    pragma solidity ^0.8.12;
                    /**
                     * manage deposits and stakes.
                     * deposit is just a balance used to pay for UserOperations (either by a paymaster or an account)
                     * stake is value locked for at least "unstakeDelay" by the staked entity.
                     */
                    interface IStakeManager {
                        event Deposited(
                            address indexed account,
                            uint256 totalDeposit
                        );
                        event Withdrawn(
                            address indexed account,
                            address withdrawAddress,
                            uint256 amount
                        );
                        /// Emitted when stake or unstake delay are modified
                        event StakeLocked(
                            address indexed account,
                            uint256 totalStaked,
                            uint256 unstakeDelaySec
                        );
                        /// Emitted once a stake is scheduled for withdrawal
                        event StakeUnlocked(
                            address indexed account,
                            uint256 withdrawTime
                        );
                        event StakeWithdrawn(
                            address indexed account,
                            address withdrawAddress,
                            uint256 amount
                        );
                        /**
                         * @param deposit the entity's deposit
                         * @param staked true if this entity is staked.
                         * @param stake actual amount of ether staked for this entity.
                         * @param unstakeDelaySec minimum delay to withdraw the stake.
                         * @param withdrawTime - first block timestamp where 'withdrawStake' will be callable, or zero if already locked
                         * @dev sizes were chosen so that (deposit,staked, stake) fit into one cell (used during handleOps)
                         *    and the rest fit into a 2nd cell.
                         *    112 bit allows for 10^15 eth
                         *    48 bit for full timestamp
                         *    32 bit allows 150 years for unstake delay
                         */
                        struct DepositInfo {
                            uint112 deposit;
                            bool staked;
                            uint112 stake;
                            uint32 unstakeDelaySec;
                            uint48 withdrawTime;
                        }
                        //API struct used by getStakeInfo and simulateValidation
                        struct StakeInfo {
                            uint256 stake;
                            uint256 unstakeDelaySec;
                        }
                        /// @return info - full deposit information of given account
                        function getDepositInfo(address account) external view returns (DepositInfo memory info);
                        /// @return the deposit (for gas payment) of the account
                        function balanceOf(address account) external view returns (uint256);
                        /**
                         * add to the deposit of the given account
                         */
                        function depositTo(address account) external payable;
                        /**
                         * add to the account's stake - amount and delay
                         * any pending unstake is first cancelled.
                         * @param _unstakeDelaySec the new lock duration before the deposit can be withdrawn.
                         */
                        function addStake(uint32 _unstakeDelaySec) external payable;
                        /**
                         * attempt to unlock the stake.
                         * the value can be withdrawn (using withdrawStake) after the unstake delay.
                         */
                        function unlockStake() external;
                        /**
                         * withdraw from the (unlocked) stake.
                         * must first call unlockStake and wait for the unstakeDelay to pass
                         * @param withdrawAddress the address to send withdrawn value.
                         */
                        function withdrawStake(address payable withdrawAddress) external;
                        /**
                         * withdraw from the deposit.
                         * @param withdrawAddress the address to send withdrawn value.
                         * @param withdrawAmount the amount to withdraw.
                         */
                        function withdrawTo(address payable withdrawAddress, uint256 withdrawAmount) external;
                    }
                    // SPDX-License-Identifier: GPL-3.0
                    pragma solidity ^0.8.12;
                    /* solhint-disable no-inline-assembly */
                    import {calldataKeccak} from "../core/Helpers.sol";
                    /**
                     * User Operation struct
                     * @param sender the sender account of this request.
                         * @param nonce unique value the sender uses to verify it is not a replay.
                         * @param initCode if set, the account contract will be created by this constructor/
                         * @param callData the method call to execute on this account.
                         * @param callGasLimit the gas limit passed to the callData method call.
                         * @param verificationGasLimit gas used for validateUserOp and validatePaymasterUserOp.
                         * @param preVerificationGas gas not calculated by the handleOps method, but added to the gas paid. Covers batch overhead.
                         * @param maxFeePerGas same as EIP-1559 gas parameter.
                         * @param maxPriorityFeePerGas same as EIP-1559 gas parameter.
                         * @param paymasterAndData if set, this field holds the paymaster address and paymaster-specific data. the paymaster will pay for the transaction instead of the sender.
                         * @param signature sender-verified signature over the entire request, the EntryPoint address and the chain ID.
                         */
                        struct UserOperation {
                            address sender;
                            uint256 nonce;
                            bytes initCode;
                            bytes callData;
                            uint256 callGasLimit;
                            uint256 verificationGasLimit;
                            uint256 preVerificationGas;
                            uint256 maxFeePerGas;
                            uint256 maxPriorityFeePerGas;
                            bytes paymasterAndData;
                            bytes signature;
                        }
                    /**
                     * Utility functions helpful when working with UserOperation structs.
                     */
                    library UserOperationLib {
                        function getSender(UserOperation calldata userOp) internal pure returns (address) {
                            address data;
                            //read sender from userOp, which is first userOp member (saves 800 gas...)
                            assembly {data := calldataload(userOp)}
                            return address(uint160(data));
                        }
                        //relayer/block builder might submit the TX with higher priorityFee, but the user should not
                        // pay above what he signed for.
                        function gasPrice(UserOperation calldata userOp) internal view returns (uint256) {
                        unchecked {
                            uint256 maxFeePerGas = userOp.maxFeePerGas;
                            uint256 maxPriorityFeePerGas = userOp.maxPriorityFeePerGas;
                            if (maxFeePerGas == maxPriorityFeePerGas) {
                                //legacy mode (for networks that don't support basefee opcode)
                                return maxFeePerGas;
                            }
                            return min(maxFeePerGas, maxPriorityFeePerGas + block.basefee);
                        }
                        }
                        function pack(UserOperation calldata userOp) internal pure returns (bytes memory ret) {
                            address sender = getSender(userOp);
                            uint256 nonce = userOp.nonce;
                            bytes32 hashInitCode = calldataKeccak(userOp.initCode);
                            bytes32 hashCallData = calldataKeccak(userOp.callData);
                            uint256 callGasLimit = userOp.callGasLimit;
                            uint256 verificationGasLimit = userOp.verificationGasLimit;
                            uint256 preVerificationGas = userOp.preVerificationGas;
                            uint256 maxFeePerGas = userOp.maxFeePerGas;
                            uint256 maxPriorityFeePerGas = userOp.maxPriorityFeePerGas;
                            bytes32 hashPaymasterAndData = calldataKeccak(userOp.paymasterAndData);
                            return abi.encode(
                                sender, nonce,
                                hashInitCode, hashCallData,
                                callGasLimit, verificationGasLimit, preVerificationGas,
                                maxFeePerGas, maxPriorityFeePerGas,
                                hashPaymasterAndData
                            );
                        }
                        function hash(UserOperation calldata userOp) internal pure returns (bytes32) {
                            return keccak256(pack(userOp));
                        }
                        function min(uint256 a, uint256 b) internal pure returns (uint256) {
                            return a < b ? a : b;
                        }
                    }
                    // SPDX-License-Identifier: LGPL-3.0-only
                    pragma solidity >=0.7.5 <0.9.0;
                    // solhint-disable no-inline-assembly
                    /**
                     * Utility functions helpful when making different kinds of contract calls in Solidity.
                     */
                    library Exec {
                        function call(
                            address to,
                            uint256 value,
                            bytes memory data,
                            uint256 txGas
                        ) internal returns (bool success) {
                            assembly {
                                success := call(txGas, to, value, add(data, 0x20), mload(data), 0, 0)
                            }
                        }
                        function staticcall(
                            address to,
                            bytes memory data,
                            uint256 txGas
                        ) internal view returns (bool success) {
                            assembly {
                                success := staticcall(txGas, to, add(data, 0x20), mload(data), 0, 0)
                            }
                        }
                        function delegateCall(
                            address to,
                            bytes memory data,
                            uint256 txGas
                        ) internal returns (bool success) {
                            assembly {
                                success := delegatecall(txGas, to, add(data, 0x20), mload(data), 0, 0)
                            }
                        }
                        // get returned data from last call or calldelegate
                        function getReturnData(uint256 maxLen) internal pure returns (bytes memory returnData) {
                            assembly {
                                let len := returndatasize()
                                if gt(len, maxLen) {
                                    len := maxLen
                                }
                                let ptr := mload(0x40)
                                mstore(0x40, add(ptr, add(len, 0x20)))
                                mstore(ptr, len)
                                returndatacopy(add(ptr, 0x20), 0, len)
                                returnData := ptr
                            }
                        }
                        // revert with explicit byte array (probably reverted info from call)
                        function revertWithData(bytes memory returnData) internal pure {
                            assembly {
                                revert(add(returnData, 32), mload(returnData))
                            }
                        }
                        function callAndRevert(address to, bytes memory data, uint256 maxLen) internal {
                            bool success = call(to,0,data,gasleft());
                            if (!success) {
                                revertWithData(getReturnData(maxLen));
                            }
                        }
                    }
                    

                    File 2 of 6: TransparentUpgradeableProxy
                    // SPDX-License-Identifier: MIT
                    // OpenZeppelin Contracts (last updated v4.9.0) (proxy/transparent/TransparentUpgradeableProxy.sol)
                    pragma solidity ^0.8.0;
                    import "../ERC1967/ERC1967Proxy.sol";
                    /**
                     * @dev Interface for {TransparentUpgradeableProxy}. In order to implement transparency, {TransparentUpgradeableProxy}
                     * does not implement this interface directly, and some of its functions are implemented by an internal dispatch
                     * mechanism. The compiler is unaware that these functions are implemented by {TransparentUpgradeableProxy} and will not
                     * include them in the ABI so this interface must be used to interact with it.
                     */
                    interface ITransparentUpgradeableProxy is IERC1967 {
                        function admin() external view returns (address);
                        function implementation() external view returns (address);
                        function changeAdmin(address) external;
                        function upgradeTo(address) external;
                        function upgradeToAndCall(address, bytes memory) external payable;
                    }
                    /**
                     * @dev This contract implements a proxy that is upgradeable by an admin.
                     *
                     * To avoid https://medium.com/nomic-labs-blog/malicious-backdoors-in-ethereum-proxies-62629adf3357[proxy selector
                     * clashing], which can potentially be used in an attack, this contract uses the
                     * https://blog.openzeppelin.com/the-transparent-proxy-pattern/[transparent proxy pattern]. This pattern implies two
                     * things that go hand in hand:
                     *
                     * 1. If any account other than the admin calls the proxy, the call will be forwarded to the implementation, even if
                     * that call matches one of the admin functions exposed by the proxy itself.
                     * 2. If the admin calls the proxy, it can access the admin functions, but its calls will never be forwarded to the
                     * implementation. If the admin tries to call a function on the implementation it will fail with an error that says
                     * "admin cannot fallback to proxy target".
                     *
                     * These properties mean that the admin account can only be used for admin actions like upgrading the proxy or changing
                     * the admin, so it's best if it's a dedicated account that is not used for anything else. This will avoid headaches due
                     * to sudden errors when trying to call a function from the proxy implementation.
                     *
                     * Our recommendation is for the dedicated account to be an instance of the {ProxyAdmin} contract. If set up this way,
                     * you should think of the `ProxyAdmin` instance as the real administrative interface of your proxy.
                     *
                     * NOTE: The real interface of this proxy is that defined in `ITransparentUpgradeableProxy`. This contract does not
                     * inherit from that interface, and instead the admin functions are implicitly implemented using a custom dispatch
                     * mechanism in `_fallback`. Consequently, the compiler will not produce an ABI for this contract. This is necessary to
                     * fully implement transparency without decoding reverts caused by selector clashes between the proxy and the
                     * implementation.
                     *
                     * WARNING: It is not recommended to extend this contract to add additional external functions. If you do so, the compiler
                     * will not check that there are no selector conflicts, due to the note above. A selector clash between any new function
                     * and the functions declared in {ITransparentUpgradeableProxy} will be resolved in favor of the new one. This could
                     * render the admin operations inaccessible, which could prevent upgradeability. Transparency may also be compromised.
                     */
                    contract TransparentUpgradeableProxy is ERC1967Proxy {
                        /**
                         * @dev Initializes an upgradeable proxy managed by `_admin`, backed by the implementation at `_logic`, and
                         * optionally initialized with `_data` as explained in {ERC1967Proxy-constructor}.
                         */
                        constructor(address _logic, address admin_, bytes memory _data) payable ERC1967Proxy(_logic, _data) {
                            _changeAdmin(admin_);
                        }
                        /**
                         * @dev Modifier used internally that will delegate the call to the implementation unless the sender is the admin.
                         *
                         * CAUTION: This modifier is deprecated, as it could cause issues if the modified function has arguments, and the
                         * implementation provides a function with the same selector.
                         */
                        modifier ifAdmin() {
                            if (msg.sender == _getAdmin()) {
                                _;
                            } else {
                                _fallback();
                            }
                        }
                        /**
                         * @dev If caller is the admin process the call internally, otherwise transparently fallback to the proxy behavior
                         */
                        function _fallback() internal virtual override {
                            if (msg.sender == _getAdmin()) {
                                bytes memory ret;
                                bytes4 selector = msg.sig;
                                if (selector == ITransparentUpgradeableProxy.upgradeTo.selector) {
                                    ret = _dispatchUpgradeTo();
                                } else if (selector == ITransparentUpgradeableProxy.upgradeToAndCall.selector) {
                                    ret = _dispatchUpgradeToAndCall();
                                } else if (selector == ITransparentUpgradeableProxy.changeAdmin.selector) {
                                    ret = _dispatchChangeAdmin();
                                } else if (selector == ITransparentUpgradeableProxy.admin.selector) {
                                    ret = _dispatchAdmin();
                                } else if (selector == ITransparentUpgradeableProxy.implementation.selector) {
                                    ret = _dispatchImplementation();
                                } else {
                                    revert("TransparentUpgradeableProxy: admin cannot fallback to proxy target");
                                }
                                assembly {
                                    return(add(ret, 0x20), mload(ret))
                                }
                            } else {
                                super._fallback();
                            }
                        }
                        /**
                         * @dev Returns the current admin.
                         *
                         * TIP: To get this value clients can read directly from the storage slot shown below (specified by EIP1967) using the
                         * https://eth.wiki/json-rpc/API#eth_getstorageat[`eth_getStorageAt`] RPC call.
                         * `0xb53127684a568b3173ae13b9f8a6016e243e63b6e8ee1178d6a717850b5d6103`
                         */
                        function _dispatchAdmin() private returns (bytes memory) {
                            _requireZeroValue();
                            address admin = _getAdmin();
                            return abi.encode(admin);
                        }
                        /**
                         * @dev Returns the current implementation.
                         *
                         * TIP: To get this value clients can read directly from the storage slot shown below (specified by EIP1967) using the
                         * https://eth.wiki/json-rpc/API#eth_getstorageat[`eth_getStorageAt`] RPC call.
                         * `0x360894a13ba1a3210667c828492db98dca3e2076cc3735a920a3ca505d382bbc`
                         */
                        function _dispatchImplementation() private returns (bytes memory) {
                            _requireZeroValue();
                            address implementation = _implementation();
                            return abi.encode(implementation);
                        }
                        /**
                         * @dev Changes the admin of the proxy.
                         *
                         * Emits an {AdminChanged} event.
                         */
                        function _dispatchChangeAdmin() private returns (bytes memory) {
                            _requireZeroValue();
                            address newAdmin = abi.decode(msg.data[4:], (address));
                            _changeAdmin(newAdmin);
                            return "";
                        }
                        /**
                         * @dev Upgrade the implementation of the proxy.
                         */
                        function _dispatchUpgradeTo() private returns (bytes memory) {
                            _requireZeroValue();
                            address newImplementation = abi.decode(msg.data[4:], (address));
                            _upgradeToAndCall(newImplementation, bytes(""), false);
                            return "";
                        }
                        /**
                         * @dev Upgrade the implementation of the proxy, and then call a function from the new implementation as specified
                         * by `data`, which should be an encoded function call. This is useful to initialize new storage variables in the
                         * proxied contract.
                         */
                        function _dispatchUpgradeToAndCall() private returns (bytes memory) {
                            (address newImplementation, bytes memory data) = abi.decode(msg.data[4:], (address, bytes));
                            _upgradeToAndCall(newImplementation, data, true);
                            return "";
                        }
                        /**
                         * @dev Returns the current admin.
                         *
                         * CAUTION: This function is deprecated. Use {ERC1967Upgrade-_getAdmin} instead.
                         */
                        function _admin() internal view virtual returns (address) {
                            return _getAdmin();
                        }
                        /**
                         * @dev To keep this contract fully transparent, all `ifAdmin` functions must be payable. This helper is here to
                         * emulate some proxy functions being non-payable while still allowing value to pass through.
                         */
                        function _requireZeroValue() private {
                            require(msg.value == 0);
                        }
                    }
                    // SPDX-License-Identifier: MIT
                    // OpenZeppelin Contracts (last updated v4.7.0) (proxy/ERC1967/ERC1967Proxy.sol)
                    pragma solidity ^0.8.0;
                    import "../Proxy.sol";
                    import "./ERC1967Upgrade.sol";
                    /**
                     * @dev This contract implements an upgradeable proxy. It is upgradeable because calls are delegated to an
                     * implementation address that can be changed. This address is stored in storage in the location specified by
                     * https://eips.ethereum.org/EIPS/eip-1967[EIP1967], so that it doesn't conflict with the storage layout of the
                     * implementation behind the proxy.
                     */
                    contract ERC1967Proxy is Proxy, ERC1967Upgrade {
                        /**
                         * @dev Initializes the upgradeable proxy with an initial implementation specified by `_logic`.
                         *
                         * If `_data` is nonempty, it's used as data in a delegate call to `_logic`. This will typically be an encoded
                         * function call, and allows initializing the storage of the proxy like a Solidity constructor.
                         */
                        constructor(address _logic, bytes memory _data) payable {
                            _upgradeToAndCall(_logic, _data, false);
                        }
                        /**
                         * @dev Returns the current implementation address.
                         */
                        function _implementation() internal view virtual override returns (address impl) {
                            return ERC1967Upgrade._getImplementation();
                        }
                    }
                    // SPDX-License-Identifier: MIT
                    // OpenZeppelin Contracts (last updated v4.6.0) (proxy/Proxy.sol)
                    pragma solidity ^0.8.0;
                    /**
                     * @dev This abstract contract provides a fallback function that delegates all calls to another contract using the EVM
                     * instruction `delegatecall`. We refer to the second contract as the _implementation_ behind the proxy, and it has to
                     * be specified by overriding the virtual {_implementation} function.
                     *
                     * Additionally, delegation to the implementation can be triggered manually through the {_fallback} function, or to a
                     * different contract through the {_delegate} function.
                     *
                     * The success and return data of the delegated call will be returned back to the caller of the proxy.
                     */
                    abstract contract Proxy {
                        /**
                         * @dev Delegates the current call to `implementation`.
                         *
                         * This function does not return to its internal call site, it will return directly to the external caller.
                         */
                        function _delegate(address implementation) internal virtual {
                            assembly {
                                // Copy msg.data. We take full control of memory in this inline assembly
                                // block because it will not return to Solidity code. We overwrite the
                                // Solidity scratch pad at memory position 0.
                                calldatacopy(0, 0, calldatasize())
                                // Call the implementation.
                                // out and outsize are 0 because we don't know the size yet.
                                let result := delegatecall(gas(), implementation, 0, calldatasize(), 0, 0)
                                // Copy the returned data.
                                returndatacopy(0, 0, returndatasize())
                                switch result
                                // delegatecall returns 0 on error.
                                case 0 {
                                    revert(0, returndatasize())
                                }
                                default {
                                    return(0, returndatasize())
                                }
                            }
                        }
                        /**
                         * @dev This is a virtual function that should be overridden so it returns the address to which the fallback function
                         * and {_fallback} should delegate.
                         */
                        function _implementation() internal view virtual returns (address);
                        /**
                         * @dev Delegates the current call to the address returned by `_implementation()`.
                         *
                         * This function does not return to its internal call site, it will return directly to the external caller.
                         */
                        function _fallback() internal virtual {
                            _beforeFallback();
                            _delegate(_implementation());
                        }
                        /**
                         * @dev Fallback function that delegates calls to the address returned by `_implementation()`. Will run if no other
                         * function in the contract matches the call data.
                         */
                        fallback() external payable virtual {
                            _fallback();
                        }
                        /**
                         * @dev Fallback function that delegates calls to the address returned by `_implementation()`. Will run if call data
                         * is empty.
                         */
                        receive() external payable virtual {
                            _fallback();
                        }
                        /**
                         * @dev Hook that is called before falling back to the implementation. Can happen as part of a manual `_fallback`
                         * call, or as part of the Solidity `fallback` or `receive` functions.
                         *
                         * If overridden should call `super._beforeFallback()`.
                         */
                        function _beforeFallback() internal virtual {}
                    }
                    // SPDX-License-Identifier: MIT
                    // OpenZeppelin Contracts (last updated v4.9.0) (proxy/ERC1967/ERC1967Upgrade.sol)
                    pragma solidity ^0.8.2;
                    import "../beacon/IBeacon.sol";
                    import "../../interfaces/IERC1967.sol";
                    import "../../interfaces/draft-IERC1822.sol";
                    import "../../utils/Address.sol";
                    import "../../utils/StorageSlot.sol";
                    /**
                     * @dev This abstract contract provides getters and event emitting update functions for
                     * https://eips.ethereum.org/EIPS/eip-1967[EIP1967] slots.
                     *
                     * _Available since v4.1._
                     */
                    abstract contract ERC1967Upgrade is IERC1967 {
                        // This is the keccak-256 hash of "eip1967.proxy.rollback" subtracted by 1
                        bytes32 private constant _ROLLBACK_SLOT = 0x4910fdfa16fed3260ed0e7147f7cc6da11a60208b5b9406d12a635614ffd9143;
                        /**
                         * @dev Storage slot with the address of the current implementation.
                         * This is the keccak-256 hash of "eip1967.proxy.implementation" subtracted by 1, and is
                         * validated in the constructor.
                         */
                        bytes32 internal constant _IMPLEMENTATION_SLOT = 0x360894a13ba1a3210667c828492db98dca3e2076cc3735a920a3ca505d382bbc;
                        /**
                         * @dev Returns the current implementation address.
                         */
                        function _getImplementation() internal view returns (address) {
                            return StorageSlot.getAddressSlot(_IMPLEMENTATION_SLOT).value;
                        }
                        /**
                         * @dev Stores a new address in the EIP1967 implementation slot.
                         */
                        function _setImplementation(address newImplementation) private {
                            require(Address.isContract(newImplementation), "ERC1967: new implementation is not a contract");
                            StorageSlot.getAddressSlot(_IMPLEMENTATION_SLOT).value = newImplementation;
                        }
                        /**
                         * @dev Perform implementation upgrade
                         *
                         * Emits an {Upgraded} event.
                         */
                        function _upgradeTo(address newImplementation) internal {
                            _setImplementation(newImplementation);
                            emit Upgraded(newImplementation);
                        }
                        /**
                         * @dev Perform implementation upgrade with additional setup call.
                         *
                         * Emits an {Upgraded} event.
                         */
                        function _upgradeToAndCall(address newImplementation, bytes memory data, bool forceCall) internal {
                            _upgradeTo(newImplementation);
                            if (data.length > 0 || forceCall) {
                                Address.functionDelegateCall(newImplementation, data);
                            }
                        }
                        /**
                         * @dev Perform implementation upgrade with security checks for UUPS proxies, and additional setup call.
                         *
                         * Emits an {Upgraded} event.
                         */
                        function _upgradeToAndCallUUPS(address newImplementation, bytes memory data, bool forceCall) internal {
                            // Upgrades from old implementations will perform a rollback test. This test requires the new
                            // implementation to upgrade back to the old, non-ERC1822 compliant, implementation. Removing
                            // this special case will break upgrade paths from old UUPS implementation to new ones.
                            if (StorageSlot.getBooleanSlot(_ROLLBACK_SLOT).value) {
                                _setImplementation(newImplementation);
                            } else {
                                try IERC1822Proxiable(newImplementation).proxiableUUID() returns (bytes32 slot) {
                                    require(slot == _IMPLEMENTATION_SLOT, "ERC1967Upgrade: unsupported proxiableUUID");
                                } catch {
                                    revert("ERC1967Upgrade: new implementation is not UUPS");
                                }
                                _upgradeToAndCall(newImplementation, data, forceCall);
                            }
                        }
                        /**
                         * @dev Storage slot with the admin of the contract.
                         * This is the keccak-256 hash of "eip1967.proxy.admin" subtracted by 1, and is
                         * validated in the constructor.
                         */
                        bytes32 internal constant _ADMIN_SLOT = 0xb53127684a568b3173ae13b9f8a6016e243e63b6e8ee1178d6a717850b5d6103;
                        /**
                         * @dev Returns the current admin.
                         */
                        function _getAdmin() internal view returns (address) {
                            return StorageSlot.getAddressSlot(_ADMIN_SLOT).value;
                        }
                        /**
                         * @dev Stores a new address in the EIP1967 admin slot.
                         */
                        function _setAdmin(address newAdmin) private {
                            require(newAdmin != address(0), "ERC1967: new admin is the zero address");
                            StorageSlot.getAddressSlot(_ADMIN_SLOT).value = newAdmin;
                        }
                        /**
                         * @dev Changes the admin of the proxy.
                         *
                         * Emits an {AdminChanged} event.
                         */
                        function _changeAdmin(address newAdmin) internal {
                            emit AdminChanged(_getAdmin(), newAdmin);
                            _setAdmin(newAdmin);
                        }
                        /**
                         * @dev The storage slot of the UpgradeableBeacon contract which defines the implementation for this proxy.
                         * This is bytes32(uint256(keccak256('eip1967.proxy.beacon')) - 1)) and is validated in the constructor.
                         */
                        bytes32 internal constant _BEACON_SLOT = 0xa3f0ad74e5423aebfd80d3ef4346578335a9a72aeaee59ff6cb3582b35133d50;
                        /**
                         * @dev Returns the current beacon.
                         */
                        function _getBeacon() internal view returns (address) {
                            return StorageSlot.getAddressSlot(_BEACON_SLOT).value;
                        }
                        /**
                         * @dev Stores a new beacon in the EIP1967 beacon slot.
                         */
                        function _setBeacon(address newBeacon) private {
                            require(Address.isContract(newBeacon), "ERC1967: new beacon is not a contract");
                            require(
                                Address.isContract(IBeacon(newBeacon).implementation()),
                                "ERC1967: beacon implementation is not a contract"
                            );
                            StorageSlot.getAddressSlot(_BEACON_SLOT).value = newBeacon;
                        }
                        /**
                         * @dev Perform beacon upgrade with additional setup call. Note: This upgrades the address of the beacon, it does
                         * not upgrade the implementation contained in the beacon (see {UpgradeableBeacon-_setImplementation} for that).
                         *
                         * Emits a {BeaconUpgraded} event.
                         */
                        function _upgradeBeaconToAndCall(address newBeacon, bytes memory data, bool forceCall) internal {
                            _setBeacon(newBeacon);
                            emit BeaconUpgraded(newBeacon);
                            if (data.length > 0 || forceCall) {
                                Address.functionDelegateCall(IBeacon(newBeacon).implementation(), data);
                            }
                        }
                    }
                    // SPDX-License-Identifier: MIT
                    // OpenZeppelin Contracts v4.4.1 (proxy/beacon/IBeacon.sol)
                    pragma solidity ^0.8.0;
                    /**
                     * @dev This is the interface that {BeaconProxy} expects of its beacon.
                     */
                    interface IBeacon {
                        /**
                         * @dev Must return an address that can be used as a delegate call target.
                         *
                         * {BeaconProxy} will check that this address is a contract.
                         */
                        function implementation() external view returns (address);
                    }
                    // SPDX-License-Identifier: MIT
                    // OpenZeppelin Contracts (last updated v4.9.0) (interfaces/IERC1967.sol)
                    pragma solidity ^0.8.0;
                    /**
                     * @dev ERC-1967: Proxy Storage Slots. This interface contains the events defined in the ERC.
                     *
                     * _Available since v4.8.3._
                     */
                    interface IERC1967 {
                        /**
                         * @dev Emitted when the implementation is upgraded.
                         */
                        event Upgraded(address indexed implementation);
                        /**
                         * @dev Emitted when the admin account has changed.
                         */
                        event AdminChanged(address previousAdmin, address newAdmin);
                        /**
                         * @dev Emitted when the beacon is changed.
                         */
                        event BeaconUpgraded(address indexed beacon);
                    }
                    // SPDX-License-Identifier: MIT
                    // OpenZeppelin Contracts (last updated v4.5.0) (interfaces/draft-IERC1822.sol)
                    pragma solidity ^0.8.0;
                    /**
                     * @dev ERC1822: Universal Upgradeable Proxy Standard (UUPS) documents a method for upgradeability through a simplified
                     * proxy whose upgrades are fully controlled by the current implementation.
                     */
                    interface IERC1822Proxiable {
                        /**
                         * @dev Returns the storage slot that the proxiable contract assumes is being used to store the implementation
                         * address.
                         *
                         * IMPORTANT: A proxy pointing at a proxiable contract should not be considered proxiable itself, because this risks
                         * bricking a proxy that upgrades to it, by delegating to itself until out of gas. Thus it is critical that this
                         * function revert if invoked through a proxy.
                         */
                        function proxiableUUID() external view returns (bytes32);
                    }
                    // SPDX-License-Identifier: MIT
                    // OpenZeppelin Contracts (last updated v4.9.0) (utils/Address.sol)
                    pragma solidity ^0.8.1;
                    /**
                     * @dev Collection of functions related to the address type
                     */
                    library Address {
                        /**
                         * @dev Returns true if `account` is a contract.
                         *
                         * [IMPORTANT]
                         * ====
                         * It is unsafe to assume that an address for which this function returns
                         * false is an externally-owned account (EOA) and not a contract.
                         *
                         * Among others, `isContract` will return false for the following
                         * types of addresses:
                         *
                         *  - an externally-owned account
                         *  - a contract in construction
                         *  - an address where a contract will be created
                         *  - an address where a contract lived, but was destroyed
                         *
                         * Furthermore, `isContract` will also return true if the target contract within
                         * the same transaction is already scheduled for destruction by `SELFDESTRUCT`,
                         * which only has an effect at the end of a transaction.
                         * ====
                         *
                         * [IMPORTANT]
                         * ====
                         * You shouldn't rely on `isContract` to protect against flash loan attacks!
                         *
                         * Preventing calls from contracts is highly discouraged. It breaks composability, breaks support for smart wallets
                         * like Gnosis Safe, and does not provide security since it can be circumvented by calling from a contract
                         * constructor.
                         * ====
                         */
                        function isContract(address account) internal view returns (bool) {
                            // This method relies on extcodesize/address.code.length, which returns 0
                            // for contracts in construction, since the code is only stored at the end
                            // of the constructor execution.
                            return account.code.length > 0;
                        }
                        /**
                         * @dev Replacement for Solidity's `transfer`: sends `amount` wei to
                         * `recipient`, forwarding all available gas and reverting on errors.
                         *
                         * https://eips.ethereum.org/EIPS/eip-1884[EIP1884] increases the gas cost
                         * of certain opcodes, possibly making contracts go over the 2300 gas limit
                         * imposed by `transfer`, making them unable to receive funds via
                         * `transfer`. {sendValue} removes this limitation.
                         *
                         * https://consensys.net/diligence/blog/2019/09/stop-using-soliditys-transfer-now/[Learn more].
                         *
                         * IMPORTANT: because control is transferred to `recipient`, care must be
                         * taken to not create reentrancy vulnerabilities. Consider using
                         * {ReentrancyGuard} or the
                         * https://solidity.readthedocs.io/en/v0.8.0/security-considerations.html#use-the-checks-effects-interactions-pattern[checks-effects-interactions pattern].
                         */
                        function sendValue(address payable recipient, uint256 amount) internal {
                            require(address(this).balance >= amount, "Address: insufficient balance");
                            (bool success, ) = recipient.call{value: amount}("");
                            require(success, "Address: unable to send value, recipient may have reverted");
                        }
                        /**
                         * @dev Performs a Solidity function call using a low level `call`. A
                         * plain `call` is an unsafe replacement for a function call: use this
                         * function instead.
                         *
                         * If `target` reverts with a revert reason, it is bubbled up by this
                         * function (like regular Solidity function calls).
                         *
                         * Returns the raw returned data. To convert to the expected return value,
                         * use https://solidity.readthedocs.io/en/latest/units-and-global-variables.html?highlight=abi.decode#abi-encoding-and-decoding-functions[`abi.decode`].
                         *
                         * Requirements:
                         *
                         * - `target` must be a contract.
                         * - calling `target` with `data` must not revert.
                         *
                         * _Available since v3.1._
                         */
                        function functionCall(address target, bytes memory data) internal returns (bytes memory) {
                            return functionCallWithValue(target, data, 0, "Address: low-level call failed");
                        }
                        /**
                         * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], but with
                         * `errorMessage` as a fallback revert reason when `target` reverts.
                         *
                         * _Available since v3.1._
                         */
                        function functionCall(
                            address target,
                            bytes memory data,
                            string memory errorMessage
                        ) internal returns (bytes memory) {
                            return functionCallWithValue(target, data, 0, errorMessage);
                        }
                        /**
                         * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
                         * but also transferring `value` wei to `target`.
                         *
                         * Requirements:
                         *
                         * - the calling contract must have an ETH balance of at least `value`.
                         * - the called Solidity function must be `payable`.
                         *
                         * _Available since v3.1._
                         */
                        function functionCallWithValue(address target, bytes memory data, uint256 value) internal returns (bytes memory) {
                            return functionCallWithValue(target, data, value, "Address: low-level call with value failed");
                        }
                        /**
                         * @dev Same as {xref-Address-functionCallWithValue-address-bytes-uint256-}[`functionCallWithValue`], but
                         * with `errorMessage` as a fallback revert reason when `target` reverts.
                         *
                         * _Available since v3.1._
                         */
                        function functionCallWithValue(
                            address target,
                            bytes memory data,
                            uint256 value,
                            string memory errorMessage
                        ) internal returns (bytes memory) {
                            require(address(this).balance >= value, "Address: insufficient balance for call");
                            (bool success, bytes memory returndata) = target.call{value: value}(data);
                            return verifyCallResultFromTarget(target, success, returndata, errorMessage);
                        }
                        /**
                         * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
                         * but performing a static call.
                         *
                         * _Available since v3.3._
                         */
                        function functionStaticCall(address target, bytes memory data) internal view returns (bytes memory) {
                            return functionStaticCall(target, data, "Address: low-level static call failed");
                        }
                        /**
                         * @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`],
                         * but performing a static call.
                         *
                         * _Available since v3.3._
                         */
                        function functionStaticCall(
                            address target,
                            bytes memory data,
                            string memory errorMessage
                        ) internal view returns (bytes memory) {
                            (bool success, bytes memory returndata) = target.staticcall(data);
                            return verifyCallResultFromTarget(target, success, returndata, errorMessage);
                        }
                        /**
                         * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
                         * but performing a delegate call.
                         *
                         * _Available since v3.4._
                         */
                        function functionDelegateCall(address target, bytes memory data) internal returns (bytes memory) {
                            return functionDelegateCall(target, data, "Address: low-level delegate call failed");
                        }
                        /**
                         * @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`],
                         * but performing a delegate call.
                         *
                         * _Available since v3.4._
                         */
                        function functionDelegateCall(
                            address target,
                            bytes memory data,
                            string memory errorMessage
                        ) internal returns (bytes memory) {
                            (bool success, bytes memory returndata) = target.delegatecall(data);
                            return verifyCallResultFromTarget(target, success, returndata, errorMessage);
                        }
                        /**
                         * @dev Tool to verify that a low level call to smart-contract was successful, and revert (either by bubbling
                         * the revert reason or using the provided one) in case of unsuccessful call or if target was not a contract.
                         *
                         * _Available since v4.8._
                         */
                        function verifyCallResultFromTarget(
                            address target,
                            bool success,
                            bytes memory returndata,
                            string memory errorMessage
                        ) internal view returns (bytes memory) {
                            if (success) {
                                if (returndata.length == 0) {
                                    // only check isContract if the call was successful and the return data is empty
                                    // otherwise we already know that it was a contract
                                    require(isContract(target), "Address: call to non-contract");
                                }
                                return returndata;
                            } else {
                                _revert(returndata, errorMessage);
                            }
                        }
                        /**
                         * @dev Tool to verify that a low level call was successful, and revert if it wasn't, either by bubbling the
                         * revert reason or using the provided one.
                         *
                         * _Available since v4.3._
                         */
                        function verifyCallResult(
                            bool success,
                            bytes memory returndata,
                            string memory errorMessage
                        ) internal pure returns (bytes memory) {
                            if (success) {
                                return returndata;
                            } else {
                                _revert(returndata, errorMessage);
                            }
                        }
                        function _revert(bytes memory returndata, string memory errorMessage) private pure {
                            // Look for revert reason and bubble it up if present
                            if (returndata.length > 0) {
                                // The easiest way to bubble the revert reason is using memory via assembly
                                /// @solidity memory-safe-assembly
                                assembly {
                                    let returndata_size := mload(returndata)
                                    revert(add(32, returndata), returndata_size)
                                }
                            } else {
                                revert(errorMessage);
                            }
                        }
                    }
                    // SPDX-License-Identifier: MIT
                    // OpenZeppelin Contracts (last updated v4.9.0) (utils/StorageSlot.sol)
                    // This file was procedurally generated from scripts/generate/templates/StorageSlot.js.
                    pragma solidity ^0.8.0;
                    /**
                     * @dev Library for reading and writing primitive types to specific storage slots.
                     *
                     * Storage slots are often used to avoid storage conflict when dealing with upgradeable contracts.
                     * This library helps with reading and writing to such slots without the need for inline assembly.
                     *
                     * The functions in this library return Slot structs that contain a `value` member that can be used to read or write.
                     *
                     * Example usage to set ERC1967 implementation slot:
                     * ```solidity
                     * contract ERC1967 {
                     *     bytes32 internal constant _IMPLEMENTATION_SLOT = 0x360894a13ba1a3210667c828492db98dca3e2076cc3735a920a3ca505d382bbc;
                     *
                     *     function _getImplementation() internal view returns (address) {
                     *         return StorageSlot.getAddressSlot(_IMPLEMENTATION_SLOT).value;
                     *     }
                     *
                     *     function _setImplementation(address newImplementation) internal {
                     *         require(Address.isContract(newImplementation), "ERC1967: new implementation is not a contract");
                     *         StorageSlot.getAddressSlot(_IMPLEMENTATION_SLOT).value = newImplementation;
                     *     }
                     * }
                     * ```
                     *
                     * _Available since v4.1 for `address`, `bool`, `bytes32`, `uint256`._
                     * _Available since v4.9 for `string`, `bytes`._
                     */
                    library StorageSlot {
                        struct AddressSlot {
                            address value;
                        }
                        struct BooleanSlot {
                            bool value;
                        }
                        struct Bytes32Slot {
                            bytes32 value;
                        }
                        struct Uint256Slot {
                            uint256 value;
                        }
                        struct StringSlot {
                            string value;
                        }
                        struct BytesSlot {
                            bytes value;
                        }
                        /**
                         * @dev Returns an `AddressSlot` with member `value` located at `slot`.
                         */
                        function getAddressSlot(bytes32 slot) internal pure returns (AddressSlot storage r) {
                            /// @solidity memory-safe-assembly
                            assembly {
                                r.slot := slot
                            }
                        }
                        /**
                         * @dev Returns an `BooleanSlot` with member `value` located at `slot`.
                         */
                        function getBooleanSlot(bytes32 slot) internal pure returns (BooleanSlot storage r) {
                            /// @solidity memory-safe-assembly
                            assembly {
                                r.slot := slot
                            }
                        }
                        /**
                         * @dev Returns an `Bytes32Slot` with member `value` located at `slot`.
                         */
                        function getBytes32Slot(bytes32 slot) internal pure returns (Bytes32Slot storage r) {
                            /// @solidity memory-safe-assembly
                            assembly {
                                r.slot := slot
                            }
                        }
                        /**
                         * @dev Returns an `Uint256Slot` with member `value` located at `slot`.
                         */
                        function getUint256Slot(bytes32 slot) internal pure returns (Uint256Slot storage r) {
                            /// @solidity memory-safe-assembly
                            assembly {
                                r.slot := slot
                            }
                        }
                        /**
                         * @dev Returns an `StringSlot` with member `value` located at `slot`.
                         */
                        function getStringSlot(bytes32 slot) internal pure returns (StringSlot storage r) {
                            /// @solidity memory-safe-assembly
                            assembly {
                                r.slot := slot
                            }
                        }
                        /**
                         * @dev Returns an `StringSlot` representation of the string storage pointer `store`.
                         */
                        function getStringSlot(string storage store) internal pure returns (StringSlot storage r) {
                            /// @solidity memory-safe-assembly
                            assembly {
                                r.slot := store.slot
                            }
                        }
                        /**
                         * @dev Returns an `BytesSlot` with member `value` located at `slot`.
                         */
                        function getBytesSlot(bytes32 slot) internal pure returns (BytesSlot storage r) {
                            /// @solidity memory-safe-assembly
                            assembly {
                                r.slot := slot
                            }
                        }
                        /**
                         * @dev Returns an `BytesSlot` representation of the bytes storage pointer `store`.
                         */
                        function getBytesSlot(bytes storage store) internal pure returns (BytesSlot storage r) {
                            /// @solidity memory-safe-assembly
                            assembly {
                                r.slot := store.slot
                            }
                        }
                    }
                    

                    File 3 of 6: TransparentUpgradeableProxy
                    // SPDX-License-Identifier: MIT
                    // OpenZeppelin Contracts (last updated v4.9.0) (proxy/transparent/TransparentUpgradeableProxy.sol)
                    pragma solidity ^0.8.0;
                    import "../ERC1967/ERC1967Proxy.sol";
                    /**
                     * @dev Interface for {TransparentUpgradeableProxy}. In order to implement transparency, {TransparentUpgradeableProxy}
                     * does not implement this interface directly, and some of its functions are implemented by an internal dispatch
                     * mechanism. The compiler is unaware that these functions are implemented by {TransparentUpgradeableProxy} and will not
                     * include them in the ABI so this interface must be used to interact with it.
                     */
                    interface ITransparentUpgradeableProxy is IERC1967 {
                        function admin() external view returns (address);
                        function implementation() external view returns (address);
                        function changeAdmin(address) external;
                        function upgradeTo(address) external;
                        function upgradeToAndCall(address, bytes memory) external payable;
                    }
                    /**
                     * @dev This contract implements a proxy that is upgradeable by an admin.
                     *
                     * To avoid https://medium.com/nomic-labs-blog/malicious-backdoors-in-ethereum-proxies-62629adf3357[proxy selector
                     * clashing], which can potentially be used in an attack, this contract uses the
                     * https://blog.openzeppelin.com/the-transparent-proxy-pattern/[transparent proxy pattern]. This pattern implies two
                     * things that go hand in hand:
                     *
                     * 1. If any account other than the admin calls the proxy, the call will be forwarded to the implementation, even if
                     * that call matches one of the admin functions exposed by the proxy itself.
                     * 2. If the admin calls the proxy, it can access the admin functions, but its calls will never be forwarded to the
                     * implementation. If the admin tries to call a function on the implementation it will fail with an error that says
                     * "admin cannot fallback to proxy target".
                     *
                     * These properties mean that the admin account can only be used for admin actions like upgrading the proxy or changing
                     * the admin, so it's best if it's a dedicated account that is not used for anything else. This will avoid headaches due
                     * to sudden errors when trying to call a function from the proxy implementation.
                     *
                     * Our recommendation is for the dedicated account to be an instance of the {ProxyAdmin} contract. If set up this way,
                     * you should think of the `ProxyAdmin` instance as the real administrative interface of your proxy.
                     *
                     * NOTE: The real interface of this proxy is that defined in `ITransparentUpgradeableProxy`. This contract does not
                     * inherit from that interface, and instead the admin functions are implicitly implemented using a custom dispatch
                     * mechanism in `_fallback`. Consequently, the compiler will not produce an ABI for this contract. This is necessary to
                     * fully implement transparency without decoding reverts caused by selector clashes between the proxy and the
                     * implementation.
                     *
                     * WARNING: It is not recommended to extend this contract to add additional external functions. If you do so, the compiler
                     * will not check that there are no selector conflicts, due to the note above. A selector clash between any new function
                     * and the functions declared in {ITransparentUpgradeableProxy} will be resolved in favor of the new one. This could
                     * render the admin operations inaccessible, which could prevent upgradeability. Transparency may also be compromised.
                     */
                    contract TransparentUpgradeableProxy is ERC1967Proxy {
                        /**
                         * @dev Initializes an upgradeable proxy managed by `_admin`, backed by the implementation at `_logic`, and
                         * optionally initialized with `_data` as explained in {ERC1967Proxy-constructor}.
                         */
                        constructor(address _logic, address admin_, bytes memory _data) payable ERC1967Proxy(_logic, _data) {
                            _changeAdmin(admin_);
                        }
                        /**
                         * @dev Modifier used internally that will delegate the call to the implementation unless the sender is the admin.
                         *
                         * CAUTION: This modifier is deprecated, as it could cause issues if the modified function has arguments, and the
                         * implementation provides a function with the same selector.
                         */
                        modifier ifAdmin() {
                            if (msg.sender == _getAdmin()) {
                                _;
                            } else {
                                _fallback();
                            }
                        }
                        /**
                         * @dev If caller is the admin process the call internally, otherwise transparently fallback to the proxy behavior
                         */
                        function _fallback() internal virtual override {
                            if (msg.sender == _getAdmin()) {
                                bytes memory ret;
                                bytes4 selector = msg.sig;
                                if (selector == ITransparentUpgradeableProxy.upgradeTo.selector) {
                                    ret = _dispatchUpgradeTo();
                                } else if (selector == ITransparentUpgradeableProxy.upgradeToAndCall.selector) {
                                    ret = _dispatchUpgradeToAndCall();
                                } else if (selector == ITransparentUpgradeableProxy.changeAdmin.selector) {
                                    ret = _dispatchChangeAdmin();
                                } else if (selector == ITransparentUpgradeableProxy.admin.selector) {
                                    ret = _dispatchAdmin();
                                } else if (selector == ITransparentUpgradeableProxy.implementation.selector) {
                                    ret = _dispatchImplementation();
                                } else {
                                    revert("TransparentUpgradeableProxy: admin cannot fallback to proxy target");
                                }
                                assembly {
                                    return(add(ret, 0x20), mload(ret))
                                }
                            } else {
                                super._fallback();
                            }
                        }
                        /**
                         * @dev Returns the current admin.
                         *
                         * TIP: To get this value clients can read directly from the storage slot shown below (specified by EIP1967) using the
                         * https://eth.wiki/json-rpc/API#eth_getstorageat[`eth_getStorageAt`] RPC call.
                         * `0xb53127684a568b3173ae13b9f8a6016e243e63b6e8ee1178d6a717850b5d6103`
                         */
                        function _dispatchAdmin() private returns (bytes memory) {
                            _requireZeroValue();
                            address admin = _getAdmin();
                            return abi.encode(admin);
                        }
                        /**
                         * @dev Returns the current implementation.
                         *
                         * TIP: To get this value clients can read directly from the storage slot shown below (specified by EIP1967) using the
                         * https://eth.wiki/json-rpc/API#eth_getstorageat[`eth_getStorageAt`] RPC call.
                         * `0x360894a13ba1a3210667c828492db98dca3e2076cc3735a920a3ca505d382bbc`
                         */
                        function _dispatchImplementation() private returns (bytes memory) {
                            _requireZeroValue();
                            address implementation = _implementation();
                            return abi.encode(implementation);
                        }
                        /**
                         * @dev Changes the admin of the proxy.
                         *
                         * Emits an {AdminChanged} event.
                         */
                        function _dispatchChangeAdmin() private returns (bytes memory) {
                            _requireZeroValue();
                            address newAdmin = abi.decode(msg.data[4:], (address));
                            _changeAdmin(newAdmin);
                            return "";
                        }
                        /**
                         * @dev Upgrade the implementation of the proxy.
                         */
                        function _dispatchUpgradeTo() private returns (bytes memory) {
                            _requireZeroValue();
                            address newImplementation = abi.decode(msg.data[4:], (address));
                            _upgradeToAndCall(newImplementation, bytes(""), false);
                            return "";
                        }
                        /**
                         * @dev Upgrade the implementation of the proxy, and then call a function from the new implementation as specified
                         * by `data`, which should be an encoded function call. This is useful to initialize new storage variables in the
                         * proxied contract.
                         */
                        function _dispatchUpgradeToAndCall() private returns (bytes memory) {
                            (address newImplementation, bytes memory data) = abi.decode(msg.data[4:], (address, bytes));
                            _upgradeToAndCall(newImplementation, data, true);
                            return "";
                        }
                        /**
                         * @dev Returns the current admin.
                         *
                         * CAUTION: This function is deprecated. Use {ERC1967Upgrade-_getAdmin} instead.
                         */
                        function _admin() internal view virtual returns (address) {
                            return _getAdmin();
                        }
                        /**
                         * @dev To keep this contract fully transparent, all `ifAdmin` functions must be payable. This helper is here to
                         * emulate some proxy functions being non-payable while still allowing value to pass through.
                         */
                        function _requireZeroValue() private {
                            require(msg.value == 0);
                        }
                    }
                    // SPDX-License-Identifier: MIT
                    // OpenZeppelin Contracts (last updated v4.7.0) (proxy/ERC1967/ERC1967Proxy.sol)
                    pragma solidity ^0.8.0;
                    import "../Proxy.sol";
                    import "./ERC1967Upgrade.sol";
                    /**
                     * @dev This contract implements an upgradeable proxy. It is upgradeable because calls are delegated to an
                     * implementation address that can be changed. This address is stored in storage in the location specified by
                     * https://eips.ethereum.org/EIPS/eip-1967[EIP1967], so that it doesn't conflict with the storage layout of the
                     * implementation behind the proxy.
                     */
                    contract ERC1967Proxy is Proxy, ERC1967Upgrade {
                        /**
                         * @dev Initializes the upgradeable proxy with an initial implementation specified by `_logic`.
                         *
                         * If `_data` is nonempty, it's used as data in a delegate call to `_logic`. This will typically be an encoded
                         * function call, and allows initializing the storage of the proxy like a Solidity constructor.
                         */
                        constructor(address _logic, bytes memory _data) payable {
                            _upgradeToAndCall(_logic, _data, false);
                        }
                        /**
                         * @dev Returns the current implementation address.
                         */
                        function _implementation() internal view virtual override returns (address impl) {
                            return ERC1967Upgrade._getImplementation();
                        }
                    }
                    // SPDX-License-Identifier: MIT
                    // OpenZeppelin Contracts (last updated v4.6.0) (proxy/Proxy.sol)
                    pragma solidity ^0.8.0;
                    /**
                     * @dev This abstract contract provides a fallback function that delegates all calls to another contract using the EVM
                     * instruction `delegatecall`. We refer to the second contract as the _implementation_ behind the proxy, and it has to
                     * be specified by overriding the virtual {_implementation} function.
                     *
                     * Additionally, delegation to the implementation can be triggered manually through the {_fallback} function, or to a
                     * different contract through the {_delegate} function.
                     *
                     * The success and return data of the delegated call will be returned back to the caller of the proxy.
                     */
                    abstract contract Proxy {
                        /**
                         * @dev Delegates the current call to `implementation`.
                         *
                         * This function does not return to its internal call site, it will return directly to the external caller.
                         */
                        function _delegate(address implementation) internal virtual {
                            assembly {
                                // Copy msg.data. We take full control of memory in this inline assembly
                                // block because it will not return to Solidity code. We overwrite the
                                // Solidity scratch pad at memory position 0.
                                calldatacopy(0, 0, calldatasize())
                                // Call the implementation.
                                // out and outsize are 0 because we don't know the size yet.
                                let result := delegatecall(gas(), implementation, 0, calldatasize(), 0, 0)
                                // Copy the returned data.
                                returndatacopy(0, 0, returndatasize())
                                switch result
                                // delegatecall returns 0 on error.
                                case 0 {
                                    revert(0, returndatasize())
                                }
                                default {
                                    return(0, returndatasize())
                                }
                            }
                        }
                        /**
                         * @dev This is a virtual function that should be overridden so it returns the address to which the fallback function
                         * and {_fallback} should delegate.
                         */
                        function _implementation() internal view virtual returns (address);
                        /**
                         * @dev Delegates the current call to the address returned by `_implementation()`.
                         *
                         * This function does not return to its internal call site, it will return directly to the external caller.
                         */
                        function _fallback() internal virtual {
                            _beforeFallback();
                            _delegate(_implementation());
                        }
                        /**
                         * @dev Fallback function that delegates calls to the address returned by `_implementation()`. Will run if no other
                         * function in the contract matches the call data.
                         */
                        fallback() external payable virtual {
                            _fallback();
                        }
                        /**
                         * @dev Fallback function that delegates calls to the address returned by `_implementation()`. Will run if call data
                         * is empty.
                         */
                        receive() external payable virtual {
                            _fallback();
                        }
                        /**
                         * @dev Hook that is called before falling back to the implementation. Can happen as part of a manual `_fallback`
                         * call, or as part of the Solidity `fallback` or `receive` functions.
                         *
                         * If overridden should call `super._beforeFallback()`.
                         */
                        function _beforeFallback() internal virtual {}
                    }
                    // SPDX-License-Identifier: MIT
                    // OpenZeppelin Contracts (last updated v4.9.0) (proxy/ERC1967/ERC1967Upgrade.sol)
                    pragma solidity ^0.8.2;
                    import "../beacon/IBeacon.sol";
                    import "../../interfaces/IERC1967.sol";
                    import "../../interfaces/draft-IERC1822.sol";
                    import "../../utils/Address.sol";
                    import "../../utils/StorageSlot.sol";
                    /**
                     * @dev This abstract contract provides getters and event emitting update functions for
                     * https://eips.ethereum.org/EIPS/eip-1967[EIP1967] slots.
                     *
                     * _Available since v4.1._
                     */
                    abstract contract ERC1967Upgrade is IERC1967 {
                        // This is the keccak-256 hash of "eip1967.proxy.rollback" subtracted by 1
                        bytes32 private constant _ROLLBACK_SLOT = 0x4910fdfa16fed3260ed0e7147f7cc6da11a60208b5b9406d12a635614ffd9143;
                        /**
                         * @dev Storage slot with the address of the current implementation.
                         * This is the keccak-256 hash of "eip1967.proxy.implementation" subtracted by 1, and is
                         * validated in the constructor.
                         */
                        bytes32 internal constant _IMPLEMENTATION_SLOT = 0x360894a13ba1a3210667c828492db98dca3e2076cc3735a920a3ca505d382bbc;
                        /**
                         * @dev Returns the current implementation address.
                         */
                        function _getImplementation() internal view returns (address) {
                            return StorageSlot.getAddressSlot(_IMPLEMENTATION_SLOT).value;
                        }
                        /**
                         * @dev Stores a new address in the EIP1967 implementation slot.
                         */
                        function _setImplementation(address newImplementation) private {
                            require(Address.isContract(newImplementation), "ERC1967: new implementation is not a contract");
                            StorageSlot.getAddressSlot(_IMPLEMENTATION_SLOT).value = newImplementation;
                        }
                        /**
                         * @dev Perform implementation upgrade
                         *
                         * Emits an {Upgraded} event.
                         */
                        function _upgradeTo(address newImplementation) internal {
                            _setImplementation(newImplementation);
                            emit Upgraded(newImplementation);
                        }
                        /**
                         * @dev Perform implementation upgrade with additional setup call.
                         *
                         * Emits an {Upgraded} event.
                         */
                        function _upgradeToAndCall(address newImplementation, bytes memory data, bool forceCall) internal {
                            _upgradeTo(newImplementation);
                            if (data.length > 0 || forceCall) {
                                Address.functionDelegateCall(newImplementation, data);
                            }
                        }
                        /**
                         * @dev Perform implementation upgrade with security checks for UUPS proxies, and additional setup call.
                         *
                         * Emits an {Upgraded} event.
                         */
                        function _upgradeToAndCallUUPS(address newImplementation, bytes memory data, bool forceCall) internal {
                            // Upgrades from old implementations will perform a rollback test. This test requires the new
                            // implementation to upgrade back to the old, non-ERC1822 compliant, implementation. Removing
                            // this special case will break upgrade paths from old UUPS implementation to new ones.
                            if (StorageSlot.getBooleanSlot(_ROLLBACK_SLOT).value) {
                                _setImplementation(newImplementation);
                            } else {
                                try IERC1822Proxiable(newImplementation).proxiableUUID() returns (bytes32 slot) {
                                    require(slot == _IMPLEMENTATION_SLOT, "ERC1967Upgrade: unsupported proxiableUUID");
                                } catch {
                                    revert("ERC1967Upgrade: new implementation is not UUPS");
                                }
                                _upgradeToAndCall(newImplementation, data, forceCall);
                            }
                        }
                        /**
                         * @dev Storage slot with the admin of the contract.
                         * This is the keccak-256 hash of "eip1967.proxy.admin" subtracted by 1, and is
                         * validated in the constructor.
                         */
                        bytes32 internal constant _ADMIN_SLOT = 0xb53127684a568b3173ae13b9f8a6016e243e63b6e8ee1178d6a717850b5d6103;
                        /**
                         * @dev Returns the current admin.
                         */
                        function _getAdmin() internal view returns (address) {
                            return StorageSlot.getAddressSlot(_ADMIN_SLOT).value;
                        }
                        /**
                         * @dev Stores a new address in the EIP1967 admin slot.
                         */
                        function _setAdmin(address newAdmin) private {
                            require(newAdmin != address(0), "ERC1967: new admin is the zero address");
                            StorageSlot.getAddressSlot(_ADMIN_SLOT).value = newAdmin;
                        }
                        /**
                         * @dev Changes the admin of the proxy.
                         *
                         * Emits an {AdminChanged} event.
                         */
                        function _changeAdmin(address newAdmin) internal {
                            emit AdminChanged(_getAdmin(), newAdmin);
                            _setAdmin(newAdmin);
                        }
                        /**
                         * @dev The storage slot of the UpgradeableBeacon contract which defines the implementation for this proxy.
                         * This is bytes32(uint256(keccak256('eip1967.proxy.beacon')) - 1)) and is validated in the constructor.
                         */
                        bytes32 internal constant _BEACON_SLOT = 0xa3f0ad74e5423aebfd80d3ef4346578335a9a72aeaee59ff6cb3582b35133d50;
                        /**
                         * @dev Returns the current beacon.
                         */
                        function _getBeacon() internal view returns (address) {
                            return StorageSlot.getAddressSlot(_BEACON_SLOT).value;
                        }
                        /**
                         * @dev Stores a new beacon in the EIP1967 beacon slot.
                         */
                        function _setBeacon(address newBeacon) private {
                            require(Address.isContract(newBeacon), "ERC1967: new beacon is not a contract");
                            require(
                                Address.isContract(IBeacon(newBeacon).implementation()),
                                "ERC1967: beacon implementation is not a contract"
                            );
                            StorageSlot.getAddressSlot(_BEACON_SLOT).value = newBeacon;
                        }
                        /**
                         * @dev Perform beacon upgrade with additional setup call. Note: This upgrades the address of the beacon, it does
                         * not upgrade the implementation contained in the beacon (see {UpgradeableBeacon-_setImplementation} for that).
                         *
                         * Emits a {BeaconUpgraded} event.
                         */
                        function _upgradeBeaconToAndCall(address newBeacon, bytes memory data, bool forceCall) internal {
                            _setBeacon(newBeacon);
                            emit BeaconUpgraded(newBeacon);
                            if (data.length > 0 || forceCall) {
                                Address.functionDelegateCall(IBeacon(newBeacon).implementation(), data);
                            }
                        }
                    }
                    // SPDX-License-Identifier: MIT
                    // OpenZeppelin Contracts v4.4.1 (proxy/beacon/IBeacon.sol)
                    pragma solidity ^0.8.0;
                    /**
                     * @dev This is the interface that {BeaconProxy} expects of its beacon.
                     */
                    interface IBeacon {
                        /**
                         * @dev Must return an address that can be used as a delegate call target.
                         *
                         * {BeaconProxy} will check that this address is a contract.
                         */
                        function implementation() external view returns (address);
                    }
                    // SPDX-License-Identifier: MIT
                    // OpenZeppelin Contracts (last updated v4.9.0) (interfaces/IERC1967.sol)
                    pragma solidity ^0.8.0;
                    /**
                     * @dev ERC-1967: Proxy Storage Slots. This interface contains the events defined in the ERC.
                     *
                     * _Available since v4.8.3._
                     */
                    interface IERC1967 {
                        /**
                         * @dev Emitted when the implementation is upgraded.
                         */
                        event Upgraded(address indexed implementation);
                        /**
                         * @dev Emitted when the admin account has changed.
                         */
                        event AdminChanged(address previousAdmin, address newAdmin);
                        /**
                         * @dev Emitted when the beacon is changed.
                         */
                        event BeaconUpgraded(address indexed beacon);
                    }
                    // SPDX-License-Identifier: MIT
                    // OpenZeppelin Contracts (last updated v4.5.0) (interfaces/draft-IERC1822.sol)
                    pragma solidity ^0.8.0;
                    /**
                     * @dev ERC1822: Universal Upgradeable Proxy Standard (UUPS) documents a method for upgradeability through a simplified
                     * proxy whose upgrades are fully controlled by the current implementation.
                     */
                    interface IERC1822Proxiable {
                        /**
                         * @dev Returns the storage slot that the proxiable contract assumes is being used to store the implementation
                         * address.
                         *
                         * IMPORTANT: A proxy pointing at a proxiable contract should not be considered proxiable itself, because this risks
                         * bricking a proxy that upgrades to it, by delegating to itself until out of gas. Thus it is critical that this
                         * function revert if invoked through a proxy.
                         */
                        function proxiableUUID() external view returns (bytes32);
                    }
                    // SPDX-License-Identifier: MIT
                    // OpenZeppelin Contracts (last updated v4.9.0) (utils/Address.sol)
                    pragma solidity ^0.8.1;
                    /**
                     * @dev Collection of functions related to the address type
                     */
                    library Address {
                        /**
                         * @dev Returns true if `account` is a contract.
                         *
                         * [IMPORTANT]
                         * ====
                         * It is unsafe to assume that an address for which this function returns
                         * false is an externally-owned account (EOA) and not a contract.
                         *
                         * Among others, `isContract` will return false for the following
                         * types of addresses:
                         *
                         *  - an externally-owned account
                         *  - a contract in construction
                         *  - an address where a contract will be created
                         *  - an address where a contract lived, but was destroyed
                         *
                         * Furthermore, `isContract` will also return true if the target contract within
                         * the same transaction is already scheduled for destruction by `SELFDESTRUCT`,
                         * which only has an effect at the end of a transaction.
                         * ====
                         *
                         * [IMPORTANT]
                         * ====
                         * You shouldn't rely on `isContract` to protect against flash loan attacks!
                         *
                         * Preventing calls from contracts is highly discouraged. It breaks composability, breaks support for smart wallets
                         * like Gnosis Safe, and does not provide security since it can be circumvented by calling from a contract
                         * constructor.
                         * ====
                         */
                        function isContract(address account) internal view returns (bool) {
                            // This method relies on extcodesize/address.code.length, which returns 0
                            // for contracts in construction, since the code is only stored at the end
                            // of the constructor execution.
                            return account.code.length > 0;
                        }
                        /**
                         * @dev Replacement for Solidity's `transfer`: sends `amount` wei to
                         * `recipient`, forwarding all available gas and reverting on errors.
                         *
                         * https://eips.ethereum.org/EIPS/eip-1884[EIP1884] increases the gas cost
                         * of certain opcodes, possibly making contracts go over the 2300 gas limit
                         * imposed by `transfer`, making them unable to receive funds via
                         * `transfer`. {sendValue} removes this limitation.
                         *
                         * https://consensys.net/diligence/blog/2019/09/stop-using-soliditys-transfer-now/[Learn more].
                         *
                         * IMPORTANT: because control is transferred to `recipient`, care must be
                         * taken to not create reentrancy vulnerabilities. Consider using
                         * {ReentrancyGuard} or the
                         * https://solidity.readthedocs.io/en/v0.8.0/security-considerations.html#use-the-checks-effects-interactions-pattern[checks-effects-interactions pattern].
                         */
                        function sendValue(address payable recipient, uint256 amount) internal {
                            require(address(this).balance >= amount, "Address: insufficient balance");
                            (bool success, ) = recipient.call{value: amount}("");
                            require(success, "Address: unable to send value, recipient may have reverted");
                        }
                        /**
                         * @dev Performs a Solidity function call using a low level `call`. A
                         * plain `call` is an unsafe replacement for a function call: use this
                         * function instead.
                         *
                         * If `target` reverts with a revert reason, it is bubbled up by this
                         * function (like regular Solidity function calls).
                         *
                         * Returns the raw returned data. To convert to the expected return value,
                         * use https://solidity.readthedocs.io/en/latest/units-and-global-variables.html?highlight=abi.decode#abi-encoding-and-decoding-functions[`abi.decode`].
                         *
                         * Requirements:
                         *
                         * - `target` must be a contract.
                         * - calling `target` with `data` must not revert.
                         *
                         * _Available since v3.1._
                         */
                        function functionCall(address target, bytes memory data) internal returns (bytes memory) {
                            return functionCallWithValue(target, data, 0, "Address: low-level call failed");
                        }
                        /**
                         * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], but with
                         * `errorMessage` as a fallback revert reason when `target` reverts.
                         *
                         * _Available since v3.1._
                         */
                        function functionCall(
                            address target,
                            bytes memory data,
                            string memory errorMessage
                        ) internal returns (bytes memory) {
                            return functionCallWithValue(target, data, 0, errorMessage);
                        }
                        /**
                         * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
                         * but also transferring `value` wei to `target`.
                         *
                         * Requirements:
                         *
                         * - the calling contract must have an ETH balance of at least `value`.
                         * - the called Solidity function must be `payable`.
                         *
                         * _Available since v3.1._
                         */
                        function functionCallWithValue(address target, bytes memory data, uint256 value) internal returns (bytes memory) {
                            return functionCallWithValue(target, data, value, "Address: low-level call with value failed");
                        }
                        /**
                         * @dev Same as {xref-Address-functionCallWithValue-address-bytes-uint256-}[`functionCallWithValue`], but
                         * with `errorMessage` as a fallback revert reason when `target` reverts.
                         *
                         * _Available since v3.1._
                         */
                        function functionCallWithValue(
                            address target,
                            bytes memory data,
                            uint256 value,
                            string memory errorMessage
                        ) internal returns (bytes memory) {
                            require(address(this).balance >= value, "Address: insufficient balance for call");
                            (bool success, bytes memory returndata) = target.call{value: value}(data);
                            return verifyCallResultFromTarget(target, success, returndata, errorMessage);
                        }
                        /**
                         * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
                         * but performing a static call.
                         *
                         * _Available since v3.3._
                         */
                        function functionStaticCall(address target, bytes memory data) internal view returns (bytes memory) {
                            return functionStaticCall(target, data, "Address: low-level static call failed");
                        }
                        /**
                         * @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`],
                         * but performing a static call.
                         *
                         * _Available since v3.3._
                         */
                        function functionStaticCall(
                            address target,
                            bytes memory data,
                            string memory errorMessage
                        ) internal view returns (bytes memory) {
                            (bool success, bytes memory returndata) = target.staticcall(data);
                            return verifyCallResultFromTarget(target, success, returndata, errorMessage);
                        }
                        /**
                         * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
                         * but performing a delegate call.
                         *
                         * _Available since v3.4._
                         */
                        function functionDelegateCall(address target, bytes memory data) internal returns (bytes memory) {
                            return functionDelegateCall(target, data, "Address: low-level delegate call failed");
                        }
                        /**
                         * @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`],
                         * but performing a delegate call.
                         *
                         * _Available since v3.4._
                         */
                        function functionDelegateCall(
                            address target,
                            bytes memory data,
                            string memory errorMessage
                        ) internal returns (bytes memory) {
                            (bool success, bytes memory returndata) = target.delegatecall(data);
                            return verifyCallResultFromTarget(target, success, returndata, errorMessage);
                        }
                        /**
                         * @dev Tool to verify that a low level call to smart-contract was successful, and revert (either by bubbling
                         * the revert reason or using the provided one) in case of unsuccessful call or if target was not a contract.
                         *
                         * _Available since v4.8._
                         */
                        function verifyCallResultFromTarget(
                            address target,
                            bool success,
                            bytes memory returndata,
                            string memory errorMessage
                        ) internal view returns (bytes memory) {
                            if (success) {
                                if (returndata.length == 0) {
                                    // only check isContract if the call was successful and the return data is empty
                                    // otherwise we already know that it was a contract
                                    require(isContract(target), "Address: call to non-contract");
                                }
                                return returndata;
                            } else {
                                _revert(returndata, errorMessage);
                            }
                        }
                        /**
                         * @dev Tool to verify that a low level call was successful, and revert if it wasn't, either by bubbling the
                         * revert reason or using the provided one.
                         *
                         * _Available since v4.3._
                         */
                        function verifyCallResult(
                            bool success,
                            bytes memory returndata,
                            string memory errorMessage
                        ) internal pure returns (bytes memory) {
                            if (success) {
                                return returndata;
                            } else {
                                _revert(returndata, errorMessage);
                            }
                        }
                        function _revert(bytes memory returndata, string memory errorMessage) private pure {
                            // Look for revert reason and bubble it up if present
                            if (returndata.length > 0) {
                                // The easiest way to bubble the revert reason is using memory via assembly
                                /// @solidity memory-safe-assembly
                                assembly {
                                    let returndata_size := mload(returndata)
                                    revert(add(32, returndata), returndata_size)
                                }
                            } else {
                                revert(errorMessage);
                            }
                        }
                    }
                    // SPDX-License-Identifier: MIT
                    // OpenZeppelin Contracts (last updated v4.9.0) (utils/StorageSlot.sol)
                    // This file was procedurally generated from scripts/generate/templates/StorageSlot.js.
                    pragma solidity ^0.8.0;
                    /**
                     * @dev Library for reading and writing primitive types to specific storage slots.
                     *
                     * Storage slots are often used to avoid storage conflict when dealing with upgradeable contracts.
                     * This library helps with reading and writing to such slots without the need for inline assembly.
                     *
                     * The functions in this library return Slot structs that contain a `value` member that can be used to read or write.
                     *
                     * Example usage to set ERC1967 implementation slot:
                     * ```solidity
                     * contract ERC1967 {
                     *     bytes32 internal constant _IMPLEMENTATION_SLOT = 0x360894a13ba1a3210667c828492db98dca3e2076cc3735a920a3ca505d382bbc;
                     *
                     *     function _getImplementation() internal view returns (address) {
                     *         return StorageSlot.getAddressSlot(_IMPLEMENTATION_SLOT).value;
                     *     }
                     *
                     *     function _setImplementation(address newImplementation) internal {
                     *         require(Address.isContract(newImplementation), "ERC1967: new implementation is not a contract");
                     *         StorageSlot.getAddressSlot(_IMPLEMENTATION_SLOT).value = newImplementation;
                     *     }
                     * }
                     * ```
                     *
                     * _Available since v4.1 for `address`, `bool`, `bytes32`, `uint256`._
                     * _Available since v4.9 for `string`, `bytes`._
                     */
                    library StorageSlot {
                        struct AddressSlot {
                            address value;
                        }
                        struct BooleanSlot {
                            bool value;
                        }
                        struct Bytes32Slot {
                            bytes32 value;
                        }
                        struct Uint256Slot {
                            uint256 value;
                        }
                        struct StringSlot {
                            string value;
                        }
                        struct BytesSlot {
                            bytes value;
                        }
                        /**
                         * @dev Returns an `AddressSlot` with member `value` located at `slot`.
                         */
                        function getAddressSlot(bytes32 slot) internal pure returns (AddressSlot storage r) {
                            /// @solidity memory-safe-assembly
                            assembly {
                                r.slot := slot
                            }
                        }
                        /**
                         * @dev Returns an `BooleanSlot` with member `value` located at `slot`.
                         */
                        function getBooleanSlot(bytes32 slot) internal pure returns (BooleanSlot storage r) {
                            /// @solidity memory-safe-assembly
                            assembly {
                                r.slot := slot
                            }
                        }
                        /**
                         * @dev Returns an `Bytes32Slot` with member `value` located at `slot`.
                         */
                        function getBytes32Slot(bytes32 slot) internal pure returns (Bytes32Slot storage r) {
                            /// @solidity memory-safe-assembly
                            assembly {
                                r.slot := slot
                            }
                        }
                        /**
                         * @dev Returns an `Uint256Slot` with member `value` located at `slot`.
                         */
                        function getUint256Slot(bytes32 slot) internal pure returns (Uint256Slot storage r) {
                            /// @solidity memory-safe-assembly
                            assembly {
                                r.slot := slot
                            }
                        }
                        /**
                         * @dev Returns an `StringSlot` with member `value` located at `slot`.
                         */
                        function getStringSlot(bytes32 slot) internal pure returns (StringSlot storage r) {
                            /// @solidity memory-safe-assembly
                            assembly {
                                r.slot := slot
                            }
                        }
                        /**
                         * @dev Returns an `StringSlot` representation of the string storage pointer `store`.
                         */
                        function getStringSlot(string storage store) internal pure returns (StringSlot storage r) {
                            /// @solidity memory-safe-assembly
                            assembly {
                                r.slot := store.slot
                            }
                        }
                        /**
                         * @dev Returns an `BytesSlot` with member `value` located at `slot`.
                         */
                        function getBytesSlot(bytes32 slot) internal pure returns (BytesSlot storage r) {
                            /// @solidity memory-safe-assembly
                            assembly {
                                r.slot := slot
                            }
                        }
                        /**
                         * @dev Returns an `BytesSlot` representation of the bytes storage pointer `store`.
                         */
                        function getBytesSlot(bytes storage store) internal pure returns (BytesSlot storage r) {
                            /// @solidity memory-safe-assembly
                            assembly {
                                r.slot := store.slot
                            }
                        }
                    }
                    

                    File 4 of 6: VerifyingPaymaster
                    // SPDX-License-Identifier: MIT
                    // OpenZeppelin Contracts (last updated v4.7.0) (access/Ownable.sol)
                    pragma solidity ^0.8.0;
                    import "../utils/Context.sol";
                    /**
                     * @dev Contract module which provides a basic access control mechanism, where
                     * there is an account (an owner) that can be granted exclusive access to
                     * specific functions.
                     *
                     * By default, the owner account will be the one that deploys the contract. This
                     * can later be changed with {transferOwnership}.
                     *
                     * This module is used through inheritance. It will make available the modifier
                     * `onlyOwner`, which can be applied to your functions to restrict their use to
                     * the owner.
                     */
                    abstract contract Ownable is Context {
                        address private _owner;
                        event OwnershipTransferred(address indexed previousOwner, address indexed newOwner);
                        /**
                         * @dev Initializes the contract setting the deployer as the initial owner.
                         */
                        constructor() {
                            _transferOwnership(_msgSender());
                        }
                        /**
                         * @dev Throws if called by any account other than the owner.
                         */
                        modifier onlyOwner() {
                            _checkOwner();
                            _;
                        }
                        /**
                         * @dev Returns the address of the current owner.
                         */
                        function owner() public view virtual returns (address) {
                            return _owner;
                        }
                        /**
                         * @dev Throws if the sender is not the owner.
                         */
                        function _checkOwner() internal view virtual {
                            require(owner() == _msgSender(), "Ownable: caller is not the owner");
                        }
                        /**
                         * @dev Leaves the contract without owner. It will not be possible to call
                         * `onlyOwner` functions anymore. Can only be called by the current owner.
                         *
                         * NOTE: Renouncing ownership will leave the contract without an owner,
                         * thereby removing any functionality that is only available to the owner.
                         */
                        function renounceOwnership() public virtual onlyOwner {
                            _transferOwnership(address(0));
                        }
                        /**
                         * @dev Transfers ownership of the contract to a new account (`newOwner`).
                         * Can only be called by the current owner.
                         */
                        function transferOwnership(address newOwner) public virtual onlyOwner {
                            require(newOwner != address(0), "Ownable: new owner is the zero address");
                            _transferOwnership(newOwner);
                        }
                        /**
                         * @dev Transfers ownership of the contract to a new account (`newOwner`).
                         * Internal function without access restriction.
                         */
                        function _transferOwnership(address newOwner) internal virtual {
                            address oldOwner = _owner;
                            _owner = newOwner;
                            emit OwnershipTransferred(oldOwner, newOwner);
                        }
                    }
                    // SPDX-License-Identifier: MIT
                    // OpenZeppelin Contracts v4.4.1 (token/ERC20/extensions/draft-IERC20Permit.sol)
                    pragma solidity ^0.8.0;
                    /**
                     * @dev Interface of the ERC20 Permit extension allowing approvals to be made via signatures, as defined in
                     * https://eips.ethereum.org/EIPS/eip-2612[EIP-2612].
                     *
                     * Adds the {permit} method, which can be used to change an account's ERC20 allowance (see {IERC20-allowance}) by
                     * presenting a message signed by the account. By not relying on {IERC20-approve}, the token holder account doesn't
                     * need to send a transaction, and thus is not required to hold Ether at all.
                     */
                    interface IERC20Permit {
                        /**
                         * @dev Sets `value` as the allowance of `spender` over ``owner``'s tokens,
                         * given ``owner``'s signed approval.
                         *
                         * IMPORTANT: The same issues {IERC20-approve} has related to transaction
                         * ordering also apply here.
                         *
                         * Emits an {Approval} event.
                         *
                         * Requirements:
                         *
                         * - `spender` cannot be the zero address.
                         * - `deadline` must be a timestamp in the future.
                         * - `v`, `r` and `s` must be a valid `secp256k1` signature from `owner`
                         * over the EIP712-formatted function arguments.
                         * - the signature must use ``owner``'s current nonce (see {nonces}).
                         *
                         * For more information on the signature format, see the
                         * https://eips.ethereum.org/EIPS/eip-2612#specification[relevant EIP
                         * section].
                         */
                        function permit(
                            address owner,
                            address spender,
                            uint256 value,
                            uint256 deadline,
                            uint8 v,
                            bytes32 r,
                            bytes32 s
                        ) external;
                        /**
                         * @dev Returns the current nonce for `owner`. This value must be
                         * included whenever a signature is generated for {permit}.
                         *
                         * Every successful call to {permit} increases ``owner``'s nonce by one. This
                         * prevents a signature from being used multiple times.
                         */
                        function nonces(address owner) external view returns (uint256);
                        /**
                         * @dev Returns the domain separator used in the encoding of the signature for {permit}, as defined by {EIP712}.
                         */
                        // solhint-disable-next-line func-name-mixedcase
                        function DOMAIN_SEPARATOR() external view returns (bytes32);
                    }
                    // SPDX-License-Identifier: MIT
                    // OpenZeppelin Contracts (last updated v4.6.0) (token/ERC20/IERC20.sol)
                    pragma solidity ^0.8.0;
                    /**
                     * @dev Interface of the ERC20 standard as defined in the EIP.
                     */
                    interface IERC20 {
                        /**
                         * @dev Emitted when `value` tokens are moved from one account (`from`) to
                         * another (`to`).
                         *
                         * Note that `value` may be zero.
                         */
                        event Transfer(address indexed from, address indexed to, uint256 value);
                        /**
                         * @dev Emitted when the allowance of a `spender` for an `owner` is set by
                         * a call to {approve}. `value` is the new allowance.
                         */
                        event Approval(address indexed owner, address indexed spender, uint256 value);
                        /**
                         * @dev Returns the amount of tokens in existence.
                         */
                        function totalSupply() external view returns (uint256);
                        /**
                         * @dev Returns the amount of tokens owned by `account`.
                         */
                        function balanceOf(address account) external view returns (uint256);
                        /**
                         * @dev Moves `amount` tokens from the caller's account to `to`.
                         *
                         * Returns a boolean value indicating whether the operation succeeded.
                         *
                         * Emits a {Transfer} event.
                         */
                        function transfer(address to, uint256 amount) external returns (bool);
                        /**
                         * @dev Returns the remaining number of tokens that `spender` will be
                         * allowed to spend on behalf of `owner` through {transferFrom}. This is
                         * zero by default.
                         *
                         * This value changes when {approve} or {transferFrom} are called.
                         */
                        function allowance(address owner, address spender) external view returns (uint256);
                        /**
                         * @dev Sets `amount` as the allowance of `spender` over the caller's tokens.
                         *
                         * Returns a boolean value indicating whether the operation succeeded.
                         *
                         * IMPORTANT: Beware that changing an allowance with this method brings the risk
                         * that someone may use both the old and the new allowance by unfortunate
                         * transaction ordering. One possible solution to mitigate this race
                         * condition is to first reduce the spender's allowance to 0 and set the
                         * desired value afterwards:
                         * https://github.com/ethereum/EIPs/issues/20#issuecomment-263524729
                         *
                         * Emits an {Approval} event.
                         */
                        function approve(address spender, uint256 amount) external returns (bool);
                        /**
                         * @dev Moves `amount` tokens from `from` to `to` using the
                         * allowance mechanism. `amount` is then deducted from the caller's
                         * allowance.
                         *
                         * Returns a boolean value indicating whether the operation succeeded.
                         *
                         * Emits a {Transfer} event.
                         */
                        function transferFrom(
                            address from,
                            address to,
                            uint256 amount
                        ) external returns (bool);
                    }
                    // SPDX-License-Identifier: MIT
                    // OpenZeppelin Contracts (last updated v4.8.0) (token/ERC20/utils/SafeERC20.sol)
                    pragma solidity ^0.8.0;
                    import "../IERC20.sol";
                    import "../extensions/draft-IERC20Permit.sol";
                    import "../../../utils/Address.sol";
                    /**
                     * @title SafeERC20
                     * @dev Wrappers around ERC20 operations that throw on failure (when the token
                     * contract returns false). Tokens that return no value (and instead revert or
                     * throw on failure) are also supported, non-reverting calls are assumed to be
                     * successful.
                     * To use this library you can add a `using SafeERC20 for IERC20;` statement to your contract,
                     * which allows you to call the safe operations as `token.safeTransfer(...)`, etc.
                     */
                    library SafeERC20 {
                        using Address for address;
                        function safeTransfer(
                            IERC20 token,
                            address to,
                            uint256 value
                        ) internal {
                            _callOptionalReturn(token, abi.encodeWithSelector(token.transfer.selector, to, value));
                        }
                        function safeTransferFrom(
                            IERC20 token,
                            address from,
                            address to,
                            uint256 value
                        ) internal {
                            _callOptionalReturn(token, abi.encodeWithSelector(token.transferFrom.selector, from, to, value));
                        }
                        /**
                         * @dev Deprecated. This function has issues similar to the ones found in
                         * {IERC20-approve}, and its usage is discouraged.
                         *
                         * Whenever possible, use {safeIncreaseAllowance} and
                         * {safeDecreaseAllowance} instead.
                         */
                        function safeApprove(
                            IERC20 token,
                            address spender,
                            uint256 value
                        ) internal {
                            // safeApprove should only be called when setting an initial allowance,
                            // or when resetting it to zero. To increase and decrease it, use
                            // 'safeIncreaseAllowance' and 'safeDecreaseAllowance'
                            require(
                                (value == 0) || (token.allowance(address(this), spender) == 0),
                                "SafeERC20: approve from non-zero to non-zero allowance"
                            );
                            _callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, value));
                        }
                        function safeIncreaseAllowance(
                            IERC20 token,
                            address spender,
                            uint256 value
                        ) internal {
                            uint256 newAllowance = token.allowance(address(this), spender) + value;
                            _callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, newAllowance));
                        }
                        function safeDecreaseAllowance(
                            IERC20 token,
                            address spender,
                            uint256 value
                        ) internal {
                            unchecked {
                                uint256 oldAllowance = token.allowance(address(this), spender);
                                require(oldAllowance >= value, "SafeERC20: decreased allowance below zero");
                                uint256 newAllowance = oldAllowance - value;
                                _callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, newAllowance));
                            }
                        }
                        function safePermit(
                            IERC20Permit token,
                            address owner,
                            address spender,
                            uint256 value,
                            uint256 deadline,
                            uint8 v,
                            bytes32 r,
                            bytes32 s
                        ) internal {
                            uint256 nonceBefore = token.nonces(owner);
                            token.permit(owner, spender, value, deadline, v, r, s);
                            uint256 nonceAfter = token.nonces(owner);
                            require(nonceAfter == nonceBefore + 1, "SafeERC20: permit did not succeed");
                        }
                        /**
                         * @dev Imitates a Solidity high-level call (i.e. a regular function call to a contract), relaxing the requirement
                         * on the return value: the return value is optional (but if data is returned, it must not be false).
                         * @param token The token targeted by the call.
                         * @param data The call data (encoded using abi.encode or one of its variants).
                         */
                        function _callOptionalReturn(IERC20 token, bytes memory data) private {
                            // We need to perform a low level call here, to bypass Solidity's return data size checking mechanism, since
                            // we're implementing it ourselves. We use {Address-functionCall} to perform this call, which verifies that
                            // the target address contains contract code and also asserts for success in the low-level call.
                            bytes memory returndata = address(token).functionCall(data, "SafeERC20: low-level call failed");
                            if (returndata.length > 0) {
                                // Return data is optional
                                require(abi.decode(returndata, (bool)), "SafeERC20: ERC20 operation did not succeed");
                            }
                        }
                    }
                    // SPDX-License-Identifier: MIT
                    // OpenZeppelin Contracts (last updated v4.8.0) (utils/Address.sol)
                    pragma solidity ^0.8.1;
                    /**
                     * @dev Collection of functions related to the address type
                     */
                    library Address {
                        /**
                         * @dev Returns true if `account` is a contract.
                         *
                         * [IMPORTANT]
                         * ====
                         * It is unsafe to assume that an address for which this function returns
                         * false is an externally-owned account (EOA) and not a contract.
                         *
                         * Among others, `isContract` will return false for the following
                         * types of addresses:
                         *
                         *  - an externally-owned account
                         *  - a contract in construction
                         *  - an address where a contract will be created
                         *  - an address where a contract lived, but was destroyed
                         * ====
                         *
                         * [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://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");
                            (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 v4.4.1 (utils/Context.sol)
                    pragma solidity ^0.8.0;
                    /**
                     * @dev Provides information about the current execution context, including the
                     * sender of the transaction and its data. While these are generally available
                     * via msg.sender and msg.data, they should not be accessed in such a direct
                     * manner, since when dealing with meta-transactions the account sending and
                     * paying for execution may not be the actual sender (as far as an application
                     * is concerned).
                     *
                     * This contract is only required for intermediate, library-like contracts.
                     */
                    abstract contract Context {
                        function _msgSender() internal view virtual returns (address) {
                            return msg.sender;
                        }
                        function _msgData() internal view virtual returns (bytes calldata) {
                            return msg.data;
                        }
                    }
                    // SPDX-License-Identifier: MIT
                    // OpenZeppelin Contracts (last updated v4.8.0) (utils/cryptography/ECDSA.sol)
                    pragma solidity ^0.8.0;
                    import "../Strings.sol";
                    /**
                     * @dev Elliptic Curve Digital Signature Algorithm (ECDSA) operations.
                     *
                     * These functions can be used to verify that a message was signed by the holder
                     * of the private keys of a given address.
                     */
                    library ECDSA {
                        enum RecoverError {
                            NoError,
                            InvalidSignature,
                            InvalidSignatureLength,
                            InvalidSignatureS,
                            InvalidSignatureV // Deprecated in v4.8
                        }
                        function _throwError(RecoverError error) private pure {
                            if (error == RecoverError.NoError) {
                                return; // no error: do nothing
                            } else if (error == RecoverError.InvalidSignature) {
                                revert("ECDSA: invalid signature");
                            } else if (error == RecoverError.InvalidSignatureLength) {
                                revert("ECDSA: invalid signature length");
                            } else if (error == RecoverError.InvalidSignatureS) {
                                revert("ECDSA: invalid signature 's' value");
                            }
                        }
                        /**
                         * @dev Returns the address that signed a hashed message (`hash`) with
                         * `signature` or error string. This address can then be used for verification purposes.
                         *
                         * The `ecrecover` EVM opcode allows for malleable (non-unique) signatures:
                         * this function rejects them by requiring the `s` value to be in the lower
                         * half order, and the `v` value to be either 27 or 28.
                         *
                         * IMPORTANT: `hash` _must_ be the result of a hash operation for the
                         * verification to be secure: it is possible to craft signatures that
                         * recover to arbitrary addresses for non-hashed data. A safe way to ensure
                         * this is by receiving a hash of the original message (which may otherwise
                         * be too long), and then calling {toEthSignedMessageHash} on it.
                         *
                         * Documentation for signature generation:
                         * - with https://web3js.readthedocs.io/en/v1.3.4/web3-eth-accounts.html#sign[Web3.js]
                         * - with https://docs.ethers.io/v5/api/signer/#Signer-signMessage[ethers]
                         *
                         * _Available since v4.3._
                         */
                        function tryRecover(bytes32 hash, bytes memory signature) internal pure returns (address, RecoverError) {
                            if (signature.length == 65) {
                                bytes32 r;
                                bytes32 s;
                                uint8 v;
                                // ecrecover takes the signature parameters, and the only way to get them
                                // currently is to use assembly.
                                /// @solidity memory-safe-assembly
                                assembly {
                                    r := mload(add(signature, 0x20))
                                    s := mload(add(signature, 0x40))
                                    v := byte(0, mload(add(signature, 0x60)))
                                }
                                return tryRecover(hash, v, r, s);
                            } else {
                                return (address(0), RecoverError.InvalidSignatureLength);
                            }
                        }
                        /**
                         * @dev Returns the address that signed a hashed message (`hash`) with
                         * `signature`. This address can then be used for verification purposes.
                         *
                         * The `ecrecover` EVM opcode allows for malleable (non-unique) signatures:
                         * this function rejects them by requiring the `s` value to be in the lower
                         * half order, and the `v` value to be either 27 or 28.
                         *
                         * IMPORTANT: `hash` _must_ be the result of a hash operation for the
                         * verification to be secure: it is possible to craft signatures that
                         * recover to arbitrary addresses for non-hashed data. A safe way to ensure
                         * this is by receiving a hash of the original message (which may otherwise
                         * be too long), and then calling {toEthSignedMessageHash} on it.
                         */
                        function recover(bytes32 hash, bytes memory signature) internal pure returns (address) {
                            (address recovered, RecoverError error) = tryRecover(hash, signature);
                            _throwError(error);
                            return recovered;
                        }
                        /**
                         * @dev Overload of {ECDSA-tryRecover} that receives the `r` and `vs` short-signature fields separately.
                         *
                         * See https://eips.ethereum.org/EIPS/eip-2098[EIP-2098 short signatures]
                         *
                         * _Available since v4.3._
                         */
                        function tryRecover(
                            bytes32 hash,
                            bytes32 r,
                            bytes32 vs
                        ) internal pure returns (address, RecoverError) {
                            bytes32 s = vs & bytes32(0x7fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff);
                            uint8 v = uint8((uint256(vs) >> 255) + 27);
                            return tryRecover(hash, v, r, s);
                        }
                        /**
                         * @dev Overload of {ECDSA-recover} that receives the `r and `vs` short-signature fields separately.
                         *
                         * _Available since v4.2._
                         */
                        function recover(
                            bytes32 hash,
                            bytes32 r,
                            bytes32 vs
                        ) internal pure returns (address) {
                            (address recovered, RecoverError error) = tryRecover(hash, r, vs);
                            _throwError(error);
                            return recovered;
                        }
                        /**
                         * @dev Overload of {ECDSA-tryRecover} that receives the `v`,
                         * `r` and `s` signature fields separately.
                         *
                         * _Available since v4.3._
                         */
                        function tryRecover(
                            bytes32 hash,
                            uint8 v,
                            bytes32 r,
                            bytes32 s
                        ) internal pure returns (address, RecoverError) {
                            // EIP-2 still allows signature malleability for ecrecover(). Remove this possibility and make the signature
                            // unique. Appendix F in the Ethereum Yellow paper (https://ethereum.github.io/yellowpaper/paper.pdf), defines
                            // the valid range for s in (301): 0 < s < secp256k1n ÷ 2 + 1, and for v in (302): v ∈ {27, 28}. Most
                            // signatures from current libraries generate a unique signature with an s-value in the lower half order.
                            //
                            // If your library generates malleable signatures, such as s-values in the upper range, calculate a new s-value
                            // with 0xFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFEBAAEDCE6AF48A03BBFD25E8CD0364141 - s1 and flip v from 27 to 28 or
                            // vice versa. If your library also generates signatures with 0/1 for v instead 27/28, add 27 to v to accept
                            // these malleable signatures as well.
                            if (uint256(s) > 0x7FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF5D576E7357A4501DDFE92F46681B20A0) {
                                return (address(0), RecoverError.InvalidSignatureS);
                            }
                            // If the signature is valid (and not malleable), return the signer address
                            address signer = ecrecover(hash, v, r, s);
                            if (signer == address(0)) {
                                return (address(0), RecoverError.InvalidSignature);
                            }
                            return (signer, RecoverError.NoError);
                        }
                        /**
                         * @dev Overload of {ECDSA-recover} that receives the `v`,
                         * `r` and `s` signature fields separately.
                         */
                        function recover(
                            bytes32 hash,
                            uint8 v,
                            bytes32 r,
                            bytes32 s
                        ) internal pure returns (address) {
                            (address recovered, RecoverError error) = tryRecover(hash, v, r, s);
                            _throwError(error);
                            return recovered;
                        }
                        /**
                         * @dev Returns an Ethereum Signed Message, created from a `hash`. This
                         * produces hash corresponding to the one signed with the
                         * https://eth.wiki/json-rpc/API#eth_sign[`eth_sign`]
                         * JSON-RPC method as part of EIP-191.
                         *
                         * See {recover}.
                         */
                        function toEthSignedMessageHash(bytes32 hash) internal pure returns (bytes32) {
                            // 32 is the length in bytes of hash,
                            // enforced by the type signature above
                            return keccak256(abi.encodePacked("\\x19Ethereum Signed Message:\
                    32", hash));
                        }
                        /**
                         * @dev Returns an Ethereum Signed Message, created from `s`. This
                         * produces hash corresponding to the one signed with the
                         * https://eth.wiki/json-rpc/API#eth_sign[`eth_sign`]
                         * JSON-RPC method as part of EIP-191.
                         *
                         * See {recover}.
                         */
                        function toEthSignedMessageHash(bytes memory s) internal pure returns (bytes32) {
                            return keccak256(abi.encodePacked("\\x19Ethereum Signed Message:\
                    ", Strings.toString(s.length), s));
                        }
                        /**
                         * @dev Returns an Ethereum Signed Typed Data, created from a
                         * `domainSeparator` and a `structHash`. This produces hash corresponding
                         * to the one signed with the
                         * https://eips.ethereum.org/EIPS/eip-712[`eth_signTypedData`]
                         * JSON-RPC method as part of EIP-712.
                         *
                         * See {recover}.
                         */
                        function toTypedDataHash(bytes32 domainSeparator, bytes32 structHash) internal pure returns (bytes32) {
                            return keccak256(abi.encodePacked("\\x19\\x01", domainSeparator, structHash));
                        }
                    }
                    // SPDX-License-Identifier: MIT
                    // OpenZeppelin Contracts (last updated v4.8.0) (utils/math/Math.sol)
                    pragma solidity ^0.8.0;
                    /**
                     * @dev Standard math utilities missing in the Solidity language.
                     */
                    library Math {
                        enum Rounding {
                            Down, // Toward negative infinity
                            Up, // Toward infinity
                            Zero // Toward zero
                        }
                        /**
                         * @dev Returns the largest of two numbers.
                         */
                        function max(uint256 a, uint256 b) internal pure returns (uint256) {
                            return a > b ? a : b;
                        }
                        /**
                         * @dev Returns the smallest of two numbers.
                         */
                        function min(uint256 a, uint256 b) internal pure returns (uint256) {
                            return a < b ? a : b;
                        }
                        /**
                         * @dev Returns the average of two numbers. The result is rounded towards
                         * zero.
                         */
                        function average(uint256 a, uint256 b) internal pure returns (uint256) {
                            // (a + b) / 2 can overflow.
                            return (a & b) + (a ^ b) / 2;
                        }
                        /**
                         * @dev Returns the ceiling of the division of two numbers.
                         *
                         * This differs from standard division with `/` in that it rounds up instead
                         * of rounding down.
                         */
                        function ceilDiv(uint256 a, uint256 b) internal pure returns (uint256) {
                            // (a + b - 1) / b can overflow on addition, so we distribute.
                            return a == 0 ? 0 : (a - 1) / b + 1;
                        }
                        /**
                         * @notice Calculates floor(x * y / denominator) with full precision. Throws if result overflows a uint256 or denominator == 0
                         * @dev Original credit to Remco Bloemen under MIT license (https://xn--2-umb.com/21/muldiv)
                         * with further edits by Uniswap Labs also under MIT license.
                         */
                        function mulDiv(
                            uint256 x,
                            uint256 y,
                            uint256 denominator
                        ) internal pure returns (uint256 result) {
                            unchecked {
                                // 512-bit multiply [prod1 prod0] = x * y. Compute the product mod 2^256 and mod 2^256 - 1, then use
                                // use the Chinese Remainder Theorem to reconstruct the 512 bit result. The result is stored in two 256
                                // variables such that product = prod1 * 2^256 + prod0.
                                uint256 prod0; // Least significant 256 bits of the product
                                uint256 prod1; // Most significant 256 bits of the product
                                assembly {
                                    let mm := mulmod(x, y, not(0))
                                    prod0 := mul(x, y)
                                    prod1 := sub(sub(mm, prod0), lt(mm, prod0))
                                }
                                // Handle non-overflow cases, 256 by 256 division.
                                if (prod1 == 0) {
                                    return prod0 / denominator;
                                }
                                // Make sure the result is less than 2^256. Also prevents denominator == 0.
                                require(denominator > prod1);
                                ///////////////////////////////////////////////
                                // 512 by 256 division.
                                ///////////////////////////////////////////////
                                // Make division exact by subtracting the remainder from [prod1 prod0].
                                uint256 remainder;
                                assembly {
                                    // Compute remainder using mulmod.
                                    remainder := mulmod(x, y, denominator)
                                    // Subtract 256 bit number from 512 bit number.
                                    prod1 := sub(prod1, gt(remainder, prod0))
                                    prod0 := sub(prod0, remainder)
                                }
                                // Factor powers of two out of denominator and compute largest power of two divisor of denominator. Always >= 1.
                                // See https://cs.stackexchange.com/q/138556/92363.
                                // Does not overflow because the denominator cannot be zero at this stage in the function.
                                uint256 twos = denominator & (~denominator + 1);
                                assembly {
                                    // Divide denominator by twos.
                                    denominator := div(denominator, twos)
                                    // Divide [prod1 prod0] by twos.
                                    prod0 := div(prod0, twos)
                                    // Flip twos such that it is 2^256 / twos. If twos is zero, then it becomes one.
                                    twos := add(div(sub(0, twos), twos), 1)
                                }
                                // Shift in bits from prod1 into prod0.
                                prod0 |= prod1 * twos;
                                // Invert denominator mod 2^256. Now that denominator is an odd number, it has an inverse modulo 2^256 such
                                // that denominator * inv = 1 mod 2^256. Compute the inverse by starting with a seed that is correct for
                                // four bits. That is, denominator * inv = 1 mod 2^4.
                                uint256 inverse = (3 * denominator) ^ 2;
                                // Use the Newton-Raphson iteration to improve the precision. Thanks to Hensel's lifting lemma, this also works
                                // in modular arithmetic, doubling the correct bits in each step.
                                inverse *= 2 - denominator * inverse; // inverse mod 2^8
                                inverse *= 2 - denominator * inverse; // inverse mod 2^16
                                inverse *= 2 - denominator * inverse; // inverse mod 2^32
                                inverse *= 2 - denominator * inverse; // inverse mod 2^64
                                inverse *= 2 - denominator * inverse; // inverse mod 2^128
                                inverse *= 2 - denominator * inverse; // inverse mod 2^256
                                // Because the division is now exact we can divide by multiplying with the modular inverse of denominator.
                                // This will give us the correct result modulo 2^256. Since the preconditions guarantee that the outcome is
                                // less than 2^256, this is the final result. We don't need to compute the high bits of the result and prod1
                                // is no longer required.
                                result = prod0 * inverse;
                                return result;
                            }
                        }
                        /**
                         * @notice Calculates x * y / denominator with full precision, following the selected rounding direction.
                         */
                        function mulDiv(
                            uint256 x,
                            uint256 y,
                            uint256 denominator,
                            Rounding rounding
                        ) internal pure returns (uint256) {
                            uint256 result = mulDiv(x, y, denominator);
                            if (rounding == Rounding.Up && mulmod(x, y, denominator) > 0) {
                                result += 1;
                            }
                            return result;
                        }
                        /**
                         * @dev Returns the square root of a number. If the number is not a perfect square, the value is rounded down.
                         *
                         * Inspired by Henry S. Warren, Jr.'s "Hacker's Delight" (Chapter 11).
                         */
                        function sqrt(uint256 a) internal pure returns (uint256) {
                            if (a == 0) {
                                return 0;
                            }
                            // For our first guess, we get the biggest power of 2 which is smaller than the square root of the target.
                            //
                            // We know that the "msb" (most significant bit) of our target number `a` is a power of 2 such that we have
                            // `msb(a) <= a < 2*msb(a)`. This value can be written `msb(a)=2**k` with `k=log2(a)`.
                            //
                            // This can be rewritten `2**log2(a) <= a < 2**(log2(a) + 1)`
                            // → `sqrt(2**k) <= sqrt(a) < sqrt(2**(k+1))`
                            // → `2**(k/2) <= sqrt(a) < 2**((k+1)/2) <= 2**(k/2 + 1)`
                            //
                            // Consequently, `2**(log2(a) / 2)` is a good first approximation of `sqrt(a)` with at least 1 correct bit.
                            uint256 result = 1 << (log2(a) >> 1);
                            // At this point `result` is an estimation with one bit of precision. We know the true value is a uint128,
                            // since it is the square root of a uint256. Newton's method converges quadratically (precision doubles at
                            // every iteration). We thus need at most 7 iteration to turn our partial result with one bit of precision
                            // into the expected uint128 result.
                            unchecked {
                                result = (result + a / result) >> 1;
                                result = (result + a / result) >> 1;
                                result = (result + a / result) >> 1;
                                result = (result + a / result) >> 1;
                                result = (result + a / result) >> 1;
                                result = (result + a / result) >> 1;
                                result = (result + a / result) >> 1;
                                return min(result, a / result);
                            }
                        }
                        /**
                         * @notice Calculates sqrt(a), following the selected rounding direction.
                         */
                        function sqrt(uint256 a, Rounding rounding) internal pure returns (uint256) {
                            unchecked {
                                uint256 result = sqrt(a);
                                return result + (rounding == Rounding.Up && result * result < a ? 1 : 0);
                            }
                        }
                        /**
                         * @dev Return the log in base 2, rounded down, of a positive value.
                         * Returns 0 if given 0.
                         */
                        function log2(uint256 value) internal pure returns (uint256) {
                            uint256 result = 0;
                            unchecked {
                                if (value >> 128 > 0) {
                                    value >>= 128;
                                    result += 128;
                                }
                                if (value >> 64 > 0) {
                                    value >>= 64;
                                    result += 64;
                                }
                                if (value >> 32 > 0) {
                                    value >>= 32;
                                    result += 32;
                                }
                                if (value >> 16 > 0) {
                                    value >>= 16;
                                    result += 16;
                                }
                                if (value >> 8 > 0) {
                                    value >>= 8;
                                    result += 8;
                                }
                                if (value >> 4 > 0) {
                                    value >>= 4;
                                    result += 4;
                                }
                                if (value >> 2 > 0) {
                                    value >>= 2;
                                    result += 2;
                                }
                                if (value >> 1 > 0) {
                                    result += 1;
                                }
                            }
                            return result;
                        }
                        /**
                         * @dev Return the log in base 2, following the selected rounding direction, of a positive value.
                         * Returns 0 if given 0.
                         */
                        function log2(uint256 value, Rounding rounding) internal pure returns (uint256) {
                            unchecked {
                                uint256 result = log2(value);
                                return result + (rounding == Rounding.Up && 1 << result < value ? 1 : 0);
                            }
                        }
                        /**
                         * @dev Return the log in base 10, rounded down, of a positive value.
                         * Returns 0 if given 0.
                         */
                        function log10(uint256 value) internal pure returns (uint256) {
                            uint256 result = 0;
                            unchecked {
                                if (value >= 10**64) {
                                    value /= 10**64;
                                    result += 64;
                                }
                                if (value >= 10**32) {
                                    value /= 10**32;
                                    result += 32;
                                }
                                if (value >= 10**16) {
                                    value /= 10**16;
                                    result += 16;
                                }
                                if (value >= 10**8) {
                                    value /= 10**8;
                                    result += 8;
                                }
                                if (value >= 10**4) {
                                    value /= 10**4;
                                    result += 4;
                                }
                                if (value >= 10**2) {
                                    value /= 10**2;
                                    result += 2;
                                }
                                if (value >= 10**1) {
                                    result += 1;
                                }
                            }
                            return result;
                        }
                        /**
                         * @dev Return the log in base 10, following the selected rounding direction, of a positive value.
                         * Returns 0 if given 0.
                         */
                        function log10(uint256 value, Rounding rounding) internal pure returns (uint256) {
                            unchecked {
                                uint256 result = log10(value);
                                return result + (rounding == Rounding.Up && 10**result < value ? 1 : 0);
                            }
                        }
                        /**
                         * @dev Return the log in base 256, rounded down, of a positive value.
                         * Returns 0 if given 0.
                         *
                         * Adding one to the result gives the number of pairs of hex symbols needed to represent `value` as a hex string.
                         */
                        function log256(uint256 value) internal pure returns (uint256) {
                            uint256 result = 0;
                            unchecked {
                                if (value >> 128 > 0) {
                                    value >>= 128;
                                    result += 16;
                                }
                                if (value >> 64 > 0) {
                                    value >>= 64;
                                    result += 8;
                                }
                                if (value >> 32 > 0) {
                                    value >>= 32;
                                    result += 4;
                                }
                                if (value >> 16 > 0) {
                                    value >>= 16;
                                    result += 2;
                                }
                                if (value >> 8 > 0) {
                                    result += 1;
                                }
                            }
                            return result;
                        }
                        /**
                         * @dev Return the log in base 10, following the selected rounding direction, of a positive value.
                         * Returns 0 if given 0.
                         */
                        function log256(uint256 value, Rounding rounding) internal pure returns (uint256) {
                            unchecked {
                                uint256 result = log256(value);
                                return result + (rounding == Rounding.Up && 1 << (result * 8) < value ? 1 : 0);
                            }
                        }
                    }
                    // SPDX-License-Identifier: MIT
                    // OpenZeppelin Contracts (last updated v4.8.0) (utils/Strings.sol)
                    pragma solidity ^0.8.0;
                    import "./math/Math.sol";
                    /**
                     * @dev String operations.
                     */
                    library Strings {
                        bytes16 private constant _SYMBOLS = "0123456789abcdef";
                        uint8 private constant _ADDRESS_LENGTH = 20;
                        /**
                         * @dev Converts a `uint256` to its ASCII `string` decimal representation.
                         */
                        function toString(uint256 value) internal pure returns (string memory) {
                            unchecked {
                                uint256 length = Math.log10(value) + 1;
                                string memory buffer = new string(length);
                                uint256 ptr;
                                /// @solidity memory-safe-assembly
                                assembly {
                                    ptr := add(buffer, add(32, length))
                                }
                                while (true) {
                                    ptr--;
                                    /// @solidity memory-safe-assembly
                                    assembly {
                                        mstore8(ptr, byte(mod(value, 10), _SYMBOLS))
                                    }
                                    value /= 10;
                                    if (value == 0) break;
                                }
                                return buffer;
                            }
                        }
                        /**
                         * @dev Converts a `uint256` to its ASCII `string` hexadecimal representation.
                         */
                        function toHexString(uint256 value) internal pure returns (string memory) {
                            unchecked {
                                return toHexString(value, Math.log256(value) + 1);
                            }
                        }
                        /**
                         * @dev Converts a `uint256` to its ASCII `string` hexadecimal representation with fixed length.
                         */
                        function toHexString(uint256 value, uint256 length) internal pure returns (string memory) {
                            bytes memory buffer = new bytes(2 * length + 2);
                            buffer[0] = "0";
                            buffer[1] = "x";
                            for (uint256 i = 2 * length + 1; i > 1; --i) {
                                buffer[i] = _SYMBOLS[value & 0xf];
                                value >>= 4;
                            }
                            require(value == 0, "Strings: hex length insufficient");
                            return string(buffer);
                        }
                        /**
                         * @dev Converts an `address` with fixed length of 20 bytes to its not checksummed ASCII `string` hexadecimal representation.
                         */
                        function toHexString(address addr) internal pure returns (string memory) {
                            return toHexString(uint256(uint160(addr)), _ADDRESS_LENGTH);
                        }
                    }
                    // SPDX-License-Identifier: GPL-3.0
                    pragma solidity ^0.8.12;
                    /* solhint-disable reason-string */
                    import "@openzeppelin/contracts/access/Ownable.sol";
                    import "../interfaces/IPaymaster.sol";
                    import "../interfaces/IEntryPoint.sol";
                    import "./Helpers.sol";
                    /**
                     * Helper class for creating a paymaster.
                     * provides helper methods for staking.
                     * validates that the postOp is called only by the entryPoint
                     */
                    abstract contract BasePaymaster is IPaymaster, Ownable {
                        IEntryPoint immutable public entryPoint;
                        constructor(IEntryPoint _entryPoint) {
                            entryPoint = _entryPoint;
                        }
                        /// @inheritdoc IPaymaster
                        function validatePaymasterUserOp(UserOperation calldata userOp, bytes32 userOpHash, uint256 maxCost)
                        external override returns (bytes memory context, uint256 validationData) {
                             _requireFromEntryPoint();
                            return _validatePaymasterUserOp(userOp, userOpHash, maxCost);
                        }
                        function _validatePaymasterUserOp(UserOperation calldata userOp, bytes32 userOpHash, uint256 maxCost)
                        internal virtual returns (bytes memory context, uint256 validationData);
                        /// @inheritdoc IPaymaster
                        function postOp(PostOpMode mode, bytes calldata context, uint256 actualGasCost) external override {
                            _requireFromEntryPoint();
                            _postOp(mode, context, actualGasCost);
                        }
                        /**
                         * post-operation handler.
                         * (verified to be called only through the entryPoint)
                         * @dev if subclass returns a non-empty context from validatePaymasterUserOp, it must also implement this method.
                         * @param mode enum with the following options:
                         *      opSucceeded - user operation succeeded.
                         *      opReverted  - user op reverted. still has to pay for gas.
                         *      postOpReverted - user op succeeded, but caused postOp (in mode=opSucceeded) to revert.
                         *                       Now this is the 2nd call, after user's op was deliberately reverted.
                         * @param context - the context value returned by validatePaymasterUserOp
                         * @param actualGasCost - actual gas used so far (without this postOp call).
                         */
                        function _postOp(PostOpMode mode, bytes calldata context, uint256 actualGasCost) internal virtual {
                            (mode,context,actualGasCost); // unused params
                            // subclass must override this method if validatePaymasterUserOp returns a context
                            revert("must override");
                        }
                        /**
                         * add a deposit for this paymaster, used for paying for transaction fees
                         */
                        function deposit() public payable {
                            entryPoint.depositTo{value : msg.value}(address(this));
                        }
                        /**
                         * withdraw value from the deposit
                         * @param withdrawAddress target to send to
                         * @param amount to withdraw
                         */
                        function withdrawTo(address payable withdrawAddress, uint256 amount) public onlyOwner {
                            entryPoint.withdrawTo(withdrawAddress, amount);
                        }
                        /**
                         * add stake for this paymaster.
                         * This method can also carry eth value to add to the current stake.
                         * @param unstakeDelaySec - the unstake delay for this paymaster. Can only be increased.
                         */
                        function addStake(uint32 unstakeDelaySec) external payable onlyOwner {
                            entryPoint.addStake{value : msg.value}(unstakeDelaySec);
                        }
                        /**
                         * return current paymaster's deposit on the entryPoint.
                         */
                        function getDeposit() public view returns (uint256) {
                            return entryPoint.balanceOf(address(this));
                        }
                        /**
                         * unlock the stake, in order to withdraw it.
                         * The paymaster can't serve requests once unlocked, until it calls addStake again
                         */
                        function unlockStake() external onlyOwner {
                            entryPoint.unlockStake();
                        }
                        /**
                         * withdraw the entire paymaster's stake.
                         * stake must be unlocked first (and then wait for the unstakeDelay to be over)
                         * @param withdrawAddress the address to send withdrawn value.
                         */
                        function withdrawStake(address payable withdrawAddress) external onlyOwner {
                            entryPoint.withdrawStake(withdrawAddress);
                        }
                        /// validate the call is made from a valid entrypoint
                        function _requireFromEntryPoint() internal virtual {
                            require(msg.sender == address(entryPoint), "Sender not EntryPoint");
                        }
                    }
                    // SPDX-License-Identifier: GPL-3.0
                    pragma solidity ^0.8.12;
                    /* solhint-disable no-inline-assembly */
                    /**
                     * returned data from validateUserOp.
                     * validateUserOp returns a uint256, with is created by `_packedValidationData` and parsed by `_parseValidationData`
                     * @param aggregator - address(0) - the account validated the signature by itself.
                     *              address(1) - the account failed to validate the signature.
                     *              otherwise - this is an address of a signature aggregator that must be used to validate the signature.
                     * @param validAfter - this UserOp is valid only after this timestamp.
                     * @param validaUntil - this UserOp is valid only up to this timestamp.
                     */
                        struct ValidationData {
                            address aggregator;
                            uint48 validAfter;
                            uint48 validUntil;
                        }
                    //extract sigFailed, validAfter, validUntil.
                    // also convert zero validUntil to type(uint48).max
                        function _parseValidationData(uint validationData) pure returns (ValidationData memory data) {
                            address aggregator = address(uint160(validationData));
                            uint48 validUntil = uint48(validationData >> 160);
                            if (validUntil == 0) {
                                validUntil = type(uint48).max;
                            }
                            uint48 validAfter = uint48(validationData >> (48 + 160));
                            return ValidationData(aggregator, validAfter, validUntil);
                        }
                    // intersect account and paymaster ranges.
                        function _intersectTimeRange(uint256 validationData, uint256 paymasterValidationData) pure returns (ValidationData memory) {
                            ValidationData memory accountValidationData = _parseValidationData(validationData);
                            ValidationData memory pmValidationData = _parseValidationData(paymasterValidationData);
                            address aggregator = accountValidationData.aggregator;
                            if (aggregator == address(0)) {
                                aggregator = pmValidationData.aggregator;
                            }
                            uint48 validAfter = accountValidationData.validAfter;
                            uint48 validUntil = accountValidationData.validUntil;
                            uint48 pmValidAfter = pmValidationData.validAfter;
                            uint48 pmValidUntil = pmValidationData.validUntil;
                            if (validAfter < pmValidAfter) validAfter = pmValidAfter;
                            if (validUntil > pmValidUntil) validUntil = pmValidUntil;
                            return ValidationData(aggregator, validAfter, validUntil);
                        }
                    /**
                     * helper to pack the return value for validateUserOp
                     * @param data - the ValidationData to pack
                     */
                        function _packValidationData(ValidationData memory data) pure returns (uint256) {
                            return uint160(data.aggregator) | (uint256(data.validUntil) << 160) | (uint256(data.validAfter) << (160 + 48));
                        }
                    /**
                     * helper to pack the return value for validateUserOp, when not using an aggregator
                     * @param sigFailed - true for signature failure, false for success
                     * @param validUntil last timestamp this UserOperation is valid (or zero for infinite)
                     * @param validAfter first timestamp this UserOperation is valid
                     */
                        function _packValidationData(bool sigFailed, uint48 validUntil, uint48 validAfter) pure returns (uint256) {
                            return (sigFailed ? 1 : 0) | (uint256(validUntil) << 160) | (uint256(validAfter) << (160 + 48));
                        }
                    /**
                     * keccak function over calldata.
                     * @dev copy calldata into memory, do keccak and drop allocated memory. Strangely, this is more efficient than letting solidity do it.
                     */
                        function calldataKeccak(bytes calldata data) pure returns (bytes32 ret) {
                            assembly {
                                let mem := mload(0x40)
                                let len := data.length
                                calldatacopy(mem, data.offset, len)
                                ret := keccak256(mem, len)
                            }
                        }
                    // SPDX-License-Identifier: GPL-3.0
                    pragma solidity ^0.8.12;
                    import "./UserOperation.sol";
                    /**
                     * Aggregated Signatures validator.
                     */
                    interface IAggregator {
                        /**
                         * validate aggregated signature.
                         * revert if the aggregated signature does not match the given list of operations.
                         */
                        function validateSignatures(UserOperation[] calldata userOps, bytes calldata signature) external view;
                        /**
                         * validate signature of a single userOp
                         * This method is should be called by bundler after EntryPoint.simulateValidation() returns (reverts) with ValidationResultWithAggregation
                         * First it validates the signature over the userOp. Then it returns data to be used when creating the handleOps.
                         * @param userOp the userOperation received from the user.
                         * @return sigForUserOp the value to put into the signature field of the userOp when calling handleOps.
                         *    (usually empty, unless account and aggregator support some kind of "multisig"
                         */
                        function validateUserOpSignature(UserOperation calldata userOp)
                        external view returns (bytes memory sigForUserOp);
                        /**
                         * aggregate multiple signatures into a single value.
                         * This method is called off-chain to calculate the signature to pass with handleOps()
                         * bundler MAY use optimized custom code perform this aggregation
                         * @param userOps array of UserOperations to collect the signatures from.
                         * @return aggregatedSignature the aggregated signature
                         */
                        function aggregateSignatures(UserOperation[] calldata userOps) external view returns (bytes memory aggregatedSignature);
                    }
                    /**
                     ** Account-Abstraction (EIP-4337) singleton EntryPoint implementation.
                     ** Only one instance required on each chain.
                     **/
                    // SPDX-License-Identifier: GPL-3.0
                    pragma solidity ^0.8.12;
                    /* solhint-disable avoid-low-level-calls */
                    /* solhint-disable no-inline-assembly */
                    /* solhint-disable reason-string */
                    import "./UserOperation.sol";
                    import "./IStakeManager.sol";
                    import "./IAggregator.sol";
                    import "./INonceManager.sol";
                    interface IEntryPoint is IStakeManager, INonceManager {
                        /***
                         * An event emitted after each successful request
                         * @param userOpHash - unique identifier for the request (hash its entire content, except signature).
                         * @param sender - the account that generates this request.
                         * @param paymaster - if non-null, the paymaster that pays for this request.
                         * @param nonce - the nonce value from the request.
                         * @param success - true if the sender transaction succeeded, false if reverted.
                         * @param actualGasCost - actual amount paid (by account or paymaster) for this UserOperation.
                         * @param actualGasUsed - total gas used by this UserOperation (including preVerification, creation, validation and execution).
                         */
                        event UserOperationEvent(bytes32 indexed userOpHash, address indexed sender, address indexed paymaster, uint256 nonce, bool success, uint256 actualGasCost, uint256 actualGasUsed);
                        /**
                         * account "sender" was deployed.
                         * @param userOpHash the userOp that deployed this account. UserOperationEvent will follow.
                         * @param sender the account that is deployed
                         * @param factory the factory used to deploy this account (in the initCode)
                         * @param paymaster the paymaster used by this UserOp
                         */
                        event AccountDeployed(bytes32 indexed userOpHash, address indexed sender, address factory, address paymaster);
                        /**
                         * An event emitted if the UserOperation "callData" reverted with non-zero length
                         * @param userOpHash the request unique identifier.
                         * @param sender the sender of this request
                         * @param nonce the nonce used in the request
                         * @param revertReason - the return bytes from the (reverted) call to "callData".
                         */
                        event UserOperationRevertReason(bytes32 indexed userOpHash, address indexed sender, uint256 nonce, bytes revertReason);
                        /**
                         * an event emitted by handleOps(), before starting the execution loop.
                         * any event emitted before this event, is part of the validation.
                         */
                        event BeforeExecution();
                        /**
                         * signature aggregator used by the following UserOperationEvents within this bundle.
                         */
                        event SignatureAggregatorChanged(address indexed aggregator);
                        /**
                         * a custom revert error of handleOps, to identify the offending op.
                         *  NOTE: if simulateValidation passes successfully, there should be no reason for handleOps to fail on it.
                         *  @param opIndex - index into the array of ops to the failed one (in simulateValidation, this is always zero)
                         *  @param reason - revert reason
                         *      The string starts with a unique code "AAmn", where "m" is "1" for factory, "2" for account and "3" for paymaster issues,
                         *      so a failure can be attributed to the correct entity.
                         *   Should be caught in off-chain handleOps simulation and not happen on-chain.
                         *   Useful for mitigating DoS attempts against batchers or for troubleshooting of factory/account/paymaster reverts.
                         */
                        error FailedOp(uint256 opIndex, string reason);
                        /**
                         * error case when a signature aggregator fails to verify the aggregated signature it had created.
                         */
                        error SignatureValidationFailed(address aggregator);
                        /**
                         * Successful result from simulateValidation.
                         * @param returnInfo gas and time-range returned values
                         * @param senderInfo stake information about the sender
                         * @param factoryInfo stake information about the factory (if any)
                         * @param paymasterInfo stake information about the paymaster (if any)
                         */
                        error ValidationResult(ReturnInfo returnInfo,
                            StakeInfo senderInfo, StakeInfo factoryInfo, StakeInfo paymasterInfo);
                        /**
                         * Successful result from simulateValidation, if the account returns a signature aggregator
                         * @param returnInfo gas and time-range returned values
                         * @param senderInfo stake information about the sender
                         * @param factoryInfo stake information about the factory (if any)
                         * @param paymasterInfo stake information about the paymaster (if any)
                         * @param aggregatorInfo signature aggregation info (if the account requires signature aggregator)
                         *      bundler MUST use it to verify the signature, or reject the UserOperation
                         */
                        error ValidationResultWithAggregation(ReturnInfo returnInfo,
                            StakeInfo senderInfo, StakeInfo factoryInfo, StakeInfo paymasterInfo,
                            AggregatorStakeInfo aggregatorInfo);
                        /**
                         * return value of getSenderAddress
                         */
                        error SenderAddressResult(address sender);
                        /**
                         * return value of simulateHandleOp
                         */
                        error ExecutionResult(uint256 preOpGas, uint256 paid, uint48 validAfter, uint48 validUntil, bool targetSuccess, bytes targetResult);
                        //UserOps handled, per aggregator
                        struct UserOpsPerAggregator {
                            UserOperation[] userOps;
                            // aggregator address
                            IAggregator aggregator;
                            // aggregated signature
                            bytes signature;
                        }
                        /**
                         * Execute a batch of UserOperation.
                         * no signature aggregator is used.
                         * if any account requires an aggregator (that is, it returned an aggregator when
                         * performing simulateValidation), then handleAggregatedOps() must be used instead.
                         * @param ops the operations to execute
                         * @param beneficiary the address to receive the fees
                         */
                        function handleOps(UserOperation[] calldata ops, address payable beneficiary) external;
                        /**
                         * Execute a batch of UserOperation with Aggregators
                         * @param opsPerAggregator the operations to execute, grouped by aggregator (or address(0) for no-aggregator accounts)
                         * @param beneficiary the address to receive the fees
                         */
                        function handleAggregatedOps(
                            UserOpsPerAggregator[] calldata opsPerAggregator,
                            address payable beneficiary
                        ) external;
                        /**
                         * generate a request Id - unique identifier for this request.
                         * the request ID is a hash over the content of the userOp (except the signature), the entrypoint and the chainid.
                         */
                        function getUserOpHash(UserOperation calldata userOp) external view returns (bytes32);
                        /**
                         * Simulate a call to account.validateUserOp and paymaster.validatePaymasterUserOp.
                         * @dev this method always revert. Successful result is ValidationResult error. other errors are failures.
                         * @dev The node must also verify it doesn't use banned opcodes, and that it doesn't reference storage outside the account's data.
                         * @param userOp the user operation to validate.
                         */
                        function simulateValidation(UserOperation calldata userOp) external;
                        /**
                         * gas and return values during simulation
                         * @param preOpGas the gas used for validation (including preValidationGas)
                         * @param prefund the required prefund for this operation
                         * @param sigFailed validateUserOp's (or paymaster's) signature check failed
                         * @param validAfter - first timestamp this UserOp is valid (merging account and paymaster time-range)
                         * @param validUntil - last timestamp this UserOp is valid (merging account and paymaster time-range)
                         * @param paymasterContext returned by validatePaymasterUserOp (to be passed into postOp)
                         */
                        struct ReturnInfo {
                            uint256 preOpGas;
                            uint256 prefund;
                            bool sigFailed;
                            uint48 validAfter;
                            uint48 validUntil;
                            bytes paymasterContext;
                        }
                        /**
                         * returned aggregated signature info.
                         * the aggregator returned by the account, and its current stake.
                         */
                        struct AggregatorStakeInfo {
                            address aggregator;
                            StakeInfo stakeInfo;
                        }
                        /**
                         * Get counterfactual sender address.
                         *  Calculate the sender contract address that will be generated by the initCode and salt in the UserOperation.
                         * this method always revert, and returns the address in SenderAddressResult error
                         * @param initCode the constructor code to be passed into the UserOperation.
                         */
                        function getSenderAddress(bytes memory initCode) external;
                        /**
                         * simulate full execution of a UserOperation (including both validation and target execution)
                         * this method will always revert with "ExecutionResult".
                         * it performs full validation of the UserOperation, but ignores signature error.
                         * an optional target address is called after the userop succeeds, and its value is returned
                         * (before the entire call is reverted)
                         * Note that in order to collect the the success/failure of the target call, it must be executed
                         * with trace enabled to track the emitted events.
                         * @param op the UserOperation to simulate
                         * @param target if nonzero, a target address to call after userop simulation. If called, the targetSuccess and targetResult
                         *        are set to the return from that call.
                         * @param targetCallData callData to pass to target address
                         */
                        function simulateHandleOp(UserOperation calldata op, address target, bytes calldata targetCallData) external;
                    }
                    // SPDX-License-Identifier: GPL-3.0
                    pragma solidity ^0.8.12;
                    interface INonceManager {
                        /**
                         * Return the next nonce for this sender.
                         * Within a given key, the nonce values are sequenced (starting with zero, and incremented by one on each userop)
                         * But UserOp with different keys can come with arbitrary order.
                         *
                         * @param sender the account address
                         * @param key the high 192 bit of the nonce
                         * @return nonce a full nonce to pass for next UserOp with this sender.
                         */
                        function getNonce(address sender, uint192 key)
                        external view returns (uint256 nonce);
                        /**
                         * Manually increment the nonce of the sender.
                         * This method is exposed just for completeness..
                         * Account does NOT need to call it, neither during validation, nor elsewhere,
                         * as the EntryPoint will update the nonce regardless.
                         * Possible use-case is call it with various keys to "initialize" their nonces to one, so that future
                         * UserOperations will not pay extra for the first transaction with a given key.
                         */
                        function incrementNonce(uint192 key) external;
                    }
                    // SPDX-License-Identifier: GPL-3.0
                    pragma solidity ^0.8.12;
                    import "./UserOperation.sol";
                    /**
                     * the interface exposed by a paymaster contract, who agrees to pay the gas for user's operations.
                     * a paymaster must hold a stake to cover the required entrypoint stake and also the gas for the transaction.
                     */
                    interface IPaymaster {
                        enum PostOpMode {
                            opSucceeded, // user op succeeded
                            opReverted, // user op reverted. still has to pay for gas.
                            postOpReverted //user op succeeded, but caused postOp to revert. Now it's a 2nd call, after user's op was deliberately reverted.
                        }
                        /**
                         * payment validation: check if paymaster agrees to pay.
                         * Must verify sender is the entryPoint.
                         * Revert to reject this request.
                         * Note that bundlers will reject this method if it changes the state, unless the paymaster is trusted (whitelisted)
                         * The paymaster pre-pays using its deposit, and receive back a refund after the postOp method returns.
                         * @param userOp the user operation
                         * @param userOpHash hash of the user's request data.
                         * @param maxCost the maximum cost of this transaction (based on maximum gas and gas price from userOp)
                         * @return context value to send to a postOp
                         *      zero length to signify postOp is not required.
                         * @return validationData signature and time-range of this operation, encoded the same as the return value of validateUserOperation
                         *      <20-byte> sigAuthorizer - 0 for valid signature, 1 to mark signature failure,
                         *         otherwise, an address of an "authorizer" contract.
                         *      <6-byte> validUntil - last timestamp this operation is valid. 0 for "indefinite"
                         *      <6-byte> validAfter - first timestamp this operation is valid
                         *      Note that the validation code cannot use block.timestamp (or block.number) directly.
                         */
                        function validatePaymasterUserOp(UserOperation calldata userOp, bytes32 userOpHash, uint256 maxCost)
                        external returns (bytes memory context, uint256 validationData);
                        /**
                         * post-operation handler.
                         * Must verify sender is the entryPoint
                         * @param mode enum with the following options:
                         *      opSucceeded - user operation succeeded.
                         *      opReverted  - user op reverted. still has to pay for gas.
                         *      postOpReverted - user op succeeded, but caused postOp (in mode=opSucceeded) to revert.
                         *                       Now this is the 2nd call, after user's op was deliberately reverted.
                         * @param context - the context value returned by validatePaymasterUserOp
                         * @param actualGasCost - actual gas used so far (without this postOp call).
                         */
                        function postOp(PostOpMode mode, bytes calldata context, uint256 actualGasCost) external;
                    }
                    // SPDX-License-Identifier: GPL-3.0-only
                    pragma solidity ^0.8.12;
                    /**
                     * manage deposits and stakes.
                     * deposit is just a balance used to pay for UserOperations (either by a paymaster or an account)
                     * stake is value locked for at least "unstakeDelay" by the staked entity.
                     */
                    interface IStakeManager {
                        event Deposited(
                            address indexed account,
                            uint256 totalDeposit
                        );
                        event Withdrawn(
                            address indexed account,
                            address withdrawAddress,
                            uint256 amount
                        );
                        /// Emitted when stake or unstake delay are modified
                        event StakeLocked(
                            address indexed account,
                            uint256 totalStaked,
                            uint256 unstakeDelaySec
                        );
                        /// Emitted once a stake is scheduled for withdrawal
                        event StakeUnlocked(
                            address indexed account,
                            uint256 withdrawTime
                        );
                        event StakeWithdrawn(
                            address indexed account,
                            address withdrawAddress,
                            uint256 amount
                        );
                        /**
                         * @param deposit the entity's deposit
                         * @param staked true if this entity is staked.
                         * @param stake actual amount of ether staked for this entity.
                         * @param unstakeDelaySec minimum delay to withdraw the stake.
                         * @param withdrawTime - first block timestamp where 'withdrawStake' will be callable, or zero if already locked
                         * @dev sizes were chosen so that (deposit,staked, stake) fit into one cell (used during handleOps)
                         *    and the rest fit into a 2nd cell.
                         *    112 bit allows for 10^15 eth
                         *    48 bit for full timestamp
                         *    32 bit allows 150 years for unstake delay
                         */
                        struct DepositInfo {
                            uint112 deposit;
                            bool staked;
                            uint112 stake;
                            uint32 unstakeDelaySec;
                            uint48 withdrawTime;
                        }
                        //API struct used by getStakeInfo and simulateValidation
                        struct StakeInfo {
                            uint256 stake;
                            uint256 unstakeDelaySec;
                        }
                        /// @return info - full deposit information of given account
                        function getDepositInfo(address account) external view returns (DepositInfo memory info);
                        /// @return the deposit (for gas payment) of the account
                        function balanceOf(address account) external view returns (uint256);
                        /**
                         * add to the deposit of the given account
                         */
                        function depositTo(address account) external payable;
                        /**
                         * add to the account's stake - amount and delay
                         * any pending unstake is first cancelled.
                         * @param _unstakeDelaySec the new lock duration before the deposit can be withdrawn.
                         */
                        function addStake(uint32 _unstakeDelaySec) external payable;
                        /**
                         * attempt to unlock the stake.
                         * the value can be withdrawn (using withdrawStake) after the unstake delay.
                         */
                        function unlockStake() external;
                        /**
                         * withdraw from the (unlocked) stake.
                         * must first call unlockStake and wait for the unstakeDelay to pass
                         * @param withdrawAddress the address to send withdrawn value.
                         */
                        function withdrawStake(address payable withdrawAddress) external;
                        /**
                         * withdraw from the deposit.
                         * @param withdrawAddress the address to send withdrawn value.
                         * @param withdrawAmount the amount to withdraw.
                         */
                        function withdrawTo(address payable withdrawAddress, uint256 withdrawAmount) external;
                    }
                    // SPDX-License-Identifier: GPL-3.0
                    pragma solidity ^0.8.12;
                    /* solhint-disable no-inline-assembly */
                    import {calldataKeccak} from "../core/Helpers.sol";
                    /**
                     * User Operation struct
                     * @param sender the sender account of this request.
                         * @param nonce unique value the sender uses to verify it is not a replay.
                         * @param initCode if set, the account contract will be created by this constructor/
                         * @param callData the method call to execute on this account.
                         * @param callGasLimit the gas limit passed to the callData method call.
                         * @param verificationGasLimit gas used for validateUserOp and validatePaymasterUserOp.
                         * @param preVerificationGas gas not calculated by the handleOps method, but added to the gas paid. Covers batch overhead.
                         * @param maxFeePerGas same as EIP-1559 gas parameter.
                         * @param maxPriorityFeePerGas same as EIP-1559 gas parameter.
                         * @param paymasterAndData if set, this field holds the paymaster address and paymaster-specific data. the paymaster will pay for the transaction instead of the sender.
                         * @param signature sender-verified signature over the entire request, the EntryPoint address and the chain ID.
                         */
                        struct UserOperation {
                            address sender;
                            uint256 nonce;
                            bytes initCode;
                            bytes callData;
                            uint256 callGasLimit;
                            uint256 verificationGasLimit;
                            uint256 preVerificationGas;
                            uint256 maxFeePerGas;
                            uint256 maxPriorityFeePerGas;
                            bytes paymasterAndData;
                            bytes signature;
                        }
                    /**
                     * Utility functions helpful when working with UserOperation structs.
                     */
                    library UserOperationLib {
                        function getSender(UserOperation calldata userOp) internal pure returns (address) {
                            address data;
                            //read sender from userOp, which is first userOp member (saves 800 gas...)
                            assembly {data := calldataload(userOp)}
                            return address(uint160(data));
                        }
                        //relayer/block builder might submit the TX with higher priorityFee, but the user should not
                        // pay above what he signed for.
                        function gasPrice(UserOperation calldata userOp) internal view returns (uint256) {
                        unchecked {
                            uint256 maxFeePerGas = userOp.maxFeePerGas;
                            uint256 maxPriorityFeePerGas = userOp.maxPriorityFeePerGas;
                            if (maxFeePerGas == maxPriorityFeePerGas) {
                                //legacy mode (for networks that don't support basefee opcode)
                                return maxFeePerGas;
                            }
                            return min(maxFeePerGas, maxPriorityFeePerGas + block.basefee);
                        }
                        }
                        function pack(UserOperation calldata userOp) internal pure returns (bytes memory ret) {
                            address sender = getSender(userOp);
                            uint256 nonce = userOp.nonce;
                            bytes32 hashInitCode = calldataKeccak(userOp.initCode);
                            bytes32 hashCallData = calldataKeccak(userOp.callData);
                            uint256 callGasLimit = userOp.callGasLimit;
                            uint256 verificationGasLimit = userOp.verificationGasLimit;
                            uint256 preVerificationGas = userOp.preVerificationGas;
                            uint256 maxFeePerGas = userOp.maxFeePerGas;
                            uint256 maxPriorityFeePerGas = userOp.maxPriorityFeePerGas;
                            bytes32 hashPaymasterAndData = calldataKeccak(userOp.paymasterAndData);
                            return abi.encode(
                                sender, nonce,
                                hashInitCode, hashCallData,
                                callGasLimit, verificationGasLimit, preVerificationGas,
                                maxFeePerGas, maxPriorityFeePerGas,
                                hashPaymasterAndData
                            );
                        }
                        function hash(UserOperation calldata userOp) internal pure returns (bytes32) {
                            return keccak256(pack(userOp));
                        }
                        function min(uint256 a, uint256 b) internal pure returns (uint256) {
                            return a < b ? a : b;
                        }
                    }
                    // SPDX-License-Identifier: GPL-3.0
                    pragma solidity ^0.8.19;
                    import { IEntryPoint } from "account-abstraction/contracts/interfaces/IEntryPoint.sol";
                    import { UserOperation } from "account-abstraction/contracts/interfaces/UserOperation.sol";
                    import { UserOperationLib } from "account-abstraction/contracts/interfaces/UserOperation.sol";
                    import { BasePaymaster } from "account-abstraction/contracts/core/BasePaymaster.sol";
                    import { calldataKeccak } from "account-abstraction/contracts/core/Helpers.sol";
                    import { ECDSA } from "@openzeppelin/contracts/utils/cryptography/ECDSA.sol";
                    import { IERC20 } from "@openzeppelin/contracts/token/ERC20/IERC20.sol";
                    import { SafeERC20 } from "@openzeppelin/contracts/token/ERC20/utils/SafeERC20.sol";
                    import { Math } from "@openzeppelin/contracts/utils/math/Math.sol";
                    import "account-abstraction/contracts/core/Helpers.sol" as Helpers;
                    /**
                     * A paymaster based on the eth-infinitism sample VerifyingPaymaster contract.
                     * It has the same functionality as the sample, but with added support for withdrawing ERC20 tokens.
                     * All withdrawn tokens will be transferred to the owner address.
                     * Note that the off-chain signer should have a strategy in place to handle a failed token withdrawal.
                     *
                     * See account-abstraction/contracts/samples/VerifyingPaymaster.sol for detailed comments.
                     */
                    contract VerifyingPaymaster is BasePaymaster {
                        using ECDSA for bytes32;
                        using UserOperationLib for UserOperation;
                        using SafeERC20 for IERC20;
                        uint256 private constant VALID_PND_OFFSET = 20;
                        uint256 private constant SIGNATURE_OFFSET = 148;
                        uint256 public constant POST_OP_GAS = 35000;
                        address public verifier;
                        address public vault;
                        constructor(IEntryPoint _entryPoint, address _owner) BasePaymaster(_entryPoint) {
                            _transferOwnership(_owner);
                            verifier = _owner;
                            vault = _owner;
                        }
                        function setVerifier(address _verifier) public onlyOwner {
                            verifier = _verifier;
                        }
                        function setVault(address _vault) public onlyOwner {
                            vault = _vault;
                        }
                        function pack(UserOperation calldata userOp) internal pure returns (bytes memory ret) {
                            return
                                abi.encode(
                                    userOp.getSender(),
                                    userOp.nonce,
                                    calldataKeccak(userOp.initCode),
                                    calldataKeccak(userOp.callData),
                                    userOp.callGasLimit,
                                    userOp.verificationGasLimit,
                                    userOp.preVerificationGas,
                                    userOp.maxFeePerGas,
                                    userOp.maxPriorityFeePerGas
                                );
                        }
                        function getHash(
                            UserOperation calldata userOp,
                            uint48 validUntil,
                            uint48 validAfter,
                            address erc20Token,
                            uint256 exchangeRate
                        ) public view returns (bytes32) {
                            return
                                keccak256(
                                    abi.encode(pack(userOp), block.chainid, address(this), validUntil, validAfter, erc20Token, exchangeRate)
                                );
                        }
                        function _validatePaymasterUserOp(
                            UserOperation calldata userOp,
                            bytes32 /*userOpHash*/,
                            uint256 requiredPreFund
                        ) internal view override returns (bytes memory context, uint256 validationData) {
                            (requiredPreFund);
                            (
                                uint48 validUntil,
                                uint48 validAfter,
                                address erc20Token,
                                uint256 exchangeRate,
                                bytes calldata signature
                            ) = parsePaymasterAndData(userOp.paymasterAndData);
                            // solhint-disable-next-line reason-string
                            require(
                                signature.length == 64 || signature.length == 65,
                                "VerifyingPaymaster: invalid signature length in paymasterAndData"
                            );
                            bytes32 hash = ECDSA.toEthSignedMessageHash(getHash(userOp, validUntil, validAfter, erc20Token, exchangeRate));
                            context = "";
                            if (erc20Token != address(0)) {
                                context = abi.encode(
                                    userOp.sender,
                                    erc20Token,
                                    exchangeRate,
                                    userOp.maxFeePerGas,
                                    userOp.maxPriorityFeePerGas
                                );
                            }
                            if (verifier != ECDSA.recover(hash, signature)) {
                                return (context, Helpers._packValidationData(true, validUntil, validAfter));
                            }
                            return (context, Helpers._packValidationData(false, validUntil, validAfter));
                        }
                        function _postOp(PostOpMode mode, bytes calldata context, uint256 actualGasCost) internal override {
                            (address sender, IERC20 token, uint256 exchangeRate, uint256 maxFeePerGas, uint256 maxPriorityFeePerGas) = abi
                                .decode(context, (address, IERC20, uint256, uint256, uint256));
                            uint256 opGasPrice;
                            unchecked {
                                if (maxFeePerGas == maxPriorityFeePerGas) {
                                    opGasPrice = maxFeePerGas;
                                } else {
                                    opGasPrice = Math.min(maxFeePerGas, maxPriorityFeePerGas + block.basefee);
                                }
                            }
                            uint256 actualTokenCost = ((actualGasCost + (POST_OP_GAS * opGasPrice)) * exchangeRate) / 1e18;
                            if (mode != PostOpMode.postOpReverted) {
                                token.safeTransferFrom(sender, vault, actualTokenCost);
                            }
                        }
                        function parsePaymasterAndData(
                            bytes calldata paymasterAndData
                        )
                            public
                            pure
                            returns (
                                uint48 validUntil,
                                uint48 validAfter,
                                address erc20Token,
                                uint256 exchangeRate,
                                bytes calldata signature
                            )
                        {
                            (validUntil, validAfter, erc20Token, exchangeRate) = abi.decode(
                                paymasterAndData[VALID_PND_OFFSET:SIGNATURE_OFFSET],
                                (uint48, uint48, address, uint256)
                            );
                            signature = paymasterAndData[SIGNATURE_OFFSET:];
                        }
                    }
                    

                    File 5 of 6: MultiSend
                    // SPDX-License-Identifier: LGPL-3.0-only
                    pragma solidity >=0.7.0 <0.9.0;
                    /// @title Multi Send - Allows to batch multiple transactions into one.
                    /// @author Nick Dodson - <nick.dodson@consensys.net>
                    /// @author Gonçalo Sá - <goncalo.sa@consensys.net>
                    /// @author Stefan George - <stefan@gnosis.io>
                    /// @author Richard Meissner - <richard@gnosis.io>
                    contract MultiSend {
                        address private immutable multisendSingleton;
                        constructor() {
                            multisendSingleton = address(this);
                        }
                        /// @dev Sends multiple transactions and reverts all if one fails.
                        /// @param transactions Encoded transactions. Each transaction is encoded as a packed bytes of
                        ///                     operation as a uint8 with 0 for a call or 1 for a delegatecall (=> 1 byte),
                        ///                     to as a address (=> 20 bytes),
                        ///                     value as a uint256 (=> 32 bytes),
                        ///                     data length as a uint256 (=> 32 bytes),
                        ///                     data as bytes.
                        ///                     see abi.encodePacked for more information on packed encoding
                        /// @notice This method is payable as delegatecalls keep the msg.value from the previous call
                        ///         If the calling method (e.g. execTransaction) received ETH this would revert otherwise
                        function multiSend(bytes memory transactions) public payable {
                            require(address(this) != multisendSingleton, "MultiSend should only be called via delegatecall");
                            // solhint-disable-next-line no-inline-assembly
                            assembly {
                                let length := mload(transactions)
                                let i := 0x20
                                for {
                                    // Pre block is not used in "while mode"
                                } lt(i, length) {
                                    // Post block is not used in "while mode"
                                } {
                                    // First byte of the data is the operation.
                                    // We shift by 248 bits (256 - 8 [operation byte]) it right since mload will always load 32 bytes (a word).
                                    // This will also zero out unused data.
                                    let operation := shr(0xf8, mload(add(transactions, i)))
                                    // We offset the load address by 1 byte (operation byte)
                                    // We shift it right by 96 bits (256 - 160 [20 address bytes]) to right-align the data and zero out unused data.
                                    let to := shr(0x60, mload(add(transactions, add(i, 0x01))))
                                    // We offset the load address by 21 byte (operation byte + 20 address bytes)
                                    let value := mload(add(transactions, add(i, 0x15)))
                                    // We offset the load address by 53 byte (operation byte + 20 address bytes + 32 value bytes)
                                    let dataLength := mload(add(transactions, add(i, 0x35)))
                                    // We offset the load address by 85 byte (operation byte + 20 address bytes + 32 value bytes + 32 data length bytes)
                                    let data := add(transactions, add(i, 0x55))
                                    let success := 0
                                    switch operation
                                        case 0 {
                                            success := call(gas(), to, value, data, dataLength, 0, 0)
                                        }
                                        case 1 {
                                            success := delegatecall(gas(), to, data, dataLength, 0, 0)
                                        }
                                    if eq(success, 0) {
                                        revert(0, 0)
                                    }
                                    // Next entry starts at 85 byte + data length
                                    i := add(i, add(0x55, dataLength))
                                }
                            }
                        }
                    }

                    File 6 of 6: MerkleDistributor
                    // SPDX-License-Identifier: BUSL-1.1
                    pragma solidity 0.8.21;
                    import { OwnableUpgradeable } from "@openzeppelin/contracts-upgradeable/access/OwnableUpgradeable.sol";
                    import { PausableUpgradeable } from "@openzeppelin/contracts-upgradeable/security/PausableUpgradeable.sol";
                    import { MerkleProofUpgradeable } from
                        "@openzeppelin/contracts-upgradeable/utils/cryptography/MerkleProofUpgradeable.sol";
                    interface IERC20 {
                        function transfer(address recipient, uint256 amount) external returns (bool);
                    }
                    interface IMerkleDistributor {
                        error ZeroValueProvided();
                        error NoTokensToClaim();
                        error AlreadyClaimed();
                        error InvalidMerkleProof();
                        error TransferFailed();
                        /// @dev returns the address of the token distributed by this contract.
                        function token() external view returns (address);
                        /// @dev Returns true if the index has been marked claimed.
                        /// @param index The index of the claim.
                        /// @param account The address to check if the claim is claimed.
                        function isClaimed(uint256 index, address account) external view returns (bool);
                        /// @dev claim the given amount of the token to the given address. Reverts if the inputs are invalid.
                        /// @param index The index of the claim.
                        /// @param account The address to send the token to.
                        /// @param cumulativeAmount The cumulative amount of the claim.
                        /// @param merkleProof The merkle proof to verify the claim.
                        function claim(uint256 index, address account, uint256 cumulativeAmount, bytes32[] calldata merkleProof) external;
                        event Claimed(uint256 index, address account, uint256 amount);
                        event MerkleRootSet(uint256 index, bytes32 currentMerkleRoot);
                    }
                    contract MerkleDistributor is IMerkleDistributor, OwnableUpgradeable, PausableUpgradeable {
                        address public override token;
                        address public protocolTreasury;
                        uint256 public feeInBPS;
                        uint256 public currentMerkleRootIndex;
                        bytes32 public currentMerkleRoot;
                        struct UserClaim {
                            uint256 lastClaimedIndex;
                            uint256 cumulativeAmount;
                        }
                        mapping(address user => UserClaim userClaim) public userClaims;
                        /// @custom:oz-upgrades-unsafe-allow constructor
                        constructor() {
                            _disableInitializers();
                        }
                        /// @dev Initializes the contract
                        function initialize(address token_, address _protocolTreasury, uint256 _feeInBPS) public initializer {
                            if (token_ == address(0) || _protocolTreasury == address(0)) {
                                revert ZeroValueProvided();
                            }
                            __Ownable_init();
                            __Pausable_init();
                            token = token_;
                            protocolTreasury = _protocolTreasury;
                            feeInBPS = _feeInBPS;
                        }
                        /// @inheritdoc IMerkleDistributor
                        function isClaimed(uint256 index, address account) public view override returns (bool) {
                            if (index == 0) revert ZeroValueProvided();
                            return userClaims[account].lastClaimedIndex >= index;
                        }
                        /// @inheritdoc IMerkleDistributor
                        function claim(
                            uint256 index,
                            address account,
                            uint256 cumulativeAmount,
                            bytes32[] calldata merkleProof
                        )
                            external
                            override
                            whenNotPaused
                        {
                            if (currentMerkleRoot == bytes32(0)) {
                                revert ZeroValueProvided();
                            }
                            if (isClaimed(index, account)) {
                                revert AlreadyClaimed();
                            }
                            // Verify the merkle proof.
                            bytes32 node = keccak256(abi.encodePacked(account, cumulativeAmount));
                            if (!MerkleProofUpgradeable.verify(merkleProof, currentMerkleRoot, node)) {
                                revert InvalidMerkleProof();
                            }
                            // Calculate the claimable amount
                            uint256 claimableAmount = cumulativeAmount - userClaims[account].cumulativeAmount;
                            // Ensure there is something to claim
                            if (claimableAmount == 0) {
                                revert NoTokensToClaim();
                            }
                            // Update user claim info, and send the token.
                            userClaims[account].lastClaimedIndex = index;
                            userClaims[account].cumulativeAmount = cumulativeAmount;
                            // Send the claimable amount to the user - deducting the fee
                            uint256 fee = (claimableAmount * feeInBPS) / 10_000;
                            uint256 amountToSend = claimableAmount - fee;
                            if (!IERC20(token).transfer(account, amountToSend)) {
                                revert TransferFailed();
                            }
                            // Send the fee to the protocol treasury
                            if (!IERC20(token).transfer(protocolTreasury, fee)) {
                                revert TransferFailed();
                            }
                            emit Claimed(index, account, claimableAmount);
                        }
                        /*//////////////////////////////////////////////////////////////
                                                Admin FUNCTION
                        //////////////////////////////////////////////////////////////*/
                        /// @dev Set the merkle root for the given index.
                        /// @dev only called by the owner.
                        /// @param _merkleRootToSet The merkle root to set.
                        function setMerkleRoot(bytes32 _merkleRootToSet) external onlyOwner {
                            if (_merkleRootToSet == bytes32(0)) {
                                revert ZeroValueProvided();
                            }
                            currentMerkleRoot = _merkleRootToSet;
                            currentMerkleRootIndex++;
                            emit MerkleRootSet(currentMerkleRootIndex, currentMerkleRoot);
                        }
                        /// @dev Set the protocol treasury address.
                        /// @dev only called by the owner.
                        /// @param _protocolTreasury The address of the protocol treasury.
                        function setProtocolTreasury(address _protocolTreasury) external onlyOwner {
                            if (_protocolTreasury == address(0)) {
                                revert ZeroValueProvided();
                            }
                            protocolTreasury = _protocolTreasury;
                        }
                        /// @dev Set the fee in BPS.
                        /// @dev only called by the owner.
                        /// @param _feeInBPS The fee in BPS.
                        function setFeeInBPS(uint256 _feeInBPS) external onlyOwner {
                            feeInBPS = _feeInBPS;
                        }
                        /// @dev Pause the contract
                        function pause() external onlyOwner {
                            _pause();
                        }
                        /// @dev Unpause the contract
                        function unpause() external onlyOwner {
                            _unpause();
                        }
                    }
                    // SPDX-License-Identifier: MIT
                    // OpenZeppelin Contracts (last updated v4.9.0) (access/Ownable.sol)
                    pragma solidity ^0.8.0;
                    import "../utils/ContextUpgradeable.sol";
                    import "../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.7.0) (security/Pausable.sol)
                    pragma solidity ^0.8.0;
                    import "../utils/ContextUpgradeable.sol";
                    import "../proxy/utils/Initializable.sol";
                    /**
                     * @dev Contract module which allows children to implement an emergency stop
                     * mechanism that can be triggered by an authorized account.
                     *
                     * This module is used through inheritance. It will make available the
                     * modifiers `whenNotPaused` and `whenPaused`, which can be applied to
                     * the functions of your contract. Note that they will not be pausable by
                     * simply including this module, only once the modifiers are put in place.
                     */
                    abstract contract PausableUpgradeable is Initializable, ContextUpgradeable {
                        /**
                         * @dev Emitted when the pause is triggered by `account`.
                         */
                        event Paused(address account);
                        /**
                         * @dev Emitted when the pause is lifted by `account`.
                         */
                        event Unpaused(address account);
                        bool private _paused;
                        /**
                         * @dev Initializes the contract in unpaused state.
                         */
                        function __Pausable_init() internal onlyInitializing {
                            __Pausable_init_unchained();
                        }
                        function __Pausable_init_unchained() internal onlyInitializing {
                            _paused = false;
                        }
                        /**
                         * @dev Modifier to make a function callable only when the contract is not paused.
                         *
                         * Requirements:
                         *
                         * - The contract must not be paused.
                         */
                        modifier whenNotPaused() {
                            _requireNotPaused();
                            _;
                        }
                        /**
                         * @dev Modifier to make a function callable only when the contract is paused.
                         *
                         * Requirements:
                         *
                         * - The contract must be paused.
                         */
                        modifier whenPaused() {
                            _requirePaused();
                            _;
                        }
                        /**
                         * @dev Returns true if the contract is paused, and false otherwise.
                         */
                        function paused() public view virtual returns (bool) {
                            return _paused;
                        }
                        /**
                         * @dev Throws if the contract is paused.
                         */
                        function _requireNotPaused() internal view virtual {
                            require(!paused(), "Pausable: paused");
                        }
                        /**
                         * @dev Throws if the contract is not paused.
                         */
                        function _requirePaused() internal view virtual {
                            require(paused(), "Pausable: not paused");
                        }
                        /**
                         * @dev Triggers stopped state.
                         *
                         * Requirements:
                         *
                         * - The contract must not be paused.
                         */
                        function _pause() internal virtual whenNotPaused {
                            _paused = true;
                            emit Paused(_msgSender());
                        }
                        /**
                         * @dev Returns to normal state.
                         *
                         * Requirements:
                         *
                         * - The contract must be paused.
                         */
                        function _unpause() internal virtual whenPaused {
                            _paused = false;
                            emit Unpaused(_msgSender());
                        }
                        /**
                         * @dev This empty reserved space is put in place to allow future versions to add new
                         * variables without shifting down storage in the inheritance chain.
                         * See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps
                         */
                        uint256[49] private __gap;
                    }
                    // SPDX-License-Identifier: MIT
                    // OpenZeppelin Contracts (last updated v4.9.2) (utils/cryptography/MerkleProof.sol)
                    pragma solidity ^0.8.0;
                    /**
                     * @dev These functions deal with verification of Merkle Tree proofs.
                     *
                     * The tree and the proofs can be generated using our
                     * https://github.com/OpenZeppelin/merkle-tree[JavaScript library].
                     * You will find a quickstart guide in the readme.
                     *
                     * WARNING: You should avoid using leaf values that are 64 bytes long prior to
                     * hashing, or use a hash function other than keccak256 for hashing leaves.
                     * This is because the concatenation of a sorted pair of internal nodes in
                     * the merkle tree could be reinterpreted as a leaf value.
                     * OpenZeppelin's JavaScript library generates merkle trees that are safe
                     * against this attack out of the box.
                     */
                    library MerkleProofUpgradeable {
                        /**
                         * @dev Returns true if a `leaf` can be proved to be a part of a Merkle tree
                         * defined by `root`. For this, a `proof` must be provided, containing
                         * sibling hashes on the branch from the leaf to the root of the tree. Each
                         * pair of leaves and each pair of pre-images are assumed to be sorted.
                         */
                        function verify(bytes32[] memory proof, bytes32 root, bytes32 leaf) internal pure returns (bool) {
                            return processProof(proof, leaf) == root;
                        }
                        /**
                         * @dev Calldata version of {verify}
                         *
                         * _Available since v4.7._
                         */
                        function verifyCalldata(bytes32[] calldata proof, bytes32 root, bytes32 leaf) internal pure returns (bool) {
                            return processProofCalldata(proof, leaf) == root;
                        }
                        /**
                         * @dev Returns the rebuilt hash obtained by traversing a Merkle tree up
                         * from `leaf` using `proof`. A `proof` is valid if and only if the rebuilt
                         * hash matches the root of the tree. When processing the proof, the pairs
                         * of leafs & pre-images are assumed to be sorted.
                         *
                         * _Available since v4.4._
                         */
                        function processProof(bytes32[] memory proof, bytes32 leaf) internal pure returns (bytes32) {
                            bytes32 computedHash = leaf;
                            for (uint256 i = 0; i < proof.length; i++) {
                                computedHash = _hashPair(computedHash, proof[i]);
                            }
                            return computedHash;
                        }
                        /**
                         * @dev Calldata version of {processProof}
                         *
                         * _Available since v4.7._
                         */
                        function processProofCalldata(bytes32[] calldata proof, bytes32 leaf) internal pure returns (bytes32) {
                            bytes32 computedHash = leaf;
                            for (uint256 i = 0; i < proof.length; i++) {
                                computedHash = _hashPair(computedHash, proof[i]);
                            }
                            return computedHash;
                        }
                        /**
                         * @dev Returns true if the `leaves` can be simultaneously proven to be a part of a merkle tree defined by
                         * `root`, according to `proof` and `proofFlags` as described in {processMultiProof}.
                         *
                         * CAUTION: Not all merkle trees admit multiproofs. See {processMultiProof} for details.
                         *
                         * _Available since v4.7._
                         */
                        function multiProofVerify(
                            bytes32[] memory proof,
                            bool[] memory proofFlags,
                            bytes32 root,
                            bytes32[] memory leaves
                        ) internal pure returns (bool) {
                            return processMultiProof(proof, proofFlags, leaves) == root;
                        }
                        /**
                         * @dev Calldata version of {multiProofVerify}
                         *
                         * CAUTION: Not all merkle trees admit multiproofs. See {processMultiProof} for details.
                         *
                         * _Available since v4.7._
                         */
                        function multiProofVerifyCalldata(
                            bytes32[] calldata proof,
                            bool[] calldata proofFlags,
                            bytes32 root,
                            bytes32[] memory leaves
                        ) internal pure returns (bool) {
                            return processMultiProofCalldata(proof, proofFlags, leaves) == root;
                        }
                        /**
                         * @dev Returns the root of a tree reconstructed from `leaves` and sibling nodes in `proof`. The reconstruction
                         * proceeds by incrementally reconstructing all inner nodes by combining a leaf/inner node with either another
                         * leaf/inner node or a proof sibling node, depending on whether each `proofFlags` item is true or false
                         * respectively.
                         *
                         * CAUTION: Not all merkle trees admit multiproofs. To use multiproofs, it is sufficient to ensure that: 1) the tree
                         * is complete (but not necessarily perfect), 2) the leaves to be proven are in the opposite order they are in the
                         * tree (i.e., as seen from right to left starting at the deepest layer and continuing at the next layer).
                         *
                         * _Available since v4.7._
                         */
                        function processMultiProof(
                            bytes32[] memory proof,
                            bool[] memory proofFlags,
                            bytes32[] memory leaves
                        ) internal pure returns (bytes32 merkleRoot) {
                            // This function rebuilds the root hash by traversing the tree up from the leaves. The root is rebuilt by
                            // consuming and producing values on a queue. The queue starts with the `leaves` array, then goes onto the
                            // `hashes` array. At the end of the process, the last hash in the `hashes` array should contain the root of
                            // the merkle tree.
                            uint256 leavesLen = leaves.length;
                            uint256 proofLen = proof.length;
                            uint256 totalHashes = proofFlags.length;
                            // Check proof validity.
                            require(leavesLen + proofLen - 1 == totalHashes, "MerkleProof: invalid multiproof");
                            // The xxxPos values are "pointers" to the next value to consume in each array. All accesses are done using
                            // `xxx[xxxPos++]`, which return the current value and increment the pointer, thus mimicking a queue's "pop".
                            bytes32[] memory hashes = new bytes32[](totalHashes);
                            uint256 leafPos = 0;
                            uint256 hashPos = 0;
                            uint256 proofPos = 0;
                            // At each step, we compute the next hash using two values:
                            // - a value from the "main queue". If not all leaves have been consumed, we get the next leaf, otherwise we
                            //   get the next hash.
                            // - depending on the flag, either another value from the "main queue" (merging branches) or an element from the
                            //   `proof` array.
                            for (uint256 i = 0; i < totalHashes; i++) {
                                bytes32 a = leafPos < leavesLen ? leaves[leafPos++] : hashes[hashPos++];
                                bytes32 b = proofFlags[i]
                                    ? (leafPos < leavesLen ? leaves[leafPos++] : hashes[hashPos++])
                                    : proof[proofPos++];
                                hashes[i] = _hashPair(a, b);
                            }
                            if (totalHashes > 0) {
                                require(proofPos == proofLen, "MerkleProof: invalid multiproof");
                                unchecked {
                                    return hashes[totalHashes - 1];
                                }
                            } else if (leavesLen > 0) {
                                return leaves[0];
                            } else {
                                return proof[0];
                            }
                        }
                        /**
                         * @dev Calldata version of {processMultiProof}.
                         *
                         * CAUTION: Not all merkle trees admit multiproofs. See {processMultiProof} for details.
                         *
                         * _Available since v4.7._
                         */
                        function processMultiProofCalldata(
                            bytes32[] calldata proof,
                            bool[] calldata proofFlags,
                            bytes32[] memory leaves
                        ) internal pure returns (bytes32 merkleRoot) {
                            // This function rebuilds the root hash by traversing the tree up from the leaves. The root is rebuilt by
                            // consuming and producing values on a queue. The queue starts with the `leaves` array, then goes onto the
                            // `hashes` array. At the end of the process, the last hash in the `hashes` array should contain the root of
                            // the merkle tree.
                            uint256 leavesLen = leaves.length;
                            uint256 proofLen = proof.length;
                            uint256 totalHashes = proofFlags.length;
                            // Check proof validity.
                            require(leavesLen + proofLen - 1 == totalHashes, "MerkleProof: invalid multiproof");
                            // The xxxPos values are "pointers" to the next value to consume in each array. All accesses are done using
                            // `xxx[xxxPos++]`, which return the current value and increment the pointer, thus mimicking a queue's "pop".
                            bytes32[] memory hashes = new bytes32[](totalHashes);
                            uint256 leafPos = 0;
                            uint256 hashPos = 0;
                            uint256 proofPos = 0;
                            // At each step, we compute the next hash using two values:
                            // - a value from the "main queue". If not all leaves have been consumed, we get the next leaf, otherwise we
                            //   get the next hash.
                            // - depending on the flag, either another value from the "main queue" (merging branches) or an element from the
                            //   `proof` array.
                            for (uint256 i = 0; i < totalHashes; i++) {
                                bytes32 a = leafPos < leavesLen ? leaves[leafPos++] : hashes[hashPos++];
                                bytes32 b = proofFlags[i]
                                    ? (leafPos < leavesLen ? leaves[leafPos++] : hashes[hashPos++])
                                    : proof[proofPos++];
                                hashes[i] = _hashPair(a, b);
                            }
                            if (totalHashes > 0) {
                                require(proofPos == proofLen, "MerkleProof: invalid multiproof");
                                unchecked {
                                    return hashes[totalHashes - 1];
                                }
                            } else if (leavesLen > 0) {
                                return leaves[0];
                            } else {
                                return proof[0];
                            }
                        }
                        function _hashPair(bytes32 a, bytes32 b) private pure returns (bytes32) {
                            return a < b ? _efficientHash(a, b) : _efficientHash(b, a);
                        }
                        function _efficientHash(bytes32 a, bytes32 b) private pure returns (bytes32 value) {
                            /// @solidity memory-safe-assembly
                            assembly {
                                mstore(0x00, a)
                                mstore(0x20, b)
                                value := keccak256(0x00, 0x40)
                            }
                        }
                    }
                    // SPDX-License-Identifier: MIT
                    // OpenZeppelin Contracts v4.4.1 (utils/Context.sol)
                    pragma solidity ^0.8.0;
                    import "../proxy/utils/Initializable.sol";
                    /**
                     * @dev Provides information about the current execution context, including the
                     * sender of the transaction and its data. While these are generally available
                     * via msg.sender and msg.data, they should not be accessed in such a direct
                     * manner, since when dealing with meta-transactions the account sending and
                     * paying for execution may not be the actual sender (as far as an application
                     * is concerned).
                     *
                     * This contract is only required for intermediate, library-like contracts.
                     */
                    abstract contract ContextUpgradeable is Initializable {
                        function __Context_init() internal onlyInitializing {
                        }
                        function __Context_init_unchained() internal onlyInitializing {
                        }
                        function _msgSender() internal view virtual returns (address) {
                            return msg.sender;
                        }
                        function _msgData() internal view virtual returns (bytes calldata) {
                            return msg.data;
                        }
                        /**
                         * @dev This empty reserved space is put in place to allow future versions to add new
                         * variables without shifting down storage in the inheritance chain.
                         * See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps
                         */
                        uint256[50] private __gap;
                    }
                    // SPDX-License-Identifier: MIT
                    // OpenZeppelin Contracts (last updated v4.9.0) (proxy/utils/Initializable.sol)
                    pragma solidity ^0.8.2;
                    import "../../utils/AddressUpgradeable.sol";
                    /**
                     * @dev This is a base contract to aid in writing upgradeable contracts, or any kind of contract that will be deployed
                     * behind a proxy. Since proxied contracts do not make use of a constructor, it's common to move constructor logic to an
                     * external initializer function, usually called `initialize`. It then becomes necessary to protect this initializer
                     * function so it can only be called once. The {initializer} modifier provided by this contract will have this effect.
                     *
                     * The initialization functions use a version number. Once a version number is used, it is consumed and cannot be
                     * reused. This mechanism prevents re-execution of each "step" but allows the creation of new initialization steps in
                     * case an upgrade adds a module that needs to be initialized.
                     *
                     * For example:
                     *
                     * [.hljs-theme-light.nopadding]
                     * ```solidity
                     * contract MyToken is ERC20Upgradeable {
                     *     function initialize() initializer public {
                     *         __ERC20_init("MyToken", "MTK");
                     *     }
                     * }
                     *
                     * contract MyTokenV2 is MyToken, ERC20PermitUpgradeable {
                     *     function initializeV2() reinitializer(2) public {
                     *         __ERC20Permit_init("MyToken");
                     *     }
                     * }
                     * ```
                     *
                     * TIP: To avoid leaving the proxy in an uninitialized state, the initializer function should be called as early as
                     * possible by providing the encoded function call as the `_data` argument to {ERC1967Proxy-constructor}.
                     *
                     * CAUTION: When used with inheritance, manual care must be taken to not invoke a parent initializer twice, or to ensure
                     * that all initializers are idempotent. This is not verified automatically as constructors are by Solidity.
                     *
                     * [CAUTION]
                     * ====
                     * Avoid leaving a contract uninitialized.
                     *
                     * An uninitialized contract can be taken over by an attacker. This applies to both a proxy and its implementation
                     * contract, which may impact the proxy. To prevent the implementation contract from being used, you should invoke
                     * the {_disableInitializers} function in the constructor to automatically lock it when it is deployed:
                     *
                     * [.hljs-theme-light.nopadding]
                     * ```
                     * /// @custom:oz-upgrades-unsafe-allow constructor
                     * constructor() {
                     *     _disableInitializers();
                     * }
                     * ```
                     * ====
                     */
                    abstract contract Initializable {
                        /**
                         * @dev Indicates that the contract has been initialized.
                         * @custom:oz-retyped-from bool
                         */
                        uint8 private _initialized;
                        /**
                         * @dev Indicates that the contract is in the process of being initialized.
                         */
                        bool private _initializing;
                        /**
                         * @dev Triggered when the contract has been initialized or reinitialized.
                         */
                        event Initialized(uint8 version);
                        /**
                         * @dev A modifier that defines a protected initializer function that can be invoked at most once. In its scope,
                         * `onlyInitializing` functions can be used to initialize parent contracts.
                         *
                         * Similar to `reinitializer(1)`, except that functions marked with `initializer` can be nested in the context of a
                         * constructor.
                         *
                         * Emits an {Initialized} event.
                         */
                        modifier initializer() {
                            bool isTopLevelCall = !_initializing;
                            require(
                                (isTopLevelCall && _initialized < 1) || (!AddressUpgradeable.isContract(address(this)) && _initialized == 1),
                                "Initializable: contract is already initialized"
                            );
                            _initialized = 1;
                            if (isTopLevelCall) {
                                _initializing = true;
                            }
                            _;
                            if (isTopLevelCall) {
                                _initializing = false;
                                emit Initialized(1);
                            }
                        }
                        /**
                         * @dev A modifier that defines a protected reinitializer function that can be invoked at most once, and only if the
                         * contract hasn't been initialized to a greater version before. In its scope, `onlyInitializing` functions can be
                         * used to initialize parent contracts.
                         *
                         * A reinitializer may be used after the original initialization step. This is essential to configure modules that
                         * are added through upgrades and that require initialization.
                         *
                         * When `version` is 1, this modifier is similar to `initializer`, except that functions marked with `reinitializer`
                         * cannot be nested. If one is invoked in the context of another, execution will revert.
                         *
                         * Note that versions can jump in increments greater than 1; this implies that if multiple reinitializers coexist in
                         * a contract, executing them in the right order is up to the developer or operator.
                         *
                         * WARNING: setting the version to 255 will prevent any future reinitialization.
                         *
                         * Emits an {Initialized} event.
                         */
                        modifier reinitializer(uint8 version) {
                            require(!_initializing && _initialized < version, "Initializable: contract is already initialized");
                            _initialized = version;
                            _initializing = true;
                            _;
                            _initializing = false;
                            emit Initialized(version);
                        }
                        /**
                         * @dev Modifier to protect an initialization function so that it can only be invoked by functions with the
                         * {initializer} and {reinitializer} modifiers, directly or indirectly.
                         */
                        modifier onlyInitializing() {
                            require(_initializing, "Initializable: contract is not initializing");
                            _;
                        }
                        /**
                         * @dev Locks the contract, preventing any future reinitialization. This cannot be part of an initializer call.
                         * Calling this in the constructor of a contract will prevent that contract from being initialized or reinitialized
                         * to any version. It is recommended to use this to lock implementation contracts that are designed to be called
                         * through proxies.
                         *
                         * Emits an {Initialized} event the first time it is successfully executed.
                         */
                        function _disableInitializers() internal virtual {
                            require(!_initializing, "Initializable: contract is initializing");
                            if (_initialized != type(uint8).max) {
                                _initialized = type(uint8).max;
                                emit Initialized(type(uint8).max);
                            }
                        }
                        /**
                         * @dev Returns the highest version that has been initialized. See {reinitializer}.
                         */
                        function _getInitializedVersion() internal view returns (uint8) {
                            return _initialized;
                        }
                        /**
                         * @dev Returns `true` if the contract is currently initializing. See {onlyInitializing}.
                         */
                        function _isInitializing() internal view returns (bool) {
                            return _initializing;
                        }
                    }
                    // SPDX-License-Identifier: MIT
                    // OpenZeppelin Contracts (last updated v4.9.0) (utils/Address.sol)
                    pragma solidity ^0.8.1;
                    /**
                     * @dev Collection of functions related to the address type
                     */
                    library AddressUpgradeable {
                        /**
                         * @dev Returns true if `account` is a contract.
                         *
                         * [IMPORTANT]
                         * ====
                         * It is unsafe to assume that an address for which this function returns
                         * false is an externally-owned account (EOA) and not a contract.
                         *
                         * Among others, `isContract` will return false for the following
                         * types of addresses:
                         *
                         *  - an externally-owned account
                         *  - a contract in construction
                         *  - an address where a contract will be created
                         *  - an address where a contract lived, but was destroyed
                         *
                         * Furthermore, `isContract` will also return true if the target contract within
                         * the same transaction is already scheduled for destruction by `SELFDESTRUCT`,
                         * which only has an effect at the end of a transaction.
                         * ====
                         *
                         * [IMPORTANT]
                         * ====
                         * You shouldn't rely on `isContract` to protect against flash loan attacks!
                         *
                         * Preventing calls from contracts is highly discouraged. It breaks composability, breaks support for smart wallets
                         * like Gnosis Safe, and does not provide security since it can be circumvented by calling from a contract
                         * constructor.
                         * ====
                         */
                        function isContract(address account) internal view returns (bool) {
                            // This method relies on extcodesize/address.code.length, which returns 0
                            // for contracts in construction, since the code is only stored at the end
                            // of the constructor execution.
                            return account.code.length > 0;
                        }
                        /**
                         * @dev Replacement for Solidity's `transfer`: sends `amount` wei to
                         * `recipient`, forwarding all available gas and reverting on errors.
                         *
                         * https://eips.ethereum.org/EIPS/eip-1884[EIP1884] increases the gas cost
                         * of certain opcodes, possibly making contracts go over the 2300 gas limit
                         * imposed by `transfer`, making them unable to receive funds via
                         * `transfer`. {sendValue} removes this limitation.
                         *
                         * https://consensys.net/diligence/blog/2019/09/stop-using-soliditys-transfer-now/[Learn more].
                         *
                         * IMPORTANT: because control is transferred to `recipient`, care must be
                         * taken to not create reentrancy vulnerabilities. Consider using
                         * {ReentrancyGuard} or the
                         * https://solidity.readthedocs.io/en/v0.8.0/security-considerations.html#use-the-checks-effects-interactions-pattern[checks-effects-interactions pattern].
                         */
                        function sendValue(address payable recipient, uint256 amount) internal {
                            require(address(this).balance >= amount, "Address: insufficient balance");
                            (bool success, ) = recipient.call{value: amount}("");
                            require(success, "Address: unable to send value, recipient may have reverted");
                        }
                        /**
                         * @dev Performs a Solidity function call using a low level `call`. A
                         * plain `call` is an unsafe replacement for a function call: use this
                         * function instead.
                         *
                         * If `target` reverts with a revert reason, it is bubbled up by this
                         * function (like regular Solidity function calls).
                         *
                         * Returns the raw returned data. To convert to the expected return value,
                         * use https://solidity.readthedocs.io/en/latest/units-and-global-variables.html?highlight=abi.decode#abi-encoding-and-decoding-functions[`abi.decode`].
                         *
                         * Requirements:
                         *
                         * - `target` must be a contract.
                         * - calling `target` with `data` must not revert.
                         *
                         * _Available since v3.1._
                         */
                        function functionCall(address target, bytes memory data) internal returns (bytes memory) {
                            return functionCallWithValue(target, data, 0, "Address: low-level call failed");
                        }
                        /**
                         * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], but with
                         * `errorMessage` as a fallback revert reason when `target` reverts.
                         *
                         * _Available since v3.1._
                         */
                        function functionCall(
                            address target,
                            bytes memory data,
                            string memory errorMessage
                        ) internal returns (bytes memory) {
                            return functionCallWithValue(target, data, 0, errorMessage);
                        }
                        /**
                         * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
                         * but also transferring `value` wei to `target`.
                         *
                         * Requirements:
                         *
                         * - the calling contract must have an ETH balance of at least `value`.
                         * - the called Solidity function must be `payable`.
                         *
                         * _Available since v3.1._
                         */
                        function functionCallWithValue(address target, bytes memory data, uint256 value) internal returns (bytes memory) {
                            return functionCallWithValue(target, data, value, "Address: low-level call with value failed");
                        }
                        /**
                         * @dev Same as {xref-Address-functionCallWithValue-address-bytes-uint256-}[`functionCallWithValue`], but
                         * with `errorMessage` as a fallback revert reason when `target` reverts.
                         *
                         * _Available since v3.1._
                         */
                        function functionCallWithValue(
                            address target,
                            bytes memory data,
                            uint256 value,
                            string memory errorMessage
                        ) internal returns (bytes memory) {
                            require(address(this).balance >= value, "Address: insufficient balance for call");
                            (bool success, bytes memory returndata) = target.call{value: value}(data);
                            return verifyCallResultFromTarget(target, success, returndata, errorMessage);
                        }
                        /**
                         * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
                         * but performing a static call.
                         *
                         * _Available since v3.3._
                         */
                        function functionStaticCall(address target, bytes memory data) internal view returns (bytes memory) {
                            return functionStaticCall(target, data, "Address: low-level static call failed");
                        }
                        /**
                         * @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`],
                         * but performing a static call.
                         *
                         * _Available since v3.3._
                         */
                        function functionStaticCall(
                            address target,
                            bytes memory data,
                            string memory errorMessage
                        ) internal view returns (bytes memory) {
                            (bool success, bytes memory returndata) = target.staticcall(data);
                            return verifyCallResultFromTarget(target, success, returndata, errorMessage);
                        }
                        /**
                         * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
                         * but performing a delegate call.
                         *
                         * _Available since v3.4._
                         */
                        function functionDelegateCall(address target, bytes memory data) internal returns (bytes memory) {
                            return functionDelegateCall(target, data, "Address: low-level delegate call failed");
                        }
                        /**
                         * @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`],
                         * but performing a delegate call.
                         *
                         * _Available since v3.4._
                         */
                        function functionDelegateCall(
                            address target,
                            bytes memory data,
                            string memory errorMessage
                        ) internal returns (bytes memory) {
                            (bool success, bytes memory returndata) = target.delegatecall(data);
                            return verifyCallResultFromTarget(target, success, returndata, errorMessage);
                        }
                        /**
                         * @dev Tool to verify that a low level call to smart-contract was successful, and revert (either by bubbling
                         * the revert reason or using the provided one) in case of unsuccessful call or if target was not a contract.
                         *
                         * _Available since v4.8._
                         */
                        function verifyCallResultFromTarget(
                            address target,
                            bool success,
                            bytes memory returndata,
                            string memory errorMessage
                        ) internal view returns (bytes memory) {
                            if (success) {
                                if (returndata.length == 0) {
                                    // only check isContract if the call was successful and the return data is empty
                                    // otherwise we already know that it was a contract
                                    require(isContract(target), "Address: call to non-contract");
                                }
                                return returndata;
                            } else {
                                _revert(returndata, errorMessage);
                            }
                        }
                        /**
                         * @dev Tool to verify that a low level call was successful, and revert if it wasn't, either by bubbling the
                         * revert reason or using the provided one.
                         *
                         * _Available since v4.3._
                         */
                        function verifyCallResult(
                            bool success,
                            bytes memory returndata,
                            string memory errorMessage
                        ) internal pure returns (bytes memory) {
                            if (success) {
                                return returndata;
                            } else {
                                _revert(returndata, errorMessage);
                            }
                        }
                        function _revert(bytes memory returndata, string memory errorMessage) private pure {
                            // Look for revert reason and bubble it up if present
                            if (returndata.length > 0) {
                                // The easiest way to bubble the revert reason is using memory via assembly
                                /// @solidity memory-safe-assembly
                                assembly {
                                    let returndata_size := mload(returndata)
                                    revert(add(32, returndata), returndata_size)
                                }
                            } else {
                                revert(errorMessage);
                            }
                        }
                    }