ETH Price: $4,153.77 (+8.26%)

Transaction Decoder

Block:
22971786 at Jul-22-2025 02:37:11 AM +UTC
Transaction Fee:
0.000919123195831176 ETH $3.82
Gas Used:
591,612 Gas / 1.553591198 Gwei

Emitted Events:

117 0xea05cdb9733d952fac4a89e636213aadcb50259c.0x69b31548dea9b3b707b4dff357d326e3e9348b24e7a6080a218a6edeeec48f9b( 0x69b31548dea9b3b707b4dff357d326e3e9348b24e7a6080a218a6edeeec48f9b, 000000000000000000000000a9d1e08c7793af67e9d92fe308d5697fb81d3e43, 00000000000000000000000000000000000000000000000001e84226172fd000, 0000000000000000000000000000000000000000000000000000000000000060, 0000000000000000000000000000000000000000000000000000000000000000 )

Account State Difference:

  Address   Before After State Difference Code
0x049F274E...237Ab631D 0.000028888777142101 Eth0.0001155925 Eth0.000086703722857899
0x06BFc707...27A7872e1 0.278196916328904541 Eth2.654538066328904541 Eth2.37634115
0x0879eccD...C746bcf88 0.000072569786327958 Eth0.0001155625 Eth0.000042992713672042
0x0aFFb9bb...4D9C94d7b 0.00008059302543476 Eth0.000114395 Eth0.00003380197456524
0x31b2921D...B46f00F4c 0 Eth0.00997852 Eth0.00997852
0x51b0390b...077930b42
0 Eth
Nonce: 0
0.06600948 Eth
Nonce: 0
0.06600948From: 0 To: 0
0x5E9d8D1D...6d4d515F2 0 Eth0.06601906 Eth0.06601906
0x661A23d3...379731b56
0 Eth
Nonce: 0
0.000114395 Eth
Nonce: 0
0.000114395From: 0 To: 0
0x6E85b058...dDAe59DdA 0.00050774563314662 Eth0.00312789563314662 Eth0.00262015
0x772DC254...dBD3Bd2D1 0.000014860766508 Eth0.006223090766508 Eth0.00620823
0x7830c87C...31FA86F43
(Coinbase: Deposit)
111.195657876875881062 Eth
Nonce: 2558559
111.194738753680049886 Eth
Nonce: 2558560
0.000919123195831176
0x8de3C7aE...9bcFfC711 0.000005788863402 Eth0.000195368863402 Eth0.00018958
0x8FdE1F59...0f85D1561
0 Eth
Nonce: 0
0.0000974275 Eth
Nonce: 0
0.0000974275From: 0 To: 0
0x9b451A1d...Fd139A9D3 0.000248757753023 Eth0.005640507753023 Eth0.00539175
0xa0327E8a...43544DE97 5.70765126105600695 Eth5.84508378105600695 Eth0.13743252
0xA9D1e08C...FB81d3E43
(Coinbase 10)
3,208.588801482924861361 Eth3,203.921223571199652491 Eth4.66757791172520887
0xB026E8bD...C49Add124
0 Eth
Nonce: 0
0.000114395 Eth
Nonce: 0
0.000114395From: 0 To: 0
0xB7A5330e...B31feb09a 0 Eth0.00621802 Eth0.00621802
0xbb924498...8e109f693
0 Eth
Nonce: 0
0.000114395 Eth
Nonce: 0
0.000114395From: 0 To: 0
0xBf1beCb9...D3eEa1A84
0 Eth
Nonce: 0
0.02660443 Eth
Nonce: 0
0.02660443From: 0 To: 0
0xc0306b55...5e84c70eE 0.000025196058181891 Eth0.0001155625 Eth0.000090366441818109
0xcEe6856c...66c563b8c
0 Eth
Nonce: 0
0.05175769 Eth
Nonce: 0
0.05175769From: 0 To: 0
0xD66f96e2...b733515db
0 Eth
Nonce: 0
0.00528295 Eth
Nonce: 0
0.00528295From: 0 To: 0
0xd6AB3C3C...ff39feacE
0 Eth
Nonce: 0
0.000114395 Eth
Nonce: 0
0.000114395From: 0 To: 0
0xd776A1Ab...5ccF7A954 0.000057427779306256 Eth0.000114395 Eth0.000056967220693744
(BuilderNet)
10.778492123727128936 Eth10.779083735727128936 Eth0.000591612
0xDFD39685...E97e2e600 0.02140094 Eth1.89759959 Eth1.87619865
0xEcF407ab...7949D919F 0.000087056392384604 Eth0.000114425 Eth0.000027368607615396
0xef33Bd0A...062c6Ff3d
0 Eth
Nonce: 0
0.00026404 Eth
Nonce: 0
0.00026404From: 0 To: 0
0xf82836De...73667B90D 0.004056208863023433 Eth0.030206318863023433 Eth0.02615011
0xFC756271...A1B801eAA 0.00005716145601356 Eth0.000114395 Eth0.00005723354398644
0xfeeDBDcF...7AD55DF51 0.000004940663385 Eth0.003966080663385 Eth0.00396114

Execution Trace

Coinbase 10.1a1da075( )
  • ETH 0.000086703722857899 0x049f274e80f6f859a3041f7d4ae2a93237ab631d.CALL( )
  • ETH 0.06600948 0x51b0390b780ec3afff7bbdf8238a026077930b42.CALL( )
  • ETH 0.000042992713672042 0x0879eccdad24d103c5c06e19131cc94c746bcf88.CALL( )
  • ETH 0.000090366441818109 0xc0306b55c5b673603b6ec499becb2ff5e84c70ee.CALL( )
  • ETH 1.87619865 0xdfd39685cfb8845c5ea689f1c137933e97e2e600.CALL( )
  • ETH 0.00997852 0x31b2921d57e68c7a6c0c2282a22163db46f00f4c.CALL( )
  • ETH 0.00026404 0xef33bd0ad8422f765077e0a2eef348a062c6ff3d.CALL( )
  • ETH 0.02660443 0xbf1becb9ab7433aa03d16cedd0dd363d3eea1a84.CALL( )
  • ETH 0.02615011 0xf82836de6aed5644c6ee6f11a4b5f5d73667b90d.CALL( )
  • ETH 2.37634115 0x06bfc707ff15b554408865994a336f227a7872e1.CALL( )
  • ETH 0.0000974275 0x8fde1f5921be5e31eb94791ae023c1b0f85d1561.CALL( )
  • ETH 0.00528295 0xd66f96e2f6f15a9902826b17e88d614b733515db.CALL( )
  • ETH 0.05175769 0xcee6856c3c83a644e67998ac701eb5466c563b8c.CALL( )
  • ETH 0.13743252 0xea05cdb9733d952fac4a89e636213aadcb50259c.CALL( )
    • ETH 0.13743252 ForwarderV4.DELEGATECALL( )
      • ETH 0.13743252 0xa0327e8a880ac7d6ce47c059204fa8843544de97.CALL( )
      • ETH 0.00621802 0xb7a5330e1f9a65256063c79571d7440b31feb09a.CALL( )
      • ETH 0.00018958 0x8de3c7ae9c53fe4ae3128ae399a32719bcffc711.CALL( )
      • ETH 0.00396114 0xfeedbdcf1b69006b91b9fa67f7c367c7ad55df51.CALL( )
      • ETH 0.00262015 0x6e85b05843f20ba62d5dec213ed5f8eddae59dda.CALL( )
      • ETH 0.06601906 0x5e9d8d1d61b4b638f0bf92f310da4b46d4d515f2.CALL( )
      • ETH 0.00539175 0x9b451a1d7b8032c61e5300f376535cbfd139a9d3.CALL( )
      • ETH 0.00620823 0x772dc2542ec21ac9ba38591035b1adfdbd3bd2d1.CALL( )
      • ETH 0.000027368607615396 0xecf407ab1f40d99104c7ba6904e00c87949d919f.CALL( )
      • ETH 0.00003380197456524 0x0affb9bb4d14aaae6e531d52b0ea4744d9c94d7b.CALL( )
      • ETH 0.000114395 0x661a23d3594115298f9718d86277857379731b56.CALL( )
      • ETH 0.000114395 0xbb924498d9efaeedbfef2f7ef06a5e28e109f693.CALL( )
      • ETH 0.000114395 0xd6ab3c3cb4cb83c2bba5b44efbabeb6ff39feace.CALL( )
      • ETH 0.00005723354398644 0xfc75627183ddfc6e701511ff499e5e9a1b801eaa.CALL( )
      • ETH 0.000114395 0xb026e8bd6ee03d281679a1912bdf741c49add124.CALL( )
      • ETH 0.000056967220693744 0xd776a1ab518865f758da18cdc9361435ccf7a954.CALL( )
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts (last updated v5.0.0) (token/ERC1155/IERC1155.sol)
        pragma solidity ^0.8.20;
        import {IERC165} from "../../utils/introspection/IERC165.sol";
        /**
         * @dev Required interface of an ERC1155 compliant contract, as defined in the
         * https://eips.ethereum.org/EIPS/eip-1155[EIP].
         */
        interface IERC1155 is IERC165 {
            /**
             * @dev Emitted when `value` amount of tokens of type `id` are transferred from `from` to `to` by `operator`.
             */
            event TransferSingle(address indexed operator, address indexed from, address indexed to, uint256 id, uint256 value);
            /**
             * @dev Equivalent to multiple {TransferSingle} events, where `operator`, `from` and `to` are the same for all
             * transfers.
             */
            event TransferBatch(
                address indexed operator,
                address indexed from,
                address indexed to,
                uint256[] ids,
                uint256[] values
            );
            /**
             * @dev Emitted when `account` grants or revokes permission to `operator` to transfer their tokens, according to
             * `approved`.
             */
            event ApprovalForAll(address indexed account, address indexed operator, bool approved);
            /**
             * @dev Emitted when the URI for token type `id` changes to `value`, if it is a non-programmatic URI.
             *
             * If an {URI} event was emitted for `id`, the standard
             * https://eips.ethereum.org/EIPS/eip-1155#metadata-extensions[guarantees] that `value` will equal the value
             * returned by {IERC1155MetadataURI-uri}.
             */
            event URI(string value, uint256 indexed id);
            /**
             * @dev Returns the value of tokens of token type `id` owned by `account`.
             *
             * Requirements:
             *
             * - `account` cannot be the zero address.
             */
            function balanceOf(address account, uint256 id) external view returns (uint256);
            /**
             * @dev xref:ROOT:erc1155.adoc#batch-operations[Batched] version of {balanceOf}.
             *
             * Requirements:
             *
             * - `accounts` and `ids` must have the same length.
             */
            function balanceOfBatch(
                address[] calldata accounts,
                uint256[] calldata ids
            ) external view returns (uint256[] memory);
            /**
             * @dev Grants or revokes permission to `operator` to transfer the caller's tokens, according to `approved`,
             *
             * Emits an {ApprovalForAll} event.
             *
             * Requirements:
             *
             * - `operator` cannot be the caller.
             */
            function setApprovalForAll(address operator, bool approved) external;
            /**
             * @dev Returns true if `operator` is approved to transfer ``account``'s tokens.
             *
             * See {setApprovalForAll}.
             */
            function isApprovedForAll(address account, address operator) external view returns (bool);
            /**
             * @dev Transfers a `value` amount of tokens of type `id` from `from` to `to`.
             *
             * WARNING: This function can potentially allow a reentrancy attack when transferring tokens
             * to an untrusted contract, when invoking {onERC1155Received} on the receiver.
             * Ensure to follow the checks-effects-interactions pattern and consider employing
             * reentrancy guards when interacting with untrusted contracts.
             *
             * Emits a {TransferSingle} event.
             *
             * Requirements:
             *
             * - `to` cannot be the zero address.
             * - If the caller is not `from`, it must have been approved to spend ``from``'s tokens via {setApprovalForAll}.
             * - `from` must have a balance of tokens of type `id` of at least `value` amount.
             * - If `to` refers to a smart contract, it must implement {IERC1155Receiver-onERC1155Received} and return the
             * acceptance magic value.
             */
            function safeTransferFrom(address from, address to, uint256 id, uint256 value, bytes calldata data) external;
            /**
             * @dev xref:ROOT:erc1155.adoc#batch-operations[Batched] version of {safeTransferFrom}.
             *
             *
             * WARNING: This function can potentially allow a reentrancy attack when transferring tokens
             * to an untrusted contract, when invoking {onERC1155BatchReceived} on the receiver.
             * Ensure to follow the checks-effects-interactions pattern and consider employing
             * reentrancy guards when interacting with untrusted contracts.
             *
             * Emits a {TransferBatch} event.
             *
             * Requirements:
             *
             * - `ids` and `values` must have the same length.
             * - If `to` refers to a smart contract, it must implement {IERC1155Receiver-onERC1155BatchReceived} and return the
             * acceptance magic value.
             */
            function safeBatchTransferFrom(
                address from,
                address to,
                uint256[] calldata ids,
                uint256[] calldata values,
                bytes calldata data
            ) external;
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts (last updated v5.0.0) (token/ERC1155/IERC1155Receiver.sol)
        pragma solidity ^0.8.20;
        import {IERC165} from "../../utils/introspection/IERC165.sol";
        /**
         * @dev Interface that must be implemented by smart contracts in order to receive
         * ERC-1155 token transfers.
         */
        interface IERC1155Receiver is IERC165 {
            /**
             * @dev Handles the receipt of a single ERC1155 token type. This function is
             * called at the end of a `safeTransferFrom` after the balance has been updated.
             *
             * NOTE: To accept the transfer, this must return
             * `bytes4(keccak256("onERC1155Received(address,address,uint256,uint256,bytes)"))`
             * (i.e. 0xf23a6e61, or its own function selector).
             *
             * @param operator The address which initiated the transfer (i.e. msg.sender)
             * @param from The address which previously owned the token
             * @param id The ID of the token being transferred
             * @param value The amount of tokens being transferred
             * @param data Additional data with no specified format
             * @return `bytes4(keccak256("onERC1155Received(address,address,uint256,uint256,bytes)"))` if transfer is allowed
             */
            function onERC1155Received(
                address operator,
                address from,
                uint256 id,
                uint256 value,
                bytes calldata data
            ) external returns (bytes4);
            /**
             * @dev Handles the receipt of a multiple ERC1155 token types. This function
             * is called at the end of a `safeBatchTransferFrom` after the balances have
             * been updated.
             *
             * NOTE: To accept the transfer(s), this must return
             * `bytes4(keccak256("onERC1155BatchReceived(address,address,uint256[],uint256[],bytes)"))`
             * (i.e. 0xbc197c81, or its own function selector).
             *
             * @param operator The address which initiated the batch transfer (i.e. msg.sender)
             * @param from The address which previously owned the token
             * @param ids An array containing ids of each token being transferred (order and length must match values array)
             * @param values An array containing amounts of each token being transferred (order and length must match ids array)
             * @param data Additional data with no specified format
             * @return `bytes4(keccak256("onERC1155BatchReceived(address,address,uint256[],uint256[],bytes)"))` if transfer is allowed
             */
            function onERC1155BatchReceived(
                address operator,
                address from,
                uint256[] calldata ids,
                uint256[] calldata values,
                bytes calldata data
            ) external returns (bytes4);
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts (last updated v5.0.0) (token/ERC1155/utils/ERC1155Holder.sol)
        pragma solidity ^0.8.20;
        import {IERC165, ERC165} from "../../../utils/introspection/ERC165.sol";
        import {IERC1155Receiver} from "../IERC1155Receiver.sol";
        /**
         * @dev Simple implementation of `IERC1155Receiver` that will allow a contract to hold ERC1155 tokens.
         *
         * IMPORTANT: When inheriting this contract, you must include a way to use the received tokens, otherwise they will be
         * stuck.
         */
        abstract contract ERC1155Holder is ERC165, IERC1155Receiver {
            /**
             * @dev See {IERC165-supportsInterface}.
             */
            function supportsInterface(bytes4 interfaceId) public view virtual override(ERC165, IERC165) returns (bool) {
                return interfaceId == type(IERC1155Receiver).interfaceId || super.supportsInterface(interfaceId);
            }
            function onERC1155Received(
                address,
                address,
                uint256,
                uint256,
                bytes memory
            ) public virtual override returns (bytes4) {
                return this.onERC1155Received.selector;
            }
            function onERC1155BatchReceived(
                address,
                address,
                uint256[] memory,
                uint256[] memory,
                bytes memory
            ) public virtual override returns (bytes4) {
                return this.onERC1155BatchReceived.selector;
            }
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts (last updated v5.0.0) (token/ERC721/IERC721.sol)
        pragma solidity ^0.8.20;
        import {IERC165} from "../../utils/introspection/IERC165.sol";
        /**
         * @dev Required interface of an ERC721 compliant contract.
         */
        interface IERC721 is IERC165 {
            /**
             * @dev Emitted when `tokenId` token is transferred from `from` to `to`.
             */
            event Transfer(address indexed from, address indexed to, uint256 indexed tokenId);
            /**
             * @dev Emitted when `owner` enables `approved` to manage the `tokenId` token.
             */
            event Approval(address indexed owner, address indexed approved, uint256 indexed tokenId);
            /**
             * @dev Emitted when `owner` enables or disables (`approved`) `operator` to manage all of its assets.
             */
            event ApprovalForAll(address indexed owner, address indexed operator, bool approved);
            /**
             * @dev Returns the number of tokens in ``owner``'s account.
             */
            function balanceOf(address owner) external view returns (uint256 balance);
            /**
             * @dev Returns the owner of the `tokenId` token.
             *
             * Requirements:
             *
             * - `tokenId` must exist.
             */
            function ownerOf(uint256 tokenId) external view returns (address owner);
            /**
             * @dev Safely transfers `tokenId` token from `from` to `to`.
             *
             * Requirements:
             *
             * - `from` cannot be the zero address.
             * - `to` cannot be the zero address.
             * - `tokenId` token must exist and be owned by `from`.
             * - If the caller is not `from`, it must be approved to move this token by either {approve} or {setApprovalForAll}.
             * - If `to` refers to a smart contract, it must implement {IERC721Receiver-onERC721Received}, which is called upon
             *   a safe transfer.
             *
             * Emits a {Transfer} event.
             */
            function safeTransferFrom(address from, address to, uint256 tokenId, bytes calldata data) external;
            /**
             * @dev Safely transfers `tokenId` token from `from` to `to`, checking first that contract recipients
             * are aware of the ERC721 protocol to prevent tokens from being forever locked.
             *
             * Requirements:
             *
             * - `from` cannot be the zero address.
             * - `to` cannot be the zero address.
             * - `tokenId` token must exist and be owned by `from`.
             * - If the caller is not `from`, it must have been allowed to move this token by either {approve} or
             *   {setApprovalForAll}.
             * - If `to` refers to a smart contract, it must implement {IERC721Receiver-onERC721Received}, which is called upon
             *   a safe transfer.
             *
             * Emits a {Transfer} event.
             */
            function safeTransferFrom(address from, address to, uint256 tokenId) external;
            /**
             * @dev Transfers `tokenId` token from `from` to `to`.
             *
             * WARNING: Note that the caller is responsible to confirm that the recipient is capable of receiving ERC721
             * or else they may be permanently lost. Usage of {safeTransferFrom} prevents loss, though the caller must
             * understand this adds an external call which potentially creates a reentrancy vulnerability.
             *
             * Requirements:
             *
             * - `from` cannot be the zero address.
             * - `to` cannot be the zero address.
             * - `tokenId` token must be owned by `from`.
             * - If the caller is not `from`, it must be approved to move this token by either {approve} or {setApprovalForAll}.
             *
             * Emits a {Transfer} event.
             */
            function transferFrom(address from, address to, uint256 tokenId) external;
            /**
             * @dev Gives permission to `to` to transfer `tokenId` token to another account.
             * The approval is cleared when the token is transferred.
             *
             * Only a single account can be approved at a time, so approving the zero address clears previous approvals.
             *
             * Requirements:
             *
             * - The caller must own the token or be an approved operator.
             * - `tokenId` must exist.
             *
             * Emits an {Approval} event.
             */
            function approve(address to, uint256 tokenId) external;
            /**
             * @dev Approve or remove `operator` as an operator for the caller.
             * Operators can call {transferFrom} or {safeTransferFrom} for any token owned by the caller.
             *
             * Requirements:
             *
             * - The `operator` cannot be the address zero.
             *
             * Emits an {ApprovalForAll} event.
             */
            function setApprovalForAll(address operator, bool approved) external;
            /**
             * @dev Returns the account approved for `tokenId` token.
             *
             * Requirements:
             *
             * - `tokenId` must exist.
             */
            function getApproved(uint256 tokenId) external view returns (address operator);
            /**
             * @dev Returns if the `operator` is allowed to manage all of the assets of `owner`.
             *
             * See {setApprovalForAll}
             */
            function isApprovedForAll(address owner, address operator) external view returns (bool);
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts (last updated v5.0.0) (token/ERC721/IERC721Receiver.sol)
        pragma solidity ^0.8.20;
        /**
         * @title ERC721 token receiver interface
         * @dev Interface for any contract that wants to support safeTransfers
         * from ERC721 asset contracts.
         */
        interface IERC721Receiver {
            /**
             * @dev Whenever an {IERC721} `tokenId` token is transferred to this contract via {IERC721-safeTransferFrom}
             * by `operator` from `from`, this function is called.
             *
             * It must return its Solidity selector to confirm the token transfer.
             * If any other value is returned or the interface is not implemented by the recipient, the transfer will be
             * reverted.
             *
             * The selector can be obtained in Solidity with `IERC721Receiver.onERC721Received.selector`.
             */
            function onERC721Received(
                address operator,
                address from,
                uint256 tokenId,
                bytes calldata data
            ) external returns (bytes4);
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts (last updated v5.0.0) (utils/Address.sol)
        pragma solidity ^0.8.20;
        /**
         * @dev Collection of functions related to the address type
         */
        library Address {
            /**
             * @dev The ETH balance of the account is not enough to perform the operation.
             */
            error AddressInsufficientBalance(address account);
            /**
             * @dev There's no code at `target` (it is not a contract).
             */
            error AddressEmptyCode(address target);
            /**
             * @dev A call to an address target failed. The target may have reverted.
             */
            error FailedInnerCall();
            /**
             * @dev Replacement for Solidity's `transfer`: sends `amount` wei to
             * `recipient`, forwarding all available gas and reverting on errors.
             *
             * https://eips.ethereum.org/EIPS/eip-1884[EIP1884] increases the gas cost
             * of certain opcodes, possibly making contracts go over the 2300 gas limit
             * imposed by `transfer`, making them unable to receive funds via
             * `transfer`. {sendValue} removes this limitation.
             *
             * https://consensys.net/diligence/blog/2019/09/stop-using-soliditys-transfer-now/[Learn more].
             *
             * IMPORTANT: because control is transferred to `recipient`, care must be
             * taken to not create reentrancy vulnerabilities. Consider using
             * {ReentrancyGuard} or the
             * https://solidity.readthedocs.io/en/v0.8.20/security-considerations.html#use-the-checks-effects-interactions-pattern[checks-effects-interactions pattern].
             */
            function sendValue(address payable recipient, uint256 amount) internal {
                if (address(this).balance < amount) {
                    revert AddressInsufficientBalance(address(this));
                }
                (bool success, ) = recipient.call{value: amount}("");
                if (!success) {
                    revert FailedInnerCall();
                }
            }
            /**
             * @dev Performs a Solidity function call using a low level `call`. A
             * plain `call` is an unsafe replacement for a function call: use this
             * function instead.
             *
             * If `target` reverts with a revert reason or custom error, it is bubbled
             * up by this function (like regular Solidity function calls). However, if
             * the call reverted with no returned reason, this function reverts with a
             * {FailedInnerCall} error.
             *
             * Returns the raw returned data. To convert to the expected return value,
             * use https://solidity.readthedocs.io/en/latest/units-and-global-variables.html?highlight=abi.decode#abi-encoding-and-decoding-functions[`abi.decode`].
             *
             * Requirements:
             *
             * - `target` must be a contract.
             * - calling `target` with `data` must not revert.
             */
            function functionCall(address target, bytes memory data) internal returns (bytes memory) {
                return functionCallWithValue(target, data, 0);
            }
            /**
             * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
             * but also transferring `value` wei to `target`.
             *
             * Requirements:
             *
             * - the calling contract must have an ETH balance of at least `value`.
             * - the called Solidity function must be `payable`.
             */
            function functionCallWithValue(address target, bytes memory data, uint256 value) internal returns (bytes memory) {
                if (address(this).balance < value) {
                    revert AddressInsufficientBalance(address(this));
                }
                (bool success, bytes memory returndata) = target.call{value: value}(data);
                return verifyCallResultFromTarget(target, success, returndata);
            }
            /**
             * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
             * but performing a static call.
             */
            function functionStaticCall(address target, bytes memory data) internal view returns (bytes memory) {
                (bool success, bytes memory returndata) = target.staticcall(data);
                return verifyCallResultFromTarget(target, success, returndata);
            }
            /**
             * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
             * but performing a delegate call.
             */
            function functionDelegateCall(address target, bytes memory data) internal returns (bytes memory) {
                (bool success, bytes memory returndata) = target.delegatecall(data);
                return verifyCallResultFromTarget(target, success, returndata);
            }
            /**
             * @dev Tool to verify that a low level call to smart-contract was successful, and reverts if the target
             * was not a contract or bubbling up the revert reason (falling back to {FailedInnerCall}) in case of an
             * unsuccessful call.
             */
            function verifyCallResultFromTarget(
                address target,
                bool success,
                bytes memory returndata
            ) internal view returns (bytes memory) {
                if (!success) {
                    _revert(returndata);
                } else {
                    // only check if target is a contract if the call was successful and the return data is empty
                    // otherwise we already know that it was a contract
                    if (returndata.length == 0 && target.code.length == 0) {
                        revert AddressEmptyCode(target);
                    }
                    return returndata;
                }
            }
            /**
             * @dev Tool to verify that a low level call was successful, and reverts if it wasn't, either by bubbling the
             * revert reason or with a default {FailedInnerCall} error.
             */
            function verifyCallResult(bool success, bytes memory returndata) internal pure returns (bytes memory) {
                if (!success) {
                    _revert(returndata);
                } else {
                    return returndata;
                }
            }
            /**
             * @dev Reverts with returndata if present. Otherwise reverts with {FailedInnerCall}.
             */
            function _revert(bytes memory returndata) private pure {
                // Look for revert reason and bubble it up if present
                if (returndata.length > 0) {
                    // The easiest way to bubble the revert reason is using memory via assembly
                    /// @solidity memory-safe-assembly
                    assembly {
                        let returndata_size := mload(returndata)
                        revert(add(32, returndata), returndata_size)
                    }
                } else {
                    revert FailedInnerCall();
                }
            }
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts (last updated v5.0.0) (utils/introspection/ERC165.sol)
        pragma solidity ^0.8.20;
        import {IERC165} from "./IERC165.sol";
        /**
         * @dev Implementation of the {IERC165} interface.
         *
         * Contracts that want to implement ERC165 should inherit from this contract and override {supportsInterface} to check
         * for the additional interface id that will be supported. For example:
         *
         * ```solidity
         * function supportsInterface(bytes4 interfaceId) public view virtual override returns (bool) {
         *     return interfaceId == type(MyInterface).interfaceId || super.supportsInterface(interfaceId);
         * }
         * ```
         */
        abstract contract ERC165 is IERC165 {
            /**
             * @dev See {IERC165-supportsInterface}.
             */
            function supportsInterface(bytes4 interfaceId) public view virtual returns (bool) {
                return interfaceId == type(IERC165).interfaceId;
            }
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts (last updated v5.0.0) (utils/introspection/IERC165.sol)
        pragma solidity ^0.8.20;
        /**
         * @dev Interface of the ERC165 standard, as defined in the
         * https://eips.ethereum.org/EIPS/eip-165[EIP].
         *
         * Implementers can declare support of contract interfaces, which can then be
         * queried by others ({ERC165Checker}).
         *
         * For an implementation, see {ERC165}.
         */
        interface IERC165 {
            /**
             * @dev Returns true if this contract implements the interface defined by
             * `interfaceId`. See the corresponding
             * https://eips.ethereum.org/EIPS/eip-165#how-interfaces-are-identified[EIP section]
             * to learn more about how these ids are created.
             *
             * This function call must use less than 30 000 gas.
             */
            function supportsInterface(bytes4 interfaceId) external view returns (bool);
        }
        // SPDX-License-Identifier: UNLICENSED
        pragma solidity 0.8.20;
        /**
         * Contract that exposes the needed erc20 token functions
         */
        abstract contract ERC20Interface {
          // Send _value amount of tokens to address _to
          function transfer(address _to, uint256 _value)
            public
            virtual
            returns (bool success);
          // Get the account balance of another account with address _owner
          function balanceOf(address _owner)
            public
            view
            virtual
            returns (uint256 balance);
        }
        // SPDX-License-Identifier: Apache-2.0
        pragma solidity 0.8.20;
        import '@openzeppelin/contracts/token/ERC1155/IERC1155.sol';
        import '@openzeppelin/contracts/token/ERC721/IERC721.sol';
        import '@openzeppelin/contracts/token/ERC721/IERC721Receiver.sol';
        import '@openzeppelin/contracts/token/ERC1155/utils/ERC1155Holder.sol';
        import './ERC20Interface.sol';
        import './TransferHelper.sol';
        import './IForwarderV4.sol';
        /**
         * @title ForwarderV4
         * @notice This contract will forward any incoming Ether or token to the parent address of the contract
         */
        contract ForwarderV4 is IERC721Receiver, ERC1155Holder, IForwarderV4 {
          /// @notice Any funds sent to this contract will be forwarded to this address
          address public parentAddress;
          /// @notice Address which is allowed to call methods on this contract alongwith the parentAddress
          address public feeAddress;
          bool public autoFlush721 = true;
          bool public autoFlush1155 = true;
          /**
           * @notice Event triggered when a deposit is received in the forwarder
           * @param from Address from which the deposit is received
           * @param value Amount of Ether received
           * @param data Data sent along with the deposit
           */
          event ForwarderDeposited(address from, uint256 value, bytes data);
          /**
           * @notice Modifier that will execute internal code block only if the sender is from the allowed addresses
           */
          modifier onlyAllowedAddress() {
            require(
              msg.sender == parentAddress || msg.sender == feeAddress,
              'Address is not allowed'
            );
            _;
          }
          /**
           * @notice Modifier that will execute internal code block only if the contract has not been initialized yet
           */
          modifier onlyUninitialized() {
            require(parentAddress == address(0x0), 'Already initialized');
            _;
          }
          /**
           * @notice Default function; Gets called when Ether is deposited with no data, and forwards it to the parent address
           */
          receive() external payable {
            flush();
          }
          /**
           * @notice Default function; Gets called when data is sent but does not match any other function
           */
          fallback() external payable {
            flush();
          }
          /**
           * @notice Initialize the contract, and sets the destination address to that of the parent address
           * @param _parentAddress Address to which the funds should be forwarded
           * @param _feeAddress Address which is allowed to call methods on this contract alongwith the parentAddress
           * @param _autoFlush721 Whether to automatically flush ERC721 tokens or not
           * @param _autoFlush1155 Whether to automatically flush ERC1155 tokens or not
           */
          function init(
            address _parentAddress,
            address _feeAddress,
            bool _autoFlush721,
            bool _autoFlush1155
          ) external onlyUninitialized {
            require(_parentAddress != address(0x0), 'Invalid parent address');
            parentAddress = _parentAddress;
            require(_feeAddress != address(0x0), 'Invalid fee address');
            feeAddress = _feeAddress;
            uint256 value = address(this).balance;
            /// @notice set whether we want to automatically flush erc721/erc1155 tokens or not
            autoFlush721 = _autoFlush721;
            autoFlush1155 = _autoFlush1155;
            if (value == 0) {
              return;
            }
            /**
             * Since we are forwarding on initialization,
             * we don't have the context of the original sender.
             * We still emit an event about the forwarding but set
             * the sender to the forwarder itself
             */
            emit ForwarderDeposited(address(this), value, msg.data);
            (bool success, ) = parentAddress.call{ value: value }('');
            require(success, 'Flush failed');
          }
          /**
           * @inheritdoc IForwarderV4
           */
          function setAutoFlush721(bool autoFlush)
            external
            virtual
            override
            onlyAllowedAddress
          {
            autoFlush721 = autoFlush;
          }
          /**
           * @inheritdoc IForwarderV4
           */
          function setAutoFlush1155(bool autoFlush)
            external
            virtual
            override
            onlyAllowedAddress
          {
            autoFlush1155 = autoFlush;
          }
          /**
           * ERC721 standard callback function for when a ERC721 is transfered. The forwarder will send the nft
           * to the base wallet once the nft contract invokes this method after transfering the nft.
           *
           * @param _operator The address which called `safeTransferFrom` function
           * @param _from The address of the sender
           * @param _tokenId The token id of the nft
           * @param data Additional data with no specified format, sent in call to `_to`
           */
          function onERC721Received(
            address _operator,
            address _from,
            uint256 _tokenId,
            bytes memory data
          ) external virtual override returns (bytes4) {
            if (autoFlush721) {
              IERC721 instance = IERC721(msg.sender);
              require(
                instance.supportsInterface(type(IERC721).interfaceId),
                'The caller does not support the ERC721 interface'
              );
              /// this won't work for ERC721 re-entrancy
              instance.safeTransferFrom(address(this), parentAddress, _tokenId, data);
            }
            return this.onERC721Received.selector;
          }
          /**
           * @notice Method to allow for calls to other contracts. This method can only be called by the parent address
           * @param target The target contract address whose method needs to be called
           * @param value The amount of Ether to be sent
           * @param data The calldata to be sent
           */
          function callFromParent(
            address target,
            uint256 value,
            bytes calldata data
          ) external returns (bytes memory) {
            require(msg.sender == parentAddress, 'Only Parent');
            (bool success, bytes memory returnedData) = target.call{ value: value }(
              data
            );
            require(success, 'Parent call execution failed');
            return returnedData;
          }
          /**
           * @inheritdoc ERC1155Holder
           */
          function onERC1155Received(
            address _operator,
            address _from,
            uint256 id,
            uint256 value,
            bytes memory data
          ) public virtual override returns (bytes4) {
            IERC1155 instance = IERC1155(msg.sender);
            require(
              instance.supportsInterface(type(IERC1155).interfaceId),
              'The caller does not support the IERC1155 interface'
            );
            if (autoFlush1155) {
              instance.safeTransferFrom(address(this), parentAddress, id, value, data);
            }
            return this.onERC1155Received.selector;
          }
          /**
           * @inheritdoc ERC1155Holder
           */
          function onERC1155BatchReceived(
            address _operator,
            address _from,
            uint256[] memory ids,
            uint256[] memory values,
            bytes memory data
          ) public virtual override returns (bytes4) {
            IERC1155 instance = IERC1155(msg.sender);
            require(
              instance.supportsInterface(type(IERC1155).interfaceId),
              'The caller does not support the IERC1155 interface'
            );
            if (autoFlush1155) {
              instance.safeBatchTransferFrom(
                address(this),
                parentAddress,
                ids,
                values,
                data
              );
            }
            return this.onERC1155BatchReceived.selector;
          }
          /**
           * @inheritdoc IForwarderV4
           */
          function flushTokens(address tokenContractAddress)
            external
            virtual
            override
            onlyAllowedAddress
          {
            ERC20Interface instance = ERC20Interface(tokenContractAddress);
            address forwarderAddress = address(this);
            uint256 forwarderBalance = instance.balanceOf(forwarderAddress);
            if (forwarderBalance == 0) {
              return;
            }
            TransferHelper.safeTransfer(
              tokenContractAddress,
              parentAddress,
              forwarderBalance
            );
          }
          /**
           * @inheritdoc IForwarderV4
           */
          function flushERC721Token(address tokenContractAddress, uint256 tokenId)
            external
            virtual
            override
            onlyAllowedAddress
          {
            IERC721 instance = IERC721(tokenContractAddress);
            require(
              instance.supportsInterface(type(IERC721).interfaceId),
              'The tokenContractAddress does not support the ERC721 interface'
            );
            address ownerAddress = instance.ownerOf(tokenId);
            instance.transferFrom(ownerAddress, parentAddress, tokenId);
          }
          /**
           * @inheritdoc IForwarderV4
           */
          function flushERC1155Tokens(address tokenContractAddress, uint256 tokenId)
            external
            virtual
            override
            onlyAllowedAddress
          {
            IERC1155 instance = IERC1155(tokenContractAddress);
            require(
              instance.supportsInterface(type(IERC1155).interfaceId),
              'The caller does not support the IERC1155 interface'
            );
            address forwarderAddress = address(this);
            uint256 forwarderBalance = instance.balanceOf(forwarderAddress, tokenId);
            instance.safeTransferFrom(
              forwarderAddress,
              parentAddress,
              tokenId,
              forwarderBalance,
              ''
            );
          }
          /**
           * @inheritdoc IForwarderV4
           */
          function batchFlushERC1155Tokens(
            address tokenContractAddress,
            uint256[] calldata tokenIds
          ) external virtual override onlyAllowedAddress {
            IERC1155 instance = IERC1155(tokenContractAddress);
            require(
              instance.supportsInterface(type(IERC1155).interfaceId),
              'The caller does not support the IERC1155 interface'
            );
            address forwarderAddress = address(this);
            uint256 length = tokenIds.length;
            uint256[] memory amounts = new uint256[](tokenIds.length);
            for (uint256 i; i < length; i++) {
              amounts[i] = instance.balanceOf(forwarderAddress, tokenIds[i]);
            }
            instance.safeBatchTransferFrom(
              forwarderAddress,
              parentAddress,
              tokenIds,
              amounts,
              ''
            );
          }
          /**
           * @inheritdoc IForwarderV4
           */
          function batchFlushERC20Tokens(address[] calldata tokenContractAddresses)
            external
            virtual
            override
            onlyAllowedAddress
          {
            address forwarderAddress = address(this);
            uint256 length = tokenContractAddresses.length;
            for (uint256 i; i < length; i++) {
              ERC20Interface instance = ERC20Interface(tokenContractAddresses[i]);
              uint256 forwarderBalance = instance.balanceOf(forwarderAddress);
              if (forwarderBalance == 0) {
                continue;
              }
              TransferHelper.safeTransfer(
                tokenContractAddresses[i],
                parentAddress,
                forwarderBalance
              );
            }
          }
          /**
           * @notice Flush the entire balance of the contract to the parent address.
           */
          function flush() public {
            uint256 value = address(this).balance;
            if (value == 0) {
              return;
            }
            emit ForwarderDeposited(msg.sender, value, msg.data);
            (bool success, ) = parentAddress.call{ value: value }('');
            require(success, 'Flush failed');
          }
          /**
           * @inheritdoc IERC165
           */
          function supportsInterface(bytes4 interfaceId)
            public
            view
            virtual
            override(ERC1155Holder, IERC165)
            returns (bool)
          {
            return
              interfaceId == type(IForwarderV4).interfaceId ||
              super.supportsInterface(interfaceId);
          }
        }
        // SPDX-License-Identifier: Apache-2.0
        pragma solidity 0.8.20;
        import '@openzeppelin/contracts/utils/introspection/IERC165.sol';
        interface IForwarderV4 is IERC165 {
          /**
           * Sets the autoflush721 parameter.
           *
           * @param autoFlush whether to autoflush erc721 tokens
           */
          function setAutoFlush721(bool autoFlush) external;
          /**
           * Sets the autoflush1155 parameter.
           *
           * @param autoFlush whether to autoflush erc1155 tokens
           */
          function setAutoFlush1155(bool autoFlush) external;
          /**
           * Execute a token transfer of the full balance from the forwarder to the parent address
           *
           * @param tokenContractAddress the address of the erc20 token contract
           */
          function flushTokens(address tokenContractAddress) external;
          /**
           * Execute a nft transfer from the forwarder to the parent address
           *
           * @param tokenContractAddress the address of the ERC721 NFT contract
           * @param tokenId The token id of the nft
           */
          function flushERC721Token(address tokenContractAddress, uint256 tokenId)
            external;
          /**
           * Execute a nft transfer from the forwarder to the parent address.
           *
           * @param tokenContractAddress the address of the ERC1155 NFT contract
           * @param tokenId The token id of the nft
           */
          function flushERC1155Tokens(address tokenContractAddress, uint256 tokenId)
            external;
          /**
           * Execute a batch nft transfer from the forwarder to the parent address.
           *
           * @param tokenContractAddress the address of the ERC1155 NFT contract
           * @param tokenIds The token ids of the nfts
           */
          function batchFlushERC1155Tokens(
            address tokenContractAddress,
            uint256[] calldata tokenIds
          ) external;
          /**
           * Execute a batch erc20 transfer from the forwarder to the parent address.
           *
           * @param tokenContractAddresses the addresses of the ERC20 token contracts
           */
          function batchFlushERC20Tokens(address[] calldata tokenContractAddresses)
            external;
        }
        // SPDX-License-Identifier: GPL-3.0-or-later
        // source: https://github.com/Uniswap/solidity-lib/blob/master/contracts/libraries/TransferHelper.sol
        pragma solidity 0.8.20;
        import '@openzeppelin/contracts/utils/Address.sol';
        // helper methods for interacting with ERC20 tokens and sending ETH that do not consistently return true/false
        library TransferHelper {
          function safeTransfer(
            address token,
            address to,
            uint256 value
          ) internal {
            // bytes4(keccak256(bytes('transfer(address,uint256)')));
            (bool success, bytes memory data) = token.call(
              abi.encodeWithSelector(0xa9059cbb, to, value)
            );
            require(
              success && (data.length == 0 || abi.decode(data, (bool))),
              'TransferHelper::safeTransfer: transfer failed'
            );
          }
          function safeTransferFrom(
            address token,
            address from,
            address to,
            uint256 value
          ) internal {
            // bytes4(keccak256(bytes('transferFrom(address,address,uint256)')));
            (bool success, bytes memory returndata) = token.call(
              abi.encodeWithSelector(0x23b872dd, from, to, value)
            );
            Address.verifyCallResult(success, returndata);
          }
        }